mirror of
https://github.com/supabase/supabase.git
synced 2026-05-12 04:16:08 +08:00
## Problem We used to have a `_Shadcn_` suffix for all the shadcn form components because we also had `formik` form components. This is not needed anymore. ## Solution - Remove the suffix - Update all usages
283 lines
9.9 KiB
TypeScript
283 lines
9.9 KiB
TypeScript
import { zodResolver } from '@hookform/resolvers/zod'
|
|
import { useForm } from 'react-hook-form'
|
|
import { toast } from 'sonner'
|
|
import {
|
|
Badge,
|
|
Button,
|
|
Dialog,
|
|
DialogContent,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogSection,
|
|
DialogSectionSeparator,
|
|
DialogTitle,
|
|
Form,
|
|
FormControl,
|
|
FormField,
|
|
Input_Shadcn_,
|
|
Select_Shadcn_,
|
|
SelectContent_Shadcn_,
|
|
SelectItem_Shadcn_,
|
|
SelectSeparator_Shadcn_,
|
|
SelectTrigger_Shadcn_,
|
|
SelectValue_Shadcn_,
|
|
} from 'ui'
|
|
import { Admonition } from 'ui-patterns'
|
|
import { FormItemLayout } from 'ui-patterns/form/FormItemLayout/FormItemLayout'
|
|
import { ShimmeringLoader } from 'ui-patterns/ShimmeringLoader'
|
|
import * as z from 'zod'
|
|
|
|
import { extensionsWithRecommendedSchemas } from './Extensions.constants'
|
|
import { DocsButton } from '@/components/ui/DocsButton'
|
|
import { useDatabaseExtensionEnableMutation } from '@/data/database-extensions/database-extension-enable-mutation'
|
|
import { type DatabaseExtension } from '@/data/database-extensions/database-extensions-query'
|
|
import { useSchemasQuery } from '@/data/database/schemas-query'
|
|
import { useIsOrioleDb, useSelectedProjectQuery } from '@/hooks/misc/useSelectedProject'
|
|
import { useProtectedSchemas } from '@/hooks/useProtectedSchemas'
|
|
import { DOCS_URL } from '@/lib/constants'
|
|
|
|
const orioleExtCallOuts = ['vector', 'postgis']
|
|
|
|
const FormSchema = z.object({ name: z.string(), schema: z.string() }).superRefine((val, ctx) => {
|
|
if (val.schema === 'custom' && val.name.length === 0) {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.custom,
|
|
path: ['name'],
|
|
message: 'Please provide a name for the schema',
|
|
})
|
|
}
|
|
})
|
|
|
|
interface EnableExtensionModalProps {
|
|
visible: boolean
|
|
extension: DatabaseExtension
|
|
onCancel: () => void
|
|
}
|
|
|
|
export const EnableExtensionModal = ({
|
|
visible,
|
|
extension,
|
|
onCancel,
|
|
}: EnableExtensionModalProps) => {
|
|
const isOrioleDb = useIsOrioleDb()
|
|
const { data: project } = useSelectedProjectQuery()
|
|
const { data: protectedSchemas } = useProtectedSchemas({ excludeSchemas: ['extensions'] })
|
|
|
|
const recommendedSchema = extensionsWithRecommendedSchemas[extension.name]
|
|
|
|
const { data: schemas = [], isPending: isLoading } = useSchemasQuery(
|
|
{
|
|
projectRef: project?.ref,
|
|
connectionString: project?.connectionString,
|
|
},
|
|
{ enabled: visible }
|
|
)
|
|
const availableSchemas = schemas.filter(
|
|
(schema) =>
|
|
schema.name === recommendedSchema ||
|
|
!protectedSchemas.some((protectedSchema) => protectedSchema.name === schema.name)
|
|
)
|
|
|
|
// [Joshen] Hard-coding pg_cron here as this is enforced on our end (Not via pg_available_extension_versions)
|
|
const defaultSchema =
|
|
extension.name === 'pg_cron' ? 'pg_catalog' : extension.default_version_schema
|
|
|
|
const { mutate: enableExtension, isPending: isEnabling } = useDatabaseExtensionEnableMutation({
|
|
onSuccess: () => {
|
|
toast.success(`Extension "${extension.name}" is now enabled`)
|
|
onCancel()
|
|
},
|
|
onError: (error) => {
|
|
toast.error(`Failed to enable ${extension.name}: ${error.message}`)
|
|
},
|
|
})
|
|
|
|
const defaultValues = { name: extension.name, schema: recommendedSchema ?? 'extensions' }
|
|
const form = useForm<z.infer<typeof FormSchema>>({
|
|
mode: 'onBlur',
|
|
reValidateMode: 'onBlur',
|
|
resolver: zodResolver(FormSchema),
|
|
defaultValues,
|
|
})
|
|
const { schema } = form.watch()
|
|
|
|
const onSubmit = async (values: z.infer<typeof FormSchema>) => {
|
|
if (project === undefined) return console.error('Project is required')
|
|
|
|
const schema =
|
|
defaultSchema !== undefined && defaultSchema !== null
|
|
? defaultSchema
|
|
: values.schema === 'custom'
|
|
? values.name
|
|
: values.schema
|
|
|
|
enableExtension({
|
|
projectRef: project.ref,
|
|
connectionString: project?.connectionString,
|
|
schema,
|
|
name: extension.name,
|
|
version: extension.default_version,
|
|
cascade: true,
|
|
createSchema: !schema.startsWith('pg_'),
|
|
})
|
|
}
|
|
|
|
return (
|
|
<Dialog
|
|
open={visible}
|
|
onOpenChange={(open: boolean) => {
|
|
if (!open) onCancel()
|
|
}}
|
|
>
|
|
<DialogContent size="small" aria-describedby={undefined}>
|
|
<DialogHeader>
|
|
<DialogTitle>Enable {extension.name}</DialogTitle>
|
|
</DialogHeader>
|
|
|
|
<DialogSectionSeparator />
|
|
|
|
{isOrioleDb && orioleExtCallOuts.includes(extension.name) && (
|
|
<Admonition
|
|
type="default"
|
|
title="Extension is limited by OrioleDB"
|
|
className="border-x-0 border-t-0 rounded-none"
|
|
>
|
|
<span className="block">
|
|
{extension.name} cannot be accelerated by indexes on tables that are using the
|
|
OrioleDB access method
|
|
</span>
|
|
<DocsButton abbrev={false} className="mt-2" href={`${DOCS_URL}`} />
|
|
</Admonition>
|
|
)}
|
|
|
|
{extension.name === 'pg_cron' && project?.cloud_provider === 'FLY' && (
|
|
<Admonition
|
|
type="warning"
|
|
title="The pg_cron extension is not fully supported for Fly projects"
|
|
className="border-x-0 border-t-0 rounded-none"
|
|
>
|
|
<p>
|
|
You can still enable the extension, but pg_cron jobs may not run due to the behavior
|
|
of Fly projects.
|
|
</p>
|
|
<DocsButton
|
|
className="mt-2"
|
|
href={`${DOCS_URL}/guides/platform/fly-postgres#limitations`}
|
|
/>
|
|
</Admonition>
|
|
)}
|
|
|
|
<DialogSection>
|
|
<Form {...form}>
|
|
<form id="enable-extensions-form" onSubmit={form.handleSubmit(onSubmit)}>
|
|
{isLoading ? (
|
|
<div className="space-y-2">
|
|
<ShimmeringLoader />
|
|
<div className="w-3/4">
|
|
<ShimmeringLoader />
|
|
</div>
|
|
</div>
|
|
) : !!defaultSchema ? (
|
|
<div className="flex flex-col gap-y-2">
|
|
<FormItemLayout
|
|
isReactForm={false}
|
|
label="Select a schema to enable the extension for"
|
|
>
|
|
<Input_Shadcn_ disabled value={defaultSchema} />
|
|
</FormItemLayout>
|
|
<p className="text-sm text-foreground-light">
|
|
Extension must be installed in the "{defaultSchema}" schema.
|
|
</p>
|
|
</div>
|
|
) : (
|
|
<div className="flex flex-col gap-y-2">
|
|
<FormField
|
|
key="schema"
|
|
name="schema"
|
|
control={form.control}
|
|
render={({ field }) => (
|
|
<FormItemLayout
|
|
name="schema"
|
|
label="Select a schema to enable the extension for"
|
|
>
|
|
<FormControl>
|
|
<Select_Shadcn_
|
|
value={field.value}
|
|
onValueChange={field.onChange}
|
|
disabled={!!defaultSchema}
|
|
>
|
|
<SelectTrigger_Shadcn_>
|
|
<SelectValue_Shadcn_ placeholder="Select a schema" />
|
|
</SelectTrigger_Shadcn_>
|
|
<SelectContent_Shadcn_>
|
|
<SelectItem_Shadcn_ value="custom">
|
|
Create a new schema{' '}
|
|
<code className="text-code-inline">{extension.name}</code>
|
|
</SelectItem_Shadcn_>
|
|
<SelectSeparator_Shadcn_ />
|
|
{availableSchemas.map((schema) => {
|
|
return (
|
|
<SelectItem_Shadcn_ key={schema.id} value={schema.name}>
|
|
{schema.name}
|
|
{schema.name === recommendedSchema ? (
|
|
<Badge className="ml-2" variant="success">
|
|
Recommended
|
|
</Badge>
|
|
) : !defaultSchema && schema.name === 'extensions' ? (
|
|
<Badge className="ml-2">Default</Badge>
|
|
) : null}
|
|
</SelectItem_Shadcn_>
|
|
)
|
|
})}
|
|
</SelectContent_Shadcn_>
|
|
</Select_Shadcn_>
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
|
|
{!!recommendedSchema && (
|
|
<p className="text-sm text-foreground-light">
|
|
Use the "{recommendedSchema}" schema for full compatibility with related
|
|
features.
|
|
</p>
|
|
)}
|
|
|
|
{schema === 'custom' && (
|
|
<FormField
|
|
key="name"
|
|
name="name"
|
|
control={form.control}
|
|
render={({ field }) => (
|
|
<FormItemLayout name="name" label="Schema name">
|
|
<FormControl>
|
|
<Input_Shadcn_ {...field} />
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
)}
|
|
</div>
|
|
)}
|
|
</form>
|
|
</Form>
|
|
</DialogSection>
|
|
|
|
<DialogFooter>
|
|
<Button type="default" disabled={isEnabling} onClick={() => onCancel()}>
|
|
Cancel
|
|
</Button>
|
|
<Button
|
|
htmlType="submit"
|
|
form="enable-extensions-form"
|
|
loading={isEnabling}
|
|
disabled={isLoading || isEnabling}
|
|
>
|
|
Enable extension
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
)
|
|
}
|