Skip to content

Latest commit

 

History

History
204 lines (168 loc) · 11.4 KB

README.md

File metadata and controls

204 lines (168 loc) · 11.4 KB

Stripe

Stripe offers an api, webhooks, and a dashboard that we can use to integrate ecommerce into apps that need it.

Table of Contents

Test Mode

Stripe has a separate testing environment with its own keys and test cards to try things out. You can even set up separate webhooks and products that can be imported to live mode whenever you're ready. Test mode will be denoted by a banner at the top of the dashboard. If you have automatic email receipts being sent to customers, this won't be active in test mode.

Subscription Management with a Hosted Checkout

This subscription tutorial covers a lot of the essentials for setting up stripe with django, and we'll be describing some specifics we can take to tailor it to our purposes. The basic flow for subscriptions will be to redirect the user to Stripe's checkout, create a webhook that listens to a few subscription related events, and save relevant data to the database after the webhook is called.

From that tutorial, it's worth following the steps for:

* Be sure to get keys and ids into a .env file as we normally do. ** The success and cancel redirects here can be anything you'd like as long as you keep the session id. In Parserator, we redirected back to the account page in order to avoid making dedicated success/cancel pages.

So you've got your keys, your models are ready, and you're sending users to a Stripe hosted checkout page for a subscription. That's great! Here are some extra things to consider.

Webhooks

At this point, the webhooks on Stripe are set up to listen for the checkout.session.complete event. This event is great for creating a new StripeCustomer object. If we're operating under the assumption of a recurring subscription, we'll need some additional events that denote updates and cancellations.

Receiving Webhook Events

Here's a list of some useful webhook events:

  • checkout.session.complete - after the checkout process successfully finishes
  • customer.subscription.updated - after a customer's subscription details change in any way
  • customer.subscription.deleted - after a customer's subscription has fully been cancelled

Testing Webhooks

The tutorial above mentions testing webhooks locally using the Stripe CLI. This is useful for quickly making sure events you need are handled correctly. However, once it's time to test on a review app, or if you're locally working on parts that communicate with Stripe after receiving a webhook request, it becomes more useful to use one of the test webhooks created on the dashboard.

Head to the Stripe developer dashboard and select the Webhooks tab while in test mode. Create a webhook with the events you need, and enter your app's url ending with the path to the webhook you've made. This will be different for each case:

  • For review apps, that could look something like https://<review-app-subdomain>.herokuapp.com/stripe-webhook/
  • For local dev, use a service like Ngrok to expose your local environment. It's easiest if you can set a consistent url that you don't have to change much.

Make sure the webhook signing secret is added to your environment variables and you're good to go! Remember to disable that webhook when you're done testing.

Configuring Stripe's Checkout

Prepopulating customer email during checkout

If your users provide emails, you can prepopulate that field by passing in a customer_email while creating the checkout session.

# views.py - checkout creation view
def get(self, request):
    ...
    checkout_session = stripe.checkout.Session.create(
        client_reference_id=request.user.id if request.user.is_authenticated else None,
        customer=<stripe customer id>,
        customer_email=<email string>,
        success_url=<success url> + '&session_id={CHECKOUT_SESSION_ID}',
        ...
    )

Note: Stripe expects the literal string {CHECKOUT_SESSION_ID} when receiving the success url so that it can replace it with an appropriate value.

Multiple Subscription tiers

In order to have the customer choose which tier they'd like to sign up for, we'll create buttons for each with their names in the id (i.e. Juniper) to help the view discern which price_id (i.e. price_123abc) to use during checkout creation.

After creating those buttons, we can pass their id to the view when clicked, and use a product map to find the price_id by name. In this example, we have a get_product_map() helper function that creates a dict with each subscription tier's price_id from Stripe as the key, and the name of the tier in Parserator as the value. Then we use that dict to find the price_id, and pass it in as the single item in the line_items argument:

