Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(clerk-js): OrgProfile billing page and new subscription methods #5423

Merged
merged 27 commits into from
Mar 26, 2025
Merged
Changes from all commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
5999289
preliminary add PricingTable to OrgProfile, filter plans by type
aeliox Mar 14, 2025
2d1bc79
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 14, 2025
fc1eaee
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 17, 2025
ceb1ab6
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 18, 2025
b6ea065
add usePlans hook, rename PlanDetailDrawer -> SubscriptionDetailDrawer
aeliox Mar 19, 2025
161e194
pipe orgId through checkout flow, where appropriate
aeliox Mar 19, 2025
93bcce5
wip canceling a sub and revalidating plans
aeliox Mar 21, 2025
9321819
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 21, 2025
c307814
merge fix for pricing table layouts
aeliox Mar 21, 2025
c30610d
wip commit
aeliox Mar 21, 2025
59f4076
wire up revalidation of plans / subs after checkout / cancelation
aeliox Mar 21, 2025
a69e6b3
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 21, 2025
96a43ee
lazy loads billng page in OrgProfile
aeliox Mar 21, 2025
0007589
bundle size bumps
aeliox Mar 21, 2025
d9f1dc5
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 24, 2025
b08efc5
fix for lazy-loading BillingPage in OrgProfile
aeliox Mar 24, 2025
ea98ef5
address some feedback
aeliox Mar 24, 2025
1408c92
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 24, 2025
aee9638
remove unnecessary staleTime on plans fetch
aeliox Mar 24, 2025
b15bad4
fix cache collisions of subscriptions and payment sources
aeliox Mar 25, 2025
2d9ba9d
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 25, 2025
ecfe4e4
handle environment commerce disabled setting where appropriate
aeliox Mar 26, 2025
2a68edf
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 26, 2025
fdd2d4e
bundlewatch size bump
aeliox Mar 26, 2025
4dfde8c
address PR feedback
aeliox Mar 26, 2025
32b92b6
Merge branch 'main' into keiran/org-profile-pricing-table
aeliox Mar 26, 2025
3fbf27d
bundle bump
aeliox Mar 26, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions .changeset/yellow-hairs-refuse.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
---
'@clerk/localizations': patch
'@clerk/clerk-js': patch
'@clerk/shared': patch
'@clerk/types': patch
---

