Tool to help you manage your Grafana dashboards using Git.

webhook.go 2.3KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  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. // Iterate over the commits descriptions from the payload
  38. for _, commit := range pl.Commits {
  39. // We don't want to process commits made by the puller
  40. if commit.Author.Email == cfg.Git.CommitsAuthor.Email {
  41. continue
  42. }
  43. // Push all added files
  44. for _, addedFile := range commit.Added {
  45. if err = pushFile(addedFile); err != nil {
  46. panic(err)
  47. }
  48. }
  49. // Push all modified files
  50. for _, modifiedFile := range commit.Modified {
  51. if err = pushFile(modifiedFile); err != nil {
  52. panic(err)
  53. }
  54. }
  55. // TODO: Remove a dashboard when its file gets deleted?
  56. }
  57. // Grafana will auto-update the version number after we pushed the new
  58. // dashboards, so we use the puller mechanic to pull the updated numbers and
  59. // commit them in the git repo.
  60. if err = puller.PullGrafanaAndCommit(grafanaClient, cfg); err != nil {
  61. panic(err)
  62. }
  63. }
  64. // pushFile pushes the content of a given file to the Grafana API in order to
  65. // create or update a dashboard.
  66. // Returns an error if there was an issue reading the file or sending its content
  67. // to the Grafana instance.
  68. func pushFile(filename string) error {
  69. filePath := cfg.Git.ClonePath + "/" + filename
  70. fileContent, err := ioutil.ReadFile(filePath)
  71. if err != nil {
  72. return err
  73. }
  74. // Remove the .json part
  75. slug := strings.Split(filename, ".json")[0]
  76. return grafanaClient.CreateOrUpdateDashboard(slug, fileContent)
  77. }