Clerk billing for B2C SaaS
Clerk billing for B2C SaaS allows you to create plans and manage subscriptions for individual users in your application. If you'd like to charge companies or organizations, see Billing for B2B SaaS. You can also combine both B2C and B2B billing in the same application.
Enable billing
To enable billing for your application, navigate to the Billing Settings page in the Clerk Dashboard. This page will guide you through enabling billing for your application.
Clerk billing costs just 0.7% per transaction, plus Stripe's transaction fees which are paid directly to Stripe. Clerk Billing is not the same as Stripe Billing. Plans and pricing are managed directly through the Clerk Dashboard and won't sync with your existing Stripe products or plans. Clerk uses Stripe only for payment processing, so you don't need to set up Stripe Billing.
Payment gateway
Once you have enabled billing, you will see the following Payment gateway options for collecting payments via Stripe:
- Clerk development gateway: A shared test Stripe account used for development instances. This allows developers to test and build billing flows in development without needing to create and configure a Stripe account.
- Stripe account: Use your own Stripe account for production. A Stripe account created for a development instance cannot be used for production. You will need to create a separate Stripe account for your production environment.
Create a plan
Subscription plans are what your users subscribe to. There is no limit to the number of plans you can create.
To create a plan, navigate to the Plans page in the Clerk Dashboard. Here, you can create, edit, and delete plans. To setup B2C billing, select the Plans for Users tab and select Add Plan. When creating a plan, you can also create features for the plan; see the next section for more information.
Add features to a plan
Features make it easy to give entitlements to your plans. You can add any number of features to a plan.
You can add a feature to a plan when you are creating a plan. To add it after a plan is created:
- Navigate to the Plans page in the Clerk Dashboard.
- Select the plan you'd like to add a feature to.
- In the Features section, select Add Feature.
Create a pricing page
You can create a pricing page by using the <PricingTable /> component. This component displays a table of plans and features that users can subscribe to. It's recommended to create a dedicated page, as shown in the following example.
import { PricingTable } from '@clerk/remix'
export default function PricingPage() {
  return (
    <div style={{ maxWidth: '800px', margin: '0 auto', padding: '0 1rem' }}>
      <PricingTable />
    </div>
  )
}Control access with features and plans
You can use Clerk's features and plans to gate access to the content. There are a few ways to do this, but the recommended approach is to either use the method or the <Protect> component.
The has() method is available for any JavaScript framework, while <Protect> is only available for React-based frameworks.
Example: Using has()
Use the has() method to test if the user has access to a plan:
const hasPremiumAccess = has({ plan: 'gold' })Or a feature:
const hasPremiumAccess = has({ feature: 'widgets' })The  method is a server-side helper that checks if the organization has been granted a specific type of access control (role, permission, feature, or plan) and returns a boolean value. has() is available on the , which you will access differently .
The following example demonstrates how to use has() to check if a user has a plan.
import { getAuth } from '@clerk/remix/ssr.server'
import { useLoaderData } from '@remix-run/react'
import type { LoaderFunctionArgs } from '@remix-run/node'
export const loader = async (args: LoaderFunctionArgs) => {
  const { has } = await getAuth(args)
  return { hasBronzePlan: has({ plan: 'bronze' }) }
}
export default function BronzeContentPage() {
  const { hasBronzePlan } = useLoaderData<typeof loader>()
  if (!hasBronzePlan) return <h1>Only subscribers to the Bronze plan can access this content.</h1>
  return <h1>For Bronze subscribers only</h1>
}The following example demonstrates how to use has() to check if a user has a feature.
import { getAuth } from '@clerk/remix/ssr.server'
import { useLoaderData } from '@remix-run/react'
import type { LoaderFunctionArgs } from '@remix-run/node'
export const loader = async (args: LoaderFunctionArgs) => {
  const { has } = await getAuth(args)
  return { hasPremiumAccess: has({ feature: 'premium_access' }) }
}
export default function PremiumContentPage() {
  const { hasPremiumAccess } = useLoaderData<typeof loader>()
  if (!hasPremiumAccess)
    return <h1>Only subscribers with the Premium Access feature can access this content.</h1>
  return <h1>Our Exclusive Content</h1>
}Example: Using <Protect>
The <Protect> component protects content or even entire routes by checking if the user has been granted a specific type of access control (role, permission, feature, or plan). You can pass a fallback prop to <Protect> that will be rendered if the user does not have the access control.
The following example demonstrates how to use <Protect> to protect a page by checking if the user has a plan.
import { Protect } from '@clerk/remix'
export default function ProtectedContentPage() {
  return (
    <Protect
      plan="bronze"
      fallback={<p>Only subscribers to the Bronze plan can access this content.</p>}
    >
      <h1>Exclusive Bronze Content</h1>
      <p>This content is only visible to Bronze subscribers.</p>
    </Protect>
  )
}The following example demonstrates how to use <Protect> to protect a page by checking if the user has a feature.
import { Protect } from '@clerk/remix'
export default function ProtectedPremiumContentPage() {
  return (
    <Protect
      feature="premium_access"
      fallback={<p>Only subscribers with the Premium Access feature can access this content.</p>}
    >
      <h1>Exclusive Premium Content</h1>
      <p>This content is only visible to users with Premium Access feature.</p>
    </Protect>
  )
}Feedback
Last updated on