datasources.go 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. package api
  2. import (
  3. "sort"
  4. "github.com/grafana/grafana/pkg/api/dtos"
  5. "github.com/grafana/grafana/pkg/bus"
  6. m "github.com/grafana/grafana/pkg/models"
  7. "github.com/grafana/grafana/pkg/plugins"
  8. "github.com/grafana/grafana/pkg/util"
  9. )
  10. func GetDataSources(c *m.ReqContext) Response {
  11. query := m.GetDataSourcesQuery{OrgId: c.OrgId}
  12. if err := bus.Dispatch(&query); err != nil {
  13. return Error(500, "Failed to query datasources", err)
  14. }
  15. dsFilterQuery := m.DatasourcesPermissionFilterQuery{
  16. User: c.SignedInUser,
  17. Datasources: query.Result,
  18. }
  19. if err := bus.Dispatch(&dsFilterQuery); err != nil {
  20. if err != bus.ErrHandlerNotFound {
  21. return Error(500, "Could not get datasources", err)
  22. }
  23. dsFilterQuery.Result = query.Result
  24. }
  25. result := make(dtos.DataSourceList, 0)
  26. for _, ds := range dsFilterQuery.Result {
  27. dsItem := dtos.DataSourceListItemDTO{
  28. OrgId: ds.OrgId,
  29. Id: ds.Id,
  30. Name: ds.Name,
  31. Url: ds.Url,
  32. Type: ds.Type,
  33. Access: ds.Access,
  34. Password: ds.Password,
  35. Database: ds.Database,
  36. User: ds.User,
  37. BasicAuth: ds.BasicAuth,
  38. IsDefault: ds.IsDefault,
  39. JsonData: ds.JsonData,
  40. ReadOnly: ds.ReadOnly,
  41. }
  42. if plugin, exists := plugins.DataSources[ds.Type]; exists {
  43. dsItem.TypeLogoUrl = plugin.Info.Logos.Small
  44. } else {
  45. dsItem.TypeLogoUrl = "public/img/icn-datasource.svg"
  46. }
  47. result = append(result, dsItem)
  48. }
  49. sort.Sort(result)
  50. return JSON(200, &result)
  51. }
  52. func hasRequiredDatasourcePermission(dsId int64, permission m.DsPermissionType, user *m.SignedInUser) Response {
  53. query := m.HasRequiredDataSourcePermissionQuery{
  54. Id: dsId,
  55. User: user,
  56. RequiredPermission: permission,
  57. }
  58. if err := bus.Dispatch(&query); err != nil {
  59. if err == bus.ErrHandlerNotFound {
  60. return nil
  61. }
  62. if err == m.ErrDataSourceAccessDenied {
  63. return Error(403, err.Error(), nil)
  64. }
  65. return Error(500, "Failed to check data source permissions", err)
  66. }
  67. return nil
  68. }
  69. func GetDataSourceById(c *m.ReqContext) Response {
  70. query := m.GetDataSourceByIdQuery{
  71. Id: c.ParamsInt64(":id"),
  72. OrgId: c.OrgId,
  73. }
  74. if err := bus.Dispatch(&query); err != nil {
  75. if err == m.ErrDataSourceNotFound {
  76. return Error(404, "Data source not found", nil)
  77. }
  78. return Error(500, "Failed to query datasources", err)
  79. }
  80. ds := query.Result
  81. dtos := convertModelToDtos(ds)
  82. return JSON(200, &dtos)
  83. }
  84. func DeleteDataSourceById(c *m.ReqContext) Response {
  85. id := c.ParamsInt64(":id")
  86. if id <= 0 {
  87. return Error(400, "Missing valid datasource id", nil)
  88. }
  89. ds, err := getRawDataSourceById(id, c.OrgId)
  90. if err != nil {
  91. return Error(400, "Failed to delete datasource", nil)
  92. }
  93. if ds.ReadOnly {
  94. return Error(403, "Cannot delete read-only data source", nil)
  95. }
  96. cmd := &m.DeleteDataSourceByIdCommand{Id: id, OrgId: c.OrgId}
  97. err = bus.Dispatch(cmd)
  98. if err != nil {
  99. return Error(500, "Failed to delete datasource", err)
  100. }
  101. return Success("Data source deleted")
  102. }
  103. func DeleteDataSourceByName(c *m.ReqContext) Response {
  104. name := c.Params(":name")
  105. if name == "" {
  106. return Error(400, "Missing valid datasource name", nil)
  107. }
  108. getCmd := &m.GetDataSourceByNameQuery{Name: name, OrgId: c.OrgId}
  109. if err := bus.Dispatch(getCmd); err != nil {
  110. if err == m.ErrDataSourceNotFound {
  111. return Error(404, "Data source not found", nil)
  112. }
  113. return Error(500, "Failed to delete datasource", err)
  114. }
  115. if getCmd.Result.ReadOnly {
  116. return Error(403, "Cannot delete read-only data source", nil)
  117. }
  118. cmd := &m.DeleteDataSourceByNameCommand{Name: name, OrgId: c.OrgId}
  119. err := bus.Dispatch(cmd)
  120. if err != nil {
  121. return Error(500, "Failed to delete datasource", err)
  122. }
  123. return Success("Data source deleted")
  124. }
  125. func AddDataSource(c *m.ReqContext, cmd m.AddDataSourceCommand) Response {
  126. cmd.OrgId = c.OrgId
  127. if err := bus.Dispatch(&cmd); err != nil {
  128. if err == m.ErrDataSourceNameExists {
  129. return Error(409, err.Error(), err)
  130. }
  131. return Error(500, "Failed to add datasource", err)
  132. }
  133. ds := convertModelToDtos(cmd.Result)
  134. return JSON(200, util.DynMap{
  135. "message": "Datasource added",
  136. "id": cmd.Result.Id,
  137. "name": cmd.Result.Name,
  138. "datasource": ds,
  139. })
  140. }
  141. func UpdateDataSource(c *m.ReqContext, cmd m.UpdateDataSourceCommand) Response {
  142. cmd.OrgId = c.OrgId
  143. cmd.Id = c.ParamsInt64(":id")
  144. err := fillWithSecureJSONData(&cmd)
  145. if err != nil {
  146. return Error(500, "Failed to update datasource", err)
  147. }
  148. err = bus.Dispatch(&cmd)
  149. if err != nil {
  150. if err == m.ErrDataSourceUpdatingOldVersion {
  151. return Error(500, "Failed to update datasource. Reload new version and try again", err)
  152. }
  153. return Error(500, "Failed to update datasource", err)
  154. }
  155. query := m.GetDataSourceByIdQuery{
  156. Id: cmd.Id,
  157. OrgId: c.OrgId,
  158. }
  159. if err := bus.Dispatch(&query); err != nil {
  160. if err == m.ErrDataSourceNotFound {
  161. return Error(404, "Data source not found", nil)
  162. }
  163. return Error(500, "Failed to query datasources", err)
  164. }
  165. dtos := convertModelToDtos(query.Result)
  166. return JSON(200, util.DynMap{
  167. "message": "Datasource updated",
  168. "id": cmd.Id,
  169. "name": cmd.Name,
  170. "datasource": dtos,
  171. })
  172. }
  173. func fillWithSecureJSONData(cmd *m.UpdateDataSourceCommand) error {
  174. if len(cmd.SecureJsonData) == 0 {
  175. return nil
  176. }
  177. ds, err := getRawDataSourceById(cmd.Id, cmd.OrgId)
  178. if err != nil {
  179. return err
  180. }
  181. if ds.ReadOnly {
  182. return m.ErrDatasourceIsReadOnly
  183. }
  184. secureJSONData := ds.SecureJsonData.Decrypt()
  185. for k, v := range secureJSONData {
  186. if _, ok := cmd.SecureJsonData[k]; !ok {
  187. cmd.SecureJsonData[k] = v
  188. }
  189. }
  190. return nil
  191. }
  192. func getRawDataSourceById(id int64, orgID int64) (*m.DataSource, error) {
  193. query := m.GetDataSourceByIdQuery{
  194. Id: id,
  195. OrgId: orgID,
  196. }
  197. if err := bus.Dispatch(&query); err != nil {
  198. return nil, err
  199. }
  200. return query.Result, nil
  201. }
  202. // Get /api/datasources/name/:name
  203. func GetDataSourceByName(c *m.ReqContext) Response {
  204. query := m.GetDataSourceByNameQuery{Name: c.Params(":name"), OrgId: c.OrgId}
  205. if err := bus.Dispatch(&query); err != nil {
  206. if err == m.ErrDataSourceNotFound {
  207. return Error(404, "Data source not found", nil)
  208. }
  209. return Error(500, "Failed to query datasources", err)
  210. }
  211. dtos := convertModelToDtos(query.Result)
  212. dtos.ReadOnly = true
  213. return JSON(200, &dtos)
  214. }
  215. // Get /api/datasources/id/:name
  216. func GetDataSourceIdByName(c *m.ReqContext) Response {
  217. query := m.GetDataSourceByNameQuery{Name: c.Params(":name"), OrgId: c.OrgId}
  218. if err := bus.Dispatch(&query); err != nil {
  219. if err == m.ErrDataSourceNotFound {
  220. return Error(404, "Data source not found", nil)
  221. }
  222. return Error(500, "Failed to query datasources", err)
  223. }
  224. ds := query.Result
  225. dtos := dtos.AnyId{
  226. Id: ds.Id,
  227. }
  228. return JSON(200, &dtos)
  229. }
  230. func convertModelToDtos(ds *m.DataSource) dtos.DataSource {
  231. dto := dtos.DataSource{
  232. Id: ds.Id,
  233. OrgId: ds.OrgId,
  234. Name: ds.Name,
  235. Url: ds.Url,
  236. Type: ds.Type,
  237. Access: ds.Access,
  238. Password: ds.Password,
  239. Database: ds.Database,
  240. User: ds.User,
  241. BasicAuth: ds.BasicAuth,
  242. BasicAuthUser: ds.BasicAuthUser,
  243. BasicAuthPassword: ds.BasicAuthPassword,
  244. WithCredentials: ds.WithCredentials,
  245. IsDefault: ds.IsDefault,
  246. JsonData: ds.JsonData,
  247. SecureJsonFields: map[string]bool{},
  248. Version: ds.Version,
  249. ReadOnly: ds.ReadOnly,
  250. }
  251. for k, v := range ds.SecureJsonData {
  252. if len(v) > 0 {
  253. dto.SecureJsonFields[k] = true
  254. }
  255. }
  256. return dto
  257. }