0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2025-01-06 20:40:08 -05:00

feat(console): add charge notification for token usage (#5153)

This commit is contained in:
Xiao Yijun 2023-12-25 17:13:07 +08:00 committed by GitHub
parent a6f12307b3
commit 12e6d288dd
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 41 additions and 4 deletions

View file

@ -26,7 +26,7 @@
"@fontsource/roboto-mono": "^5.0.0", "@fontsource/roboto-mono": "^5.0.0",
"@jest/types": "^29.5.0", "@jest/types": "^29.5.0",
"@logto/app-insights": "workspace:^1.3.1", "@logto/app-insights": "workspace:^1.3.1",
"@logto/cloud": "0.2.5-5deb133", "@logto/cloud": "0.2.5-d28a065",
"@logto/connector-kit": "workspace:^2.0.0", "@logto/connector-kit": "workspace:^2.0.0",
"@logto/core-kit": "workspace:^2.2.1", "@logto/core-kit": "workspace:^2.2.1",
"@logto/language-kit": "workspace:^1.0.0", "@logto/language-kit": "workspace:^1.0.0",

View file

@ -82,6 +82,7 @@ const defaultTenantResponse: TenantResponse = {
usage: { usage: {
activeUsers: 0, activeUsers: 0,
cost: 0, cost: 0,
tokenUsage: 0,
}, },
openInvoices: [], openInvoices: [],
isSuspended: false, isSuspended: false,

View file

@ -1,11 +1,16 @@
import { cond } from '@silverhand/essentials';
import { type SubscriptionUsage, type Subscription } from '@/cloud/types/router'; import { type SubscriptionUsage, type Subscription } from '@/cloud/types/router';
import BillInfo from '@/components/BillInfo'; import BillInfo from '@/components/BillInfo';
import ChargeNotification from '@/components/ChargeNotification';
import FormCard from '@/components/FormCard'; import FormCard from '@/components/FormCard';
import PlanDescription from '@/components/PlanDescription'; import PlanDescription from '@/components/PlanDescription';
import PlanName from '@/components/PlanName'; import PlanName from '@/components/PlanName';
import PlanUsage from '@/components/PlanUsage'; import PlanUsage from '@/components/PlanUsage';
import { isDevFeaturesEnabled } from '@/consts/env';
import FormField from '@/ds-components/FormField'; import FormField from '@/ds-components/FormField';
import { type SubscriptionPlan } from '@/types/subscriptions'; import { type SubscriptionPlan } from '@/types/subscriptions';
import { hasSurpassedQuotaLimit } from '@/utils/quota';
import MauLimitExceedNotification from './MauLimitExceededNotification'; import MauLimitExceedNotification from './MauLimitExceededNotification';
import PaymentOverdueNotification from './PaymentOverdueNotification'; import PaymentOverdueNotification from './PaymentOverdueNotification';
@ -18,7 +23,17 @@ type Props = {
}; };
function CurrentPlan({ subscription, subscriptionPlan, subscriptionUsage }: Props) { function CurrentPlan({ subscription, subscriptionPlan, subscriptionUsage }: Props) {
const { id, name } = subscriptionPlan; const {
id,
name,
quota: { tokenLimit },
} = subscriptionPlan;
const hasTokenSurpassedLimit = hasSurpassedQuotaLimit({
quotaKey: 'tokenLimit',
usage: subscriptionUsage.tokenUsage,
plan: subscriptionPlan,
});
return ( return (
<FormCard title="subscription.current_plan" description="subscription.current_plan_description"> <FormCard title="subscription.current_plan" description="subscription.current_plan_description">
@ -48,6 +63,16 @@ function CurrentPlan({ subscription, subscriptionPlan, subscriptionUsage }: Prop
currentPlan={subscriptionPlan} currentPlan={subscriptionPlan}
className={styles.notification} className={styles.notification}
/> />
{/* Todo @xiaoyijun [Pricing] Remove feature flag */}
{isDevFeaturesEnabled && (
<ChargeNotification
hasSurpassedLimit={hasTokenSurpassedLimit}
quotaItemPhraseKey="tokens"
checkedFlagKey="token"
className={styles.notification}
quotaLimit={cond(typeof tokenLimit === 'number' && tokenLimit)}
/>
)}
<PaymentOverdueNotification className={styles.notification} /> <PaymentOverdueNotification className={styles.notification} />
</FormCard> </FormCard>
); );

View file

@ -60,6 +60,7 @@ export const adminConsoleDataGuard = z.object({
.optional(), .optional(),
checkedChargeNotification: z checkedChargeNotification: z
.object({ .object({
token: z.boolean().optional(),
apiResource: z.boolean().optional(), apiResource: z.boolean().optional(),
machineToMachineApp: z.boolean().optional(), machineToMachineApp: z.boolean().optional(),
}) })

View file

@ -2840,8 +2840,8 @@ importers:
specifier: workspace:^1.3.1 specifier: workspace:^1.3.1
version: link:../app-insights version: link:../app-insights
'@logto/cloud': '@logto/cloud':
specifier: 0.2.5-5deb133 specifier: 0.2.5-d28a065
version: 0.2.5-5deb133(zod@3.22.4) version: 0.2.5-d28a065(zod@3.22.4)
'@logto/connector-kit': '@logto/connector-kit':
specifier: workspace:^2.0.0 specifier: workspace:^2.0.0
version: link:../toolkit/connector-kit version: link:../toolkit/connector-kit
@ -7603,6 +7603,16 @@ packages:
- zod - zod
dev: true dev: true
/@logto/cloud@0.2.5-d28a065(zod@3.22.4):
resolution: {integrity: sha512-04uwAaE9bIixt4nHAeqQsOzXxB0yFNDUYArJWjTo8xcmTQGBQttq8xyVKWVWCWPp7hSEZhb74h5b45my0IOMPA==}
engines: {node: ^20.9.0}
dependencies:
'@silverhand/essentials': 2.8.6
'@withtyped/server': 0.12.9(zod@3.22.4)
transitivePeerDependencies:
- zod
dev: true
/@logto/js@3.0.1: /@logto/js@3.0.1:
resolution: {integrity: sha512-vsU6mH5oiiW3k00pMyVA4V31K2Bd0rOT9qWch2l5e5o1yCQLJ3zUIOjGjChu3m2TRu1d920iiUpZU3Lzf6Pwdw==} resolution: {integrity: sha512-vsU6mH5oiiW3k00pMyVA4V31K2Bd0rOT9qWch2l5e5o1yCQLJ3zUIOjGjChu3m2TRu1d920iiUpZU3Lzf6Pwdw==}
dependencies: dependencies: