mirror of
https://github.com/supabase/supabase.git
synced 2026-06-13 01:39:53 +08:00
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>
207 lines
6.9 KiB
TypeScript
207 lines
6.9 KiB
TypeScript
import { PermissionAction } from '@supabase/shared-types/out/constants'
|
|
import { useFlag } from 'common'
|
|
import { ChevronDown, RefreshCw } from 'lucide-react'
|
|
import { useRouter } from 'next/router'
|
|
import { useState } from 'react'
|
|
import { toast } from 'sonner'
|
|
import {
|
|
Button,
|
|
cn,
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuTrigger,
|
|
} from 'ui'
|
|
import ConfirmationModal from 'ui-patterns/Dialogs/ConfirmationModal'
|
|
|
|
import { ButtonTooltip } from '@/components/ui/ButtonTooltip'
|
|
import { useSetProjectStatus } from '@/data/projects/project-detail-query'
|
|
import { useProjectRestartMutation } from '@/data/projects/project-restart-mutation'
|
|
import { useProjectRestartServicesMutation } from '@/data/projects/project-restart-services-mutation'
|
|
import { useAsyncCheckPermissions } from '@/hooks/misc/useCheckPermissions'
|
|
import { useIsFeatureEnabled } from '@/hooks/misc/useIsFeatureEnabled'
|
|
import {
|
|
useIsAwsK8sCloudProvider,
|
|
useIsProjectActive,
|
|
useSelectedProjectQuery,
|
|
} from '@/hooks/misc/useSelectedProject'
|
|
import { PROJECT_STATUS } from '@/lib/constants'
|
|
import { type ResponseError } from '@/types'
|
|
|
|
const RestartServerButton = () => {
|
|
const router = useRouter()
|
|
const { data: project } = useSelectedProjectQuery()
|
|
const isProjectActive = useIsProjectActive()
|
|
const canRestart = isProjectActive || project?.status === PROJECT_STATUS.ACTIVE_UNHEALTHY
|
|
const isAwsK8s = useIsAwsK8sCloudProvider()
|
|
const { setProjectStatus } = useSetProjectStatus()
|
|
|
|
const [serviceToRestart, setServiceToRestart] = useState<'project' | 'database'>()
|
|
|
|
const { projectSettingsRestartProject } = useIsFeatureEnabled([
|
|
'project_settings:restart_project',
|
|
])
|
|
|
|
const projectRef = project?.ref ?? ''
|
|
const projectRegion = project?.region ?? ''
|
|
|
|
const projectRestartDisabled = useFlag('disableProjectRestarts')
|
|
const { can: canRestartProject } = useAsyncCheckPermissions(
|
|
PermissionAction.INFRA_EXECUTE,
|
|
'reboot'
|
|
)
|
|
|
|
const { mutate: restartProject, isPending: isRestartingProject } = useProjectRestartMutation({
|
|
onSuccess: () => {
|
|
onRestartSuccess()
|
|
},
|
|
onError: (error) => {
|
|
onRestartFailed(error, 'project')
|
|
},
|
|
})
|
|
const { mutate: restartProjectServices, isPending: isRestartingServices } =
|
|
useProjectRestartServicesMutation({
|
|
onSuccess: () => {
|
|
onRestartSuccess()
|
|
},
|
|
onError: (error) => {
|
|
onRestartFailed(error, 'database')
|
|
},
|
|
})
|
|
|
|
const isLoading = isRestartingProject || isRestartingServices
|
|
const hasRestartDropdown = canRestartProject && canRestart && !projectRestartDisabled
|
|
|
|
const requestProjectRestart = () => {
|
|
if (!canRestartProject) {
|
|
return toast.error('You do not have the required permissions to restart this project')
|
|
}
|
|
restartProject({ ref: projectRef })
|
|
}
|
|
|
|
const requestDatabaseRestart = async () => {
|
|
if (!canRestartProject) {
|
|
return toast.error('You do not have the required permissions to restart this project')
|
|
}
|
|
restartProjectServices({ ref: projectRef, region: projectRegion, services: ['postgresql'] })
|
|
}
|
|
|
|
const onRestartFailed = (error: ResponseError, type: string) => {
|
|
toast.error(`Unable to restart ${type}: ${error.message}`)
|
|
setServiceToRestart(undefined)
|
|
}
|
|
|
|
const onRestartSuccess = () => {
|
|
setProjectStatus({ ref: projectRef, status: PROJECT_STATUS.RESTARTING })
|
|
toast.success('Restarting server...')
|
|
router.push(`/project/${projectRef}`)
|
|
setServiceToRestart(undefined)
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{projectSettingsRestartProject ? (
|
|
<div className="flex w-full @lg:w-auto">
|
|
<ButtonTooltip
|
|
type="default"
|
|
className={cn(
|
|
'flex-1 px-3 hover:z-10 @lg:flex-none',
|
|
canRestartProject && canRestart ? 'rounded-r-none' : ''
|
|
)}
|
|
disabled={
|
|
project === undefined ||
|
|
!canRestartProject ||
|
|
!canRestart ||
|
|
projectRestartDisabled ||
|
|
isAwsK8s
|
|
}
|
|
onClick={() => setServiceToRestart('project')}
|
|
tooltip={{
|
|
content: {
|
|
side: 'bottom',
|
|
text: projectRestartDisabled
|
|
? 'Project restart is currently disabled'
|
|
: !canRestartProject
|
|
? 'You need additional permissions to restart this project'
|
|
: !canRestart
|
|
? 'Unable to restart project as project is not active'
|
|
: isAwsK8s
|
|
? 'Project restart is not supported for AWS (Revamped) projects'
|
|
: undefined,
|
|
},
|
|
}}
|
|
>
|
|
Restart project
|
|
</ButtonTooltip>
|
|
{hasRestartDropdown && (
|
|
<DropdownMenu>
|
|
<DropdownMenuTrigger asChild>
|
|
<Button
|
|
type="default"
|
|
className="shrink-0 rounded-l-none px-[4px] py-[5px] -ml-px"
|
|
icon={<ChevronDown />}
|
|
disabled={!canRestartProject}
|
|
/>
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent align="end" side="bottom">
|
|
<DropdownMenuItem
|
|
key="database"
|
|
disabled={isLoading}
|
|
onClick={() => {
|
|
setServiceToRestart('database')
|
|
}}
|
|
>
|
|
<div className="space-y-0.5">
|
|
<p className="block text-foreground">Fast database reboot</p>
|
|
<p className="block text-foreground-light">
|
|
Restarts only the database. Faster, but may not be able to recover from all
|
|
failure modes.
|
|
</p>
|
|
</div>
|
|
</DropdownMenuItem>
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
)}
|
|
</div>
|
|
) : (
|
|
<Button
|
|
type="default"
|
|
icon={<RefreshCw />}
|
|
className="w-full @lg:w-auto"
|
|
disabled={isLoading}
|
|
onClick={() => {
|
|
setServiceToRestart('database')
|
|
}}
|
|
>
|
|
Restart database
|
|
</Button>
|
|
)}
|
|
|
|
<ConfirmationModal
|
|
visible={serviceToRestart !== undefined}
|
|
variant="destructive"
|
|
title={`Restart ${serviceToRestart}`}
|
|
description={
|
|
<>
|
|
Are you sure you want to restart your {serviceToRestart}? There will be a few minutes of
|
|
downtime.
|
|
</>
|
|
}
|
|
confirmLabel="Restart"
|
|
confirmLabelLoading="Restarting"
|
|
loading={isLoading}
|
|
onCancel={() => setServiceToRestart(undefined)}
|
|
onConfirm={async () => {
|
|
if (serviceToRestart === 'project') {
|
|
await requestProjectRestart()
|
|
} else if (serviceToRestart === 'database') {
|
|
await requestDatabaseRestart()
|
|
}
|
|
}}
|
|
/>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default RestartServerButton
|