mirror of
https://github.com/documenso/documenso.git
synced 2025-11-13 08:13:56 +10:00
249 lines
6.9 KiB
TypeScript
249 lines
6.9 KiB
TypeScript
import { NextApiRequest, NextApiResponse } from 'next';
|
|
|
|
import { buffer } from 'micro';
|
|
import { match } from 'ts-pattern';
|
|
|
|
import { Stripe, stripe } from '@documenso/lib/server-only/stripe';
|
|
import { getFlag } from '@documenso/lib/universal/get-feature-flag';
|
|
import { prisma } from '@documenso/prisma';
|
|
|
|
import { onSubscriptionDeleted } from './on-subscription-deleted';
|
|
import { onSubscriptionUpdated } from './on-subscription-updated';
|
|
|
|
type StripeWebhookResponse = {
|
|
success: boolean;
|
|
message: string;
|
|
};
|
|
|
|
export const stripeWebhookHandler = async (
|
|
req: NextApiRequest,
|
|
res: NextApiResponse<StripeWebhookResponse>,
|
|
) => {
|
|
try {
|
|
const isBillingEnabled = await getFlag('app_billing');
|
|
|
|
if (!isBillingEnabled) {
|
|
return res.status(500).json({
|
|
success: false,
|
|
message: 'Billing is disabled',
|
|
});
|
|
}
|
|
|
|
const signature =
|
|
typeof req.headers['stripe-signature'] === 'string' ? req.headers['stripe-signature'] : '';
|
|
|
|
if (!signature) {
|
|
return res.status(400).json({
|
|
success: false,
|
|
message: 'No signature found in request',
|
|
});
|
|
}
|
|
|
|
const body = await buffer(req);
|
|
|
|
const event = stripe.webhooks.constructEvent(
|
|
body,
|
|
signature,
|
|
process.env.NEXT_PRIVATE_STRIPE_WEBHOOK_SECRET,
|
|
);
|
|
|
|
await match(event.type)
|
|
.with('checkout.session.completed', async () => {
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
const session = event.data.object as Stripe.Checkout.Session;
|
|
|
|
const customerId =
|
|
typeof session.customer === 'string' ? session.customer : session.customer?.id;
|
|
|
|
let userId = Number(session.client_reference_id);
|
|
|
|
if (!userId && customerId) {
|
|
const result = await prisma.subscription.findFirst({
|
|
select: {
|
|
userId: true,
|
|
},
|
|
where: {
|
|
customerId,
|
|
},
|
|
});
|
|
|
|
if (!result?.userId) {
|
|
return res.status(500).json({
|
|
success: false,
|
|
message: 'User not found',
|
|
});
|
|
}
|
|
|
|
userId = result.userId;
|
|
}
|
|
|
|
const subscriptionId =
|
|
typeof session.subscription === 'string'
|
|
? session.subscription
|
|
: session.subscription?.id;
|
|
|
|
if (!subscriptionId || Number.isNaN(userId)) {
|
|
return res.status(500).json({
|
|
success: false,
|
|
message: 'Invalid session',
|
|
});
|
|
}
|
|
|
|
const subscription = await stripe.subscriptions.retrieve(subscriptionId);
|
|
|
|
await onSubscriptionUpdated({ userId, subscription });
|
|
|
|
return res.status(200).json({
|
|
success: true,
|
|
message: 'Webhook received',
|
|
});
|
|
})
|
|
.with('customer.subscription.updated', async () => {
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
const subscription = event.data.object as Stripe.Subscription;
|
|
|
|
const customerId =
|
|
typeof subscription.customer === 'string'
|
|
? subscription.customer
|
|
: subscription.customer.id;
|
|
|
|
const result = await prisma.subscription.findFirst({
|
|
select: {
|
|
userId: true,
|
|
},
|
|
where: {
|
|
customerId,
|
|
},
|
|
});
|
|
|
|
if (!result?.userId) {
|
|
return res.status(500).json({
|
|
success: false,
|
|
message: 'User not found',
|
|
});
|
|
}
|
|
|
|
await onSubscriptionUpdated({ userId: result.userId, subscription });
|
|
|
|
return res.status(200).json({
|
|
success: true,
|
|
message: 'Webhook received',
|
|
});
|
|
})
|
|
.with('invoice.payment_succeeded', async () => {
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
const invoice = event.data.object as Stripe.Invoice;
|
|
|
|
if (invoice.billing_reason !== 'subscription_cycle') {
|
|
return res.status(200).json({
|
|
success: true,
|
|
message: 'Webhook received',
|
|
});
|
|
}
|
|
|
|
const customerId =
|
|
typeof invoice.customer === 'string' ? invoice.customer : invoice.customer?.id;
|
|
|
|
const subscriptionId =
|
|
typeof invoice.subscription === 'string'
|
|
? invoice.subscription
|
|
: invoice.subscription?.id;
|
|
|
|
if (!customerId || !subscriptionId) {
|
|
return res.status(500).json({
|
|
success: false,
|
|
message: 'Invalid invoice',
|
|
});
|
|
}
|
|
|
|
const subscription = await stripe.subscriptions.retrieve(subscriptionId);
|
|
|
|
const result = await prisma.subscription.findFirst({
|
|
select: {
|
|
userId: true,
|
|
},
|
|
where: {
|
|
customerId,
|
|
},
|
|
});
|
|
|
|
if (!result?.userId) {
|
|
return res.status(500).json({
|
|
success: false,
|
|
message: 'User not found',
|
|
});
|
|
}
|
|
|
|
await onSubscriptionUpdated({ userId: result.userId, subscription });
|
|
|
|
return res.status(200).json({
|
|
success: true,
|
|
message: 'Webhook received',
|
|
});
|
|
})
|
|
.with('invoice.payment_failed', async () => {
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
const invoice = event.data.object as Stripe.Invoice;
|
|
|
|
const customerId =
|
|
typeof invoice.customer === 'string' ? invoice.customer : invoice.customer?.id;
|
|
|
|
const subscriptionId =
|
|
typeof invoice.subscription === 'string'
|
|
? invoice.subscription
|
|
: invoice.subscription?.id;
|
|
|
|
if (!customerId || !subscriptionId) {
|
|
return res.status(500).json({
|
|
success: false,
|
|
message: 'Invalid invoice',
|
|
});
|
|
}
|
|
|
|
const subscription = await stripe.subscriptions.retrieve(subscriptionId);
|
|
|
|
const result = await prisma.subscription.findFirst({
|
|
select: {
|
|
userId: true,
|
|
},
|
|
where: {
|
|
customerId,
|
|
},
|
|
});
|
|
|
|
if (!result?.userId) {
|
|
return res.status(500).json({
|
|
success: false,
|
|
message: 'User not found',
|
|
});
|
|
}
|
|
|
|
await onSubscriptionUpdated({ userId: result.userId, subscription });
|
|
})
|
|
.with('customer.subscription.deleted', async () => {
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
const subscription = event.data.object as Stripe.Subscription;
|
|
|
|
await onSubscriptionDeleted({ subscription });
|
|
|
|
return res.status(200).json({
|
|
success: true,
|
|
message: 'Webhook received',
|
|
});
|
|
})
|
|
.otherwise(() => {
|
|
return res.status(200).json({
|
|
success: true,
|
|
message: 'Webhook received',
|
|
});
|
|
});
|
|
} catch (err) {
|
|
console.error(err);
|
|
|
|
res.status(500).json({
|
|
success: false,
|
|
message: 'Unknown error',
|
|
});
|
|
}
|
|
};
|