datasources.go 6.6 KB

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