Tool to help you manage your Grafana dashboards using Git.

webhook.go 2.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. package main
  2. import (
  3. "io/ioutil"
  4. "strings"
  5. "config"
  6. "git"
  7. puller "puller"
  8. "gopkg.in/go-playground/webhooks.v3"
  9. "gopkg.in/go-playground/webhooks.v3/gitlab"
  10. )
  11. // SetupWebhook creates and exposes a GitLab webhook using a given configuration.
  12. // Returns an error if the webhook couldn't be set up.
  13. func SetupWebhook(cfg *config.Config) error {
  14. hook := gitlab.New(&gitlab.Config{
  15. Secret: cfg.Webhook.Secret,
  16. })
  17. hook.RegisterEvents(HandlePush, gitlab.PushEvents)
  18. return webhooks.Run(
  19. hook,
  20. cfg.Webhook.Interface+":"+cfg.Webhook.Port,
  21. cfg.Webhook.Path,
  22. )
  23. }
  24. // HandlePush is called each time a push event is sent by GitLab on the webhook.
  25. func HandlePush(payload interface{}, header webhooks.Header) {
  26. var err error
  27. // Process the payload using the right structure
  28. pl := payload.(gitlab.PushEventPayload)
  29. // Clone or pull the repository
  30. if _, err = git.Sync(cfg.Git); err != nil {
  31. panic(err)
  32. }
  33. // Only push changes made on master to Grafana
  34. if pl.Ref != "refs/heads/master" {
  35. return
  36. }
  37. // Files to push are stored in a map before being pushed to the Grafana API.
  38. // We don't push them in the loop iterating over commits because, in the
  39. // case a file is successively updated by two commits pushed at the same
  40. // time, it would push the same file several time, which isn't an optimised
  41. // behaviour.
  42. filesToPush := make(map[string]bool)
  43. // Iterate over the commits descriptions from the payload
  44. for _, commit := range pl.Commits {
  45. // We don't want to process commits made by the puller
  46. if commit.Author.Email == cfg.Git.CommitsAuthor.Email {
  47. continue
  48. }
  49. // Push all added files
  50. for _, addedFile := range commit.Added {
  51. filesToPush[addedFile] = true
  52. }
  53. // Push all modified files
  54. for _, modifiedFile := range commit.Modified {
  55. filesToPush[modifiedFile] = true
  56. }
  57. // TODO: Remove a dashboard when its file gets deleted?
  58. }
  59. // Push all files to the Grafana API
  60. for fileToPush := range filesToPush {
  61. if err = pushFile(fileToPush); err != nil {
  62. panic(err)
  63. }
  64. }
  65. // Grafana will auto-update the version number after we pushed the new
  66. // dashboards, so we use the puller mechanic to pull the updated numbers and
  67. // commit them in the git repo.
  68. if err = puller.PullGrafanaAndCommit(grafanaClient, cfg); err != nil {
  69. panic(err)
  70. }
  71. }
  72. // pushFile pushes the content of a given file to the Grafana API in order to
  73. // create or update a dashboard.
  74. // Returns an error if there was an issue reading the file or sending its content
  75. // to the Grafana instance.
  76. func pushFile(filename string) error {
  77. filePath := cfg.Git.ClonePath + "/" + filename
  78. fileContent, err := ioutil.ReadFile(filePath)
  79. if err != nil {
  80. return err
  81. }
  82. // Remove the .json part
  83. slug := strings.Split(filename, ".json")[0]
  84. return grafanaClient.CreateOrUpdateDashboard(slug, fileContent)
  85. }