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: Add self serve billing management UI #5431

Merged
merged 80 commits into from
Oct 28, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
80 commits
Select commit Hold shift + click to select a range
2440f32
Add basic plan info
AdityaHegde Aug 12, 2024
acb2da5
Add actions for switching plans
AdityaHegde Aug 19, 2024
44a38ab
Add payment section
AdityaHegde Aug 19, 2024
475dbd0
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Aug 21, 2024
441e0bf
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Aug 26, 2024
662bfee
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Sep 19, 2024
418e180
Use new APIs
AdityaHegde Sep 19, 2024
376cbaa
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Sep 23, 2024
96d26a0
Add banners for trial and billing issues
AdityaHegde Sep 23, 2024
72e6508
Improve trial end math
AdityaHegde Sep 24, 2024
1f71f66
Add TRIAL_ENDING_SOON state
AdityaHegde Sep 24, 2024
fade30f
Remove TRIAL_ENDING_SOON
AdityaHegde Sep 30, 2024
83ae354
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Sep 30, 2024
e5af9dd
Add plan pill
AdityaHegde Sep 30, 2024
c337e64
Add start team plan variants
AdityaHegde Sep 30, 2024
a16c1f7
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 1, 2024
6ba3913
Handle payment issues
AdityaHegde Oct 1, 2024
b60caca
Handle some edge cases
AdityaHegde Oct 1, 2024
368dd77
Open stripe page before plan upgrade
AdityaHegde Oct 2, 2024
ebc8e6e
Use billing issue to show trial dates
AdityaHegde Oct 3, 2024
7730520
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 3, 2024
5a3105c
Avoid using ListPlans API
AdityaHegde Oct 3, 2024
c4ea1c3
Add null checks
AdityaHegde Oct 3, 2024
8310b96
Fix trial grace period end check
AdityaHegde Oct 4, 2024
df1f20b
Handle cancelled subscriptions
AdityaHegde Oct 4, 2024
9752b05
UXQA fixes - pass 1
AdityaHegde Oct 5, 2024
fc7e934
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 7, 2024
3dabb6a
Use RenewBillingSubscription
AdityaHegde Oct 7, 2024
1d61d74
Handle no subscription case
AdityaHegde Oct 8, 2024
e993973
Always fetch stripe page
AdityaHegde Oct 8, 2024
1e5f1dd
Embed Orb customer portal
AdityaHegde Oct 8, 2024
d4b790c
Open upgrader in same page
AdityaHegde Oct 8, 2024
79f4874
Tweaks
AdityaHegde Oct 8, 2024
883738d
Fix misc issues
AdityaHegde Oct 8, 2024
2198ea1
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 9, 2024
599d568
UXQA - Pass 2
AdityaHegde Oct 9, 2024
7560fa8
Handle viewer cases
AdityaHegde Oct 9, 2024
3613cc3
Refactor to use billing issues message in hibernating message
AdityaHegde Oct 10, 2024
9c2b883
Support PAYMENT_FAILED_ISSUE
AdityaHegde Oct 10, 2024
cfe2460
Add wake all projects
AdityaHegde Oct 10, 2024
98e75c8
UXQA - Pass 3
AdityaHegde Oct 11, 2024
187bb17
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 11, 2024
d2148b1
Hibernating message improvements
AdityaHegde Oct 14, 2024
c4d546b
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 14, 2024
e382acc
Fix hibernate sub cancel check
AdityaHegde Oct 14, 2024
c6389c2
minor changes (#5890)
pjain1 Oct 14, 2024
e49a3d8
Revert asset deletion
AdityaHegde Oct 15, 2024
24acbb9
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 16, 2024
d65af0d
UXQA - Pass 4
AdityaHegde Oct 16, 2024
3d0718c
Add back quotas
AdityaHegde Oct 16, 2024
de504fe
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 17, 2024
f6fc368
Hardcode plan names
AdityaHegde Oct 17, 2024
ab7e0f1
Add CTAs to existing email formats
AdityaHegde Oct 17, 2024
323699b
UI Review comments
AdityaHegde Oct 18, 2024
472e5de
PR comments pass 2
AdityaHegde Oct 21, 2024
dfd796b
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 22, 2024
1be4252
Fix upgrade callback url to stripe
AdityaHegde Oct 22, 2024
4c04d87
PR comments pass 3
AdityaHegde Oct 23, 2024
a671dd4
Styling updates
AdityaHegde Oct 24, 2024
5674229
Add spinner to iframe loading
AdityaHegde Oct 24, 2024
8ec01f5
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 24, 2024
6d57a5c
Add POC plan support
AdityaHegde Oct 24, 2024
4b87ed9
Tweaks
AdityaHegde Oct 24, 2024
8dcd55b
PR comments pass 4
AdityaHegde Oct 25, 2024
f4301e5
fix start trial race condition (#5971)
pjain1 Oct 25, 2024
a8e41c2
web-auth: use svelte and vite-plugin-singlefile (#5965)
briangregoryholmes Oct 24, 2024
d6ce8c4
Share Project Popover (#5887)
lovincyrus Oct 24, 2024
0d774ea
disable preview button when explore resource is reconciling (#5962)
briangregoryholmes Oct 24, 2024
06c444c
fix: display null values with italics in TDD (#5945)
briangregoryholmes Oct 25, 2024
c535b01
completely remove org from orb and stripe (#5955)
pjain1 Oct 25, 2024
035155b
Don't log request cancellation as internal errors in the Github APIs …
begelundmuller Oct 25, 2024
87b95ae
trigger org repair manually (#5936)
pjain1 Oct 25, 2024
2af7c7d
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 28, 2024
4b14bf3
Fix public URLs
AdityaHegde Oct 28, 2024
7c5ba24
Fix upgrade emails
AdityaHegde Oct 28, 2024
52df6b5
Merge branch 'main' into adityahegde/billing-ui
AdityaHegde Oct 28, 2024
0dc2a39
Fix lint
AdityaHegde Oct 28, 2024
691c99a
PR comments pass 5
AdityaHegde Oct 28, 2024
1deb7fb
check org billing init in start trial job (#5983)
pjain1 Oct 28, 2024
15132a4
Add back commented email
AdityaHegde Oct 28, 2024
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
Prev Previous commit
Next Next commit
Handle cancelled subscriptions
  • Loading branch information
AdityaHegde committed Oct 4, 2024
commit df1f20b4c566b8c71959b42dcc48fb958908eeeb
11 changes: 11 additions & 0 deletions web-admin/src/features/billing/banner/handleBillingIssues.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { V1BillingIssueType } from "@rilldata/web-admin/client";

Check failure on line 1 in web-admin/src/features/billing/banner/handleBillingIssues.ts

View workflow job for this annotation

GitHub Actions / build

'V1BillingIssueType' is defined but never used
import type {
V1BillingIssue,
V1Subscription,
Expand All @@ -6,6 +7,10 @@
handlePaymentIssues,
PaymentBillingIssueTypes,
} from "@rilldata/web-admin/features/billing/banner/handlePaymentBillingIssues";
import {
getCancelledSubIssue,
handleSubscriptionIssues,
} from "@rilldata/web-admin/features/billing/banner/handleSubscriptionIssues";
import { handleTrialPlan } from "@rilldata/web-admin/features/billing/banner/handleTrialPlan";
import { isTrialPlan } from "@rilldata/web-admin/features/billing/plans/utils";
import { eventBus } from "@rilldata/web-common/lib/event-bus/event-bus";
Expand All @@ -15,6 +20,12 @@
subscription: V1Subscription,
issues: V1BillingIssue[],
) {
const cancelledSubIssue = getCancelledSubIssue(issues);
if (cancelledSubIssue) {
eventBus.emit("banner", handleSubscriptionIssues(cancelledSubIssue));
return;
}

if (isTrialPlan(subscription.plan)) {
eventBus.emit("banner", handleTrialPlan(issues));
return;
Expand Down
52 changes: 52 additions & 0 deletions web-admin/src/features/billing/banner/handleSubscriptionIssues.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
import { V1BillingIssueType } from "@rilldata/web-admin/client";
import type { V1BillingIssue } from "@rilldata/web-admin/client";
import {
showUpgradeDialog,
upgradeDialogType,
} from "@rilldata/web-admin/features/billing/banner/bannerCTADialogs";
import type { BannerMessage } from "@rilldata/web-common/lib/event-bus/events";
import { DateTime } from "luxon";

export function getCancelledSubIssue(issues: V1BillingIssue[]) {
return issues.find(
(i) =>
i.type === V1BillingIssueType.BILLING_ISSUE_TYPE_SUBSCRIPTION_CANCELLED,
);
}

export function handleSubscriptionIssues(cancelledSubIssue: V1BillingIssue) {
let accessTimeout = "";

if (cancelledSubIssue.metadata.subscriptionCancelled?.endDate) {
const endDate = DateTime.fromJSDate(
new Date(cancelledSubIssue.metadata.subscriptionCancelled?.endDate),
);
if (endDate.isValid) {
accessTimeout = `but you still have access through ${endDate.toLocaleString(DateTime.DATE_MED)}`;
}
}

return <BannerMessage>{
type: "warning",
message: `Your plan was canceled${accessTimeout}. To maintain access, renew your plan.`,
iconType: "alert",
cta: {
text: "Renew ->",
type: "button",
onClick: () => {
showUpgradeDialog.set(true);
upgradeDialogType.set("renew");
},
},
};
}

export function cancelledSubscriptionHasEnded(
cancelledSubIssue: V1BillingIssue,
) {
const endDate = new Date(
cancelledSubIssue.metadata?.subscriptionCancelled?.endDate,
);
const endTime = endDate.getTime();
return Number.isNaN(endTime) || endTime < Date.now();
}
22 changes: 18 additions & 4 deletions web-admin/src/features/billing/banner/handleTrialPlan.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,10 @@ import {
type V1BillingIssue,
V1BillingIssueType,
} from "@rilldata/web-admin/client";
import { showUpgradeDialog } from "@rilldata/web-admin/features/billing/banner/bannerCTADialogs";
import {
showUpgradeDialog,
upgradeDialogType,
} from "@rilldata/web-admin/features/billing/banner/bannerCTADialogs";
import type { BannerMessage } from "@rilldata/web-common/lib/event-bus/events";
import { shiftToLargest } from "@rilldata/web-common/lib/time/ranges/iso-ranges";
import { DateTime, type Duration } from "luxon";
Expand All @@ -12,7 +15,10 @@ const WarningPeriodInDays = 7;
const cta: BannerMessage["cta"] = {
text: "Upgrade ->",
type: "button",
onClick: () => showUpgradeDialog.set(true),
onClick: () => {
showUpgradeDialog.set(true);
upgradeDialogType.set("base");
},
};

export function getTrialIssue(issues: V1BillingIssue[]) {
Expand All @@ -27,8 +33,8 @@ export function handleTrialPlan(issues: V1BillingIssue[]): BannerMessage {
const trialIssue = getTrialIssue(issues);

const endDateStr =
trialIssue.metadata?.onTrial?.endDate ??
trialIssue.metadata?.trialEnded?.gracePeriodEndDate ??
trialIssue?.metadata?.onTrial?.endDate ??
trialIssue?.metadata?.trialEnded?.gracePeriodEndDate ??
"";

const today = DateTime.now();
Expand Down Expand Up @@ -81,6 +87,14 @@ export function getTrialMessageForDays(diff: Duration) {
return `Your trial expires in ${humanizeDuration(diff)}.`;
}

export function trialHasPastGracePeriod(trialEndedIssue: V1BillingIssue) {
const gracePeriodDate = new Date(
trialEndedIssue.metadata?.trialEnded?.gracePeriodEndDate,
);
const gracePeriodTime = gracePeriodDate.getTime();
return Number.isNaN(gracePeriodTime) || gracePeriodTime < Date.now();
}

function humanizeDuration(dur: Duration) {
dur = shiftToLargest(dur, ["seconds", "minutes", "hours", "days"]);
return dur.toHuman({ unitDisplay: "short" });
Expand Down
83 changes: 24 additions & 59 deletions web-admin/src/features/billing/plans/EndedTeamPlan.svelte
ericpgreen2 marked this conversation as resolved.
Show resolved Hide resolved
AdityaHegde marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -1,41 +1,30 @@
<script lang="ts">
import {
createAdminServiceUpdateBillingSubscription,
createAdminServiceListOrganizationBillingIssues,
type V1Subscription,
} from "@rilldata/web-admin/client";
import { getCancelledSubIssue } from "@rilldata/web-admin/features/billing/banner/handleSubscriptionIssues";
import PlanQuotas from "@rilldata/web-admin/features/billing/plans/PlanQuotas.svelte";
import { getCategorisedPlans } from "@rilldata/web-admin/features/billing/plans/selectors";
import StartTeamPlanDialog from "@rilldata/web-admin/features/billing/plans/StartTeamPlanDialog.svelte";
import PricingDetails from "@rilldata/web-admin/features/billing/PricingDetails.svelte";
import SettingsContainer from "@rilldata/web-admin/features/organizations/settings/SettingsContainer.svelte";
import {
AlertDialog,
AlertDialogContent,
AlertDialogDescription,
AlertDialogFooter,
AlertDialogHeader,
AlertDialogTitle,
AlertDialogTrigger,
} from "@rilldata/web-common/components/alert-dialog";
import { Button } from "@rilldata/web-common/components/button";
import { DateTime } from "luxon";

export let organization: string;
export let subscription: V1Subscription;

$: plan = subscription.plan;

$: categorisedPlans = getCategorisedPlans();
$: trialPlan = $categorisedPlans.data.trialPlan;

$: planUpdater = createAdminServiceUpdateBillingSubscription();
async function handleUpgradePlan() {
if (!trialPlan) return;

await $planUpdater.mutateAsync({
organization,
data: {
planName: trialPlan.name,
},
});
$: issues = createAdminServiceListOrganizationBillingIssues(organization);
$: cancelledSubIssue = getCancelledSubIssue($issues.data?.issues ?? []);

let willEndOnText = "";
$: if (cancelledSubIssue?.metadata.subscriptionCancelled?.endDate) {
const endDate = DateTime.fromJSDate(
new Date(cancelledSubIssue.metadata.subscriptionCancelled?.endDate),
);
if (endDate.isValid)
willEndOnText = endDate.toLocaleString(DateTime.DATE_MED);
}

let open = false;
Expand All @@ -44,7 +33,11 @@
<SettingsContainer title={plan.displayName ?? plan.name} titleIcon="info">
<div slot="body">
<div>
Your subscription ends on {subscription.currentBillingCycleEndDate}
{#if willEndOnText}
Your subscription ends on {willEndOnText}
{:else}
Your subscription has ended.
{/if}
<PricingDetails />
</div>
<PlanQuotas {organization} quotas={plan.quotas} />
Expand All @@ -56,37 +49,9 @@
</Button>
</svelte:fragment>

<AlertDialog bind:open slot="action">
<AlertDialogTrigger asChild let:builder>
<Button builders={[builder]} type="primary">Cancel plan</Button>
</AlertDialogTrigger>
<AlertDialogContent>
<AlertDialogHeader>
<AlertDialogTitle>Are you sure you want to cancel?</AlertDialogTitle>

<AlertDialogDescription>
If you cancel your plan, you’ll still be able to access your account
through {subscription.currentBillingCycleEndDate}.
</AlertDialogDescription>

{#if $planUpdater.error}
<div class="text-red-500 text-sm py-px">
{$planUpdater.error.message}
</div>
{/if}
</AlertDialogHeader>
<AlertDialogFooter class="mt-3">
<Button
type="secondary"
on:click={handleUpgradePlan}
loading={$planUpdater.isLoading}
>
Cancel plan
</Button>
<Button type="primary" on:click={() => (open = false)}>
Keep plan
</Button>
</AlertDialogFooter>
</AlertDialogContent>
</AlertDialog>
<Button type="primary" slot="action" on:click={() => (open = true)}>
Renew Team plan
</Button>
</SettingsContainer>

<StartTeamPlanDialog bind:open {organization} type="renew" />
7 changes: 3 additions & 4 deletions web-admin/src/features/billing/plans/Plan.svelte
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
import EndedTeamPlan from "@rilldata/web-admin/features/billing/plans/EndedTeamPlan.svelte";
import EnterprisePlan from "@rilldata/web-admin/features/billing/plans/EnterprisePlan.svelte";
import TeamPlan from "@rilldata/web-admin/features/billing/plans/TeamPlan.svelte";
import TrialPlan from "@rilldata/web-admin/features/billing/plans/TrialPlan.svelte";

Check failure on line 6 in web-admin/src/features/billing/plans/Plan.svelte

View workflow job for this annotation

GitHub Actions / build

'TrialPlan' is defined but never used
import { isTrialPlan } from "@rilldata/web-admin/features/billing/plans/utils";

export let organization: string;
Expand All @@ -18,12 +18,11 @@

{#if subscription}
{#if isTrial}
<TrialPlan {organization} {subscription} />
<!-- <TrialPlan {organization} {subscription} />-->
<EndedTeamPlan {organization} {subscription} />
{:else if isBilled}
<TeamPlan {organization} {subscription} />
{:else if hasEnded}
<EndedTeamPlan {organization} {subscription} />
{:else}
{:else if hasEnded}{:else}

Check failure on line 25 in web-admin/src/features/billing/plans/Plan.svelte

View workflow job for this annotation

GitHub Actions / build

Empty block(empty-block)
<EnterprisePlan {organization} plan={subscription.plan} />
{/if}
{/if}
29 changes: 16 additions & 13 deletions web-admin/src/features/billing/plans/StartTeamPlanDialog.svelte
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
<script lang="ts" context="module">
/**
* 1. base - When user chooses to upgrade from a trial plan.
* 2. size - When user hits the size limit and wants to upgrade.
* 3. org - When user hits the organization limit and wants to upgrade.
* 4. proj - When user hits the project limit and wants to upgrade.
* 1. base - When user chooses to upgrade from a trial plan.
* 2. size - When user hits the size limit and wants to upgrade.
* 3. org - When user hits the organization limit and wants to upgrade.
* 4. proj - When user hits the project limit and wants to upgrade.
* 5. renew - After user cancels a subscription and wants to renew.
*/
export type TeamPlanDialogTypes = "base" | "size" | "org" | "proj";
export type TeamPlanDialogTypes = "base" | "size" | "org" | "proj" | "renew";
</script>

<script lang="ts">
Expand Down Expand Up @@ -61,11 +62,18 @@
case "proj":
title = "To deploy a second project, start a Team plan";
break;

case "renew":
title = "Renew Team plan";
// TODO resume
description = `Your billing cycle will resume on TODO. Pricing is based on amount of data ingested (and compressed) into Rill`;
buttonText = "Continue";
break;
}
}
$: setCopyBasedOnType(type);

$: categorisedPlans = getCategorisedPlans(open);
$: categorisedPlans = getCategorisedPlans(open); // only fetch when the dialog is opened
$: teamPlan = $categorisedPlans.data?.teamPlan;
$: paymentIssues = getPaymentIssues(organization);
$: paymentUrl = createAdminServiceGetPaymentsPortalURL(organization, {
Expand All @@ -91,7 +99,7 @@
}

$: loading =
$categorisedPlans.isLoading ||
$categorisedPlans.isLoading || // TODO: wait for this in handleUpgradePlan instead of add to spinner
$paymentIssues.isLoading ||
$paymentUrl.isLoading ||
$planUpdater.isLoading;
Expand Down Expand Up @@ -126,12 +134,7 @@
</AlertDialogHeader>
<AlertDialogFooter class="mt-3">
<Button type="secondary" on:click={() => (open = false)}>Close</Button>
<Button
type="primary"
on:click={handleUpgradePlan}
{loading}
disabled={loading}
>
<Button type="primary" on:click={handleUpgradePlan} {loading}>
{buttonText}
</Button>
</AlertDialogFooter>
Expand Down
31 changes: 30 additions & 1 deletion web-admin/src/features/billing/selectors.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,11 @@
import { createAdminServiceGetBillingSubscription } from "@rilldata/web-admin/client";
import {
createAdminServiceGetBillingSubscription,
createAdminServiceListOrganizationBillingIssues,
V1BillingIssueType,
} from "@rilldata/web-admin/client";
import { PaymentBillingIssueTypes } from "@rilldata/web-admin/features/billing/banner/handlePaymentBillingIssues";
import { cancelledSubscriptionHasEnded } from "@rilldata/web-admin/features/billing/banner/handleSubscriptionIssues";
import { trialHasPastGracePeriod } from "@rilldata/web-admin/features/billing/banner/handleTrialPlan";

export function getPlanForOrg(org: string, enabled = true) {
return createAdminServiceGetBillingSubscription(org, {
Expand All @@ -8,3 +15,25 @@ export function getPlanForOrg(org: string, enabled = true) {
},
});
}

export function getOrgBlockerIssues(org: string) {
return createAdminServiceListOrganizationBillingIssues(org, {
query: {
select: (data) =>
data.issues?.map((i) => {
switch (i.type) {
case V1BillingIssueType.BILLING_ISSUE_TYPE_TRIAL_ENDED:
return trialHasPastGracePeriod(i) ? "Trial has ended." : "";
case V1BillingIssueType.BILLING_ISSUE_TYPE_SUBSCRIPTION_CANCELLED:
return cancelledSubscriptionHasEnded(i)
? "Subscription cancelled."
: "";
default:
return i.type in PaymentBillingIssueTypes
? "Invoice payment failed"
: "";
}
})?.[0],
},
});
}
Loading
Loading