datasources.go 7.6 KB

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