TotalUsage() — supabase Function Reference
Architecture documentation for the TotalUsage() function in TotalUsage.tsx from the supabase codebase.
Entity Profile
Relationship Graph
Source Code
apps/studio/components/interfaces/Organization/Usage/TotalUsage.tsx lines 42–251
export const TotalUsage = ({
orgSlug,
projectRef,
subscription,
startDate,
endDate,
currentBillingCycleSelected,
}: ComputeProps) => {
const isMobile = useBreakpoint('md')
const isUsageBillingEnabled = subscription?.usage_billing_enabled
const { billingAll } = useIsFeatureEnabled(['billing:all'])
const {
data: usage,
error: usageError,
isPending: isLoadingUsage,
isError: isErrorUsage,
isSuccess: isSuccessUsage,
} = useOrgUsageQuery({
orgSlug,
projectRef,
start: !currentBillingCycleSelected && startDate ? new Date(startDate) : undefined,
end: !currentBillingCycleSelected && endDate ? new Date(endDate) : undefined,
})
// When the user filters by project ref or selects a custom timeframe, we only display usage+project breakdown, but no costs/limits
const showRelationToSubscription = currentBillingCycleSelected && !projectRef
const isOnHigherPlan = ['team', 'enterprise', 'platform'].includes(subscription?.plan.id ?? '')
const hasExceededAnyLimits =
showRelationToSubscription &&
Boolean(
usage?.usages.find(
(usageItem) =>
// Filter out compute as compute has no quota and is always being charged for
!usageItem.metric.startsWith('COMPUTE_') &&
!usageItem.unlimited &&
usageItem.usage > (usageItem?.pricing_free_units ?? 0)
)
)
const sortedBillingMetrics = useMemo(() => {
if (!usage) return []
const breakdownMetrics = BILLING_BREAKDOWN_METRICS.filter((metric) =>
usage.usages.some((usage) => usage.metric === metric.key)
).filter((metric) => {
if (!METRICS_TO_HIDE_WITH_NO_USAGE.includes(metric.key as PricingMetric)) return true
const metricUsage = usage.usages.find((it) => it.metric === metric.key)
return metricUsage && metricUsage.usage > 0
})
return breakdownMetrics.slice().sort((a, b) => {
const usageMetaA = usage.usages.find((x) => x.metric === a.key)
const usageRatioA =
typeof usageMetaA !== 'number'
? (usageMetaA?.usage ?? 0) / (usageMetaA?.pricing_free_units ?? 0)
: 0
const usageMetaB = usage.usages.find((x) => x.metric === b.key)
const usageRatioB =
typeof usageMetaB !== 'number'
? (usageMetaB?.usage ?? 0) / (usageMetaB?.pricing_free_units ?? 0)
: 0
return (
// Sort unavailable features to bottom
Number(usageMetaB?.available_in_plan) - Number(usageMetaA?.available_in_plan) ||
// Sort high-usage features to top
usageRatioB - usageRatioA
)
})
}, [usage])
const computeMetrics = (usage?.usages || [])
.filter((it) => it.metric.startsWith('COMPUTE'))
.map((it) => it.metric) as ComputeUsageMetric[]
return (
<div id="summary">
<SectionContent
section={{
name: 'Usage Summary',
description: isUsageBillingEnabled
? `Your plan includes a limited amount of usage. If exceeded, you will be charged for the overages. It may take up to 1 hour to refresh.`
: `Your plan includes a limited amount of usage. If exceeded, you may experience restrictions, as you are currently not billed for overages. It may take up to 1 hour to refresh.`,
links: billingAll
? [
{
name: 'How billing works',
url: `${DOCS_URL}/guides/platform/billing-on-supabase`,
},
{
name: 'Supabase Plans',
url: 'https://supabase.com/pricing',
},
]
: [],
}}
>
{isLoadingUsage && (
<div className="space-y-2">
<ShimmeringLoader />
<ShimmeringLoader className="w-3/4" />
<ShimmeringLoader className="w-1/2" />
</div>
)}
{isErrorUsage && <AlertError subject="Failed to retrieve usage data" error={usageError} />}
{isSuccessUsage && subscription && (
<div>
{showRelationToSubscription && !isOnHigherPlan && (
<p className="text-sm">
{!hasExceededAnyLimits ? (
<span>
You have not exceeded your{' '}
<span className="font-medium">{subscription?.plan.name}</span> Plan quota in
this billing cycle.
</span>
) : hasExceededAnyLimits && subscription?.plan?.id === 'free' ? (
<span>
You have exceeded your{' '}
<span className="font-medium">{subscription?.plan.name}</span> Plan quota in
this billing cycle. Upgrade your plan to continue using Supabase without
restrictions.
</span>
) : hasExceededAnyLimits &&
subscription?.usage_billing_enabled === false &&
subscription?.plan?.id === 'pro' ? (
<span>
You have exceeded your{' '}
<span className="font-medium">{subscription?.plan.name}</span> Plan quota in
this billing cycle. Disable your spend cap to continue using Supabase without
restrictions.
</span>
) : hasExceededAnyLimits && subscription?.usage_billing_enabled === true ? (
<span>
You have exceeded your{' '}
<span className="font-medium">{subscription?.plan.name}</span> Plan quota in
this billing cycle and will be charged for over-usage.
</span>
) : (
<span>
You have not exceeded your{' '}
<span className="font-medium">{subscription?.plan.name}</span> Plan quota in
this billing cycle.
</span>
)}
</p>
)}
<div className="grid grid-cols-2 mt-3 gap-[1px] bg-border">
{sortedBillingMetrics.map((metric, i) => {
return (
<div
key={metric.key}
className={cn('col-span-2 md:col-span-1 bg-sidebar space-y-4 py-4')}
>
<BillingMetric
idx={i}
slug={orgSlug}
metric={metric}
usage={usage}
subscription={subscription!}
relativeToSubscription={showRelationToSubscription}
className={cn(i % 2 === 0 ? 'md:pr-4' : 'md:pl-4')}
/>
</div>
)
})}
{computeMetrics.map((metric, i) => {
return (
<div
key={metric}
className={cn('col-span-2 md:col-span-1 bg-sidebar space-y-4 py-4')}
>
<ComputeMetric
slug={orgSlug}
metric={{
key: metric,
name: computeUsageMetricLabel(metric) + ' Compute Hours' || metric,
units: 'hours',
anchor: 'compute',
category: 'Compute',
unitName: 'GB',
}}
relativeToSubscription={showRelationToSubscription}
usage={usage}
className={cn(
(i + sortedBillingMetrics.length) % 2 === 0 ? 'md:pr-4' : 'md:pl-4'
)}
/>
</div>
)
})}
{!isMobile && (sortedBillingMetrics.length + computeMetrics.length) % 2 === 1 && (
<div className="col-span-2 md:col-span-1 bg-sidebar" />
)}
</div>
</div>
)}
</SectionContent>
</div>
)
}
Domain
Subdomains
Source
Analyze Your Own Codebase
Get architecture documentation, dependency graphs, and domain analysis for your codebase in minutes.
Try Supermodel Free