π File detail
components/messages/RateLimitMessage.tsx
π― Use case
This file lives under βcomponents/β, which covers shared React UI pieces. On the API surface it exposes getUpsellMessage and RateLimitMessage β mainly functions, hooks, or classes. Dependencies touch React UI and src. It composes internal code from MessageResponse (relative imports).
Generated from folder role, exports, dependency roots, and inline comments β not hand-reviewed for every path.
π§ Inline summary
import { c as _c } from "react/compiler-runtime"; import React, { useEffect, useMemo, useState } from 'react'; import { extraUsage } from 'src/commands/extra-usage/index.js'; import { Box, Text } from 'src/ink.js'; import { useClaudeAiLimits } from 'src/services/claudeAiLimitsHook.js';
π€ Exports (heuristic)
getUpsellMessageRateLimitMessage
π External import roots
Package roots from from "β¦" (relative paths omitted).
reactsrc
π₯οΈ Source preview
import { c as _c } from "react/compiler-runtime";
import React, { useEffect, useMemo, useState } from 'react';
import { extraUsage } from 'src/commands/extra-usage/index.js';
import { Box, Text } from 'src/ink.js';
import { useClaudeAiLimits } from 'src/services/claudeAiLimitsHook.js';
import { shouldProcessMockLimits } from 'src/services/rateLimitMocking.js'; // Used for /mock-limits command
import { getRateLimitTier, getSubscriptionType, isClaudeAISubscriber } from 'src/utils/auth.js';
import { hasClaudeAiBillingAccess } from 'src/utils/billing.js';
import { MessageResponse } from '../MessageResponse.js';
type UpsellParams = {
shouldShowUpsell: boolean;
isMax20x: boolean;
isExtraUsageCommandEnabled: boolean;
shouldAutoOpenRateLimitOptionsMenu: boolean;
isTeamOrEnterprise: boolean;
hasBillingAccess: boolean;
};
export function getUpsellMessage({
shouldShowUpsell,
isMax20x,
isExtraUsageCommandEnabled,
shouldAutoOpenRateLimitOptionsMenu,
isTeamOrEnterprise,
hasBillingAccess
}: UpsellParams): string | null {
if (!shouldShowUpsell) return null;
if (isMax20x) {
if (isExtraUsageCommandEnabled) {
return '/extra-usage to finish what you\u2019re working on.';
}
return '/login to switch to an API usage-billed account.';
}
if (shouldAutoOpenRateLimitOptionsMenu) {
return 'Opening your options\u2026';
}
if (!isTeamOrEnterprise && !isExtraUsageCommandEnabled) {
return '/upgrade to increase your usage limit.';
}
if (isTeamOrEnterprise) {
if (!isExtraUsageCommandEnabled) return null;
if (hasBillingAccess) {
return '/extra-usage to finish what you\u2019re working on.';
}
return '/extra-usage to request more usage from your admin.';
}
return '/upgrade or /extra-usage to finish what you\u2019re working on.';
}
type RateLimitMessageProps = {
text: string;
onOpenRateLimitOptions?: () => void;
};
export function RateLimitMessage(t0) {
const $ = _c(16);
const {
text,
onOpenRateLimitOptions
} = t0;
let t1;
if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
t1 = getSubscriptionType();
$[0] = t1;
} else {
t1 = $[0];
}
const subscriptionType = t1;
let t2;
if ($[1] === Symbol.for("react.memo_cache_sentinel")) {
t2 = getRateLimitTier();
$[1] = t2;
} else {
t2 = $[1];
}
const rateLimitTier = t2;
const isTeamOrEnterprise = subscriptionType === "team" || subscriptionType === "enterprise";
const isMax20x = rateLimitTier === "default_claude_max_20x";
let t3;
if ($[2] === Symbol.for("react.memo_cache_sentinel")) {
t3 = shouldProcessMockLimits() || isClaudeAISubscriber();
$[2] = t3;
} else {
t3 = $[2];
}
const shouldShowUpsell = t3;
const canSeeRateLimitOptionsUpsell = shouldShowUpsell && !isMax20x;
const [hasOpenedInteractiveMenu, setHasOpenedInteractiveMenu] = useState(false);
const claudeAiLimits = useClaudeAiLimits();
const isCurrentlyRateLimited = claudeAiLimits.status === "rejected" && claudeAiLimits.resetsAt !== undefined && !claudeAiLimits.isUsingOverage;
const shouldAutoOpenRateLimitOptionsMenu = canSeeRateLimitOptionsUpsell && !hasOpenedInteractiveMenu && isCurrentlyRateLimited && onOpenRateLimitOptions;
let t4;
let t5;
if ($[3] !== onOpenRateLimitOptions || $[4] !== shouldAutoOpenRateLimitOptionsMenu) {
t4 = () => {
if (shouldAutoOpenRateLimitOptionsMenu) {
setHasOpenedInteractiveMenu(true);
onOpenRateLimitOptions();
}
};
t5 = [shouldAutoOpenRateLimitOptionsMenu, onOpenRateLimitOptions];
$[3] = onOpenRateLimitOptions;
$[4] = shouldAutoOpenRateLimitOptionsMenu;
$[5] = t4;
$[6] = t5;
} else {
t4 = $[5];
t5 = $[6];
}
useEffect(t4, t5);
let t6;
bb0: {
let t7;
if ($[7] !== shouldAutoOpenRateLimitOptionsMenu) {
t7 = getUpsellMessage({
shouldShowUpsell,
isMax20x,
isExtraUsageCommandEnabled: extraUsage.isEnabled(),
shouldAutoOpenRateLimitOptionsMenu: !!shouldAutoOpenRateLimitOptionsMenu,
isTeamOrEnterprise,
hasBillingAccess: hasClaudeAiBillingAccess()
});
$[7] = shouldAutoOpenRateLimitOptionsMenu;
$[8] = t7;
} else {
t7 = $[8];
}
const message = t7;
if (!message) {
t6 = null;
break bb0;
}
let t8;
if ($[9] !== message) {
t8 = <Text dimColor={true}>{message}</Text>;
$[9] = message;
$[10] = t8;
} else {
t8 = $[10];
}
t6 = t8;
}
const upsell = t6;
let t7;
if ($[11] !== text) {
t7 = <Text color="error">{text}</Text>;
$[11] = text;
$[12] = t7;
} else {
t7 = $[12];
}
const t8 = hasOpenedInteractiveMenu ? null : upsell;
let t9;
if ($[13] !== t7 || $[14] !== t8) {
t9 = <MessageResponse><Box flexDirection="column">{t7}{t8}</Box></MessageResponse>;
$[13] = t7;
$[14] = t8;
$[15] = t9;
} else {
t9 = $[15];
}
return t9;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useEffect","useMemo","useState","extraUsage","Box","Text","useClaudeAiLimits","shouldProcessMockLimits","getRateLimitTier","getSubscriptionType","isClaudeAISubscriber","hasClaudeAiBillingAccess","MessageResponse","UpsellParams","shouldShowUpsell","isMax20x","isExtraUsageCommandEnabled","shouldAutoOpenRateLimitOptionsMenu","isTeamOrEnterprise","hasBillingAccess","getUpsellMessage","RateLimitMessageProps","text","onOpenRateLimitOptions","RateLimitMessage","t0","$","_c","t1","Symbol","for","subscriptionType","t2","rateLimitTier","t3","canSeeRateLimitOptionsUpsell","hasOpenedInteractiveMenu","setHasOpenedInteractiveMenu","claudeAiLimits","isCurrentlyRateLimited","status","resetsAt","undefined","isUsingOverage","t4","t5","t6","bb0","t7","isEnabled","message","t8","upsell","t9"],"sources":["RateLimitMessage.tsx"],"sourcesContent":["import React, { useEffect, useMemo, useState } from 'react'\nimport { extraUsage } from 'src/commands/extra-usage/index.js'\nimport { Box, Text } from 'src/ink.js'\nimport { useClaudeAiLimits } from 'src/services/claudeAiLimitsHook.js'\nimport { shouldProcessMockLimits } from 'src/services/rateLimitMocking.js' // Used for /mock-limits command\nimport {\n  getRateLimitTier,\n  getSubscriptionType,\n  isClaudeAISubscriber,\n} from 'src/utils/auth.js'\nimport { hasClaudeAiBillingAccess } from 'src/utils/billing.js'\nimport { MessageResponse } from '../MessageResponse.js'\n\ntype UpsellParams = {\n  shouldShowUpsell: boolean\n  isMax20x: boolean\n  isExtraUsageCommandEnabled: boolean\n  shouldAutoOpenRateLimitOptionsMenu: boolean\n  isTeamOrEnterprise: boolean\n  hasBillingAccess: boolean\n}\n\nexport function getUpsellMessage({\n  shouldShowUpsell,\n  isMax20x,\n  isExtraUsageCommandEnabled,\n  shouldAutoOpenRateLimitOptionsMenu,\n  isTeamOrEnterprise,\n  hasBillingAccess,\n}: UpsellParams): string | null {\n  if (!shouldShowUpsell) return null\n\n  if (isMax20x) {\n    if (isExtraUsageCommandEnabled) {\n      return '/extra-usage to finish what you\\u2019re working on.'\n    }\n    return '/login to switch to an API usage-billed account.'\n  }\n\n  if (shouldAutoOpenRateLimitOptionsMenu) {\n    return 'Opening your options\\u2026'\n  }\n\n  if (!isTeamOrEnterprise && !isExtraUsageCommandEnabled) {\n    return '/upgrade to increase your usage limit.'\n  }\n\n  if (isTeamOrEnterprise) {\n    if (!isExtraUsageCommandEnabled) return null\n\n    if (hasBillingAccess) {\n      return '/extra-usage to finish what you\\u2019re working on.'\n    }\n\n    return '/extra-usage to request more usage from your admin.'\n  }\n\n  return '/upgrade or /extra-usage to finish what you\\u2019re working on.'\n}\n\ntype RateLimitMessageProps = {\n  text: string\n  onOpenRateLimitOptions?: () => void\n}\n\nexport function RateLimitMessage({\n  text,\n  onOpenRateLimitOptions,\n}: RateLimitMessageProps): React.ReactNode {\n  const subscriptionType = getSubscriptionType()\n  const rateLimitTier = getRateLimitTier()\n  const isTeamOrEnterprise =\n    subscriptionType === 'team' || subscriptionType === 'enterprise'\n  const isMax20x = rateLimitTier === 'default_claude_max_20x'\n  // Always show upsell when using /mock-limits command, otherwise show for subscribers\n  const shouldShowUpsell = shouldProcessMockLimits() || isClaudeAISubscriber()\n\n  const canSeeRateLimitOptionsUpsell = shouldShowUpsell && !isMax20x\n\n  const [hasOpenedInteractiveMenu, setHasOpenedInteractiveMenu] =\n    useState(false)\n\n  // Check actual rate limit status - only auto-open if user is currently rate limited\n  // AND we've verified this with the API (resetsAt is only set after API response).\n  // This prevents false alerts when resuming sessions with old rate limit messages.\n  const claudeAiLimits = useClaudeAiLimits()\n  const isCurrentlyRateLimited =\n    claudeAiLimits.status === 'rejected' &&\n    claudeAiLimits.resetsAt !== undefined &&\n    !claudeAiLimits.isUsingOverage\n\n  const shouldAutoOpenRateLimitOptionsMenu =\n    canSeeRateLimitOptionsUpsell &&\n    !hasOpenedInteractiveMenu &&\n    isCurrentlyRateLimited &&\n    onOpenRateLimitOptions\n\n  useEffect(() => {\n    if (shouldAutoOpenRateLimitOptionsMenu) {\n      setHasOpenedInteractiveMenu(true)\n      onOpenRateLimitOptions()\n    }\n  }, [shouldAutoOpenRateLimitOptionsMenu, onOpenRateLimitOptions])\n\n  const upsell = useMemo(() => {\n    const message = getUpsellMessage({\n      shouldShowUpsell,\n      isMax20x,\n      isExtraUsageCommandEnabled: extraUsage.isEnabled(),\n      shouldAutoOpenRateLimitOptionsMenu: !!shouldAutoOpenRateLimitOptionsMenu,\n      isTeamOrEnterprise,\n      hasBillingAccess: hasClaudeAiBillingAccess(),\n    })\n    if (!message) return null\n    return <Text dimColor>{message}</Text>\n  }, [\n    shouldShowUpsell,\n    isMax20x,\n    isTeamOrEnterprise,\n    shouldAutoOpenRateLimitOptionsMenu,\n  ])\n\n  return (\n    <MessageResponse>\n      <Box flexDirection=\"column\">\n        <Text color=\"error\">{text}</Text>\n        {hasOpenedInteractiveMenu ? null : upsell}\n      </Box>\n    </MessageResponse>\n  )\n}\n"],"mappings":"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","ignoreList":[]}