Laravel subscriptions with a ready-to-use Filament UI.
Replace days of custom billing logic with a working setup on top of Stripe (Cashier).
- Skip custom subscription logic
- Skip building pricing UI
- Skip wiring Stripe webhooks manually
SubKit provides a Filament admin panel to manage plans and plan sets, themeable Blade components for your pricing page and subscription dashboard, and a clean PHP API for subscription lifecycle operations.
Why use SubKit? While Laravel Cashier is incredibly powerful, building the actual UI and admin panel for subscriptions takes days. SubKit bridges this gap by providing a "Lickable UI" out of the box and a powerful Filament admin to manage it all without writing boilerplate code.
- Integrates with Stripe via Laravel Cashier (webhooks, checkout sessions, billing portal)
- Provides a Filament admin panel to manage Plans, Plan Sets, and Provider Prices
- Provides Blade components: a pricing table and a subscription management UI, with multiple themes
- Exposes a PHP facade and REST API for subscription operations (checkout, cancel, resume, billing portal)
Try it in action: subkit.noxls.net
- Register a test account to explore the customer billing flow.
- Once registered, navigate to
/adminto check out the Filament control panel.
- Process payments or store card data
- Replace Stripe or Laravel Cashier — it orchestrates on top of them
- Handle invoicing, taxes, or compliance
- Manage user authentication or access control
- PHP 8.4+
- Laravel 11+
- Laravel Cashier (
laravel/cashier^16.5) installed and configured - Filament (
filament/filament^3.2) - MySQL 8+ (or MariaDB 10.5+)
- A Stripe account
composer require karpovigorok/subkitPublish the config and run migrations:
php artisan vendor:publish --tag=subkit-config
php artisan vendor:publish --tag=subkit-migrations
php artisan migrateRegister the plugin in your Filament panel provider:
use SubKit\Filament\SubKitPlugin;
public function panel(Panel $panel): Panel
{
return $panel
// ...
->plugin(SubKitPlugin::make());
}php artisan vendor:publish --tag=subkit-configAdd to your .env:
STRIPE_KEY=pk_live_...
STRIPE_SECRET=sk_live_...
STRIPE_WEBHOOK_SECRET=whsec_...
# Optional
EASY_SUB_CURRENCY_CODE=USD
EASY_SUB_CURRENCY_SYMBOL=$Your User model must use Cashier's Billable trait:
use Laravel\Cashier\Billable;
class User extends Authenticatable
{
use Billable;
}Navigate to your Filament admin panel (usually /admin) → Plans → Create a plan. After creating a plan, add a Stripe Price ID in the Provider Prices tab on the plan edit page.
Point your Stripe webhook to Cashier's built-in route:
https://your-app.com/stripe/webhook
Events to enable in Stripe:
customer.subscription.createdcustomer.subscription.updatedcustomer.subscription.deletedinvoice.paidinvoice.payment_failed
Drop the pricing table into any Blade view. The component uses the currently authenticated user automatically — no user ID needed.
<x-subkit::pricing-table
provider="stripe"
success-url="{{ route('dashboard') }}"
cancel-url="{{ route('pricing') }}"
/>With a plan set (for multiple landing pages or A/B testing):
<x-subkit::pricing-table
set="homepage_2024"
provider="stripe"
success-url="{{ route('dashboard') }}"
cancel-url="{{ route('pricing') }}"
/>| Prop | Type | Default | Description |
|---|---|---|---|
set |
string|null |
null |
Plan set code. If omitted, shows all active plans. |
theme |
string|null |
'default' |
UI theme (default, dark, modern, or a custom theme). |
provider |
string |
'stripe' |
Payment provider. |
success-url |
string |
'' |
Redirect after successful checkout. Accepts a route name, relative path, or full URL. |
cancel-url |
string |
'' |
Redirect when the user cancels checkout. |
free-url |
string |
'' |
CTA destination for $0 plans (authenticated users). |
guest-redirect-url |
string|null |
null |
Where unauthenticated visitors are sent. Defaults to /register. |
company-id |
string|null |
null |
For B2B: attaches the subscription to a company rather than a user. |
subscribe-label |
string|null |
null |
Override the "Get Started" button text. |
free-label |
string|null |
null |
Override the "Get Started Free" button text. |
guest-label |
string|null |
null |
Override the "Create Account to Subscribe" button text. |
URL props accept a route name (e.g. 'dashboard'), a relative path (e.g. '/thanks?utm_source=fb'), or a full URL. Route names are resolved automatically. URLs set in the admin panel (per Plan Set) serve as defaults when the prop is omitted.
For the best UX, point Free Plan URL to a route that automatically creates a $0 subscription for the authenticated user.
Button labels follow a three-tier fallback: Blade prop → Plan Set admin setting → translation string.
Drop the manage component into your dashboard or account page:
<x-subkit::manage-subscriptions
return-url="{{ route('dashboard') }}"
/>This renders the user's active subscriptions: plan name, status badge, trial/renewal dates, and action buttons (Cancel, Resume, Manage Billing). Renders nothing if the user has no subscriptions.
| Prop | Type | Default | Description |
|---|---|---|---|
theme |
string|null |
'default' |
UI theme. |
return-url |
string |
'' |
URL to return to from the Stripe billing portal. |
guest-redirect-url |
string|null |
null |
Where guests are sent. Renders a redirect link instead of the subscription UI. |
use SubKit\Facades\SubKit;
if (SubKit::hasAccess((string) auth()->id())) {
// User has an active or trialing subscription
}hasAccess() returns true for active and trialing states.
$subscriptions = SubKit::forUser((string) auth()->id());
$active = SubKit::activeForUser((string) auth()->id()); // returns Cashier Subscription or null// Cancel at period end (access continues until the billing period ends)
SubKit::cancel($subscriptionId);
// Cancel immediately
SubKit::cancel($subscriptionId, immediately: true);SubKit::resume($subscriptionId);Redirect the user to the Stripe-hosted billing portal to manage payment methods and invoices:
$url = SubKit::billingPortal($subscriptionId, route('dashboard'));
return redirect()->away($url);Plan Sets let you curate groups of plans for specific contexts — landing pages, A/B tests, regional pricing, etc.
Create a plan set in the admin panel under Plan Sets, assign plans to it, and reference it by code:
<x-subkit::pricing-table set="startup_annual" />Per Plan Set you can configure:
- Theme — override the default UI theme
- Description — subtitle shown above the pricing table
- URLs — default success, cancel, free, and guest redirect URLs
- Button Labels — override button text per set
SubKit includes a normalized, Many-to-Many feature management system. Instead of hardcoding features in your Blade files, you can manage a global library of features (e.g., "Priority Support", "Unlimited Projects") directly in the Filament admin panel.
Simply attach features to your Plans using the intuitive Filament interface, and SubKit's pricing tables will automatically render them with beautiful checkmarks inside the pricing cards.
For company-level subscriptions, pass company-id:
<x-subkit::pricing-table
:company-id="(string) $company->id"
provider="stripe"
success-url="{{ route('dashboard') }}"
cancel-url="{{ route('pricing') }}"
/>company-id is a plain string with no foreign key constraint — it can reference any table in your app (teams, organizations, workspaces, etc.).
Three themes are bundled: default, dark, and light. Specify the theme per component or per Plan Set.
To create a custom theme, publish the views and add a new folder:
php artisan vendor:publish --tag=subkit-viewsCreate resources/views/vendor/subkit/themes/{your-theme}/pricing-table.blade.php. The theme will appear automatically in the admin panel's theme selector.
| Method | URL | Description |
|---|---|---|
POST |
/api/subkit/checkout |
Create a Stripe Checkout session |
GET |
/api/subkit/subscriptions/user |
List subscriptions for the authenticated user |
GET |
/api/subkit/subscriptions/company |
List subscriptions for a company |
POST |
/api/subkit/subscriptions/{id}/cancel |
Cancel a subscription |
POST |
/api/subkit/subscriptions/{id}/resume |
Resume a subscription |
Add authentication middleware in config/subkit.php:
'api' => [
'middleware' => ['api', 'auth:sanctum'],
'prefix' => 'api/subkit',
],Subscription state is owned by Cashier and sourced from Stripe's stripe_status field:
| State | Meaning |
|---|---|
trialing |
In a free trial period |
active |
Paid and active |
past_due |
Payment failed, awaiting retry |
paused |
Paused by the customer |
canceled |
Canceled (may still have access until period end) |
incomplete |
Checkout started but not completed |
SubKit delegates all webhook processing to Cashier. To react to lifecycle changes, listen to Cashier's WebhookHandled event in your AppServiceProvider:
use Laravel\Cashier\Events\WebhookHandled;
Event::listen(WebhookHandled::class, function (WebhookHandled $event) {
if ($event->payload['type'] === 'customer.subscription.deleted') {
// revoke access, send email, etc.
}
});| Tag | Publishes |
|---|---|
subkit-config |
config/subkit.php |
subkit-migrations |
All package migrations |
subkit-views |
Blade views (for customization) |
subkit-lang |
Translation strings |
composer testMIT
