mirror of
https://github.com/documenso/documenso.git
synced 2025-11-13 00:03:33 +10:00
## Description Previously we assumed that there can only be 1 subscription per user. However, that will soon no longer the case with the introduction of the Teams subscription. This PR will apply the required migrations to support multiple subscriptions. ## Changes Made - Updated the Prisma schema to allow for multiple `Subscriptions` per `User` - Added a Stripe `customerId` field to the `User` model - Updated relevant billing sections to support multiple subscriptions ## Testing Performed - Tested running the Prisma migration on a demo database created on the main branch Will require a lot of additional testing. ## Checklist - [ ] I have tested these changes locally and they work as expected. - [ ] I have added/updated tests that prove the effectiveness of these changes. - [X] I have followed the project's coding style guidelines. ## Additional Notes Added the following custom SQL statement to the migration: > DELETE FROM "Subscription" WHERE "planId" IS NULL OR "priceId" IS NULL; Prior to deployment this will require changes to Stripe products: - Adding `type` meta attribute --------- Co-authored-by: Lucas Smith <me@lucasjamessmith.me>
59 lines
1.9 KiB
TypeScript
59 lines
1.9 KiB
TypeScript
import type Stripe from 'stripe';
|
|
|
|
import { stripe } from '@documenso/lib/server-only/stripe';
|
|
|
|
// Utility type to handle usage of the `expand` option.
|
|
type PriceWithProduct = Stripe.Price & { product: Stripe.Product };
|
|
|
|
export type PriceIntervals = Record<Stripe.Price.Recurring.Interval, PriceWithProduct[]>;
|
|
|
|
export type GetPricesByIntervalOptions = {
|
|
/**
|
|
* Filter products by their meta 'type' attribute.
|
|
*/
|
|
type?: 'individual';
|
|
};
|
|
|
|
export const getPricesByInterval = async ({ type }: GetPricesByIntervalOptions = {}) => {
|
|
let { data: prices } = await stripe.prices.search({
|
|
query: `active:'true' type:'recurring'`,
|
|
expand: ['data.product'],
|
|
limit: 100,
|
|
});
|
|
|
|
prices = prices.filter((price) => {
|
|
// We use `expand` to get the product, but it's not typed as part of the Price type.
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
const product = price.product as Stripe.Product;
|
|
|
|
const filter = !type || product.metadata?.type === type;
|
|
|
|
// Filter out prices for products that are not active.
|
|
return product.active && filter;
|
|
});
|
|
|
|
const intervals: PriceIntervals = {
|
|
day: [],
|
|
week: [],
|
|
month: [],
|
|
year: [],
|
|
};
|
|
|
|
// Add each price to the correct interval.
|
|
for (const price of prices) {
|
|
if (price.recurring?.interval) {
|
|
// We use `expand` to get the product, but it's not typed as part of the Price type.
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
intervals[price.recurring.interval].push(price as PriceWithProduct);
|
|
}
|
|
}
|
|
|
|
// Order all prices by unit_amount.
|
|
intervals.day.sort((a, b) => Number(a.unit_amount) - Number(b.unit_amount));
|
|
intervals.week.sort((a, b) => Number(a.unit_amount) - Number(b.unit_amount));
|
|
intervals.month.sort((a, b) => Number(a.unit_amount) - Number(b.unit_amount));
|
|
intervals.year.sort((a, b) => Number(a.unit_amount) - Number(b.unit_amount));
|
|
|
|
return intervals;
|
|
};
|