datasources.go 6.9 KB

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