webhook.go 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. package billing
  2. import (
  3. "errors"
  4. "fmt"
  5. "io/ioutil"
  6. "net/http"
  7. "strconv"
  8. "github.com/porter-dev/porter/api/server/authz"
  9. "github.com/porter-dev/porter/api/server/handlers"
  10. "github.com/porter-dev/porter/api/server/shared"
  11. "github.com/porter-dev/porter/api/server/shared/apierrors"
  12. "github.com/porter-dev/porter/api/server/shared/config"
  13. "gorm.io/gorm"
  14. )
  15. type BillingWebhookHandler struct {
  16. handlers.PorterHandlerReadWriter
  17. authz.KubernetesAgentGetter
  18. }
  19. func NewBillingWebhookHandler(
  20. config *config.Config,
  21. decoderValidator shared.RequestDecoderValidator,
  22. ) http.Handler {
  23. return &BillingWebhookHandler{
  24. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, nil),
  25. }
  26. }
  27. func (c *BillingWebhookHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  28. payload, err := ioutil.ReadAll(r.Body)
  29. if err != nil {
  30. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  31. return
  32. }
  33. // verify webhook secret
  34. signature := r.Header.Get("x-signature")
  35. if !c.Config().BillingManager.VerifySignature(signature, payload) {
  36. c.HandleAPIError(w, r, apierrors.NewErrForbidden(
  37. fmt.Errorf("could not verify signature for billing webhook"),
  38. ))
  39. return
  40. }
  41. // parse usage and update project
  42. newUsage, features, err := c.Config().BillingManager.ParseProjectUsageFromWebhook(payload)
  43. if err != nil {
  44. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  45. return
  46. }
  47. // newUsage will be nil if webhook event type is not "subscription", so return without
  48. // updating usage in this case
  49. if newUsage == nil {
  50. return
  51. }
  52. // update the project's usage
  53. existingUsage, err := c.Repo().ProjectUsage().ReadProjectUsage(newUsage.ProjectID)
  54. notFound := errors.Is(err, gorm.ErrRecordNotFound)
  55. if !notFound && err != nil {
  56. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  57. return
  58. }
  59. if notFound {
  60. _, err = c.Repo().ProjectUsage().CreateProjectUsage(newUsage)
  61. } else {
  62. newUsage.ID = existingUsage.ID
  63. _, err = c.Repo().ProjectUsage().UpdateProjectUsage(newUsage)
  64. }
  65. if err != nil {
  66. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  67. return
  68. }
  69. // update the feature flags
  70. project, err := c.Repo().Project().ReadProject(newUsage.ProjectID)
  71. if err != nil {
  72. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  73. return
  74. }
  75. if managedDatabasesEnabled, err := strconv.ParseBool(features.ManagedDatabasesEnabled); err == nil {
  76. project.RDSDatabasesEnabled = managedDatabasesEnabled
  77. }
  78. if managedInfraEnabled, err := strconv.ParseBool(features.ManagedInfraEnabled); err == nil {
  79. project.ManagedInfraEnabled = managedInfraEnabled
  80. }
  81. if stacksEnabled, err := strconv.ParseBool(features.StacksEnabled); err == nil {
  82. project.StacksEnabled = stacksEnabled
  83. }
  84. if previewEnvsEnabled, err := strconv.ParseBool(features.PreviewEnvironmentsEnabled); err == nil {
  85. project.PreviewEnvsEnabled = previewEnvsEnabled
  86. }
  87. _, err = c.Repo().Project().UpdateProject(project)
  88. if err != nil {
  89. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  90. return
  91. }
  92. }