Tool to help you manage your Grafana dashboards using Git.

webhook.go 4.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. package main
  2. import (
  3. "encoding/json"
  4. "io/ioutil"
  5. "strings"
  6. "config"
  7. "git"
  8. puller "puller"
  9. "gopkg.in/go-playground/webhooks.v3"
  10. "gopkg.in/go-playground/webhooks.v3/gitlab"
  11. )
  12. // SetupWebhook creates and exposes a GitLab webhook using a given configuration.
  13. // Returns an error if the webhook couldn't be set up.
  14. func SetupWebhook(cfg *config.Config) error {
  15. hook := gitlab.New(&gitlab.Config{
  16. Secret: cfg.Webhook.Secret,
  17. })
  18. hook.RegisterEvents(HandlePush, gitlab.PushEvents)
  19. return webhooks.Run(
  20. hook,
  21. cfg.Webhook.Interface+":"+cfg.Webhook.Port,
  22. cfg.Webhook.Path,
  23. )
  24. }
  25. // HandlePush is called each time a push event is sent by GitLab on the webhook.
  26. func HandlePush(payload interface{}, header webhooks.Header) {
  27. var err error
  28. // Process the payload using the right structure
  29. pl := payload.(gitlab.PushEventPayload)
  30. // Only push changes made on master to Grafana
  31. if pl.Ref != "refs/heads/master" {
  32. return
  33. }
  34. // Clone or pull the repository
  35. if _, err = git.Sync(cfg.Git); err != nil {
  36. panic(err)
  37. }
  38. // Files to push are stored in a map before being pushed to the Grafana API.
  39. // We don't push them in the loop iterating over commits because, in the
  40. // case a file is successively updated by two commits pushed at the same
  41. // time, it would push the same file several time, which isn't an optimised
  42. // behaviour.
  43. filesToPush := make(map[string]bool)
  44. // Iterate over the commits descriptions from the payload
  45. for _, commit := range pl.Commits {
  46. // We don't want to process commits made by the puller
  47. if commit.Author.Email == cfg.Git.CommitsAuthor.Email {
  48. continue
  49. }
  50. // Push all added files, except the ones describing a dashboard which
  51. // name starts with a the prefix specified in the configuration file.
  52. for _, addedFile := range commit.Added {
  53. ignored, err := isIgnored(addedFile)
  54. if err != nil {
  55. panic(err)
  56. }
  57. if !ignored {
  58. filesToPush[addedFile] = true
  59. }
  60. }
  61. // Push all modified files, except the ones describing a dashboard which
  62. // name starts with a the prefix specified in the configuration file.
  63. for _, modifiedFile := range commit.Modified {
  64. ignored, err := isIgnored(modifiedFile)
  65. if err != nil {
  66. panic(err)
  67. }
  68. if !ignored {
  69. filesToPush[modifiedFile] = true
  70. }
  71. }
  72. // TODO: Remove a dashboard when its file gets deleted?
  73. }
  74. // Push all files to the Grafana API
  75. for fileToPush := range filesToPush {
  76. if err = pushFile(fileToPush); err != nil {
  77. panic(err)
  78. }
  79. }
  80. // Grafana will auto-update the version number after we pushed the new
  81. // dashboards, so we use the puller mechanic to pull the updated numbers and
  82. // commit them in the git repo.
  83. if err = puller.PullGrafanaAndCommit(grafanaClient, cfg); err != nil {
  84. panic(err)
  85. }
  86. }
  87. // pushFile pushes the content of a given file to the Grafana API in order to
  88. // create or update a dashboard.
  89. // Returns an error if there was an issue reading the file or sending its content
  90. // to the Grafana instance.
  91. func pushFile(filename string) error {
  92. filePath := cfg.Git.ClonePath + "/" + filename
  93. fileContent, err := ioutil.ReadFile(filePath)
  94. if err != nil {
  95. return err
  96. }
  97. // Remove the .json part
  98. slug := strings.Split(filename, ".json")[0]
  99. return grafanaClient.CreateOrUpdateDashboard(slug, fileContent)
  100. }
  101. // isIgnored checks whether the file must be ignored, by checking if there's an
  102. // prefix for ignored files set in the configuration file, and if the dashboard
  103. // described in the file has a name that starts with this prefix. Returns an
  104. // error if there was an issue reading or decoding the file.
  105. // TODO: Optimise this part of the workflow, as all files get open twice (here
  106. // and in pushFile)
  107. func isIgnored(filename string) (bool, error) {
  108. // If there's no prefix set, no file is ignored
  109. if len(cfg.Grafana.IgnorePrefix) == 0 {
  110. return false, nil
  111. }
  112. // Read the file's content
  113. fileContent, err := ioutil.ReadFile(filename)
  114. if err != nil {
  115. return false, err
  116. }
  117. // Parse the file's content to find the dashboard's name
  118. var dashboardName struct {
  119. Name string `json:"title"`
  120. }
  121. if err = json.Unmarshal(fileContent, &dashboardName); err != nil {
  122. return false, err
  123. }
  124. // Compare the lower case dashboar name to the prefix (which has already
  125. // been lower cased when loading the configuration file)
  126. lowerCaseName := strings.ToLower(dashboardName.Name)
  127. if strings.HasPrefix(lowerCaseName, cfg.Grafana.IgnorePrefix) {
  128. return true, nil
  129. }
  130. return false, nil
  131. }