mirror of
https://github.com/supabase/supabase.git
synced 2026-06-10 04:26:19 +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>
167 lines
5.6 KiB
TypeScript
167 lines
5.6 KiB
TypeScript
import { ArrowUpRight } from 'lucide-react'
|
|
import Link from 'next/link'
|
|
import { parseAsInteger, parseAsStringEnum, useQueryState } from 'nuqs'
|
|
import { useEffect } from 'react'
|
|
import { toast } from 'sonner'
|
|
import {
|
|
Button,
|
|
cn,
|
|
DialogSectionSeparator,
|
|
Sheet,
|
|
SheetContent,
|
|
SheetDescription,
|
|
SheetHeader,
|
|
SheetSection,
|
|
SheetTitle,
|
|
} from 'ui'
|
|
|
|
import { EnableReplicationCallout } from '../EnableReplicationCallout'
|
|
import { PipelineStatusName } from '../Replication.constants'
|
|
import { useDestinationInformation } from '../useDestinationInformation'
|
|
import { useIsETLPrivateAlpha } from '../useIsETLPrivateAlpha'
|
|
import { DestinationForm } from './DestinationForm'
|
|
import { DestinationType } from './DestinationPanel.types'
|
|
import { DestinationTypeSelection } from './DestinationTypeSelection'
|
|
import { ReadReplicaForm } from './ReadReplicaForm'
|
|
import { DocsButton } from '@/components/ui/DocsButton'
|
|
import { useCheckEntitlements } from '@/hooks/misc/useCheckEntitlements'
|
|
import { DOCS_URL } from '@/lib/constants'
|
|
|
|
interface DestinationPanelProps {
|
|
onSuccessCreateReadReplica?: () => void
|
|
}
|
|
|
|
export const DestinationPanel = ({ onSuccessCreateReadReplica }: DestinationPanelProps) => {
|
|
const enablePgReplicate = useIsETLPrivateAlpha()
|
|
const { hasAccess: hasETLReplicationAccess } = useCheckEntitlements('replication.etl')
|
|
|
|
const [urlDestinationType, setDestinationType] = useQueryState(
|
|
'destinationType',
|
|
parseAsStringEnum<DestinationType>([
|
|
'Read Replica',
|
|
'BigQuery',
|
|
'Analytics Bucket',
|
|
'DuckLake',
|
|
]).withOptions({
|
|
history: 'push',
|
|
clearOnDefault: true,
|
|
})
|
|
)
|
|
|
|
const [edit, setEdit] = useQueryState(
|
|
'edit',
|
|
parseAsInteger.withOptions({
|
|
history: 'push',
|
|
clearOnDefault: true,
|
|
})
|
|
)
|
|
|
|
const visible = urlDestinationType !== null || edit !== null
|
|
const editMode = edit !== null
|
|
|
|
const {
|
|
sourceId,
|
|
pipeline,
|
|
statusName,
|
|
replicationNotEnabled,
|
|
type: existingDestinationType,
|
|
destinationFetcher,
|
|
} = useDestinationInformation({ id: edit })
|
|
const destinationType = existingDestinationType ?? urlDestinationType
|
|
const invalidExistingDestination = destinationFetcher.error?.code === 404
|
|
|
|
const existingDestination = editMode
|
|
? {
|
|
sourceId,
|
|
destinationId: edit,
|
|
pipelineId: pipeline?.id,
|
|
statusName,
|
|
enabled:
|
|
statusName === PipelineStatusName.STARTED || statusName === PipelineStatusName.FAILED,
|
|
}
|
|
: undefined
|
|
|
|
const onClose = () => {
|
|
setDestinationType(null)
|
|
setEdit(null)
|
|
}
|
|
|
|
useEffect(() => {
|
|
if (edit !== null && invalidExistingDestination) {
|
|
toast(`Unable to find destination ID ${edit}`)
|
|
setEdit(null)
|
|
}
|
|
}, [edit, invalidExistingDestination, setEdit])
|
|
|
|
return (
|
|
<>
|
|
<Sheet open={visible} onOpenChange={onClose}>
|
|
<SheetContent size="default" showClose={false} className="md:w-[850px]!">
|
|
<div className="flex flex-col h-full" tabIndex={-1}>
|
|
<SheetHeader>
|
|
<SheetTitle>{editMode ? 'Edit destination' : 'Create a new destination'}</SheetTitle>
|
|
<SheetDescription>
|
|
{editMode
|
|
? 'Update the configuration for this destination'
|
|
: 'A destination can be a read replica or an external platform that receives your database changes in real time.'}
|
|
</SheetDescription>
|
|
</SheetHeader>
|
|
|
|
<DestinationTypeSelection />
|
|
|
|
<DialogSectionSeparator />
|
|
|
|
{destinationType === 'Read Replica' ? (
|
|
<ReadReplicaForm onClose={onClose} onSuccess={() => onSuccessCreateReadReplica?.()} />
|
|
) : !enablePgReplicate ? (
|
|
<SheetSection>
|
|
<div className={cn('border rounded-md p-6 flex flex-col gap-y-4')}>
|
|
<div className="flex flex-col gap-y-1">
|
|
<h4>Replicate data to external destinations in real-time</h4>
|
|
<p className="text-sm text-foreground-light">
|
|
We are currently in <span className="text-foreground">private alpha</span> and
|
|
slowly onboarding new customers to ensure stable data pipelines. Request
|
|
access below to join the waitlist. Read replicas are available now.
|
|
</p>
|
|
</div>
|
|
<div className="flex gap-x-2">
|
|
<Button
|
|
asChild
|
|
type="secondary"
|
|
iconRight={<ArrowUpRight size={16} strokeWidth={1.5} />}
|
|
>
|
|
<Link
|
|
href="https://forms.supabase.com/pg_replicate"
|
|
target="_blank"
|
|
rel="noreferrer"
|
|
>
|
|
Request alpha access
|
|
</Link>
|
|
</Button>
|
|
<DocsButton href={`${DOCS_URL}/guides/database/replication#replication`} />
|
|
</div>
|
|
</div>
|
|
</SheetSection>
|
|
) : replicationNotEnabled ? (
|
|
<SheetSection>
|
|
<EnableReplicationCallout
|
|
className="p-6!"
|
|
type={destinationType}
|
|
hasAccess={hasETLReplicationAccess}
|
|
/>
|
|
</SheetSection>
|
|
) : (
|
|
<DestinationForm
|
|
visible={visible}
|
|
selectedType={destinationType ?? 'Read Replica'}
|
|
existingDestination={existingDestination}
|
|
onClose={onClose}
|
|
/>
|
|
)}
|
|
</div>
|
|
</SheetContent>
|
|
</Sheet>
|
|
</>
|
|
)
|
|
}
|