Tool to help you manage your Grafana dashboards using Git.

poller.go 5.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. package poller
  2. import (
  3. "time"
  4. "config"
  5. "git"
  6. "grafana"
  7. puller "puller"
  8. "pusher/common"
  9. "github.com/sirupsen/logrus"
  10. )
  11. // Setup loads (and synchronise if needed) the Git repository mentioned in the
  12. // configuration file, then creates the poller that will pull from the Git
  13. // repository on a regular basis and push all the changes to Grafana.
  14. // Returns an error if the poller encountered one.
  15. func Setup(cfg *config.Config, client *grafana.Client, delRemoved bool) error {
  16. // Load the Git repository.
  17. r, needsSync, err := git.NewRepository(cfg.Git)
  18. if err != nil {
  19. return err
  20. }
  21. // Synchronise the repository if needed.
  22. if needsSync {
  23. if err = r.Sync(false); err != nil {
  24. return err
  25. }
  26. }
  27. errs := make(chan error, 1)
  28. // In the future we may want to poll from several Git repositories, so we
  29. // run the poller in a go routine.
  30. go func() {
  31. if err = poller(cfg, r, client, delRemoved); err != nil {
  32. errs <- err
  33. return
  34. }
  35. }()
  36. err = <-errs
  37. return err
  38. }
  39. // poller gets the current status of the Git repository that has previously been
  40. // loaded, and then starts an infinite loop that will pull from the Git
  41. // remote, then, if there was any new commit, retrieve the contents of the
  42. // modified and added files to push them to Grafana. If set by the user via
  43. // a command-line flag, it will also check for removed files and delete the
  44. // corresponding dashboards from Grafana. It then sleeps for the time specified
  45. // in the configuration file, before starting its next iteration.
  46. // Returns an error if there was an issue checking the Git repository status,
  47. // synchronising it, reading the files' contents or discussing with the Grafana
  48. // API
  49. func poller(
  50. cfg *config.Config, repo *git.Repository, client *grafana.Client,
  51. delRemoved bool,
  52. ) (err error) {
  53. // Get current state of the repo.
  54. // This is mainly to give an initial value to variables that will see their
  55. // content changed with every iteration of the loop.
  56. latestCommit, err := repo.GetLatestCommit()
  57. if err != nil {
  58. return
  59. }
  60. filesContents, err := repo.GetFilesContentsAtCommit(latestCommit)
  61. if err != nil {
  62. return
  63. }
  64. // We'll need to know the previous commit in order to compare its hash with
  65. // the one from the most recent commit after we pull from the remote, se we
  66. // know if there was any new commit.
  67. previousCommit := latestCommit
  68. // We need to store the content of the files from the previous iteration of
  69. // the loop in order to manage removed files which contents won't be
  70. // accessible anymore.
  71. previousFilesContents := filesContents
  72. // Start looping
  73. for {
  74. // Synchronise the repository (i.e. pull from remote).
  75. if err = repo.Sync(true); err != nil {
  76. return
  77. }
  78. // Retrieve the latest commit in order to compare its hash with the
  79. // previous one.
  80. latestCommit, err = repo.GetLatestCommit()
  81. if err != nil {
  82. return
  83. }
  84. // If there is at least one new commit, handle the changes it introduces.
  85. if previousCommit.Hash.String() != latestCommit.Hash.String() {
  86. logrus.WithFields(logrus.Fields{
  87. "previous_hash": previousCommit.Hash.String(),
  88. "new_hash": latestCommit.Hash.String(),
  89. }).Info("New commit(s) detected")
  90. // Get the updated files contents.
  91. filesContents, err = repo.GetFilesContentsAtCommit(latestCommit)
  92. if err != nil {
  93. return err
  94. }
  95. // Get the name of the files that have been added/modified and
  96. // removed between the two iterations.
  97. modified, removed, err := repo.GetModifiedAndRemovedFiles(previousCommit, latestCommit)
  98. if err != nil {
  99. return err
  100. }
  101. // Get a map containing the latest known content of each added,
  102. // modified and removed file.
  103. mergedContents := mergeContents(modified, removed, filesContents, previousFilesContents)
  104. // Push the contents of the files that were added or modified to the
  105. // Grafana API.
  106. common.PushFiles(modified, mergedContents, client)
  107. // If the user requested it, delete all dashboards that were removed
  108. // from the repository.
  109. if delRemoved {
  110. common.DeleteDashboards(removed, mergedContents, client)
  111. }
  112. // Grafana will auto-update the version number after we pushed the new
  113. // dashboards, so we use the puller mechanic to pull the updated numbers and
  114. // commit them in the git repo.
  115. if err = puller.PullGrafanaAndCommit(client, cfg); err != nil {
  116. logrus.WithFields(logrus.Fields{
  117. "error": err,
  118. "repo": cfg.Git.User + "@" + cfg.Git.URL,
  119. "clone_path": cfg.Git.ClonePath,
  120. }).Error("Call to puller returned an error")
  121. }
  122. }
  123. // Update the commit and files contents to prepare for the next iteration.
  124. previousCommit = latestCommit
  125. previousFilesContents = filesContents
  126. // Sleep before the next iteration.
  127. time.Sleep(time.Duration(cfg.Pusher.Config.Interval) * time.Second)
  128. }
  129. }
  130. // mergeContents will take as arguments a list of names of files that have been
  131. // added/modified, a list of names of files that have been removed from the Git
  132. // repository, the current contents of the files in the Git repository, and the
  133. // contents of the files in the Git repository as they were at the previous
  134. // iteration of the poller's loop.
  135. // It will create and return a map contaning the current content of all
  136. // added/modified file, and the previous content of all removed file (since
  137. // they are no longer accessible on disk). All files in this map is either added,
  138. // modified or removed on the Git repository.
  139. func mergeContents(
  140. modified []string, removed []string,
  141. filesContents map[string][]byte, previousFilesContents map[string][]byte,
  142. ) (merged map[string][]byte) {
  143. merged = make(map[string][]byte)
  144. // Load the added/modified files' contents
  145. for _, modifiedFile := range modified {
  146. merged[modifiedFile] = filesContents[modifiedFile]
  147. }
  148. // Load the removed files' contents
  149. for _, removedFile := range removed {
  150. merged[removedFile] = previousFilesContents[removedFile]
  151. }
  152. return
  153. }