Files
Momento/memento-note/lib/billing/sync-subscription-from-stripe.ts
Antigravity ee70e74bf5
All checks were successful
CI / Lint, Unit Tests & Build (push) Successful in 5m39s
CI / Deploy production (on server) (push) Successful in 22s
fix: 5 bugs critiques de l'éditeur (Phase 1 audit)
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)
2026-06-20 15:48:18 +00:00

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;
}