mirror of
https://github.com/supabase/supabase.git
synced 2026-05-09 02:09:50 -04:00
56de26fe22
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>
442 lines
16 KiB
TypeScript
442 lines
16 KiB
TypeScript
import { zodResolver } from '@hookform/resolvers/zod'
|
|
import { useParams } from 'common'
|
|
import { useEffect, useRef, useState } from 'react'
|
|
import { useForm, type SubmitHandler } from 'react-hook-form'
|
|
import { toast } from 'sonner'
|
|
import {
|
|
Button,
|
|
Dialog,
|
|
DialogContent,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogSection,
|
|
DialogSectionSeparator,
|
|
DialogTitle,
|
|
Form,
|
|
FormControl,
|
|
FormField,
|
|
FormMessage,
|
|
Input_Shadcn_,
|
|
Select_Shadcn_,
|
|
SelectContent_Shadcn_,
|
|
SelectItem_Shadcn_,
|
|
SelectTrigger_Shadcn_,
|
|
SelectValue_Shadcn_,
|
|
Switch,
|
|
} from 'ui'
|
|
import { Admonition } from 'ui-patterns'
|
|
import { FormItemLayout } from 'ui-patterns/form/FormItemLayout/FormItemLayout'
|
|
import { z } from 'zod'
|
|
|
|
import { StorageSizeUnits } from '@/components/interfaces/Storage/StorageSettings/StorageSettings.constants'
|
|
import {
|
|
convertFromBytes,
|
|
convertToBytes,
|
|
} from '@/components/interfaces/Storage/StorageSettings/StorageSettings.utils'
|
|
import { InlineLink } from '@/components/ui/InlineLink'
|
|
import { useProjectStorageConfigQuery } from '@/data/config/project-storage-config-query'
|
|
import { useBucketUpdateMutation } from '@/data/storage/bucket-update-mutation'
|
|
import { Bucket } from '@/data/storage/buckets-query'
|
|
import { DOCS_URL, IS_PLATFORM } from '@/lib/constants'
|
|
|
|
export interface EditBucketModalProps {
|
|
visible: boolean
|
|
bucket: Bucket
|
|
onClose: () => void
|
|
}
|
|
|
|
const BucketSchema = z.object({
|
|
name: z.string(),
|
|
public: z.boolean().default(false),
|
|
has_file_size_limit: z.boolean().default(false),
|
|
formatted_size_limit: z.coerce
|
|
.number()
|
|
.min(0, 'File size upload limit has to be at least 0')
|
|
.optional(),
|
|
allowed_mime_types: z.string().trim().default(''),
|
|
})
|
|
|
|
const formId = 'edit-storage-bucket-form'
|
|
|
|
export const EditBucketModal = ({ visible, bucket, onClose }: EditBucketModalProps) => {
|
|
const { ref } = useParams()
|
|
|
|
const { data } = useProjectStorageConfigQuery({ projectRef: ref }, { enabled: IS_PLATFORM })
|
|
const { value, unit } = convertFromBytes(data?.fileSizeLimit ?? 0)
|
|
const formattedGlobalUploadLimit = `${value} ${unit}`
|
|
|
|
const bucketIdRef = useRef<string | null>(null)
|
|
const [selectedUnit, setSelectedUnit] = useState<string>(StorageSizeUnits.MB)
|
|
const { value: fileSizeLimit } = convertFromBytes(bucket?.file_size_limit ?? 0)
|
|
|
|
const { mutate: updateBucket, isPending: isUpdating } = useBucketUpdateMutation({
|
|
onSuccess: () => {
|
|
toast.success(`Successfully updated bucket "${bucket?.name}"`)
|
|
onClose()
|
|
},
|
|
onError: (error) => {
|
|
// Handle specific error cases for inline display
|
|
const errorMessage = error.message?.toLowerCase() || ''
|
|
|
|
if (
|
|
errorMessage.includes('exceeded the maximum allowed size') ||
|
|
errorMessage.includes('maximum allowed size') ||
|
|
errorMessage.includes('entity too large') ||
|
|
errorMessage.includes('payload too large')
|
|
) {
|
|
// Set form error for the file size limit field
|
|
form.setError('formatted_size_limit', {
|
|
type: 'manual',
|
|
message: `Exceeds global limit of ${formattedGlobalUploadLimit}.`,
|
|
})
|
|
} else if (
|
|
errorMessage.includes('mime type') &&
|
|
(errorMessage.includes('is not supported') || errorMessage.includes('not supported'))
|
|
) {
|
|
// Set form error for the MIME types field
|
|
form.setError('allowed_mime_types', {
|
|
type: 'manual',
|
|
message: 'Invalid MIME type format. Please check your input.',
|
|
})
|
|
} else {
|
|
// For other errors, show a toast as fallback
|
|
toast.error(`Failed to update bucket: ${error.message || 'Unknown error'}`)
|
|
}
|
|
},
|
|
})
|
|
|
|
const defaultValues = {
|
|
name: bucket?.name ?? '',
|
|
public: bucket?.public,
|
|
has_file_size_limit: Boolean(bucket?.file_size_limit),
|
|
formatted_size_limit: bucket?.file_size_limit ? (fileSizeLimit ?? 0) : undefined,
|
|
allowed_mime_types: (bucket?.allowed_mime_types ?? []).join(', '),
|
|
}
|
|
|
|
const form = useForm<z.infer<typeof BucketSchema>>({
|
|
resolver: zodResolver(BucketSchema),
|
|
defaultValues,
|
|
values: defaultValues,
|
|
mode: 'onSubmit',
|
|
})
|
|
const { formatted_size_limit: formattedSizeLimitError } = form.formState.errors
|
|
|
|
const isPublicBucket = form.watch('public')
|
|
const hasFileSizeLimit = form.watch('has_file_size_limit')
|
|
const [hasAllowedMimeTypes, setHasAllowedMimeTypes] = useState(
|
|
Boolean(bucket?.allowed_mime_types?.length)
|
|
)
|
|
|
|
const isChangingBucketVisibility = bucket?.public !== isPublicBucket
|
|
const isMakingBucketPrivate = bucket?.public && !isPublicBucket
|
|
const isMakingBucketPublic = !bucket?.public && isPublicBucket
|
|
|
|
const closeModal = () => {
|
|
form.reset()
|
|
onClose()
|
|
}
|
|
|
|
const onSubmit: SubmitHandler<z.infer<typeof BucketSchema>> = async (values) => {
|
|
if (bucket === undefined) return console.error('Bucket is required')
|
|
if (ref === undefined) return console.error('Project ref is required')
|
|
|
|
// Client-side validation: Check if bucket limit exceeds global limit
|
|
// [Joshen] Should shift this into superRefine in the form schema
|
|
if (
|
|
values.has_file_size_limit &&
|
|
values.formatted_size_limit !== undefined &&
|
|
data?.fileSizeLimit
|
|
) {
|
|
const bucketLimitInBytes = convertToBytes(
|
|
values.formatted_size_limit,
|
|
selectedUnit as StorageSizeUnits
|
|
)
|
|
|
|
if (bucketLimitInBytes > data.fileSizeLimit) {
|
|
return form.setError('formatted_size_limit', {
|
|
type: 'manual',
|
|
message: 'exceed_global_limit',
|
|
})
|
|
}
|
|
}
|
|
|
|
updateBucket({
|
|
projectRef: ref,
|
|
id: bucket.id,
|
|
isPublic: values.public,
|
|
file_size_limit:
|
|
values.has_file_size_limit && values.formatted_size_limit
|
|
? convertToBytes(values.formatted_size_limit, selectedUnit as StorageSizeUnits)
|
|
: null,
|
|
allowed_mime_types: hasAllowedMimeTypes
|
|
? values.allowed_mime_types.length > 0
|
|
? values.allowed_mime_types.split(',').map((x: string) => x.trim())
|
|
: null
|
|
: null,
|
|
})
|
|
}
|
|
|
|
useEffect(() => {
|
|
if (visible && bucket) {
|
|
// Only set the selectedUnit when the bucket changes (different bucket ID)
|
|
// This preserves the user's unit selection when reopening the modal for the same bucket
|
|
if (bucketIdRef.current !== bucket.id && bucket.file_size_limit) {
|
|
const { unit } = convertFromBytes(bucket.file_size_limit)
|
|
setSelectedUnit(unit)
|
|
bucketIdRef.current = bucket.id
|
|
}
|
|
}
|
|
}, [visible, bucket, form])
|
|
|
|
return (
|
|
<Dialog
|
|
open={visible}
|
|
onOpenChange={(open) => {
|
|
if (!open) closeModal()
|
|
}}
|
|
>
|
|
<DialogContent>
|
|
<DialogHeader>
|
|
<DialogTitle>{`Edit bucket “${bucket?.name}”`}</DialogTitle>
|
|
</DialogHeader>
|
|
|
|
<DialogSectionSeparator />
|
|
|
|
<Form {...form}>
|
|
<form id={formId} onSubmit={form.handleSubmit(onSubmit)}>
|
|
<DialogSection className="space-y-6">
|
|
<FormField
|
|
key="name"
|
|
name="name"
|
|
control={form.control}
|
|
render={({ field }) => (
|
|
<FormItemLayout
|
|
hideMessage
|
|
name="name"
|
|
label="Bucket name"
|
|
labelOptional="Cannot be changed after creation"
|
|
>
|
|
<FormControl>
|
|
<Input_Shadcn_ id="name" {...field} disabled />
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
|
|
<div className="flex flex-col gap-y-3">
|
|
<FormField
|
|
key="public"
|
|
name="public"
|
|
control={form.control}
|
|
render={({ field }) => (
|
|
<FormItemLayout
|
|
hideMessage
|
|
name="public"
|
|
label="Public bucket"
|
|
description="Allow anyone to read objects without authorization"
|
|
layout="flex"
|
|
>
|
|
<FormControl>
|
|
<Switch
|
|
id="public"
|
|
size="large"
|
|
checked={field.value}
|
|
onCheckedChange={field.onChange}
|
|
/>
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
|
|
{isChangingBucketVisibility && (
|
|
<Admonition
|
|
type="warning"
|
|
title={`Warning: Making bucket ${isMakingBucketPublic ? 'public' : 'private'}`}
|
|
description={
|
|
<>
|
|
{isMakingBucketPublic && (
|
|
<p>This will make all objects in your bucket publicly accessible.</p>
|
|
)}
|
|
|
|
{isMakingBucketPrivate && (
|
|
<>
|
|
<p className="mb-2 leading-normal!">
|
|
All objects in your bucket will only accessible via signed URLs, or
|
|
downloaded with the right authorization headers.
|
|
</p>
|
|
<p className="leading-normal!">
|
|
Assets cached in the CDN may still be publicly accessible. You can
|
|
consider{' '}
|
|
<InlineLink
|
|
href={`${DOCS_URL}/guides/storage/cdn/smart-cdn#cache-eviction`}
|
|
>
|
|
purging the cache
|
|
</InlineLink>{' '}
|
|
or moving your assets to a new bucket.
|
|
</p>
|
|
</>
|
|
)}
|
|
</>
|
|
}
|
|
/>
|
|
)}
|
|
</div>
|
|
</DialogSection>
|
|
|
|
<DialogSectionSeparator />
|
|
|
|
<DialogSection className="space-y-2">
|
|
<FormField
|
|
key="has_file_size_limit"
|
|
name="has_file_size_limit"
|
|
control={form.control}
|
|
render={({ field }) => (
|
|
<FormItemLayout
|
|
name="has_file_size_limit"
|
|
label="Restrict file size"
|
|
description="Prevent uploading of files larger than a specified limit"
|
|
layout="flex"
|
|
>
|
|
<FormControl>
|
|
<Switch
|
|
id="has_file_size_limit"
|
|
size="large"
|
|
checked={field.value}
|
|
onCheckedChange={field.onChange}
|
|
/>
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
{hasFileSizeLimit && (
|
|
<div>
|
|
<FormField
|
|
key="formatted_size_limit"
|
|
name="formatted_size_limit"
|
|
control={form.control}
|
|
render={({ field }) => (
|
|
<FormItemLayout
|
|
hideMessage
|
|
name="formatted_size_limit"
|
|
label="File size limit"
|
|
>
|
|
<div className="grid grid-cols-12 gap-x-2">
|
|
<div className="col-span-8">
|
|
<FormControl>
|
|
<Input_Shadcn_
|
|
id="formatted_size_limit"
|
|
aria-label="File size limit"
|
|
type="number"
|
|
min={0}
|
|
placeholder="0"
|
|
{...field}
|
|
/>
|
|
</FormControl>
|
|
</div>
|
|
<div className="col-span-4">
|
|
<Select_Shadcn_ value={selectedUnit} onValueChange={setSelectedUnit}>
|
|
<SelectTrigger_Shadcn_ aria-label="File size limit unit" size="small">
|
|
<SelectValue_Shadcn_>{selectedUnit}</SelectValue_Shadcn_>
|
|
</SelectTrigger_Shadcn_>
|
|
<SelectContent_Shadcn_>
|
|
{Object.values(StorageSizeUnits).map((unit: string) => (
|
|
<SelectItem_Shadcn_ key={unit} value={unit} className="text-xs">
|
|
{unit}
|
|
</SelectItem_Shadcn_>
|
|
))}
|
|
</SelectContent_Shadcn_>
|
|
</Select_Shadcn_>
|
|
</div>
|
|
</div>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
{formattedSizeLimitError?.message === 'exceed_global_limit' && (
|
|
<FormMessage className="mt-2">
|
|
Exceeds global limit of {formattedGlobalUploadLimit}. Increase limit in{' '}
|
|
<InlineLink
|
|
className="text-destructive decoration-destructive-500 hover:decoration-destructive"
|
|
href={`/project/${ref}/storage/settings`}
|
|
onClick={onClose}
|
|
>
|
|
Storage Settings
|
|
</InlineLink>{' '}
|
|
first.
|
|
</FormMessage>
|
|
)}
|
|
|
|
{IS_PLATFORM && (
|
|
<p className="text-sm text-foreground-lighter mt-2">
|
|
This project has a{' '}
|
|
<InlineLink
|
|
className="text-foreground-light hover:text-foreground"
|
|
href={`/project/${ref}/storage/settings`}
|
|
onClick={onClose}
|
|
>
|
|
global file size limit
|
|
</InlineLink>{' '}
|
|
of {formattedGlobalUploadLimit}.
|
|
</p>
|
|
)}
|
|
</div>
|
|
)}
|
|
</DialogSection>
|
|
|
|
<DialogSectionSeparator />
|
|
|
|
<DialogSection className="space-y-2">
|
|
<FormItemLayout
|
|
name="has_allowed_mime_types"
|
|
label="Restrict MIME types"
|
|
description="Allow only certain types of files to be uploaded"
|
|
layout="flex"
|
|
>
|
|
<FormControl>
|
|
<Switch
|
|
id="has_allowed_mime_types"
|
|
size="large"
|
|
checked={hasAllowedMimeTypes}
|
|
onCheckedChange={setHasAllowedMimeTypes}
|
|
/>
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
{hasAllowedMimeTypes && (
|
|
<FormField
|
|
key="allowed_mime_types"
|
|
name="allowed_mime_types"
|
|
control={form.control}
|
|
render={({ field }) => (
|
|
<FormItemLayout
|
|
name="allowed_mime_types"
|
|
label="Allowed MIME types"
|
|
labelOptional="Comma separated values"
|
|
description="Wildcards are allowed, e.g. image/*."
|
|
>
|
|
<FormControl>
|
|
<Input_Shadcn_
|
|
id="allowed_mime_types"
|
|
{...field}
|
|
placeholder="e.g image/jpeg, image/png, audio/mpeg, video/mp4, etc"
|
|
/>
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
)}
|
|
</DialogSection>
|
|
</form>
|
|
</Form>
|
|
|
|
<DialogFooter>
|
|
<Button type="default" disabled={isUpdating} onClick={closeModal}>
|
|
Cancel
|
|
</Button>
|
|
<Button form={formId} htmlType="submit" loading={isUpdating}>
|
|
Save
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
)
|
|
}
|