Files
bakery-ia/shared/clients/stripe_client.py

1022 lines
49 KiB
Python
Raw Normal View History

2025-09-25 14:30:47 +02:00
"""
Stripe Payment Provider Implementation
This module implements the PaymentProvider interface for Stripe
"""
import stripe
import structlog
2026-01-11 07:50:34 +01:00
import uuid
2025-09-25 14:30:47 +02:00
from typing import Dict, Any, Optional
from datetime import datetime
from .payment_client import PaymentProvider, PaymentCustomer, PaymentMethod, Subscription, Invoice
logger = structlog.get_logger()
2026-01-14 13:15:48 +01:00
class PaymentVerificationError(Exception):
"""Exception raised when payment method verification fails"""
pass
2025-09-25 14:30:47 +02:00
class StripeProvider(PaymentProvider):
"""
Stripe implementation of the PaymentProvider interface
"""
def __init__(self, api_key: str, webhook_secret: Optional[str] = None):
"""
Initialize the Stripe provider with API key
"""
stripe.api_key = api_key
self.webhook_secret = webhook_secret
async def create_customer(self, customer_data: Dict[str, Any]) -> PaymentCustomer:
"""
2026-01-11 07:50:34 +01:00
Create a customer in Stripe with idempotency key
2025-09-25 14:30:47 +02:00
"""
try:
2026-01-11 07:50:34 +01:00
idempotency_key = f"create_customer_{uuid.uuid4()}"
logger.info("Creating Stripe customer",
email=customer_data.get('email'),
customer_name=customer_data.get('name'))
2025-09-25 14:30:47 +02:00
stripe_customer = stripe.Customer.create(
email=customer_data.get('email'),
name=customer_data.get('name'),
phone=customer_data.get('phone'),
2026-01-11 07:50:34 +01:00
metadata=customer_data.get('metadata', {}),
idempotency_key=idempotency_key
2025-09-25 14:30:47 +02:00
)
2026-01-11 07:50:34 +01:00
logger.info("Stripe customer created successfully",
customer_id=stripe_customer.id)
2025-09-25 14:30:47 +02:00
return PaymentCustomer(
id=stripe_customer.id,
email=stripe_customer.email,
name=stripe_customer.name,
created_at=datetime.fromtimestamp(stripe_customer.created)
)
except stripe.error.StripeError as e:
2026-01-11 07:50:34 +01:00
logger.error("Failed to create Stripe customer",
error=str(e),
error_type=type(e).__name__,
email=customer_data.get('email'))
2025-09-25 14:30:47 +02:00
raise e
2026-01-14 13:15:48 +01:00
async def create_subscription(self, customer_id: str, plan_id: str, payment_method_id: str, trial_period_days: Optional[int] = None) -> Dict[str, Any]:
2025-09-25 14:30:47 +02:00
"""
2026-01-11 07:50:34 +01:00
Create a subscription in Stripe with idempotency and enhanced error handling
2026-01-14 13:15:48 +01:00
Returns:
Dictionary containing subscription details and any required authentication actions
2025-09-25 14:30:47 +02:00
"""
try:
2026-01-11 07:50:34 +01:00
subscription_idempotency_key = f"create_subscription_{uuid.uuid4()}"
payment_method_idempotency_key = f"attach_pm_{uuid.uuid4()}"
customer_update_idempotency_key = f"update_customer_{uuid.uuid4()}"
2026-01-14 13:15:48 +01:00
logger.info("Creating Stripe subscription",
2026-01-11 07:50:34 +01:00
customer_id=customer_id,
plan_id=plan_id,
payment_method_id=payment_method_id)
2026-01-14 13:15:48 +01:00
2026-01-13 22:22:38 +01:00
# Attach payment method to customer with idempotency and error handling
try:
stripe.PaymentMethod.attach(
payment_method_id,
customer=customer_id,
idempotency_key=payment_method_idempotency_key
)
logger.info("Payment method attached to customer",
customer_id=customer_id,
payment_method_id=payment_method_id)
except stripe.error.InvalidRequestError as e:
# Payment method may already be attached
if 'already been attached' in str(e):
logger.warning("Payment method already attached to customer",
customer_id=customer_id,
payment_method_id=payment_method_id)
else:
raise
2026-01-14 13:15:48 +01:00
2026-01-11 07:50:34 +01:00
# Set customer's default payment method with idempotency
2025-09-25 14:30:47 +02:00
stripe.Customer.modify(
customer_id,
invoice_settings={
'default_payment_method': payment_method_id
2026-01-11 07:50:34 +01:00
},
idempotency_key=customer_update_idempotency_key
2025-09-25 14:30:47 +02:00
)
2026-01-14 13:15:48 +01:00
logger.info("Customer default payment method updated",
2026-01-11 07:50:34 +01:00
customer_id=customer_id)
2026-01-14 13:15:48 +01:00
# Verify payment method before creating subscription (especially important for trial periods)
# This ensures the card is valid and can be charged after the trial
# Use SetupIntent for card verification without immediate payment
#
# CRITICAL FOR 3DS SUPPORT:
# We do NOT confirm the SetupIntent here because:
# 1. If 3DS is required, we need the frontend to handle the redirect
# 2. The frontend will confirm the SetupIntent with the return_url
# 3. After 3DS completion, frontend will call us again with the verified payment method
#
# This prevents creating subscriptions with unverified payment methods.
setup_intent_params = {
'customer': customer_id,
'payment_method': payment_method_id,
'usage': 'off_session', # Allow charging without customer presence after verification
'idempotency_key': f"verify_payment_{uuid.uuid4()}",
'expand': ['payment_method'],
'confirm': False # Explicitly don't confirm yet - frontend will handle 3DS
}
try:
# Create SetupIntent WITHOUT confirming
# Frontend will confirm with return_url if 3DS is needed
verification_intent = stripe.SetupIntent.create(**setup_intent_params)
logger.info("SetupIntent created for payment method verification",
setup_intent_id=verification_intent.id,
status=verification_intent.status,
payment_method_id=payment_method_id)
# ALWAYS return the SetupIntent for frontend to confirm
# Frontend will handle 3DS if needed, or confirm immediately if not
# This ensures proper 3DS flow for all cards
return {
'requires_action': True, # Frontend must always confirm
'action_type': 'setup_intent_confirmation',
'client_secret': verification_intent.client_secret,
'setup_intent_id': verification_intent.id,
'status': verification_intent.status,
'customer_id': customer_id,
'payment_method_id': payment_method_id,
'plan_id': plan_id,
'trial_period_days': trial_period_days,
'message': 'Payment method verification required. Frontend must confirm SetupIntent.'
}
except stripe.error.CardError as e:
logger.error("Payment method verification failed",
error=str(e),
code=e.code,
decline_code=e.decline_code)
raise PaymentVerificationError(f"Card verification failed: {e.user_message or str(e)}")
except Exception as e:
logger.error("Unexpected error during payment verification",
error=str(e))
raise PaymentVerificationError(f"Payment verification error: {str(e)}")
2025-09-25 14:30:47 +02:00
# Create subscription with trial period if specified
subscription_params = {
'customer': customer_id,
'items': [{'price': plan_id}],
'default_payment_method': payment_method_id,
2026-01-11 07:50:34 +01:00
'idempotency_key': subscription_idempotency_key,
'expand': ['latest_invoice.payment_intent']
2025-09-25 14:30:47 +02:00
}
2026-01-14 13:15:48 +01:00
2025-09-25 14:30:47 +02:00
if trial_period_days:
subscription_params['trial_period_days'] = trial_period_days
2026-01-14 13:15:48 +01:00
logger.info("Subscription includes trial period",
2026-01-11 07:50:34 +01:00
trial_period_days=trial_period_days)
2026-01-14 13:15:48 +01:00
2025-09-25 14:30:47 +02:00
stripe_subscription = stripe.Subscription.create(**subscription_params)
2026-01-13 22:22:38 +01:00
# Handle period dates for trial vs active subscriptions
2026-01-14 13:15:48 +01:00
if stripe_subscription.status == 'trialing' and hasattr(stripe_subscription, 'items') and stripe_subscription.items:
# Access items properly - stripe_subscription.items is typically a list-like object
items_list = stripe_subscription.items.data if hasattr(stripe_subscription.items, 'data') else stripe_subscription.items
if items_list and len(items_list) > 0:
first_item = items_list[0] if isinstance(items_list, list) else items_list
current_period_start = first_item.current_period_start
current_period_end = first_item.current_period_end
logger.info("Stripe trial subscription created successfully",
subscription_id=stripe_subscription.id,
status=stripe_subscription.status,
trial_end=stripe_subscription.trial_end,
current_period_end=current_period_end)
2026-01-13 22:22:38 +01:00
else:
current_period_start = stripe_subscription.current_period_start
current_period_end = stripe_subscription.current_period_end
logger.info("Stripe subscription created successfully",
subscription_id=stripe_subscription.id,
status=stripe_subscription.status,
current_period_end=current_period_end)
2026-01-14 13:15:48 +01:00
# Check if payment requires action (3D Secure, SCA)
requires_action = False
client_secret = None
payment_intent_status = None
if stripe_subscription.latest_invoice:
if hasattr(stripe_subscription.latest_invoice, 'payment_intent') and stripe_subscription.latest_invoice.payment_intent:
payment_intent = stripe_subscription.latest_invoice.payment_intent
payment_intent_status = payment_intent.status
if payment_intent.status in ['requires_action', 'requires_source_action']:
requires_action = True
client_secret = payment_intent.client_secret
logger.info("Subscription payment requires authentication",
subscription_id=stripe_subscription.id,
payment_intent_id=payment_intent.id,
status=payment_intent.status)
# Calculate trial end if this is a trial subscription
trial_end_timestamp = None
if stripe_subscription.status == 'trialing' and hasattr(stripe_subscription, 'trial_end'):
trial_end_timestamp = stripe_subscription.trial_end
subscription_obj = Subscription(
2025-09-25 14:30:47 +02:00
id=stripe_subscription.id,
customer_id=stripe_subscription.customer,
2026-01-14 13:15:48 +01:00
plan_id=plan_id,
2025-09-25 14:30:47 +02:00
status=stripe_subscription.status,
2026-01-13 22:22:38 +01:00
current_period_start=datetime.fromtimestamp(current_period_start),
current_period_end=datetime.fromtimestamp(current_period_end),
2026-01-14 13:15:48 +01:00
created_at=datetime.fromtimestamp(stripe_subscription.created),
billing_cycle_anchor=datetime.fromtimestamp(stripe_subscription.billing_cycle_anchor) if hasattr(stripe_subscription, 'billing_cycle_anchor') else None,
cancel_at_period_end=stripe_subscription.cancel_at_period_end if hasattr(stripe_subscription, 'cancel_at_period_end') else None,
payment_intent_id=stripe_subscription.latest_invoice.payment_intent.id if (hasattr(stripe_subscription.latest_invoice, 'payment_intent') and stripe_subscription.latest_invoice.payment_intent) else None,
payment_intent_status=payment_intent_status,
payment_intent_client_secret=client_secret,
requires_action=requires_action,
trial_end=datetime.fromtimestamp(trial_end_timestamp) if trial_end_timestamp else None,
billing_interval="monthly" # Default, should be extracted from plan
2025-09-25 14:30:47 +02:00
)
2026-01-14 13:15:48 +01:00
return {
'subscription': subscription_obj,
'requires_action': requires_action,
'client_secret': client_secret,
'payment_intent_status': payment_intent_status
}
2026-01-11 07:50:34 +01:00
except stripe.error.CardError as e:
logger.error("Card error during subscription creation",
error=str(e),
error_code=e.code,
decline_code=e.decline_code,
customer_id=customer_id)
raise e
except stripe.error.InvalidRequestError as e:
logger.error("Invalid request during subscription creation",
error=str(e),
param=e.param,
customer_id=customer_id)
raise e
2025-09-25 14:30:47 +02:00
except stripe.error.StripeError as e:
2026-01-11 07:50:34 +01:00
logger.error("Failed to create Stripe subscription",
error=str(e),
error_type=type(e).__name__,
customer_id=customer_id,
plan_id=plan_id)
2025-09-25 14:30:47 +02:00
raise e
2026-01-14 13:15:48 +01:00
async def complete_subscription_after_setup_intent(
self,
setup_intent_id: str,
customer_id: str,
plan_id: str,
payment_method_id: str,
trial_period_days: Optional[int] = None
) -> Dict[str, Any]:
"""
Complete subscription creation after SetupIntent has been confirmed by frontend
This method should be called after the frontend successfully confirms a SetupIntent
(with or without 3DS). It verifies the SetupIntent is in 'succeeded' status and
then creates the subscription with the now-verified payment method.
Args:
setup_intent_id: The SetupIntent ID that was confirmed
customer_id: Stripe customer ID
plan_id: Subscription plan/price ID
payment_method_id: Payment method ID (should match SetupIntent)
trial_period_days: Optional trial period
Returns:
Dictionary containing subscription details
"""
try:
subscription_idempotency_key = f"complete_subscription_{uuid.uuid4()}"
logger.info("Completing subscription after SetupIntent confirmation",
setup_intent_id=setup_intent_id,
customer_id=customer_id,
plan_id=plan_id)
# Verify the SetupIntent is in succeeded status
setup_intent = stripe.SetupIntent.retrieve(setup_intent_id)
if setup_intent.status != 'succeeded':
logger.error("SetupIntent not in succeeded status",
setup_intent_id=setup_intent_id,
status=setup_intent.status)
raise PaymentVerificationError(
f"SetupIntent must be in 'succeeded' status. Current status: {setup_intent.status}"
)
logger.info("SetupIntent verified as succeeded, creating subscription",
setup_intent_id=setup_intent_id)
# Payment method is already attached and verified via SetupIntent
# Now create the subscription
subscription_params = {
'customer': customer_id,
'items': [{'price': plan_id}],
'default_payment_method': payment_method_id,
'idempotency_key': subscription_idempotency_key,
'expand': ['latest_invoice.payment_intent']
}
if trial_period_days:
subscription_params['trial_period_days'] = trial_period_days
logger.info("Subscription includes trial period",
trial_period_days=trial_period_days)
stripe_subscription = stripe.Subscription.create(**subscription_params)
# Handle period dates for trial vs active subscriptions
if stripe_subscription.status == 'trialing' and hasattr(stripe_subscription, 'items') and stripe_subscription.items:
items_list = stripe_subscription.items.data if hasattr(stripe_subscription.items, 'data') else stripe_subscription.items
if items_list and len(items_list) > 0:
first_item = items_list[0] if isinstance(items_list, list) else items_list
current_period_start = first_item.current_period_start
current_period_end = first_item.current_period_end
logger.info("Stripe trial subscription created after SetupIntent",
subscription_id=stripe_subscription.id,
status=stripe_subscription.status,
trial_end=stripe_subscription.trial_end,
current_period_end=current_period_end)
else:
current_period_start = stripe_subscription.current_period_start
current_period_end = stripe_subscription.current_period_end
logger.info("Stripe subscription created after SetupIntent",
subscription_id=stripe_subscription.id,
status=stripe_subscription.status,
current_period_end=current_period_end)
# Calculate trial end if this is a trial subscription
trial_end_timestamp = None
if stripe_subscription.status == 'trialing' and hasattr(stripe_subscription, 'trial_end'):
trial_end_timestamp = stripe_subscription.trial_end
subscription_obj = Subscription(
id=stripe_subscription.id,
customer_id=stripe_subscription.customer,
plan_id=plan_id,
status=stripe_subscription.status,
current_period_start=datetime.fromtimestamp(current_period_start),
current_period_end=datetime.fromtimestamp(current_period_end),
created_at=datetime.fromtimestamp(stripe_subscription.created),
billing_cycle_anchor=datetime.fromtimestamp(stripe_subscription.billing_cycle_anchor) if hasattr(stripe_subscription, 'billing_cycle_anchor') else None,
cancel_at_period_end=stripe_subscription.cancel_at_period_end if hasattr(stripe_subscription, 'cancel_at_period_end') else None,
payment_intent_id=stripe_subscription.latest_invoice.payment_intent.id if (hasattr(stripe_subscription.latest_invoice, 'payment_intent') and stripe_subscription.latest_invoice.payment_intent) else None,
payment_intent_status=None,
payment_intent_client_secret=None,
requires_action=False,
trial_end=datetime.fromtimestamp(trial_end_timestamp) if trial_end_timestamp else None,
billing_interval="monthly"
)
return {
'subscription': subscription_obj,
'requires_action': False,
'client_secret': None,
'payment_intent_status': None,
'setup_intent_id': setup_intent_id
}
except stripe.error.StripeError as e:
logger.error("Failed to complete subscription after SetupIntent",
error=str(e),
setup_intent_id=setup_intent_id,
customer_id=customer_id)
raise e
async def update_payment_method(self, customer_id: str, payment_method_id: str) -> Dict[str, Any]:
2025-09-25 14:30:47 +02:00
"""
Update the payment method for a customer in Stripe
2026-01-14 13:15:48 +01:00
Returns:
Dictionary containing payment method details and any required authentication actions
2025-09-25 14:30:47 +02:00
"""
try:
2026-01-13 22:22:38 +01:00
# Attach payment method to customer with error handling
try:
stripe.PaymentMethod.attach(
payment_method_id,
customer=customer_id,
)
logger.info("Payment method attached for update",
customer_id=customer_id,
payment_method_id=payment_method_id)
except stripe.error.InvalidRequestError as e:
# Payment method may already be attached
if 'already been attached' in str(e):
logger.warning("Payment method already attached, skipping attach",
customer_id=customer_id,
payment_method_id=payment_method_id)
else:
raise
2025-09-25 14:30:47 +02:00
# Set as default payment method
2026-01-14 13:15:48 +01:00
customer = stripe.Customer.modify(
2025-09-25 14:30:47 +02:00
customer_id,
invoice_settings={
'default_payment_method': payment_method_id
}
)
2026-01-14 13:15:48 +01:00
2025-09-25 14:30:47 +02:00
stripe_payment_method = stripe.PaymentMethod.retrieve(payment_method_id)
2026-01-14 13:15:48 +01:00
# Get any active subscriptions that might need payment confirmation
subscriptions = stripe.Subscription.list(customer=customer_id, status='active', limit=1)
requires_action = False
client_secret = None
payment_intent_status = None
# Check if there's a subscription with pending payment that requires action
if subscriptions.data:
subscription = subscriptions.data[0]
if subscription.latest_invoice:
invoice = stripe.Invoice.retrieve(
subscription.latest_invoice,
expand=['payment_intent']
)
if invoice.payment_intent:
payment_intent = invoice.payment_intent
payment_intent_status = payment_intent.status
if payment_intent.status in ['requires_action', 'requires_source_action']:
requires_action = True
client_secret = payment_intent.client_secret
logger.info("Payment requires authentication",
customer_id=customer_id,
payment_intent_id=payment_intent.id,
status=payment_intent.status)
payment_method_details = getattr(stripe_payment_method, stripe_payment_method.type, {})
return {
'payment_method': {
'id': stripe_payment_method.id,
'type': stripe_payment_method.type,
'brand': payment_method_details.get('brand') if isinstance(payment_method_details, dict) else getattr(payment_method_details, 'brand', None),
'last4': payment_method_details.get('last4') if isinstance(payment_method_details, dict) else getattr(payment_method_details, 'last4', None),
'exp_month': payment_method_details.get('exp_month') if isinstance(payment_method_details, dict) else getattr(payment_method_details, 'exp_month', None),
'exp_year': payment_method_details.get('exp_year') if isinstance(payment_method_details, dict) else getattr(payment_method_details, 'exp_year', None),
},
'requires_action': requires_action,
'client_secret': client_secret,
'payment_intent_status': payment_intent_status
}
2025-09-25 14:30:47 +02:00
except stripe.error.StripeError as e:
logger.error("Failed to update Stripe payment method", error=str(e))
raise e
2026-01-13 22:22:38 +01:00
async def cancel_subscription(
self,
subscription_id: str,
cancel_at_period_end: bool = True
) -> Subscription:
2025-09-25 14:30:47 +02:00
"""
Cancel a subscription in Stripe
2026-01-13 22:22:38 +01:00
Args:
subscription_id: Stripe subscription ID
cancel_at_period_end: If True, subscription continues until end of billing period.
If False, cancels immediately.
Returns:
Updated Subscription object
2025-09-25 14:30:47 +02:00
"""
try:
2026-01-13 22:22:38 +01:00
if cancel_at_period_end:
# Cancel at end of billing period (graceful cancellation)
stripe_subscription = stripe.Subscription.modify(
subscription_id,
cancel_at_period_end=True
)
logger.info("Subscription set to cancel at period end",
subscription_id=subscription_id,
cancel_at=stripe_subscription.trial_end if stripe_subscription.status == 'trialing' else stripe_subscription.current_period_end)
else:
# Cancel immediately
stripe_subscription = stripe.Subscription.delete(subscription_id)
logger.info("Subscription cancelled immediately",
subscription_id=subscription_id)
# Handle period dates for trial vs active subscriptions
2026-01-14 13:15:48 +01:00
if stripe_subscription.status == 'trialing' and hasattr(stripe_subscription, 'items') and stripe_subscription.items:
items_list = stripe_subscription.items.data if hasattr(stripe_subscription.items, 'data') else stripe_subscription.items
if items_list and len(items_list) > 0:
first_item = items_list[0] if isinstance(items_list, list) else items_list
current_period_start = first_item.current_period_start
current_period_end = first_item.current_period_end
2026-01-13 22:22:38 +01:00
else:
current_period_start = stripe_subscription.current_period_start
current_period_end = stripe_subscription.current_period_end
2025-09-25 14:30:47 +02:00
return Subscription(
id=stripe_subscription.id,
customer_id=stripe_subscription.customer,
2026-01-13 22:22:38 +01:00
plan_id=subscription_id,
2025-09-25 14:30:47 +02:00
status=stripe_subscription.status,
2026-01-13 22:22:38 +01:00
current_period_start=datetime.fromtimestamp(current_period_start),
current_period_end=datetime.fromtimestamp(current_period_end),
2025-09-25 14:30:47 +02:00
created_at=datetime.fromtimestamp(stripe_subscription.created)
)
except stripe.error.StripeError as e:
logger.error("Failed to cancel Stripe subscription", error=str(e))
raise e
async def get_invoices(self, customer_id: str) -> list[Invoice]:
"""
Get invoices for a customer from Stripe
"""
try:
stripe_invoices = stripe.Invoice.list(customer=customer_id, limit=100)
2025-10-31 11:54:19 +01:00
2025-09-25 14:30:47 +02:00
invoices = []
for stripe_invoice in stripe_invoices:
2025-10-31 11:54:19 +01:00
# Create base invoice object
invoice = Invoice(
2025-09-25 14:30:47 +02:00
id=stripe_invoice.id,
customer_id=stripe_invoice.customer,
subscription_id=stripe_invoice.subscription,
amount=stripe_invoice.amount_paid / 100.0, # Convert from cents
currency=stripe_invoice.currency,
status=stripe_invoice.status,
created_at=datetime.fromtimestamp(stripe_invoice.created),
due_date=datetime.fromtimestamp(stripe_invoice.due_date) if stripe_invoice.due_date else None,
description=stripe_invoice.description
2025-10-31 11:54:19 +01:00
)
# Add Stripe-specific URLs as custom attributes
invoice.invoice_pdf = stripe_invoice.invoice_pdf if hasattr(stripe_invoice, 'invoice_pdf') else None
invoice.hosted_invoice_url = stripe_invoice.hosted_invoice_url if hasattr(stripe_invoice, 'hosted_invoice_url') else None
invoices.append(invoice)
2025-09-25 14:30:47 +02:00
return invoices
except stripe.error.StripeError as e:
logger.error("Failed to retrieve Stripe invoices", error=str(e))
raise e
async def get_subscription(self, subscription_id: str) -> Subscription:
"""
Get subscription details from Stripe
"""
try:
stripe_subscription = stripe.Subscription.retrieve(subscription_id)
2026-01-13 22:22:38 +01:00
# Get the actual plan ID from the subscription items
plan_id = subscription_id # Default fallback
2026-01-14 13:15:48 +01:00
if hasattr(stripe_subscription, 'items') and stripe_subscription.items:
items_list = stripe_subscription.items.data if hasattr(stripe_subscription.items, 'data') else stripe_subscription.items
if items_list and len(items_list) > 0:
first_item = items_list[0] if isinstance(items_list, list) else items_list
plan_id = first_item.price.id
2026-01-13 22:22:38 +01:00
# Handle period dates for trial vs active subscriptions
# During trial: current_period_* fields are only in subscription items, not root
# After trial: current_period_* fields are at root level
2026-01-14 13:15:48 +01:00
if stripe_subscription.status == 'trialing' and hasattr(stripe_subscription, 'items') and stripe_subscription.items:
items_list = stripe_subscription.items.data if hasattr(stripe_subscription.items, 'data') else stripe_subscription.items
if items_list and len(items_list) > 0:
# For trial subscriptions, get period from first subscription item
first_item = items_list[0] if isinstance(items_list, list) else items_list
current_period_start = first_item.current_period_start
current_period_end = first_item.current_period_end
2026-01-13 22:22:38 +01:00
else:
# For active subscriptions, get period from root level
current_period_start = stripe_subscription.current_period_start
current_period_end = stripe_subscription.current_period_end
2025-09-25 14:30:47 +02:00
return Subscription(
id=stripe_subscription.id,
customer_id=stripe_subscription.customer,
2026-01-13 22:22:38 +01:00
plan_id=plan_id,
2025-09-25 14:30:47 +02:00
status=stripe_subscription.status,
2026-01-13 22:22:38 +01:00
current_period_start=datetime.fromtimestamp(current_period_start),
current_period_end=datetime.fromtimestamp(current_period_end),
created_at=datetime.fromtimestamp(stripe_subscription.created),
billing_cycle_anchor=datetime.fromtimestamp(stripe_subscription.billing_cycle_anchor) if stripe_subscription.billing_cycle_anchor else None,
cancel_at_period_end=stripe_subscription.cancel_at_period_end
2025-09-25 14:30:47 +02:00
)
except stripe.error.StripeError as e:
logger.error("Failed to retrieve Stripe subscription", error=str(e))
raise e
2026-01-13 22:22:38 +01:00
async def update_subscription(
self,
subscription_id: str,
new_price_id: str,
proration_behavior: str = "create_prorations",
billing_cycle_anchor: str = "unchanged",
payment_behavior: str = "error_if_incomplete",
immediate_change: bool = False
) -> Subscription:
"""
Update a subscription in Stripe with proration support
Args:
subscription_id: Stripe subscription ID
new_price_id: New Stripe price ID to switch to
proration_behavior: How to handle prorations ('create_prorations', 'none', 'always_invoice')
billing_cycle_anchor: When to apply changes ('unchanged', 'now')
payment_behavior: Payment behavior ('error_if_incomplete', 'allow_incomplete')
immediate_change: Whether to apply changes immediately or at period end
Returns:
Updated Subscription object
"""
try:
logger.info("Updating Stripe subscription",
subscription_id=subscription_id,
new_price_id=new_price_id,
proration_behavior=proration_behavior,
immediate_change=immediate_change)
# Get current subscription to preserve settings
current_subscription = stripe.Subscription.retrieve(subscription_id)
# Build update parameters
2026-01-14 13:15:48 +01:00
items_list = current_subscription.items.data if hasattr(current_subscription.items, 'data') else current_subscription.items
2026-01-13 22:22:38 +01:00
update_params = {
'items': [{
2026-01-14 13:15:48 +01:00
'id': items_list[0].id if isinstance(items_list, list) else items_list.id,
2026-01-13 22:22:38 +01:00
'price': new_price_id,
}],
'proration_behavior': proration_behavior,
'billing_cycle_anchor': billing_cycle_anchor,
'payment_behavior': payment_behavior,
'expand': ['latest_invoice.payment_intent']
}
# If not immediate change, set cancel_at_period_end to False
# and let Stripe handle the transition
if not immediate_change:
update_params['cancel_at_period_end'] = False
update_params['proration_behavior'] = 'none' # No proration for end-of-period changes
# Update the subscription
updated_subscription = stripe.Subscription.modify(
subscription_id,
**update_params
)
logger.info("Stripe subscription updated successfully",
subscription_id=updated_subscription.id,
new_price_id=new_price_id,
status=updated_subscription.status)
# Get the actual plan ID from the subscription items
plan_id = new_price_id
2026-01-14 13:15:48 +01:00
if hasattr(updated_subscription, 'items') and updated_subscription.items:
items_list = updated_subscription.items.data if hasattr(updated_subscription.items, 'data') else updated_subscription.items
if items_list and len(items_list) > 0:
first_item = items_list[0] if isinstance(items_list, list) else items_list
plan_id = first_item.price.id
2026-01-13 22:22:38 +01:00
# Handle period dates for trial vs active subscriptions
2026-01-14 13:15:48 +01:00
if updated_subscription.status == 'trialing' and hasattr(updated_subscription, 'items') and updated_subscription.items:
items_list = updated_subscription.items.data if hasattr(updated_subscription.items, 'data') else updated_subscription.items
if items_list and len(items_list) > 0:
first_item = items_list[0] if isinstance(items_list, list) else items_list
2026-01-13 22:22:38 +01:00
current_period_start = first_item.current_period_start
current_period_end = first_item.current_period_end
else:
current_period_start = updated_subscription.current_period_start
current_period_end = updated_subscription.current_period_end
return Subscription(
id=updated_subscription.id,
customer_id=updated_subscription.customer,
plan_id=plan_id,
status=updated_subscription.status,
current_period_start=datetime.fromtimestamp(current_period_start),
current_period_end=datetime.fromtimestamp(current_period_end),
created_at=datetime.fromtimestamp(updated_subscription.created),
billing_cycle_anchor=datetime.fromtimestamp(updated_subscription.billing_cycle_anchor) if updated_subscription.billing_cycle_anchor else None,
cancel_at_period_end=updated_subscription.cancel_at_period_end
)
except stripe.error.StripeError as e:
logger.error("Failed to update Stripe subscription",
error=str(e),
subscription_id=subscription_id,
new_price_id=new_price_id)
raise e
async def calculate_proration(
self,
subscription_id: str,
new_price_id: str,
proration_behavior: str = "create_prorations"
) -> Dict[str, Any]:
"""
Calculate proration amounts for a subscription change
Args:
subscription_id: Stripe subscription ID
new_price_id: New Stripe price ID
proration_behavior: Proration behavior to use
Returns:
Dictionary with proration details including amount, currency, and description
"""
try:
logger.info("Calculating proration for subscription change",
subscription_id=subscription_id,
new_price_id=new_price_id)
2026-01-14 13:15:48 +01:00
2026-01-13 22:22:38 +01:00
# Get current subscription
current_subscription = stripe.Subscription.retrieve(subscription_id)
2026-01-14 13:15:48 +01:00
items_list = current_subscription.items.data if hasattr(current_subscription.items, 'data') else current_subscription.items
first_item = items_list[0] if isinstance(items_list, list) else items_list
current_price_id = first_item.price.id
2026-01-13 22:22:38 +01:00
# Get current and new prices
current_price = stripe.Price.retrieve(current_price_id)
new_price = stripe.Price.retrieve(new_price_id)
# Calculate time remaining in current billing period
current_period_end = datetime.fromtimestamp(current_subscription.current_period_end)
current_period_start = datetime.fromtimestamp(current_subscription.current_period_start)
now = datetime.now(timezone.utc)
total_period_days = (current_period_end - current_period_start).days
remaining_days = (current_period_end - now).days
used_days = (now - current_period_start).days
# Calculate prorated amounts
current_price_amount = current_price.unit_amount / 100.0 # Convert from cents
new_price_amount = new_price.unit_amount / 100.0
# Calculate daily rates
current_daily_rate = current_price_amount / total_period_days
new_daily_rate = new_price_amount / total_period_days
# Calculate proration based on behavior
if proration_behavior == "create_prorations":
# Calculate credit for unused time on current plan
unused_current_amount = current_daily_rate * remaining_days
# Calculate charge for remaining time on new plan
prorated_new_amount = new_daily_rate * remaining_days
# Net amount (could be positive or negative)
net_amount = prorated_new_amount - unused_current_amount
return {
"current_price_amount": current_price_amount,
"new_price_amount": new_price_amount,
"unused_current_amount": unused_current_amount,
"prorated_new_amount": prorated_new_amount,
"net_amount": net_amount,
"currency": current_price.currency.upper(),
"remaining_days": remaining_days,
"used_days": used_days,
"total_period_days": total_period_days,
"description": f"Proration for changing from {current_price_id} to {new_price_id}",
"is_credit": net_amount < 0
}
elif proration_behavior == "none":
return {
"current_price_amount": current_price_amount,
"new_price_amount": new_price_amount,
"net_amount": 0,
"currency": current_price.currency.upper(),
"description": "No proration - changes apply at period end",
"is_credit": False
}
else:
return {
"current_price_amount": current_price_amount,
"new_price_amount": new_price_amount,
"net_amount": new_price_amount - current_price_amount,
"currency": current_price.currency.upper(),
"description": "Full amount difference - immediate billing",
"is_credit": False
}
except stripe.error.StripeError as e:
logger.error("Failed to calculate proration",
error=str(e),
subscription_id=subscription_id,
new_price_id=new_price_id)
raise e
async def change_billing_cycle(
self,
subscription_id: str,
new_billing_cycle: str,
proration_behavior: str = "create_prorations"
) -> Subscription:
"""
Change billing cycle (monthly yearly) for a subscription
Args:
subscription_id: Stripe subscription ID
new_billing_cycle: New billing cycle ('monthly' or 'yearly')
proration_behavior: Proration behavior to use
Returns:
Updated Subscription object
"""
try:
logger.info("Changing billing cycle for subscription",
subscription_id=subscription_id,
new_billing_cycle=new_billing_cycle)
2026-01-14 13:15:48 +01:00
2026-01-13 22:22:38 +01:00
# Get current subscription
current_subscription = stripe.Subscription.retrieve(subscription_id)
2026-01-14 13:15:48 +01:00
items_list = current_subscription.items.data if hasattr(current_subscription.items, 'data') else current_subscription.items
first_item = items_list[0] if isinstance(items_list, list) else items_list
current_price_id = first_item.price.id
2026-01-13 22:22:38 +01:00
# Get current price to determine the plan
current_price = stripe.Price.retrieve(current_price_id)
product_id = current_price.product
# Find the corresponding price for the new billing cycle
# This assumes you have price IDs set up for both monthly and yearly
# You would need to map this based on your product catalog
prices = stripe.Price.list(product=product_id, active=True)
new_price_id = None
for price in prices:
if price.recurring and price.recurring.interval == new_billing_cycle:
new_price_id = price.id
break
if not new_price_id:
raise ValueError(f"No {new_billing_cycle} price found for product {product_id}")
# Update the subscription with the new price
return await self.update_subscription(
subscription_id,
new_price_id,
proration_behavior=proration_behavior,
billing_cycle_anchor="now",
immediate_change=True
)
except stripe.error.StripeError as e:
logger.error("Failed to change billing cycle",
error=str(e),
subscription_id=subscription_id,
new_billing_cycle=new_billing_cycle)
raise e
2025-09-25 14:30:47 +02:00
async def get_customer(self, customer_id: str) -> PaymentCustomer:
"""
Get customer details from Stripe
"""
try:
stripe_customer = stripe.Customer.retrieve(customer_id)
return PaymentCustomer(
id=stripe_customer.id,
email=stripe_customer.email,
name=stripe_customer.name,
created_at=datetime.fromtimestamp(stripe_customer.created)
)
except stripe.error.StripeError as e:
logger.error("Failed to retrieve Stripe customer", error=str(e))
raise e
async def create_setup_intent(self) -> Dict[str, Any]:
"""
Create a setup intent for saving payment methods in Stripe
"""
try:
setup_intent = stripe.SetupIntent.create()
return {
'client_secret': setup_intent.client_secret,
'id': setup_intent.id
}
except stripe.error.StripeError as e:
logger.error("Failed to create Stripe setup intent", error=str(e))
raise e
async def create_payment_intent(self, amount: float, currency: str, customer_id: str, payment_method_id: str) -> Dict[str, Any]:
"""
Create a payment intent for one-time payments in Stripe
"""
try:
payment_intent = stripe.PaymentIntent.create(
amount=int(amount * 100), # Convert to cents
currency=currency,
customer=customer_id,
payment_method=payment_method_id,
confirm=True
)
2026-01-14 13:15:48 +01:00
2025-09-25 14:30:47 +02:00
return {
'id': payment_intent.id,
'client_secret': payment_intent.client_secret,
'status': payment_intent.status
}
except stripe.error.StripeError as e:
logger.error("Failed to create Stripe payment intent", error=str(e))
raise e
2026-01-14 13:15:48 +01:00
async def get_customer_payment_method(self, customer_id: str) -> Optional[PaymentMethod]:
"""
Get the default payment method for a customer from Stripe
Args:
customer_id: Stripe customer ID
Returns:
PaymentMethod object or None if no payment method exists
"""
try:
logger.info("Retrieving customer payment method", customer_id=customer_id)
# Retrieve the customer to get default payment method
stripe_customer = stripe.Customer.retrieve(customer_id)
# Get the default payment method ID
default_payment_method_id = None
if stripe_customer.invoice_settings and stripe_customer.invoice_settings.default_payment_method:
default_payment_method_id = stripe_customer.invoice_settings.default_payment_method
elif stripe_customer.default_source:
default_payment_method_id = stripe_customer.default_source
if not default_payment_method_id:
logger.info("No default payment method found for customer", customer_id=customer_id)
return None
# Retrieve the payment method details
stripe_payment_method = stripe.PaymentMethod.retrieve(default_payment_method_id)
# Extract payment method details based on type
payment_method_details = getattr(stripe_payment_method, stripe_payment_method.type, {})
logger.info("Customer payment method retrieved successfully",
customer_id=customer_id,
payment_method_id=stripe_payment_method.id,
payment_method_type=stripe_payment_method.type)
return PaymentMethod(
id=stripe_payment_method.id,
type=stripe_payment_method.type,
brand=payment_method_details.get('brand') if isinstance(payment_method_details, dict) else getattr(payment_method_details, 'brand', None),
last4=payment_method_details.get('last4') if isinstance(payment_method_details, dict) else getattr(payment_method_details, 'last4', None),
exp_month=payment_method_details.get('exp_month') if isinstance(payment_method_details, dict) else getattr(payment_method_details, 'exp_month', None),
exp_year=payment_method_details.get('exp_year') if isinstance(payment_method_details, dict) else getattr(payment_method_details, 'exp_year', None),
)
except stripe.error.StripeError as e:
logger.error("Failed to retrieve customer payment method",
error=str(e),
customer_id=customer_id)
raise e
async def get_setup_intent(self, setup_intent_id: str) -> stripe.SetupIntent:
"""
Retrieve a SetupIntent from Stripe
Args:
setup_intent_id: SetupIntent ID to retrieve
Returns:
stripe.SetupIntent object
Raises:
stripe.error.StripeError: If retrieval fails
"""
try:
logger.info("Retrieving SetupIntent from Stripe",
setup_intent_id=setup_intent_id)
# Retrieve SetupIntent from Stripe
setup_intent = stripe.SetupIntent.retrieve(setup_intent_id)
logger.info("SetupIntent retrieved successfully",
setup_intent_id=setup_intent.id,
status=setup_intent.status,
customer_id=setup_intent.customer)
return setup_intent
except stripe.error.StripeError as e:
logger.error("Failed to retrieve SetupIntent from Stripe",
error=str(e),
setup_intent_id=setup_intent_id,
error_type=type(e).__name__)
raise e