Claude-skill-registry billing-system-expert
Expert knowledge on Stripe integration, subscription plans (Glow Up, Viral Surge, Fame Flex), trial logic, plan enforcement, webhooks, and billing synchronization. Use this skill when user asks about "subscription", "billing", "stripe", "payment", "plan limits", "trial", "upgrade", "downgrade", "webhook", or "plan enforcement".
git clone https://github.com/majiayu000/claude-skill-registry
T=$(mktemp -d) && git clone --depth=1 https://github.com/majiayu000/claude-skill-registry "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/data/billing-system-expert" ~/.claude/skills/majiayu000-claude-skill-registry-billing-system-expert && rm -rf "$T"
skills/data/billing-system-expert/SKILL.mdBilling System Expert
You are an expert in the billing and subscription system for this influencer discovery platform. This skill provides comprehensive knowledge about Stripe integration, subscription plans, trial management, plan enforcement, and webhook handling.
When To Use This Skill
This skill activates when users:
- Ask about subscription plans or pricing
- Need to debug billing issues or sync problems
- Work with Stripe webhooks or payment flows
- Implement plan limit enforcement
- Debug trial period activation or conversion
- Investigate upgrade/downgrade flows
- Need to understand payment method handling
- Troubleshoot stuck onboarding or billing states
Core Knowledge
Subscription Plans
The platform offers three paid tiers plus a free tier:
Plan Structure:
// From /lib/db/schema.ts - subscription_plans table { planKey: 'glow_up' | 'viral_surge' | 'fame_flex' | 'free', campaignsLimit: number, // -1 = unlimited creatorsLimit: number, // -1 = unlimited features: jsonb, priceMonthly: number, priceYearly: number }
Plan Limits (from /lib/services/plan-enforcement.ts):
-
Glow Up (Entry Level)
- Campaigns: 3
- Creators: 1,000/month
- Stripe Price IDs:
- Monthly:
process.env.STRIPE_GLOW_UP_MONTHLY_PRICE_ID - Yearly:
process.env.STRIPE_GLOW_UP_YEARLY_PRICE_ID
- Monthly:
-
Viral Surge (Pro Level)
- Campaigns: 10
- Creators: 10,000/month
- Stripe Price IDs:
- Monthly:
process.env.STRIPE_VIRAL_SURGE_MONTHLY_PRICE_ID - Yearly:
process.env.STRIPE_VIRAL_SURGE_YEARLY_PRICE_ID
- Monthly:
-
Fame Flex (Unlimited)
- Campaigns: Unlimited (-1)
- Creators: Unlimited (-1)
- Stripe Price IDs:
- Monthly:
process.env.STRIPE_FAME_FLEX_MONTHLY_PRICE_ID - Yearly:
process.env.STRIPE_FAME_FLEX_YEARLY_PRICE_ID
- Monthly:
-
Free Tier (Default)
- Campaigns: 1 (or 0, check implementation)
- Creators: 50
- No Stripe subscription required
Plan Enforcement Logic
Service:
/lib/services/plan-enforcement.ts
Key Functions:
class PlanEnforcementService { // Get user's plan limits static async getPlanLimits(userId: string): Promise<PlanLimits | null> // Get current usage static async getCurrentUsage(userId: string): Promise<UsageInfo | null> // Validate campaign creation static async validateCampaignCreation(userId: string): Promise<{ allowed: boolean; reason?: string; usage?: UsageInfo; }> // Validate job creation (creator searches) static async validateJobCreation(userId: string, expectedCreators: number): Promise<{ allowed: boolean; reason?: string; usage?: UsageInfo; adjustedLimit?: number; }> // Track campaign creation static async trackCampaignCreated(userId: string): Promise<void> // Track creators found static async trackCreatorsFound(userId: string, creatorCount: number): Promise<void> }
Usage Tracking:
- Campaigns: Total count (not monthly reset)
- Creators: Monthly count (resets first day of month)
Example Enforcement:
// Before creating campaign const validation = await PlanEnforcementService.validateCampaignCreation(userId); if (!validation.allowed) { return NextResponse.json( { error: validation.reason, usage: validation.usage }, { status: 403 } ); } // Create campaign... // Track usage await PlanEnforcementService.trackCampaignCreated(userId);
Dev Bypass (Non-Production Only):
// Environment variable bypass PLAN_VALIDATION_BYPASS=all // or "campaigns,creators" // Request header bypass headers: { 'x-plan-bypass': 'all' // or "campaigns,creators" }
Stripe Integration
Stripe Service:
/lib/stripe/stripe-service.ts
Webhook Handler: /app/api/stripe/webhook/route.ts
Key Webhook Events:
-
checkout.session.completed
- Triggered after successful checkout
- Finalizes onboarding
- Links Stripe customer to user
- Triggers trial activation
-
customer.subscription.created
- Triggered when subscription is created
- Updates user plan in database
- Sets plan limits
- Activates trial if applicable
- CRITICAL: Must resolve plan from price ID
-
customer.subscription.updated
- Triggered on plan changes or status updates
- Handles trial → paid conversion
- Updates plan limits on upgrades
- Handles cancellation scheduling
-
customer.subscription.deleted
- Triggered when subscription ends
- Resets user to free tier
- Clears plan limits
-
customer.subscription.trial_will_end
- Triggered 3 days before trial ends
- Can trigger reminder emails
-
invoice.payment_succeeded
- Triggered on successful payment
- Updates billing sync status
-
invoice.payment_failed
- Triggered on failed payment
- Can trigger dunning emails
-
setup_intent.succeeded
- Triggered when payment method is set up
- Links payment method to customer
-
payment_method.attached
- Triggered when card is added
- Stores card details (last4, brand, exp)
Price ID to Plan Mapping
Critical Logic (from webhook handler):
function getPlanFromPriceId(priceId: string): string { const priceIdToplan = { [process.env.STRIPE_GLOW_UP_MONTHLY_PRICE_ID!]: 'glow_up', [process.env.STRIPE_GLOW_UP_YEARLY_PRICE_ID!]: 'glow_up', [process.env.STRIPE_VIRAL_SURGE_MONTHLY_PRICE_ID!]: 'viral_surge', [process.env.STRIPE_VIRAL_SURGE_YEARLY_PRICE_ID!]: 'viral_surge', [process.env.STRIPE_FAME_FLEX_MONTHLY_PRICE_ID!]: 'fame_flex', [process.env.STRIPE_FAME_FLEX_YEARLY_PRICE_ID!]: 'fame_flex', }; return priceIdToplan[priceId] || 'unknown'; }
CRITICAL: Never use arbitrary fallback plans. If plan cannot be determined, throw error and retry webhook.
Trial System
Trial Logic:
/lib/services/trial-status-calculator.ts
Trial States:
: No trial startedinactive
: Currently in trial periodactive
: Trial ended without conversionexpired
: Trial converted to paid subscriptionconverted
Trial Activation:
// During subscription creation webhook if (subscription.trial_end && subscription.status === 'trialing') { await updateUserProfile(userId, { trialStatus: 'active', trialStartDate: new Date(), trialEndDate: new Date(subscription.trial_end * 1000), onboardingStep: 'completed' }); }
Trial Conversion:
// During subscription update webhook if (subscription.status === 'active' && user.trialStatus === 'active') { await updateUserProfile(userId, { trialStatus: 'converted', trialConversionDate: new Date() }); }
Billing Sync States
Field:
billingSyncStatus in user_profiles table
Possible Values:
- Subscription created successfullywebhook_subscription_created
- Subscription updatedwebhook_subscription_updated
- Subscription cancelledwebhook_subscription_deleted
- Trial ending soonwebhook_trial_will_end
- Payment successfulwebhook_payment_succeeded
- Payment failedwebhook_payment_failed
- Payment method addedwebhook_setup_intent_succeeded
- Card attachedwebhook_payment_method_attached
- Webhook failed, used fallbackwebhook_emergency_fallback
Checking Sync Status:
node scripts/inspect-user-state.js --email user@example.com
Common Patterns
Pattern 1: Enforcing Plan Limits Before Action
// Good: Always validate before expensive operations export async function POST(req: Request) { const { userId } = await getAuthOrTest(); // Validate BEFORE creating campaign const validation = await PlanEnforcementService.validateCampaignCreation(userId); if (!validation.allowed) { return NextResponse.json( { error: validation.reason, usage: validation.usage, upgrade_required: true }, { status: 403 } ); } // Create campaign... const campaign = await db.insert(campaigns).values({ /* ... */ }); // Track usage AFTER success await PlanEnforcementService.trackCampaignCreated(userId); return NextResponse.json({ campaign }); }
When to use: Before any action that counts against limits
Pattern 2: Webhook Signature Verification
// Good: Always verify webhook signatures in production export async function POST(req: NextRequest) { const body = await req.text(); const signature = req.headers.get('stripe-signature'); if (!signature) { return NextResponse.json({ error: 'No signature' }, { status: 400 }); } // Validate signature using Stripe SDK const event = StripeService.validateWebhookSignature(body, signature); // Process webhook event... switch (event.type) { case 'customer.subscription.created': await handleSubscriptionCreated(event.data.object); break; // ... } return NextResponse.json({ received: true }); }
When to use: All Stripe webhook endpoints
Pattern 3: Resolving Plan from Subscription
// Good: Multiple fallback strategies for plan resolution async function resolvePlanFromSubscription(subscription: Stripe.Subscription): Promise<string> { // Strategy 1: Check metadata let planId = subscription.metadata.plan || subscription.metadata.planId; // Strategy 2: Derive from price ID if (!planId || planId === 'unknown') { const priceId = subscription.items.data[0]?.price?.id; if (priceId) { planId = getPlanFromPriceId(priceId); } } // Strategy 3: Throw error and retry webhook if (!planId || planId === 'unknown') { throw new Error( `Cannot determine plan for subscription ${subscription.id}. Will retry.` ); } return planId; }
When to use: Processing subscription webhooks
Anti-Patterns (Avoid These)
Anti-Pattern 1: Using Arbitrary Fallback Plans
// BAD: Can cause upgrade bugs where users get wrong plan function getPlanFromPriceId(priceId: string): string { const mapping = { /* ... */ }; return mapping[priceId] || 'glow_up'; // WRONG! }
Why it's bad: User pays for Fame Flex but gets Glow Up limits
Do this instead:
// GOOD: Throw error and retry webhook function getPlanFromPriceId(priceId: string): string { const mapping = { /* ... */ }; const plan = mapping[priceId]; if (!plan) { throw new Error(`Unknown price ID: ${priceId}. Webhook will retry.`); } return plan; }
Anti-Pattern 2: Tracking Usage Before Validation
// BAD: User exceeds limit but usage is tracked anyway await PlanEnforcementService.trackCampaignCreated(userId); const validation = await PlanEnforcementService.validateCampaignCreation(userId); if (!validation.allowed) { return NextResponse.json({ error: 'Limit exceeded' }, { status: 403 }); }
Why it's bad: Usage counter increases even when action fails
Do this instead:
// GOOD: Validate → Action → Track const validation = await PlanEnforcementService.validateCampaignCreation(userId); if (!validation.allowed) { return NextResponse.json({ error: 'Limit exceeded' }, { status: 403 }); } const campaign = await createCampaign(/* ... */); await PlanEnforcementService.trackCampaignCreated(userId);
Anti-Pattern 3: Skipping Webhook Verification
// BAD: Accepting unauthenticated webhooks export async function POST(req: Request) { const event = await req.json(); // Process without verification - DANGEROUS! await handleSubscriptionCreated(event.data.object); }
Why it's bad: Anyone can forge webhooks and manipulate plans
Do this instead:
// GOOD: Always verify signatures const body = await req.text(); const signature = req.headers.get('stripe-signature'); if (!signature) { return NextResponse.json({ error: 'No signature' }, { status: 400 }); } const event = StripeService.validateWebhookSignature(body, signature);
Troubleshooting Guide
Problem: User Plan Not Updating After Payment
Symptoms:
- User completed checkout but still shows free plan
- Stripe dashboard shows active subscription
- User cannot access paid features
Diagnosis:
- Check webhook delivery in Stripe dashboard
- Verify webhook endpoint is accessible
- Check
in databasebilling_sync_status - Look for errors in webhook logs
# Check user state node scripts/inspect-user-state.js --email user@example.com # Check webhook logs (if available) grep "STRIPE-WEBHOOK" logs/app.log | grep "ERROR"
Solution:
# Manual sync (use admin endpoint or script) curl -X POST http://localhost:3000/api/billing/sync-stripe \ -H "x-dev-auth: dev-bypass" \ -H "Content-Type: application/json" \ -d '{"userId": "user_xxx"}'
Problem: Plan Limits Not Enforced
Symptoms:
- User exceeds campaign limit but can create more
- Creator count not tracked
- No "upgrade required" error
Diagnosis:
- Check if validation is called before action
- Verify
is not set in productionPLAN_VALIDATION_BYPASS - Check plan limits in
tablesubscription_plans - Verify usage tracking is called after action
Solution:
// Add enforcement to endpoint import { PlanEnforcementService } from '@/lib/services/plan-enforcement'; export async function POST(req: Request) { const { userId } = await getAuthOrTest(); // ADD THIS const validation = await PlanEnforcementService.validateCampaignCreation(userId); if (!validation.allowed) { return NextResponse.json({ error: validation.reason }, { status: 403 }); } // Create campaign... // ADD THIS await PlanEnforcementService.trackCampaignCreated(userId); return NextResponse.json({ success: true }); }
Problem: Trial Not Activating After Checkout
Symptoms:
- User completed checkout with trial
istrial_statusinactive
notonboarding_stepcompleted
Diagnosis:
- Check if
webhook firedcheckout.session.completed - Verify subscription has
timestamptrial_end - Check
was calledfinalizeOnboarding - Look for errors in webhook logs
Solution:
# Manually complete onboarding node scripts/complete-onboarding-and-activate-plan.js user_xxx
Or trigger via API:
curl -X POST http://localhost:3000/api/onboarding/complete \ -H "x-dev-auth: dev-bypass" \ -H "x-dev-user-id: user_xxx"
Problem: Webhook Failing with "Unknown Price ID"
Symptoms:
- Webhook returns 500 error
- Logs show "Cannot determine plan"
- User plan not updated
Diagnosis:
- Check if price ID exists in Stripe dashboard
- Verify
has all.env
variablesSTRIPE_*_PRICE_ID - Check for typos in environment variables
- Ensure webhook uses correct price ID mapping
Solution:
# Verify environment variables grep "STRIPE_.*PRICE_ID" .env.local # Expected output: STRIPE_GLOW_UP_MONTHLY_PRICE_ID=price_xxx STRIPE_GLOW_UP_YEARLY_PRICE_ID=price_yyy # ... etc
If missing, add to
.env.local and restart server.
Problem: User Upgraded But Still Has Old Limits
Symptoms:
- User paid for Viral Surge but has Glow Up limits
is correct butcurrent_plan
is wrongplan_campaigns_limit- Can't create more campaigns despite upgrade
Diagnosis:
- Check
webhook firedsubscription.updated - Verify plan limits are fetched from
tablesubscription_plans - Check webhook sets
andplanCampaignsLimitplanCreatorsLimit
Solution:
// In webhook handler, ensure limits are updated: const planDetails = await db.query.subscriptionPlans.findFirst({ where: eq(subscriptionPlans.planKey, planId) }); await updateUserProfile(userId, { currentPlan: planId, planCampaignsLimit: planDetails?.campaignsLimit || 0, planCreatorsLimit: planDetails?.creatorsLimit || 0 });
Related Files
- Plan validation and usage tracking/lib/services/plan-enforcement.ts
- Billing operations/lib/services/billing-service.ts
- Stripe client wrapper/lib/stripe/stripe-service.ts
- Webhook event handlers/app/api/stripe/webhook/route.ts
- Get billing status/app/api/billing/status/route.ts
- Manual sync endpoint/app/api/billing/sync-stripe/route.ts
- Campaign validation endpoint/app/api/campaigns/can-create/route.ts
- Diagnostic script/scripts/inspect-user-state.js
- Fix script/scripts/fix-user-billing-state.js
Testing & Validation
Test Plan Enforcement:
# Create user with specific plan node scripts/complete-onboarding-and-activate-plan.js user_xxx glow_up # Try creating campaigns curl -X POST http://localhost:3000/api/campaigns \ -H "x-dev-user-id: user_xxx" \ -d '{"name": "Test Campaign 1"}' # Check usage curl http://localhost:3000/api/billing/status \ -H "x-dev-user-id: user_xxx"
Test Stripe Webhooks Locally:
# Install Stripe CLI stripe listen --forward-to localhost:3000/api/stripe/webhook # Trigger test webhook stripe trigger customer.subscription.created
Expected Behavior:
- Webhook received and verified
- User plan updated in database
- Plan limits set correctly
- Billing sync status updated
- No errors in logs
Subscription Flow Diagram
User Checkout ↓ Stripe Checkout Session ↓ checkout.session.completed (webhook) ↓ Link Stripe Customer to User ↓ customer.subscription.created (webhook) ↓ Resolve Plan from Price ID ↓ Update user_profiles: - current_plan - plan_campaigns_limit - plan_creators_limit - stripe_subscription_id - subscription_status - trial_status (if trial) ↓ Finalize Onboarding ↓ User Can Access Platform
Additional Resources
- Stripe Webhooks Documentation
- Stripe Subscriptions Guide
- Stripe Testing
- Internal:
(if exists)/docs/upgrade-user.md