shagag
Frontend Engineer
by shagag

Integrating Stripe Payments

10/28/2024

word count:1094

estimated reading time:6 minutes

This guide is still a work in progress

Stripe is a payment processing platform that allows you to accept payments online and in mobile apps. It is a popular choice for businesses of all sizes due to its ease of use, flexibility, and powerful features. This guide will walk you through the process of integrating Stripe payments into your Next.js app.

Environment Variables -

Let’s start by setting up the environment variables. Create a .env.local file in the root of your project and add the following variables:

# STRIPE
STRIPE_API_KEY="replace_me"
STRIPE_WEBHOOK_SECRET="replace_me"
NEXT_PUBLIC_STRIPE_KEY="replace_me"
NEXT_PUBLIC_STRIPE_MANAGE_URL="replace_me"
NEXT_PUBLIC_PRICE_ID_BASIC="replace_me"
NEXT_PUBLIC_PRICE_ID_PREMIUM="replace_me"

Webhook service

        "stripe:listen": "stripe listen --forward-to http://localhost:3000/api/webhooks/stripe"

This command requires the stripe cli, and can run it by using: pnpm stripe:listen It will set up a locally running service that is going to listen to events on your stripe account. The frontend api endpoint is api/webhooks/stripe and the file route.ts handles the post requests from the locally running service:

export async function POST(req: Request) {
  const body = await req.text();
  const signature = headers().get("Stripe-Signature") as string;

  let event: Stripe.Event;

  try {
    event = stripe.webhooks.constructEvent(
      body,
      signature,
      env.STRIPE_WEBHOOK_SECRET,
    );
  } catch (error) {
    return new Response(
      `Webhook Error: ${error instanceof Error ? error.message : "Unknown error"}`,
      { status: 400 },
    );
  }
  // If stripe sends a checkout.session.completed event, create a subscription
  if (event.type === "checkout.session.completed") {
    const session = event.data.object as Stripe.Checkout.Session;
    // This retrieves the subscription object from stripe
    const subscription = await stripe.subscriptions.retrieve(
      session.subscription as string,
    );

    // Add the subscription to our database
    await createSubscriptionUseCase({
      user_id: session.metadata!.userId,
      stripeSubscriptionId: subscription.id,
      stripeCustomerId: subscription.customer as string,
      stripePriceId: subscription.items.data[0]?.price.id,
      stripeCurrentPeriodEnd: new Date(subscription.current_period_end * 1000),
    });
  } else if (
    // Events for upgrading or updating a subscription
    ["customer.subscription.created", "customer.subscription.updated"].includes(
      event.type,
    )
  ) {
    const subscription = event.data.object as Stripe.Subscription;
    // if this fails due to race conditions where checkout.session.completed was not fired first, stripe will retry
    // this will update the subscription in our database
    await updateSubscriptionUseCase({
      stripePriceId: subscription.items.data[0]?.price.id,
      stripeSubscriptionId: subscription.id,
      stripeCurrentPeriodEnd: new Date(subscription.current_period_end * 1000),
    });
  }

  return new Response(null, { status: 200 });
}

For each request that comes in we are validating the stripe-signature header against the webhook secret we created to make sure this request is valid. Then depending on the type of event stripe sent, we are adding/modifying the data inside subscriptions table:

Schema:

CREATE TABLE `subscriptions` (
	`id` int AUTO_INCREMENT NOT NULL,
	`user_id` char(36) NOT NULL,
	`stripeSubscriptionId` varchar(255) NOT NULL,
	`stripeCustomerId` varchar(255) NOT NULL,
	`stripePriceId` varchar(255) NOT NULL,
	`expires` timestamp NOT NULL,
	CONSTRAINT `subscriptions_id` PRIMARY KEY(`id`),
	CONSTRAINT `subscriptions_user_id_unique` UNIQUE(`user_id`),
	CONSTRAINT `userId_idx` UNIQUE(`user_id`)
);

How does the subscription works?

Frontend flow

The following is a suggestion for implementing the stripe payment flow in your frontend:

app\dashboard\settings\subscription\page.tsx

This is the page where the user can manage his subscription, it shows the current plan the user is on, and a button that redirects to the stripes dashboard url, where the user can manage his subscription.


export default async function SubscriptionPage() {
  const user = await getCurrentUser();

  if (!user) {
    throw new Error("User not found");
  }

  const currrentPlan = await getUserPlanUseCase(user.user.id);

  return (
    currrentPlan !== "free" && (
      <ConfigurationPanel title="Manage Subscription">
        <div className="flex flex-col gap-4">
          <div>
            You are currently using the{" "}
            <span className="text-bold text-blue-400">{currrentPlan}</span>{" "}
            plan.
          </div>
          <div>You can upgrade or cancel your subscription below</div>
          <Button className="max-w-fit" asChild>
            <Link
              href={env.NEXT_PUBLIC_STRIPE_MANAGE_URL}
              target="_blank"
              rel="noreferrer"
            >
              Manage Subscription
            </Link>
          </Button>
        </div>
      </ConfigurationPanel>
    )
  );
}

use-cases/subscription.ts

This file is an example of what functions you might need to manage the user’s subscription, notice each use-case is using server actions, such as getSubscription, createSubscription, updateSubscription, these are generics, and you should implement them using your ORM or db of choice.

export type Plan = "free" | "basic" | "premium";

export async function getUserPlanUseCase(userId: string): Promise<Plan> {
  const subscription = await getSubscription(userId);

  if (!subscription) {
    return "free";
  } else {
    // Test if the subscription is premium or basic
    return subscription.stripePriceId === env.NEXT_PUBLIC_PRICE_ID_PREMIUM
      ? "premium"
      : "basic";
  }
}

export async function createSubscriptionUseCase(subscription: {
  user_id: string;
  stripeSubscriptionId: string;
  stripeCustomerId: string;
  stripePriceId: string;
  stripeCurrentPeriodEnd: Date;
}) {
  await createSubscription(subscription);
}

export async function updateSubscriptionUseCase(subscription: {
  stripeSubscriptionId: string;
  stripePriceId: string;
  stripeCurrentPeriodEnd: Date;
}) {
  await updateSubscription(subscription);
}

export function isSubscriptionActive(subscription?: Subscription) {
  if (!subscription) return false;
  return subscription.stripeCurrentPeriodEnd >= new Date();
}

checkout-button.tsx

An example of the checkout button component, using useServerAction from ZSA to trigger the server action that generates the stripe session.

export function CheckoutButton({
  className,
  children,
  priceId,
}: {
  className?: string;
  children: ReactNode;
  priceId: string;
}) {
  const { execute, isPending } = useServerAction(generateStripeSessionAction);

  return (
    <form
      onSubmit={(e) => {
        e.preventDefault();
        execute({ priceId });
      }}
    >
      <LoaderButton isLoading={isPending} className={className}>
        {children}
      </LoaderButton>
    </form>
  );
}
const stripeSession = await stripe.checkout.sessions.create({
  success_url: `${env.HOST_NAME}/success`,
  cancel_url: `${env.HOST_NAME}/cancel`,
  payment_method_types: ["card"],
  customer_email: email ? email : undefined,
  mode: "subscription",
  line_items: [
    {
      price: priceId,
      quantity: 1,
    },
  ],
  metadata: {
    userId: userId,
  },
});

redirect(stripeSession.url!);

Debugging

Visit events in the dashboard, this page shows all the events that went through the webhook, and the events that occurred on our stripe service.

You can also visit Workbench which is a more intuitive way to view the data regarding our payment system. This is critical for refunds, history and user management.