// stripe_config.js
subscriptionBtns.forEach((btn) => {
    btn.addEventListener("click", () => {
        // Get Checkout Session ID
        fetch(checkout + '?' + new URLSearchParams({
            subscription_name: btn.id
        }))
            .then((result) => {
                return result.json();
            })
            .then((data) => {
                // Redirect to Stripe Checkout
                return stripe.redirectToCheckout({sessionId: data.sessionId})
            })
            .then((res) => {
                console.log(res);
            });
    });
})
# views.py - checkout creation view
def get(self, request):
    ...
    sub_name = request.GET.get('subscription_name')

    # Get price_id from subscription name
    product_map = get_product_map()
    price_id = [id for id in product_map if sub_name in product_map[id]][0] if sub_name else None

    success_url = request.build_absolute_uri(
            reverse('account-detail')
            + f'?tier={product_map[price_id]}'
        )
    ...

    checkout_session = stripe.checkout.Session.create(
        client_reference_id=request.user.id if request.user.is_authenticated else None,
        ...
        payment_method_types=['card'],
        mode='subscription',
        line_items=[
            {
                'price': price_id,
                'quantity': 1,
            },
        ]
    )

    return JsonResponse({'sessionId': checkout_session['id']})

Customer Portal

Customers can use a customizable portal to manage their subscriptions. Change settings for this portal by heading to the Stripe dashboard, and clicking Settings > Billing > Customer portal. Here you can change the portal's header, choose whether or not cancelled subscriptions take effect immediately, allow customers to change subscriptions, and more.

Once the portal is configured, we'll need to set up a view to create the appropriate url in order for the customer to be redirected to their version of this portal. In this example, we're returning the url from the view, then using a script on the page to request the resulting url and redirect:

# views.py - portal creation view
def get(self, request):
    ...
    portal_session = stripe.billing_portal.Session.create(
        customer=<stripe customer id>,
        return_url=request.build_absolute_uri(reverse('account-detail')),
    )
    return JsonResponse({'sessionUrl': portal_session['url']})
// stripe_config.js
manageBtn.addEventListener("click", () => {
    // Get Portal Session url
    fetch(<url for our portal creation view>)
        .then((result) => {
            return result.json();
        })
        .then((data) => {
            // Redirect to Stripe Portal
            window.location.href = data.sessionUrl
        });
});

Subscription Cancellations

When customizing the portal, you can choose whether customer subscription cancellations take effect immediately, or after the billing period ends. When a customer's subscription ends without resubscription, Stripe sends a customer.subscription.deleted event.

If you choose for cancellations to take effect immediately, this event will also be sent immediately. If you choose for cancellations to take effect at the end of the billing period, this event will not be sent until the billing period ends. What Stripe will instead send immediately is a customer.subscription.updated event with a field called cancel_at_period_end set to true. Only at the end of the period, when the subscription gets officially cancelled will Stripe send the usual subscription deleted event. If a customer changes their mind and renews their subscription before it gets cancelled, another updated event will be sent with the cancel_at_period_end field set to false.

Models

The above tutorial gets you started with the StripeCustomer model that has fields for the user, stripe customer id, and stripe subscription id. Here are some other fields/models that you may want to consider:

StripeCustomer

A sub_status field to locally keep track of a customer's subscription status:

STATUS_CHOICES = [
        (None, ""),
        ("active", "active"),
        ("cancelled", "cancelled"),
        ("pending_cancellation", "pending cancellation"),
    ]
    sub_status = models.CharField(max_length=255, choices=STATUS_CHOICES, null=True, default=None)

Some billing_date fields to help easily display that info within your app, and save the customer from needing to check the Stripe portal everytime:

next_billing_date = models.DateField(null=True)
last_billing_date = models.DateField(null=True)

StripeInvoice

A dedicated model for any invoices could also help keep local records up to date:

class StripeInvoice(models.Model):
    customer = models.ForeignKey(StripeCustomer, on_delete=models.CASCADE, related_name='invoices')
    stripe_invoice_id = models.CharField(max_length=255)
    date = models.DateTimeField()
    amount_paid = models.IntegerField()
    event_data = models.JSONField()