Tool to help you manage your Grafana dashboards using Git.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. package main
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "io/ioutil"
  6. "os"
  7. "git"
  8. "grafana"
  9. gogit "gopkg.in/src-d/go-git.v4"
  10. )
  11. // diffVersion represents a dashboard version diff.
  12. type diffVersion struct {
  13. oldVersion int
  14. newVersion int
  15. }
  16. // PullGrafanaAndCommit pulls all the dashboards from Grafana then commits each
  17. // of them to Git except for those that have a newer or equal version number
  18. // already versionned in the repo
  19. func PullGrafanaAndCommit(
  20. client *grafana.Client,
  21. repoURL string, clonePath string, privateKeyPath string,
  22. ) error {
  23. // Clone or pull the repo
  24. repo, err := git.Sync(repoURL, clonePath, privateKeyPath)
  25. if err != nil {
  26. return err
  27. }
  28. w, err := repo.Worktree()
  29. if err != nil {
  30. return err
  31. }
  32. // Get URIs for all known dashboards
  33. uris, err := client.GetDashboardsURIs()
  34. if err != nil {
  35. return err
  36. }
  37. dv := make(map[string]diffVersion)
  38. // Load versions
  39. dbVersions, err := getDashboardsVersions()
  40. if err != nil {
  41. return err
  42. }
  43. // Iterate over the dashboards URIs
  44. for _, uri := range uris {
  45. // Retrieve the dashboard JSON
  46. dashboard, err := client.GetDashboard(uri)
  47. if err != nil {
  48. return err
  49. }
  50. // Check if there's a version for this dashboard in the data loaded from
  51. // the "versions.json" file. If there's a version and it's older (lower
  52. // version number) than the version we just retrieved from the Grafana
  53. // API, or if there's no known version (ok will be false), write the
  54. // changes in the repo and add the modified file to the git index.
  55. version, ok := dbVersions[dashboard.Slug]
  56. if !ok || dashboard.Version > version {
  57. if err = addDashboardChangesToRepo(dashboard, w); err != nil {
  58. return err
  59. }
  60. // We don't need to check for the value of ok because if ok is false
  61. // version will be initialised to the 0-value of the int type, which
  62. // is 0, so the previous version number will be considered to be 0,
  63. // which is the behaviour we want.
  64. dv[dashboard.Slug] = diffVersion{
  65. oldVersion: version,
  66. newVersion: dashboard.Version,
  67. }
  68. }
  69. }
  70. status, err := w.Status()
  71. if err != nil {
  72. return err
  73. }
  74. // Check if there's uncommited changes, and if that's the case, commit them.
  75. if !status.IsClean() {
  76. if err = commitNewVersions(dbVersions, dv, w); err != nil {
  77. return err
  78. }
  79. }
  80. // Push the changes (we don't do it in the if clause above in case there are
  81. // pending commits in the local repo that haven't been pushed yet).
  82. if err = git.Push(repo, privateKeyPath); err != nil {
  83. return err
  84. }
  85. return nil
  86. }
  87. // addDashboardChangesToRepo writes a dashboard content in a file, then adds the
  88. // file to the git index so it can be comitted afterwards.
  89. // Returns an error if there was an issue with either of the steps.
  90. func addDashboardChangesToRepo(
  91. dashboard *grafana.Dashboard, worktree *gogit.Worktree,
  92. ) error {
  93. slugExt := dashboard.Slug + ".json"
  94. if err := rewriteFile(
  95. *clonePath+"/"+slugExt,
  96. dashboard.RawJSON,
  97. ); err != nil {
  98. return err
  99. }
  100. if _, err := worktree.Add(slugExt); err != nil {
  101. return err
  102. }
  103. return nil
  104. }
  105. // rewriteFile removes a given file and re-creates it with a new content. The
  106. // content is provided as JSON, and is then indented before being written down.
  107. // We need the whole "remove then recreate" thing because, if the file already
  108. // exists, ioutil.WriteFile will append the content to it. However, we want to
  109. // replace the oldest version with another (so git can diff it), so we re-create
  110. // the file with the changed content.
  111. // Returns an error if there was an issue when removing or writing the file, or
  112. // indenting the JSON content.
  113. func rewriteFile(filename string, content []byte) error {
  114. if err := os.Remove(filename); err != nil {
  115. pe, ok := err.(*os.PathError)
  116. if !ok || pe.Err.Error() != "no such file or directory" {
  117. return err
  118. }
  119. }
  120. indentedContent, err := indent(content)
  121. if err != nil {
  122. return err
  123. }
  124. return ioutil.WriteFile(filename, indentedContent, 0644)
  125. }
  126. // indent indents a given JSON content with tabs.
  127. // We need to indent the content as the Grafana API returns a one-lined JSON
  128. // string, which isn't great to work with.
  129. // Returns an error if there was an issue with the process.
  130. func indent(srcJSON []byte) (indentedJSON []byte, err error) {
  131. buf := bytes.NewBuffer(nil)
  132. if err = json.Indent(buf, srcJSON, "", "\t"); err != nil {
  133. return
  134. }
  135. indentedJSON, err = ioutil.ReadAll(buf)
  136. return
  137. }