Payments¶
Setup¶
-
Sign up for a Stripe account
-
Configure your Stripe account settings:
- Add website URL in [Settings] > [Public Details]
- Set logo & colors in [Settings] > [Branding]
- Enable payment & refund emails in [Settings] > [Customer Emails]
-
Enable customer portal in [Settings] > [Customer Portal]
-
Enable Test Mode
-
Create a product and copy its price ID (format: price_1ju5GD464564) to
STRIPE_PLANS
insettings.py
. The first price_id is used in production, the second in development/DEBUG=True
mode.config/settings.pySTRIPE_PLANS = [ { 'name': 'Your App Starter', 'description': 'For people who want to get started with Your App', 'price_anchor': 199, 'price': 99, 'features': [ '✅ Django boilerplate', '✅ SEO & Blog', '✅ Mailgun emails', '✅ Stripe', '✅ SQLite optimized', '✅ Google Oauth & Magic Links', '✅ Components & animations', '❌ Lifetime updates', ], 'price_id': 'price_1Px8W3HoFkHhzrx3ldgQnapM' if not DEBUG else 'price_1234', 'coupon_id': '8acSm2ml' if not DEBUG else 'asdf', }, # ...
Bulk Creation via API
Use the scripts in
example.ipynb
to create stripe products, prices and coupons based on yoursettings.py
plans. Copy the generatedprice_id
s andcoupon_id
s back tosettings.py
. -
Get API keys from [Developers] section:
- Add public key to
STRIPE_PUBLIC_KEY
in.env.local
-
Add private key to
STRIPE_SECRET_KEY
in.env.local
-
Set up local webhook:
Add the signing secret to
STRIPE_WEBHOOK_SECRET
in.env.local
Production Setup
- Disable Test Mode
- Add production API keys to
.env.prod
: STRIPE_PUBLIC_KEY
STRIPE_SECRET_KEY
- Configure production webhook:
- URL: https://your-domain.com/webhook/ - do not forget the trailing slash
- Required event: checkout.session.completed
- Add webhook signing secret to
STRIPE_WEBHOOK_SECRET
- Optional: Set monthly payout date in [Balance] > [Manage Payouts]
- Optional: Enable customer emails in [Settings] > [Customers Emails]
Stripe Events¶
Stripe webhook event handling
match event['type']:
case 'checkout.session.completed':
# First payment successful - enable access
case 'checkout.session.expired':
# Session expired before completion
# Optional: Send recovery email to encourage completion
case 'customer.subscription.updated':
# Subscription plan changes (upgrade/downgrade/pending cancellation)
case 'customer.subscription.deleted':
# Subscription ended - revoke access
case 'invoice.paid':
# Successful payment of an invoice (e.g. a recurring payment)
# Grant access to the product, make sure the invoice is price_id matches the plan the user is subscribed to (user.price_id)
# Note: Preferred over 'customer.subscription.created' for subscription status
# Ref: https://x.com/levelsio/status/1853529566490583408
case 'invoice.payment_failed':
# Failed payment handling options:
# 1. Immediate access revocation
# 2. Grace period:
# - Let Stripe handle automatic retry emails (Smart Retries)
# - Wait for 'customer.subscription.deleted' after all retries
case _:
# Unhandled event type