Add billing page to `OrgProfile`, use new `usePlans` hook, and adds new subscription methods
4 changes: 2 additions & 2 deletions packages/clerk-js/bundlewatch.config.json
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{
"files": [
{ "path": "./dist/clerk.js", "maxSize": "580.7kB" },
{ "path": "./dist/clerk.browser.js", "maxSize": "79.25kB" },
{ "path": "./dist/clerk.js", "maxSize": "581.5kB" },
{ "path": "./dist/clerk.browser.js", "maxSize": "79.30kB" },
{ "path": "./dist/clerk.headless.js", "maxSize": "55KB" },
{ "path": "./dist/ui-common*.js", "maxSize": "96KB" },
{ "path": "./dist/vendors*.js", "maxSize": "30KB" },
9 changes: 9 additions & 0 deletions packages/clerk-js/src/core/clerk.ts
Original file line number Diff line number Diff line change
@@ -83,6 +83,7 @@ import {
createAllowedRedirectOrigins,
createBeforeUnloadTracker,
createPageLifecycle,
disabledCommerceFeature,
disabledOrganizationsFeature,
errorThrower,
generateSignatureWithCoinbaseWallet,
@@ -921,6 +922,14 @@ export class Clerk implements ClerkInterface {

public __experimental_mountPricingTable = (node: HTMLDivElement, props?: __experimental_PricingTableProps): void => {
this.assertComponentsReady(this.#componentControls);
if (disabledCommerceFeature(this, this.environment)) {
if (this.#instanceType === 'development') {
throw new ClerkRuntimeError(warnings.cannotRenderAnyCommerceComponent('PricingTable'), {
code: 'cannot_render_commerce_disabled',
});
}
return;
}
void this.#componentControls.ensureMounted({ preloadHint: 'PricingTable' }).then(controls =>
controls.mountComponent({
name: 'PricingTable',
27 changes: 24 additions & 3 deletions packages/clerk-js/src/core/modules/commerce/CommerceBilling.ts
Original file line number Diff line number Diff line change
@@ -3,26 +3,47 @@ import type {
__experimental_CommerceCheckoutJSON,
__experimental_CommercePlanResource,
__experimental_CommerceProductJSON,
__experimental_CommerceSubscriptionJSON,
__experimental_CommerceSubscriptionResource,
__experimental_CreateCheckoutParams,
__experimental_GetPlansParams,
ClerkPaginatedResponse,
} from '@clerk/types';

import { convertPageToOffsetSearchParams } from '../../../utils/convertPageToOffsetSearchParams';
import { __experimental_CommerceCheckout, __experimental_CommercePlan, BaseResource } from '../../resources/internal';
import {
__experimental_CommerceCheckout,
__experimental_CommercePlan,
__experimental_CommerceSubscription,
BaseResource,
} from '../../resources/internal';

export class __experimental_CommerceBilling implements __experimental_CommerceBillingNamespace {
getPlans = async (params?: __experimental_GetPlansParams): Promise<__experimental_CommercePlanResource[]> => {
const { data: products } = (await BaseResource._fetch({
path: `/commerce/products`,
method: 'GET',
search: convertPageToOffsetSearchParams(params),
search: { payerType: params?.subscriberType || '' },
})) as unknown as ClerkPaginatedResponse<__experimental_CommerceProductJSON>;

const defaultProduct = products.find(product => product.is_default);
return defaultProduct?.plans.map(plan => new __experimental_CommercePlan(plan)) || [];
};

getSubscriptions = async (): Promise<ClerkPaginatedResponse<__experimental_CommerceSubscriptionResource>> => {
return await BaseResource._fetch({
path: `/me/subscriptions`,
method: 'GET',
}).then(res => {
const { data: subscriptions, total_count } =
res?.response as unknown as ClerkPaginatedResponse<__experimental_CommerceSubscriptionJSON>;

return {
total_count,
data: subscriptions.map(subscription => new __experimental_CommerceSubscription(subscription)),
};
});
};

startCheckout = async (params: __experimental_CreateCheckoutParams) => {
const json = (
await BaseResource._fetch<__experimental_CommerceCheckoutJSON>({
3 changes: 2 additions & 1 deletion packages/clerk-js/src/core/resources/CommerceCheckout.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import type {
__experimental_CommerceCheckoutJSON,
__experimental_CommerceCheckoutResource,
__experimental_CommerceSubscriptionPlanPeriod,
__experimental_CommerceTotals,
__experimental_ConfirmCheckoutParams,
} from '@clerk/types';
@@ -23,7 +24,7 @@ export class __experimental_CommerceCheckout extends BaseResource implements __e
invoice?: __experimental_CommerceInvoice;
paymentSource?: __experimental_CommercePaymentSource;
plan!: __experimental_CommercePlan;
planPeriod!: string;
planPeriod!: __experimental_CommerceSubscriptionPlanPeriod;
status!: string;
subscription?: __experimental_CommerceSubscription;
totals!: __experimental_CommerceTotals;
3 changes: 1 addition & 2 deletions packages/clerk-js/src/core/resources/CommercePlan.ts
Original file line number Diff line number Diff line change
@@ -12,14 +12,14 @@ export class __experimental_CommercePlan extends BaseResource implements __exper
currencySymbol!: string;
currency!: string;
description!: string;
isActiveForPayer!: boolean;
isRecurring!: boolean;
hasBaseFee!: boolean;
payerType!: string[];
publiclyVisible!: boolean;
slug!: string;
avatarUrl!: string;
features!: __experimental_CommerceFeature[];
subscriptionIdForCurrentSubscriber: string | undefined;

constructor(data: __experimental_CommercePlanJSON) {
super();
@@ -40,7 +40,6 @@ export class __experimental_CommercePlan extends BaseResource implements __exper
this.currencySymbol = data.currency_symbol;
this.currency = data.currency;
this.description = data.description;
this.isActiveForPayer = data.is_active_for_payer;
this.isRecurring = data.is_recurring;
this.hasBaseFee = data.has_base_fee;
this.payerType = data.payer_type;
3 changes: 3 additions & 0 deletions packages/clerk-js/src/core/resources/CommerceSettings.ts
Original file line number Diff line number Diff line change
@@ -11,6 +11,7 @@ import { BaseResource } from './internal';
*/
export class __experimental_CommerceSettings extends BaseResource implements __experimental_CommerceSettingsResource {
stripePublishableKey: string = '';
enabled: boolean = false;

public constructor(
data: __experimental_CommerceSettingsJSON | __experimental_CommerceSettingsJSONSnapshot | null = null,
@@ -26,12 +27,14 @@ export class __experimental_CommerceSettings extends BaseResource implements __e
return this;
}
this.stripePublishableKey = data.stripe_publishable_key;
this.enabled = data.enabled;
return this;
}

public __internal_toSnapshot(): __experimental_CommerceSettingsJSONSnapshot {
return {
stripe_publishable_key: this.stripePublishableKey,
enabled: this.enabled,
} as unknown as __experimental_CommerceSettingsJSONSnapshot;
}
}
14 changes: 9 additions & 5 deletions packages/clerk-js/src/core/resources/CommerceSubscription.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,12 @@
import type {
__experimental_CommerceSubscriptionJSON,
__experimental_CommerceSubscriptionPlanPeriod,
__experimental_CommerceSubscriptionResource,
__experimental_CommerceSubscriptionStatus,
DeletedObjectJSON,
} from '@clerk/types';

import { __experimental_CommercePlan, BaseResource } from './internal';
import { __experimental_CommercePlan, BaseResource, DeletedObject } from './internal';

export class __experimental_CommerceSubscription
extends BaseResource
@@ -12,8 +15,8 @@ export class __experimental_CommerceSubscription
id!: string;
paymentSourceId!: string;
plan!: __experimental_CommercePlan;
planPeriod!: string;
status!: string;
planPeriod!: __experimental_CommerceSubscriptionPlanPeriod;
status!: __experimental_CommerceSubscriptionStatus;

constructor(data: __experimental_CommerceSubscriptionJSON) {
super();
@@ -40,7 +43,8 @@ export class __experimental_CommerceSubscription
path: `/me/commerce/subscriptions/${this.id}`,
method: 'DELETE',
})
)?.response;
return json;
)?.response as unknown as DeletedObjectJSON;

return new DeletedObject(json);
}
}
33 changes: 32 additions & 1 deletion packages/clerk-js/src/core/resources/Organization.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,7 @@
import type {
__experimental_CommerceSubscriptionJSON,
__experimental_CommerceSubscriptionResource,
__experimental_GetSubscriptionsParams,
AddMemberParams,
ClerkPaginatedResponse,
ClerkResourceReloadParams,
@@ -28,7 +31,12 @@ import type {

import { convertPageToOffsetSearchParams } from '../../utils/convertPageToOffsetSearchParams';
import { unixEpochToDate } from '../../utils/date';
import { BaseResource, OrganizationInvitation, OrganizationMembership } from './internal';
import {
__experimental_CommerceSubscription,
BaseResource,
OrganizationInvitation,
OrganizationMembership,
} from './internal';
import { OrganizationDomain } from './OrganizationDomain';
import { OrganizationMembershipRequest } from './OrganizationMembershipRequest';
import { Role } from './Role';
@@ -229,6 +237,29 @@ export class Organization extends BaseResource implements OrganizationResource {
}).then(res => new OrganizationMembership(res?.response as OrganizationMembershipJSON));
};

__experimental_getSubscriptions = async (
getSubscriptionsParams?: __experimental_GetSubscriptionsParams,
): Promise<ClerkPaginatedResponse<__experimental_CommerceSubscriptionResource>> => {
return await BaseResource._fetch(
{
path: `/organizations/${this.id}/subscriptions`,
method: 'GET',
search: convertPageToOffsetSearchParams(getSubscriptionsParams),
},
{
forceUpdateClient: true,
},
).then(res => {
const { data: subscriptions, total_count } =
res?.response as unknown as ClerkPaginatedResponse<__experimental_CommerceSubscriptionJSON>;

return {
total_count,
data: subscriptions.map(subscription => new __experimental_CommerceSubscription(subscription)),
};
});
};

destroy = async (): Promise<void> => {
return this._baseDelete();
};
Original file line number Diff line number Diff line change
@@ -10,6 +10,7 @@ exports[`Environment __internal_toSnapshot() 1`] = `
"single_session_mode": true,
},
"commerce_settings": {
"enabled": false,
"stripe_publishable_key": "",
},
"display_config": {
@@ -271,6 +272,7 @@ exports[`Environment __internal_toSnapshot() 1`] = `
exports[`Environment defaults values when instantiated without arguments 1`] = `
Environment {
"__experimental_commerceSettings": __experimental_CommerceSettings {
"enabled": false,
"pathRoot": "",
"stripePublishableKey": "",
},
@@ -497,6 +499,7 @@ Environment {
exports[`Environment has the same initial properties 1`] = `
Environment {
"__experimental_commerceSettings": __experimental_CommerceSettings {
"enabled": false,
"pathRoot": "",
"stripePublishableKey": "",
},
Original file line number Diff line number Diff line change
@@ -2,6 +2,7 @@

exports[`Organization has the same initial properties 1`] = `
Organization {
"__experimental_getSubscriptions": [Function],
"addMember": [Function],
"adminDeleteEnabled": true,
"createDomain": [Function],
Original file line number Diff line number Diff line change
@@ -6,6 +6,7 @@ OrganizationMembership {
"destroy": [Function],
"id": "test_id",
"organization": Organization {
"__experimental_getSubscriptions": [Function],
"addMember": [Function],
"adminDeleteEnabled": true,
"createDomain": [Function],
6 changes: 6 additions & 0 deletions packages/clerk-js/src/core/warnings.ts
Original file line number Diff line number Diff line change
@@ -11,6 +11,11 @@ const createMessageForDisabledOrganizations = (
`The <${componentName}/> cannot be rendered when the feature is turned off. Visit 'dashboard.clerk.com' to enable the feature. Since the feature is turned off, this is no-op.`,
);
};
const createMessageForDisabledCommerce = (componentName: 'PricingTable' | 'Checkout') => {
return formatWarning(
`The <${componentName}/> component cannot be rendered when commerce is disabled. Visit 'https://dashboard.clerk.com/last-active?path=commerce/settings' to follow the necessary steps to enable commerce. Since commerce is disabled, this is no-op.`,
);
};
const warnings = {
cannotRenderComponentWhenSessionExists:
'The <SignUp/> and <SignIn/> components cannot render when a user is already signed in, unless the application allows multiple sessions. Since a user is signed in and this application only allows a single session, Clerk is redirecting to the Home URL instead.',
@@ -26,6 +31,7 @@ const warnings = {
'<UserProfile/> cannot render unless a user is signed in. Since no user is signed in, this is no-op.',
cannotRenderComponentWhenOrgDoesNotExist: `<OrganizationProfile/> cannot render unless an organization is active. Since no organization is currently active, this is no-op.`,
cannotRenderAnyOrganizationComponent: createMessageForDisabledOrganizations,
cannotRenderAnyCommerceComponent: createMessageForDisabledCommerce,
cannotOpenUserProfile:
'The UserProfile modal cannot render unless a user is signed in. Since no user is signed in, this is no-op.',
cannotOpenSignInOrSignUp:
Original file line number Diff line number Diff line change
@@ -125,7 +125,7 @@ const CheckoutFormElements = ({
const [isSubmitting, setIsSubmitting] = useState(false);
const [submitError, setSubmitError] = useState<ClerkRuntimeError | ClerkAPIError | string | undefined>();

const { data } = useFetch(__experimental_commerce?.getPaymentSources, {});
const { data } = useFetch(__experimental_commerce?.getPaymentSources, 'commerce-payment-sources');
const { data: paymentSources } = data || { data: [] };

const didExpandStripePaymentMethods = useCallback(() => {
12 changes: 9 additions & 3 deletions packages/clerk-js/src/ui/components/Checkout/CheckoutPage.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import type { __experimental_CheckoutProps } from '@clerk/types';
import type { __experimental_CheckoutProps, __experimental_CommerceCheckoutResource } from '@clerk/types';
import type { Stripe } from '@stripe/stripe-js';
import { loadStripe } from '@stripe/stripe-js';
import { useEffect, useRef, useState } from 'react';
@@ -10,14 +10,15 @@ import { CheckoutComplete } from './CheckoutComplete';
import { CheckoutForm } from './CheckoutForm';

export const CheckoutPage = (props: __experimental_CheckoutProps) => {
const { planId, planPeriod } = props;
const { planId, planPeriod, orgId, onSubscriptionComplete } = props;
const stripePromiseRef = useRef<Promise<Stripe | null> | null>(null);
const [stripe, setStripe] = useState<Stripe | null>(null);
const { __experimental_commerceSettings } = useEnvironment();

const { checkout, updateCheckout, isLoading } = useCheckout({
planId,
planPeriod,
orgId,
});

useEffect(() => {
@@ -35,6 +36,11 @@ export const CheckoutPage = (props: __experimental_CheckoutProps) => {
}
}, [checkout?.externalGatewayId, __experimental_commerceSettings]);

const onCheckoutComplete = (newCheckout: __experimental_CommerceCheckoutResource) => {
updateCheckout(newCheckout);
onSubscriptionComplete?.();
};

if (isLoading) {
return (
<Spinner
@@ -69,7 +75,7 @@ export const CheckoutPage = (props: __experimental_CheckoutProps) => {
<CheckoutForm
stripe={stripe}
checkout={checkout}
onCheckoutComplete={updateCheckout}
onCheckoutComplete={onCheckoutComplete}
/>
);
};
Loading
Oops, something went wrong.