Tool to help you manage your Grafana dashboards using Git.

puller.go 5.2KB

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