Sven Efftinge 1b76bca17d [public-api] handle customer.subscription.deleted event
This event is fired by Stripe when a customer cancels their subscription
2022-09-16 16:24:17 +02:00

93 lines
2.6 KiB
Go

// Copyright (c) 2022 Gitpod GmbH. All rights reserved.
// Licensed under the GNU Affero General Public License (AGPL).
// See License-AGPL.txt in the project root for license information.
package webhooks
import (
"io"
"net/http"
"github.com/gitpod-io/gitpod/common-go/log"
"github.com/gitpod-io/gitpod/public-api-server/pkg/billingservice"
"github.com/stripe/stripe-go/v72/webhook"
)
const maxBodyBytes = int64(65536)
type webhookHandler struct {
billingService billingservice.Interface
stripeWebhookSignature string
}
func NewStripeWebhookHandler(billingService billingservice.Interface, stripeWebhookSignature string) *webhookHandler {
return &webhookHandler{
billingService: billingService,
stripeWebhookSignature: stripeWebhookSignature,
}
}
func (h *webhookHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
if req.Method != http.MethodPost {
log.Errorf("Bad HTTP method: %s", req.Method)
w.WriteHeader(http.StatusMethodNotAllowed)
return
}
stripeSignature := req.Header.Get("Stripe-Signature")
if stripeSignature == "" {
w.WriteHeader(http.StatusBadRequest)
return
}
req.Body = http.MaxBytesReader(w, req.Body, maxBodyBytes)
payload, err := io.ReadAll(req.Body)
if err != nil {
log.WithError(err).Error("Failed to read payload body.")
w.WriteHeader(http.StatusBadRequest)
return
}
// https://stripe.com/docs/webhooks/signatures#verify-official-libraries
event, err := webhook.ConstructEvent(payload, req.Header.Get("Stripe-Signature"), h.stripeWebhookSignature)
if err != nil {
log.WithError(err).Error("Failed to verify webhook signature.")
w.WriteHeader(http.StatusBadRequest)
return
}
switch event.Type {
case "invoice.finalized":
invoiceId, ok := event.Data.Object["id"].(string)
if !ok {
log.Error("failed to find invoice id in Stripe event payload")
w.WriteHeader(http.StatusBadRequest)
}
err = h.billingService.FinalizeInvoice(req.Context(), invoiceId)
if err != nil {
log.WithError(err).Error("Failed to finalize invoice")
w.WriteHeader(http.StatusInternalServerError)
return
}
case "customer.subscription.deleted":
subscriptionId, ok := event.Data.Object["id"].(string)
if !ok {
log.Error("failed to find subscriptionId id in Stripe event payload")
w.WriteHeader(http.StatusBadRequest)
}
err = h.billingService.CancelSubscription(req.Context(), subscriptionId)
if err != nil {
log.WithError(err).Error("Failed to cancel subscription")
w.WriteHeader(http.StatusInternalServerError)
return
}
default:
log.Errorf("Unexpected Stripe event type: %s", event.Type)
w.WriteHeader(http.StatusBadRequest)
return
}
}