;
isDowngrade?: boolean;
};
-function NotEligibleSwitchPlanModalContent({ targetPlan, isDowngrade = false }: Props) {
+function NotEligibleSwitchPlanModalContent({
+ targetPlan,
+ exceededQuotaKeys,
+ isDowngrade = false,
+}: Props) {
const { t } = useTranslation(undefined, {
keyPrefix: 'admin_console.subscription.not_eligible_modal',
});
@@ -42,11 +47,12 @@ function NotEligibleSwitchPlanModalContent({ targetPlan, isDowngrade = false }:
// eslint-disable-next-line no-restricted-syntax
const entries = Object.entries(quota) as SubscriptionPlanQuotaEntries;
return entries
+ .filter(([quotaKey]) => exceededQuotaKeys.includes(quotaKey))
.slice()
.sort(([preQuotaKey], [nextQuotaKey]) =>
sortBy(planQuotaItemOrder)(preQuotaKey, nextQuotaKey)
);
- }, [quota]);
+ }, [quota, exceededQuotaKeys]);
return (
diff --git a/packages/console/src/utils/subscription.ts b/packages/console/src/utils/subscription.ts
index a4f872e8a..12458887c 100644
--- a/packages/console/src/utils/subscription.ts
+++ b/packages/console/src/utils/subscription.ts
@@ -1,3 +1,4 @@
+import { conditional, trySafe } from '@silverhand/essentials';
import { ResponseError } from '@withtyped/client';
import dayjs from 'dayjs';
@@ -10,7 +11,7 @@ import {
ticketSupportResponseTimeMap,
} from '@/consts/plan-quotas';
import { featuredPlanIdOrder, featuredPlanIds } from '@/consts/subscriptions';
-import { type SubscriptionPlan } from '@/types/subscriptions';
+import { type SubscriptionPlanQuota, type SubscriptionPlan } from '@/types/subscriptions';
export const addSupportQuotaToPlan = (subscriptionPlanResponse: SubscriptionPlanResponse) => {
const { id, quota } = subscriptionPlanResponse;
@@ -51,17 +52,46 @@ export const formatPeriod = ({ periodStart, periodEnd, displayYear }: FormatPeri
};
/**
- * Note: this is a temporary solution to handle the case when the user tries to downgrade but the quota limit is exceeded.
- * Need a better solution to handle this case by sharing the error type between the console and cloud. - LOG-6608
+ * Parse the error data from the server if the error is caused by exceeding the quota limit.
+ * This is used to handle cases where users attempt to switch subscription plans, but the quota limit is exceeded.
+ *
+ * @param error - The error object from the server.
+ *
+ * @returns If the error is caused by exceeding the quota limit, returns `[true, exceededQuotaKeys]`, otherwise `[false]`.
+ *
+ * @remarks
+ * - This function parses the exceeded quota data from the error message string, since the server which uses `withtyped`
+ * only supports to return a `message` field in the error response body.
+ * - The choice to return exceeded quota keys instead of the entire data object is intentional.
+ * The data returned from the server is quota usage data, but what we want is quota limit data, so we will read quota limits from subscription plans.
*/
-export const isExceededQuotaLimitError = async (error: unknown) => {
+export const parseExceededQuotaLimitError = async (
+ error: unknown
+): Promise<[false] | [true, Array]> => {
if (!(error instanceof ResponseError)) {
- return false;
+ return [false];
}
const { message } = (await tryReadResponseErrorBody(error)) ?? {};
- return Boolean(message?.includes('Exceeded quota limit'));
+ const match = message?.match(/Status exception: Exceeded quota limit\. (.+)$/);
+
+ if (!match) {
+ return [false];
+ }
+
+ const data = match[1];
+ const exceededQuota = conditional(
+ // eslint-disable-next-line no-restricted-syntax -- trust the type from the server if error message matches
+ data && trySafe(() => JSON.parse(data) as Partial)
+ );
+
+ if (!exceededQuota) {
+ return [false];
+ }
+
+ // eslint-disable-next-line no-restricted-syntax
+ return [true, Object.keys(exceededQuota) as Array];
};
export const pickupFeaturedPlans = (plans: SubscriptionPlan[]): SubscriptionPlan[] =>