Tool to help you manage your Grafana dashboards using Git.

webhooks.go 3.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. package webhooks
  2. import (
  3. "fmt"
  4. "net/http"
  5. )
  6. // Header provides http.Header to minimize imports
  7. type Header http.Header
  8. // Provider defines the type of webhook
  9. type Provider int
  10. func (p Provider) String() string {
  11. switch p {
  12. case GitHub:
  13. return "GitHub"
  14. case Bitbucket:
  15. return "Bitbucket"
  16. case GitLab:
  17. return "GitLab"
  18. default:
  19. return "Unknown"
  20. }
  21. }
  22. // webhooks available providers
  23. const (
  24. GitHub Provider = iota
  25. Bitbucket
  26. GitLab
  27. )
  28. // Webhook interface defines a webhook to receive events
  29. type Webhook interface {
  30. Provider() Provider
  31. ParsePayload(w http.ResponseWriter, r *http.Request)
  32. }
  33. type server struct {
  34. hook Webhook
  35. path string
  36. includePathCheck bool
  37. }
  38. // ProcessPayloadFunc is a common function for payload return values
  39. type ProcessPayloadFunc func(payload interface{}, header Header)
  40. // Handler returns the webhook http.Handler for use in your own Mux implementation
  41. func Handler(hook Webhook) http.Handler {
  42. return &server{
  43. hook: hook,
  44. }
  45. }
  46. // Run runs a server
  47. func Run(hook Webhook, addr string, path string) error {
  48. srv := &server{
  49. hook: hook,
  50. path: path,
  51. includePathCheck: true,
  52. }
  53. s := &http.Server{Addr: addr, Handler: srv}
  54. DefaultLog.Info(fmt.Sprintf("Listening on addr: %s path: %s", addr, path))
  55. return s.ListenAndServe()
  56. }
  57. // RunServer runs a custom server.
  58. func RunServer(s *http.Server, hook Webhook, path string) error {
  59. srv := &server{
  60. hook: hook,
  61. path: path,
  62. includePathCheck: true,
  63. }
  64. s.Handler = srv
  65. DefaultLog.Info(fmt.Sprintf("Listening on addr: %s path: %s", s.Addr, path))
  66. return s.ListenAndServe()
  67. }
  68. // RunTLSServer runs a custom server with TLS configuration.
  69. // NOTE: http.Server Handler will be overridden by this library, just set it to nil.
  70. // Setting the Certificates can be done in the http.Server.TLSConfig.Certificates
  71. // see example here: https://github.com/go-playground/webhooks/blob/v2/webhooks_test.go#L178
  72. func RunTLSServer(s *http.Server, hook Webhook, path string) error {
  73. srv := &server{
  74. hook: hook,
  75. path: path,
  76. includePathCheck: true,
  77. }
  78. s.Handler = srv
  79. DefaultLog.Info(fmt.Sprintf("Listening on addr: %s path: %s", s.Addr, path))
  80. return s.ListenAndServeTLS("", "")
  81. }
  82. // ServeHTTP is the Handler for every posted WebHook Event
  83. func (s *server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  84. defer r.Body.Close()
  85. DefaultLog.Info("Webhook received")
  86. if r.Method != "POST" {
  87. DefaultLog.Error(fmt.Sprintf("405 Method not allowed, attempt made using Method: %s", r.Method))
  88. http.Error(w, "405 Method not allowed", http.StatusMethodNotAllowed)
  89. return
  90. }
  91. DefaultLog.Debug(fmt.Sprintf("Include path check: %t", s.includePathCheck))
  92. if s.includePathCheck {
  93. if r.URL.Path != s.path {
  94. DefaultLog.Error(fmt.Sprintf("404 Not found, POST made using path: %s, but expected %s", r.URL.Path, s.path))
  95. http.Error(w, "404 Not found", http.StatusNotFound)
  96. return
  97. }
  98. }
  99. s.hook.ParsePayload(w, r)
  100. }