Files
supabase/apps/studio/components/layouts/ProjectLayout/PausingState.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

130 lines
5.2 KiB
TypeScript

import { SupportCategories } from '@supabase/shared-types/out/constants'
import { LOCAL_STORAGE_KEYS, useParams } from 'common'
import { Circle, Loader } from 'lucide-react'
import { useEffect, useState } from 'react'
import { Badge, Button } from 'ui'
import { SupportLink } from '@/components/interfaces/Support/SupportLink'
import { useInvalidateProjectsInfiniteQuery } from '@/data/projects/org-projects-infinite-query'
import { Project, useInvalidateProjectDetailsQuery } from '@/data/projects/project-detail-query'
import { useProjectStatusQuery } from '@/data/projects/project-status-query'
import { useLongRunningTransitionState } from '@/hooks/misc/useLongRunningTransitionState'
import { PROJECT_STATUS } from '@/lib/constants'
import {
clearPersistedTransitionStartTime,
FALLBACK_LONG_RUNNING_STATE_THRESHOLD_MINUTES,
minutesToMilliseconds,
} from '@/lib/project-transition-state'
const LONG_RUNNING_STATE_THRESHOLD_MINUTES = FALLBACK_LONG_RUNNING_STATE_THRESHOLD_MINUTES
const LONG_RUNNING_STATE_THRESHOLD_MS = minutesToMilliseconds(LONG_RUNNING_STATE_THRESHOLD_MINUTES)
export interface PausingStateProps {
project: Project
}
export const PausingState = ({ project }: PausingStateProps) => {
const { ref } = useParams()
const [startPolling, setStartPolling] = useState(false)
const pauseStateStartStorageKey = ref ? LOCAL_STORAGE_KEYS.PROJECT_PAUSING_STARTED_AT(ref) : null
const isTakingLongerThanExpected = useLongRunningTransitionState({
storageKey: pauseStateStartStorageKey,
thresholdMs: LONG_RUNNING_STATE_THRESHOLD_MS,
})
const { invalidateProjectsQuery } = useInvalidateProjectsInfiniteQuery()
const { invalidateProjectDetailsQuery } = useInvalidateProjectDetailsQuery()
const { data: projectStatusData, isSuccess: isProjectStatusSuccess } = useProjectStatusQuery(
{ projectRef: ref },
{
enabled: startPolling,
refetchInterval: (query) => {
const data = query.state.data
return data?.status === PROJECT_STATUS.INACTIVE ||
data?.status === PROJECT_STATUS.PAUSE_FAILED
? false
: 2000
},
}
)
useEffect(() => {
if (!isProjectStatusSuccess) return
if (
projectStatusData?.status === PROJECT_STATUS.INACTIVE ||
projectStatusData?.status === PROJECT_STATUS.PAUSE_FAILED
) {
if (pauseStateStartStorageKey) {
clearPersistedTransitionStartTime(pauseStateStartStorageKey)
}
if (ref) invalidateProjectDetailsQuery(ref)
invalidateProjectsQuery()
}
}, [
isProjectStatusSuccess,
projectStatusData,
pauseStateStartStorageKey,
ref,
invalidateProjectDetailsQuery,
invalidateProjectsQuery,
])
useEffect(() => {
const timeoutId = setTimeout(() => setStartPolling(true), 4000)
return () => clearTimeout(timeoutId)
}, [])
return (
<div className="mx-auto my-16 w-full max-w-7xl space-y-16">
<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>
<Badge>
<div className="flex items-center gap-2">
<Loader className="animate-spin" size={12} />
<span>Pausing project</span>
</div>
</Badge>
</div>
</div>
<div className="mx-auto mt-8 mb-16 w-full max-w-7xl">
<div className="flex h-[500px] items-center justify-center rounded-sm border border-overlay bg-surface-100 p-8">
<div className="grid w-[380px] gap-4">
<div className="relative mx-auto max-w-[300px]">
<div className="absolute flex h-full w-full items-center justify-center">
<Loader className="animate-spin" size={20} strokeWidth={2} />
</div>
<Circle className="text-foreground-lighter" size={50} strokeWidth={1.5} />
</div>
<p className="text-center">Pausing {project.name}</p>
<p className="text-center text-sm text-foreground-light">
{isTakingLongerThanExpected
? `This is taking longer than usual. Contact support if your project is still pausing after ${LONG_RUNNING_STATE_THRESHOLD_MINUTES} minutes.`
: 'Your project is being paused now. This usually takes a few minutes. While paused, your data stays safe, and you can turn the project back on anytime.'}
</p>
{isTakingLongerThanExpected && (
<div className="flex justify-center">
<Button asChild type="default">
<SupportLink
queryParams={{
category: SupportCategories.DATABASE_UNRESPONSIVE,
projectRef: project.ref,
subject: 'Project stuck in pausing state',
message: `Project "${project.name}" has remained in a pausing state for over ${LONG_RUNNING_STATE_THRESHOLD_MINUTES} minutes.`,
}}
>
Contact support
</SupportLink>
</Button>
</div>
)}
</div>
</div>
</div>
</div>
</div>
)
}