1. replaceAll (Find & Replace) — une seule transaction ProseMirror au lieu d'un forEach cassé. Tous les matchs sont maintenant remplacés. 2. Link Preview unwrap — deleteNode() au lieu de clearer les attrs qui laissaient un nœud fantôme invisible dans le document. 3. Conversion Markdown → richtext — breaks: true dans marked.parse() Les simple newlines sont maintenant convertis en <br>. + préserve les blocs custom (toggle, callout, math, columns, outline, link-preview) en commentaires HTML lors de l'export MD. 4. emitNoteChange exercices — shape corrigée (type:'created' attend un objet Note, pas noteId/notebookId séparés). 5. Raccourcis clavier sans conflit : Cmd+Shift+C → Cmd+Alt+C (callout, avant: copier) Cmd+Shift+O → Cmd+Alt+O (outline, avant: historique/signets) Cmd+Shift+L → Cmd+Alt+L (colonnes, avant: lock screen macOS)
125 lines
4.3 KiB
TypeScript
125 lines
4.3 KiB
TypeScript
import type Stripe from 'stripe';
|
|
import { prisma } from '@/lib/prisma';
|
|
import { priceIdToTier } from '@/lib/billing/stripe-prices';
|
|
import { SubscriptionStatus, SubscriptionTier } from '@prisma/client';
|
|
|
|
function mapStripeStatus(stripeStatus: Stripe.Subscription.Status): SubscriptionStatus {
|
|
switch (stripeStatus) {
|
|
case 'active':
|
|
return SubscriptionStatus.ACTIVE;
|
|
case 'trialing':
|
|
return SubscriptionStatus.TRIALING;
|
|
case 'past_due':
|
|
return SubscriptionStatus.PAST_DUE;
|
|
case 'canceled':
|
|
case 'unpaid':
|
|
return SubscriptionStatus.CANCELED;
|
|
case 'incomplete':
|
|
case 'incomplete_expired':
|
|
return SubscriptionStatus.INACTIVE;
|
|
default:
|
|
return SubscriptionStatus.INACTIVE;
|
|
}
|
|
}
|
|
|
|
export async function syncSubscriptionFromStripe(
|
|
subscription: Stripe.Subscription,
|
|
userId: string,
|
|
): Promise<void> {
|
|
const priceId = subscription.items.data[0]?.price?.id ?? null;
|
|
const resolvedTier = priceId ? await priceIdToTier(priceId) : null;
|
|
|
|
const tierFromMetadata =
|
|
(subscription.metadata?.tier as string | undefined) ??
|
|
(subscription.metadata?.tier as string | undefined);
|
|
|
|
let tier: SubscriptionTier;
|
|
if (resolvedTier) {
|
|
tier = resolvedTier as SubscriptionTier;
|
|
} else if (tierFromMetadata === 'PRO' || tierFromMetadata === 'BUSINESS' || tierFromMetadata === 'ENTERPRISE') {
|
|
tier = tierFromMetadata as SubscriptionTier;
|
|
} else {
|
|
tier = SubscriptionTier.BASIC;
|
|
}
|
|
|
|
const status = mapStripeStatus(subscription.status);
|
|
|
|
const currentPeriodStartTimestamp =
|
|
(subscription as any).current_period_start ??
|
|
(subscription as any).items?.data?.[0]?.current_period_start ??
|
|
(subscription as any).start_date ??
|
|
Math.floor(Date.now() / 1000);
|
|
|
|
const currentPeriodEndTimestamp =
|
|
(subscription as any).current_period_end ??
|
|
(subscription as any).items?.data?.[0]?.current_period_end ??
|
|
(currentPeriodStartTimestamp + 30 * 24 * 3600);
|
|
|
|
const currentPeriodStart = new Date(currentPeriodStartTimestamp * 1000);
|
|
const currentPeriodEnd = new Date(currentPeriodEndTimestamp * 1000);
|
|
|
|
await prisma.subscription.upsert({
|
|
where: { stripeSubscriptionId: subscription.id },
|
|
update: {
|
|
tier,
|
|
status,
|
|
stripeCustomerId: typeof subscription.customer === 'string' ? subscription.customer : subscription.customer.id,
|
|
stripePriceId: priceId,
|
|
currentPeriodStart,
|
|
currentPeriodEnd,
|
|
cancelAtPeriodEnd: subscription.cancel_at_period_end,
|
|
canceledAt: subscription.canceled_at ? new Date(subscription.canceled_at * 1000) : null,
|
|
trialEndsAt: subscription.trial_end ? new Date(subscription.trial_end * 1000) : null,
|
|
updatedAt: new Date(),
|
|
},
|
|
create: {
|
|
userId,
|
|
tier,
|
|
status,
|
|
stripeCustomerId: typeof subscription.customer === 'string' ? subscription.customer : subscription.customer.id,
|
|
stripeSubscriptionId: subscription.id,
|
|
stripePriceId: priceId,
|
|
currentPeriodStart,
|
|
currentPeriodEnd,
|
|
cancelAtPeriodEnd: subscription.cancel_at_period_end,
|
|
canceledAt: subscription.canceled_at ? new Date(subscription.canceled_at * 1000) : null,
|
|
trialEndsAt: subscription.trial_end ? new Date(subscription.trial_end * 1000) : null,
|
|
},
|
|
});
|
|
}
|
|
|
|
export async function handleSubscriptionDeleted(subscription: Stripe.Subscription): Promise<void> {
|
|
const existing = await prisma.subscription.findUnique({
|
|
where: { stripeSubscriptionId: subscription.id },
|
|
});
|
|
if (!existing) return;
|
|
|
|
await prisma.subscription.update({
|
|
where: { stripeSubscriptionId: subscription.id },
|
|
data: {
|
|
status: SubscriptionStatus.CANCELED,
|
|
cancelAtPeriodEnd: false,
|
|
canceledAt: subscription.canceled_at ? new Date(subscription.canceled_at * 1000) : new Date(),
|
|
updatedAt: new Date(),
|
|
},
|
|
});
|
|
}
|
|
|
|
export async function resolveUserIdFromStripeEvent(
|
|
subscription: Stripe.Subscription,
|
|
): Promise<string | null> {
|
|
const metaUserId = subscription.metadata?.userId as string | undefined;
|
|
if (metaUserId) return metaUserId;
|
|
|
|
const customerId = typeof subscription.customer === 'string'
|
|
? subscription.customer
|
|
: subscription.customer.id;
|
|
|
|
const existing = await prisma.subscription.findFirst({
|
|
where: { stripeCustomerId: customerId },
|
|
select: { userId: true },
|
|
});
|
|
|
|
return existing?.userId ?? null;
|
|
}
|