webhook.go 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. package billing
  2. import (
  3. "errors"
  4. "fmt"
  5. "io/ioutil"
  6. "net/http"
  7. "github.com/porter-dev/porter/api/server/authz"
  8. "github.com/porter-dev/porter/api/server/handlers"
  9. "github.com/porter-dev/porter/api/server/shared"
  10. "github.com/porter-dev/porter/api/server/shared/apierrors"
  11. "github.com/porter-dev/porter/api/server/shared/config"
  12. "gorm.io/gorm"
  13. )
  14. type BillingWebhookHandler struct {
  15. handlers.PorterHandlerReadWriter
  16. authz.KubernetesAgentGetter
  17. }
  18. func NewBillingWebhookHandler(
  19. config *config.Config,
  20. decoderValidator shared.RequestDecoderValidator,
  21. ) http.Handler {
  22. return &BillingWebhookHandler{
  23. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, nil),
  24. }
  25. }
  26. func (c *BillingWebhookHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  27. payload, err := ioutil.ReadAll(r.Body)
  28. if err != nil {
  29. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  30. return
  31. }
  32. // verify webhook secret
  33. signature := r.Header.Get("x-signature")
  34. if !c.Config().BillingManager.VerifySignature(signature, payload) {
  35. c.HandleAPIError(w, r, apierrors.NewErrForbidden(
  36. fmt.Errorf("could not verify signature for billing webhook"),
  37. ))
  38. return
  39. }
  40. // parse usage and update project
  41. newUsage, err := c.Config().BillingManager.ParseProjectUsageFromWebhook(payload)
  42. if err != nil {
  43. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  44. return
  45. }
  46. // newUsage will be nil if webhook event type is not "subscription", so return without
  47. // updating usage in this case
  48. if newUsage == nil {
  49. return
  50. }
  51. // update the project's usage
  52. existingUsage, err := c.Repo().ProjectUsage().ReadProjectUsage(newUsage.ProjectID)
  53. notFound := errors.Is(err, gorm.ErrRecordNotFound)
  54. if !notFound && err != nil {
  55. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  56. return
  57. }
  58. if notFound {
  59. _, err = c.Repo().ProjectUsage().CreateProjectUsage(newUsage)
  60. } else {
  61. newUsage.ID = existingUsage.ID
  62. _, err = c.Repo().ProjectUsage().UpdateProjectUsage(newUsage)
  63. }
  64. if err != nil {
  65. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  66. return
  67. }
  68. }