Tool to help you manage your Grafana dashboards using Git.

dashboards.go 4.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. package grafana
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. )
  6. // dbSearchResponse represents an element of the response to a dashboard search
  7. // query
  8. type dbSearchResponse struct {
  9. ID int `json:"id"`
  10. Title string `json:"title"`
  11. URI string `json:"uri"`
  12. Type string `json:"type"`
  13. Tags []string `json:"tags"`
  14. Starred bool `json:"isStarred"`
  15. }
  16. // dbCreateOrUpdateRequest represents the request sent to create or update a
  17. // dashboard
  18. type dbCreateOrUpdateRequest struct {
  19. Dashboard rawJSON `json:"dashboard"`
  20. Overwrite bool `json:"overwrite"`
  21. }
  22. // dbCreateOrUpdateResponse represents the response sent by the Grafana API to
  23. // a dashboard creation or update. All fields described from the Grafana
  24. // documentation aren't located in this structure because there are some we
  25. // don't need.
  26. type dbCreateOrUpdateResponse struct {
  27. Status string `json:"success"`
  28. Version int `json:"version,omitempty"`
  29. Message string `json:"message,omitempty"`
  30. }
  31. // Dashboard represents a Grafana dashboard, with its JSON definition, slug and
  32. // current version.
  33. type Dashboard struct {
  34. RawJSON []byte
  35. Slug string
  36. Version int
  37. }
  38. // UnmarshalJSON tells the JSON parser how to unmarshal JSON data into an
  39. // instance of the Dashboard structure.
  40. // Returns an error if there was an issue unmarshalling the JSON.
  41. func (d *Dashboard) UnmarshalJSON(b []byte) (err error) {
  42. // Define the structure of what we want to parse
  43. var body struct {
  44. Dashboard rawJSON `json:"dashboard"`
  45. Meta struct {
  46. Slug string `json:"slug"`
  47. Version int `json:"version"`
  48. } `json:"meta"`
  49. }
  50. // Unmarshal the JSON into the newly defined structure
  51. if err = json.Unmarshal(b, &body); err != nil {
  52. return
  53. }
  54. // Define all fields with their corresponding value.
  55. d.Slug = body.Meta.Slug
  56. d.Version = body.Meta.Version
  57. d.RawJSON = body.Dashboard
  58. return
  59. }
  60. // GetDashboardsURIs requests the Grafana API for the list of all dashboards,
  61. // then returns the dashboards' URIs. An URI will look like "db/[dashboard slug]".
  62. // Returns an error if there was an issue requesting the URIs or parsing the
  63. // response body.
  64. func (c *Client) GetDashboardsURIs() (URIs []string, err error) {
  65. resp, err := c.request("GET", "search", nil)
  66. var respBody []dbSearchResponse
  67. if err = json.Unmarshal(resp, &respBody); err != nil {
  68. return
  69. }
  70. URIs = make([]string, 0)
  71. for _, db := range respBody {
  72. URIs = append(URIs, db.URI)
  73. }
  74. return
  75. }
  76. // GetDashboard requests the Grafana API for a dashboard identified by a given
  77. // URI (using the same format as GetDashboardsURIs).
  78. // Returns the dashboard as an instance of the Dashboard structure.
  79. // Returns an error if there was an issue requesting the dashboard or parsing
  80. // the response body.
  81. func (c *Client) GetDashboard(URI string) (db *Dashboard, err error) {
  82. body, err := c.request("GET", "dashboards/"+URI, nil)
  83. if err != nil {
  84. return
  85. }
  86. db = new(Dashboard)
  87. err = json.Unmarshal(body, db)
  88. return
  89. }
  90. // CreateOrUpdateDashboard takes a given JSON content (as []byte) and create the
  91. // dashboard if it doesn't exist on the Grafana instance, else updates the
  92. // existing one. The Grafana API decides whether to create or update based on the
  93. // "id" attribute in the dashboard's JSON: If it's unkown or null, it's a
  94. // creation, else it's an update.
  95. // The slug is only used for error reporting, and can differ from the
  96. // dashboard's actual slug in its JSON content. However, it's recommended to use
  97. // the same in both places.
  98. // Returns an error if there was an issue generating the request body, performing
  99. // the request or decoding the response's body.
  100. func (c *Client) CreateOrUpdateDashboard(slug string, contentJSON []byte) (err error) {
  101. reqBody := dbCreateOrUpdateRequest{
  102. Dashboard: rawJSON(contentJSON),
  103. Overwrite: true,
  104. }
  105. reqBodyJSON, err := json.Marshal(reqBody)
  106. if err != nil {
  107. return
  108. }
  109. var httpError *httpUnkownError
  110. var isHttpUnknownError bool
  111. respBodyJSON, err := c.request("POST", "dashboards/db", reqBodyJSON)
  112. if err != nil {
  113. httpError, isHttpUnknownError = err.(*httpUnkownError)
  114. // We process httpUnkownError errors below, after we decoded the body
  115. if !isHttpUnknownError {
  116. return
  117. }
  118. }
  119. var respBody dbCreateOrUpdateResponse
  120. if err = json.Unmarshal(respBodyJSON, &respBody); err != nil {
  121. return
  122. }
  123. if respBody.Status != "success" && isHttpUnknownError {
  124. return fmt.Errorf(
  125. "Failed to update dashboard %s (%d %s): %s",
  126. slug, httpError.StatusCode, respBody.Status, respBody.Message,
  127. )
  128. }
  129. return
  130. }