Files
supabase/apps/studio/components/layouts/ProjectLayout/UpgradingState/index.tsx
Ivan Vasilov 56de26fe22 chore: Migrate the monorepo to use Tailwind v4 (#45318)
This PR migrates the whole monorepo to use Tailwind v4:
- Removed `@tailwindcss/container-queries` plugin since it's included by
default in v4,
- Bump all instances of Tailwind to v4. Made minimal changes to the
shared config to remove non-supported features (`alpha` mentions),
- Migrate all apps to be compatible with v4 configs,
- Fix the `typography.css` import in 3 apps,
- Add missing rules which were included by default in v3,
- Run `pnpm dlx @tailwindcss/upgrade` on all apps, which renames a lot
of classes
- Rename all misnamed classes according to
https://tailwindcss.com/docs/upgrade-guide#renamed-utilities in all
apps.

---------

Co-authored-by: Jordi Enric <jordi.err@gmail.com>
2026-04-30 10:53:24 +00:00

251 lines
11 KiB
TypeScript

import { SupportCategories } from '@supabase/shared-types/out/constants'
import { DatabaseUpgradeProgress, DatabaseUpgradeStatus } from '@supabase/shared-types/out/events'
import { useParams } from 'common'
import dayjs from 'dayjs'
import {
AlertCircle,
Check,
CheckCircle,
Circle,
Loader,
Maximize2,
Minimize2,
Settings,
} from 'lucide-react'
import { useSearchParams } from 'next/navigation'
import { useState } from 'react'
import { Button, Tooltip, TooltipContent, TooltipTrigger } from 'ui'
import { DATABASE_UPGRADE_MESSAGES } from './UpgradingState.constants'
import { SupportLink } from '@/components/interfaces/Support/SupportLink'
import { useProjectUpgradingStatusQuery } from '@/data/config/project-upgrade-status-query'
import { useInvalidateProjectDetailsQuery } from '@/data/projects/project-detail-query'
import { useSelectedProjectQuery } from '@/hooks/misc/useSelectedProject'
import { IS_PLATFORM } from '@/lib/constants'
export const UpgradingState = () => {
const { ref } = useParams()
const queryParams = useSearchParams()
const { data: project } = useSelectedProjectQuery()
const [loading, setLoading] = useState(false)
const [isExpanded, setIsExpanded] = useState(false)
const { invalidateProjectDetailsQuery } = useInvalidateProjectDetailsQuery()
const { data } = useProjectUpgradingStatusQuery(
{
projectRef: ref,
projectStatus: project?.status,
trackingId: queryParams?.get('trackingId'),
},
{
enabled: IS_PLATFORM,
}
)
const { initiated_at, status, progress, target_version, error } =
data?.databaseUpgradeStatus ?? {}
const progressStage = Number((progress || '').split('_')[0])
const isFailed = status === DatabaseUpgradeStatus.Failed
const isCompleted = status === DatabaseUpgradeStatus.Upgraded
const isPerformingFullPhysicalBackup =
status === DatabaseUpgradeStatus.Upgrading &&
progress === DatabaseUpgradeProgress.CompletedUpgrade
const initiatedAtUTC = dayjs.utc(initiated_at ?? 0).format('DD MMM YYYY HH:mm:ss')
const initiatedAt = dayjs
.utc(initiated_at ?? 0)
.local()
.format('DD MMM YYYY HH:mm:ss (ZZ)')
const refetchProjectDetails = async () => {
setLoading(true)
if (ref) await invalidateProjectDetailsQuery(ref)
}
const subject = 'Upgrade failed for project'
const message = `Upgrade information:\n• Initiated at: ${initiated_at}\n• Target Version: ${target_version}\n• Error: ${error}`
return (
<div className="w-full mx-auto my-16 space-y-16 max-w-7xl">
<div className="mx-6 space-y-16">
<div className="flex flex-col space-y-4 lg:flex-row lg:items-center lg:space-y-0 lg:space-x-6">
<h1 className="text-3xl">{project?.name}</h1>
</div>
<div className="w-full mx-auto mt-8 mb-16 max-w-7xl">
<div className="flex h-[500px] items-center justify-center rounded-sm border border-muted bg-surface-100 p-8">
{isCompleted ? (
<div className="grid gap-4">
<div className="relative mx-auto max-w-[300px]">
<CheckCircle className="text-brand" size={40} strokeWidth={1.5} />
</div>
<div className="space-y-2">
<p className="text-center">Upgrade completed!</p>
<p className="mt-4 text-center text-sm text-foreground-light w-[300px] mx-auto">
Your project has been successfully upgraded to Postgres {target_version} and is
now back online.
</p>
</div>
<div className="mx-auto">
<Button loading={loading} disabled={loading} onClick={refetchProjectDetails}>
Return to project
</Button>
</div>
</div>
) : isFailed ? (
<div className="grid gap-4">
<div className="relative mx-auto max-w-[300px]">
<AlertCircle className="text-amber-900" size={40} strokeWidth={1.5} />
</div>
<div className="space-y-2">
<p className="text-center">We ran into an issue while upgrading your project</p>
<p className="mt-4 text-center text-sm text-foreground-light w-full md:w-[450px] mx-auto">
Your project is back online and its data is not affected. Please reach out to us
via our support form for assistance with the upgrade.
</p>
</div>
<div className="flex items-center mx-auto space-x-2">
<Button asChild type="default">
<SupportLink
queryParams={{
category: SupportCategories.DATABASE_UNRESPONSIVE,
projectRef: ref,
subject,
message,
}}
>
Contact support
</SupportLink>
</Button>
<Button loading={loading} disabled={loading} onClick={refetchProjectDetails}>
Return to project
</Button>
</div>
</div>
) : (
<div className="grid w-[480px] gap-4">
<div className="relative mx-auto max-w-[300px]">
<div className="absolute flex items-center justify-center w-full h-full">
<Settings className="animate-spin" size={20} strokeWidth={2} />
</div>
<Circle className="text-foreground-lighter" size={50} strokeWidth={1.5} />
</div>
<div className="space-y-2">
{isPerformingFullPhysicalBackup ? (
<div>
<p className="text-center">Performing a full backup</p>
<p className="text-sm text-center text-foreground-light">
Upgrade is now complete, and your project is online. A full backup is now
being performed to ensure that there is a proper base backup available
post-upgrade. This can take from a few minutes up to several hours depending
on the size of your database.
</p>
</div>
) : (
<div>
<p className="text-center">Upgrading in progress</p>
<p className="text-sm text-center text-foreground-light">
Upgrades can take from a few minutes up to several hours depending on the
size of your database. Your project will be offline while it is being
upgraded.
</p>
</div>
)}
<div
className="mt-4! mb-2! py-3 px-4 transition-all overflow-hidden border rounded-sm relative"
style={{ maxHeight: isExpanded ? '500px' : '110px' }}
>
{isExpanded ? (
<Minimize2
size={14}
strokeWidth={2}
className="absolute z-10 cursor-pointer top-3 right-3"
onClick={() => setIsExpanded(false)}
/>
) : (
<Maximize2
size={14}
strokeWidth={2}
className="absolute z-10 cursor-pointer top-3 right-3"
onClick={() => setIsExpanded(true)}
/>
)}
<div
className="space-y-2 transition-all"
style={{
translate: isExpanded
? '0px 0px'
: `0px ${
(progressStage - 2 <= 0
? 0
: progressStage > 6
? 5
: progressStage - 2) * -28
}px`,
}}
>
{DATABASE_UPGRADE_MESSAGES.map((message, idx: number) => {
const isCurrent = message.key === progress
const isCompleted = progressStage > idx
return (
<div key={message.key} className="flex items-center space-x-4">
{isCurrent ? (
<div className="flex items-center justify-center w-5 h-5 rounded-full">
<Loader
size={20}
className="animate-spin text-foreground-light"
strokeWidth={2}
/>
</div>
) : isCompleted ? (
<div className="flex items-center justify-center w-5 h-5 border rounded-full bg-brand border-brand">
<Check size={12} className="text-white" strokeWidth={3} />
</div>
) : (
<div className="flex items-center justify-center w-5 h-5 border rounded-full bg-overlay-hover" />
)}
<p
className={`text-sm ${
isCurrent
? 'text-foreground'
: isCompleted
? 'text-foreground-light'
: 'text-foreground-lighter'
} hover:text-foreground transition`}
>
{isCurrent
? message.progress
: isCompleted
? message.completed
: message.initial}
</p>
</div>
)
})}
</div>
</div>
{initiated_at !== undefined && (
<Tooltip>
<TooltipTrigger>
<p className="text-sm text-center text-foreground-light">
Started on: {initiatedAtUTC} (UTC)
</p>
</TooltipTrigger>
<TooltipContent side="bottom">{initiatedAt}</TooltipContent>
</Tooltip>
)}
</div>
</div>
)}
</div>
</div>
</div>
</div>
)
}