mirror of
https://github.com/supabase/supabase.git
synced 2026-06-22 05:52:48 +08:00
202 lines
6.4 KiB
TypeScript
202 lines
6.4 KiB
TypeScript
import Table from 'components/to-be-cleaned/Table'
|
|
import AlertError from 'components/ui/AlertError'
|
|
import { ReplicationPipelinesData } from 'data/replication/pipelines-query'
|
|
import { ResponseError } from 'types'
|
|
import ShimmeringLoader from 'ui-patterns/ShimmeringLoader'
|
|
import RowMenu from './RowMenu'
|
|
import PipelineStatus from './PipelineStatus'
|
|
import { useParams } from 'common'
|
|
import { useReplicationPipelineStatusQuery } from 'data/replication/pipeline-status-query'
|
|
import { useState } from 'react'
|
|
import { toast } from 'sonner'
|
|
import { useStartPipelineMutation } from 'data/replication/start-pipeline-mutation'
|
|
import { useStopPipelineMutation } from 'data/replication/stop-pipeline-mutation'
|
|
import { useDeleteDestinationMutation } from 'data/replication/delete-destination-mutation'
|
|
import DeleteDestination from './DeleteDestination'
|
|
import DestinationPanel from './DestinationPanel'
|
|
|
|
export type Pipeline = ReplicationPipelinesData['pipelines'][0]
|
|
|
|
interface DestinationRowProps {
|
|
sourceId: number | undefined
|
|
destinationId: number
|
|
destinationName: string
|
|
type: string
|
|
pipeline: Pipeline | undefined
|
|
error: ResponseError | null
|
|
isLoading: boolean
|
|
isError: boolean
|
|
isSuccess: boolean
|
|
}
|
|
|
|
const DestinationRow = ({
|
|
sourceId,
|
|
destinationId,
|
|
destinationName,
|
|
type,
|
|
pipeline,
|
|
error: pipelineError,
|
|
isLoading: isPipelineLoading,
|
|
isError: isPipelineError,
|
|
isSuccess: isPipelineSuccess,
|
|
}: DestinationRowProps) => {
|
|
const { ref: projectRef } = useParams()
|
|
const [refetchInterval, setRefetchInterval] = useState<number | false>(false)
|
|
const [showDeleteDestinationForm, setShowDeleteDestinationForm] = useState(false)
|
|
const [showEditDestinationPanel, setShowEditDestinationPanel] = useState(false)
|
|
|
|
const {
|
|
data: pipelineStatusData,
|
|
error: pipelineStatusError,
|
|
isLoading: isPipelineStatusLoading,
|
|
isError: isPipelineStatusError,
|
|
isSuccess: isPipelineStatusSuccess,
|
|
} = useReplicationPipelineStatusQuery(
|
|
{
|
|
projectRef,
|
|
pipelineId: pipeline?.id,
|
|
},
|
|
{ refetchInterval }
|
|
)
|
|
const [requestStatus, setRequestStatus] = useState<
|
|
'None' | 'EnableRequested' | 'DisableRequested'
|
|
>('None')
|
|
const { mutateAsync: startPipeline } = useStartPipelineMutation()
|
|
const { mutateAsync: stopPipeline } = useStopPipelineMutation()
|
|
const pipelineStatus = pipelineStatusData?.status
|
|
if (
|
|
(requestStatus === 'EnableRequested' && pipelineStatus === 'Started') ||
|
|
(requestStatus === 'DisableRequested' && pipelineStatus === 'Stopped')
|
|
) {
|
|
setRefetchInterval(false)
|
|
setRequestStatus('None')
|
|
}
|
|
|
|
const onEnableClick = async () => {
|
|
if (!projectRef) {
|
|
console.error('Project ref is required')
|
|
return
|
|
}
|
|
if (!pipeline) {
|
|
toast.error('No pipeline found')
|
|
return
|
|
}
|
|
|
|
try {
|
|
await startPipeline({ projectRef, pipelineId: pipeline.id })
|
|
} catch (error) {
|
|
toast.error('Failed to enable destination')
|
|
}
|
|
setRequestStatus('EnableRequested')
|
|
setRefetchInterval(5000)
|
|
}
|
|
const onDisableClick = async () => {
|
|
if (!projectRef) {
|
|
console.error('Project ref is required')
|
|
return
|
|
}
|
|
if (!pipeline) {
|
|
toast.error('No pipeline found')
|
|
return
|
|
}
|
|
|
|
try {
|
|
await stopPipeline({ projectRef, pipelineId: pipeline.id })
|
|
} catch (error) {
|
|
toast.error('Failed to disable destination')
|
|
}
|
|
setRequestStatus('DisableRequested')
|
|
setRefetchInterval(5000)
|
|
}
|
|
const { mutateAsync: deleteDestination } = useDeleteDestinationMutation({})
|
|
|
|
const onDeleteClick = async () => {
|
|
if (!projectRef) {
|
|
console.error('Project ref is required')
|
|
return
|
|
}
|
|
if (!pipeline) {
|
|
toast.error('No pipeline found')
|
|
return
|
|
}
|
|
|
|
try {
|
|
await stopPipeline({ projectRef, pipelineId: pipeline.id })
|
|
// deleting the destination also deletes the pipeline because of cascade delete
|
|
// so we don't need to call deletePipeline explicitly
|
|
await deleteDestination({ projectRef, destinationId: destinationId })
|
|
} catch (error) {
|
|
toast.error('Failed to delete destination')
|
|
}
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{isPipelineError && (
|
|
<AlertError error={pipelineError} subject="Failed to retrieve pipeline" />
|
|
)}
|
|
{isPipelineSuccess && (
|
|
<Table.tr>
|
|
<Table.td>
|
|
{isPipelineLoading ? <ShimmeringLoader></ShimmeringLoader> : destinationName}
|
|
</Table.td>
|
|
<Table.td>{isPipelineLoading ? <ShimmeringLoader></ShimmeringLoader> : type}</Table.td>
|
|
<Table.td>
|
|
{isPipelineLoading || !pipeline ? (
|
|
<ShimmeringLoader></ShimmeringLoader>
|
|
) : (
|
|
<PipelineStatus
|
|
pipelineStatus={pipelineStatusData?.status}
|
|
error={pipelineStatusError}
|
|
isLoading={isPipelineStatusLoading}
|
|
isError={isPipelineStatusError}
|
|
isSuccess={isPipelineStatusSuccess}
|
|
requestStatus={requestStatus}
|
|
></PipelineStatus>
|
|
)}
|
|
</Table.td>
|
|
<Table.td>
|
|
{isPipelineLoading || !pipeline ? (
|
|
<ShimmeringLoader></ShimmeringLoader>
|
|
) : (
|
|
pipeline.publication_name
|
|
)}
|
|
</Table.td>
|
|
<Table.td>
|
|
<RowMenu
|
|
pipelineStatus={pipelineStatusData?.status}
|
|
error={pipelineStatusError}
|
|
isLoading={isPipelineStatusLoading}
|
|
isError={isPipelineStatusError}
|
|
onEnableClick={onEnableClick}
|
|
onDisableClick={onDisableClick}
|
|
onDeleteClick={() => setShowDeleteDestinationForm(true)}
|
|
onEditClick={() => setShowEditDestinationPanel(true)}
|
|
></RowMenu>
|
|
</Table.td>
|
|
</Table.tr>
|
|
)}
|
|
<DeleteDestination
|
|
visible={showDeleteDestinationForm}
|
|
setVisible={setShowDeleteDestinationForm}
|
|
onDelete={onDeleteClick}
|
|
isLoading={isPipelineStatusLoading}
|
|
name={destinationName}
|
|
/>
|
|
<DestinationPanel
|
|
visible={showEditDestinationPanel}
|
|
onClose={() => setShowEditDestinationPanel(false)}
|
|
sourceId={sourceId}
|
|
existingDestination={{
|
|
sourceId,
|
|
destinationId: destinationId,
|
|
pipelineId: pipeline?.id,
|
|
enabled: pipelineStatusData?.status === 'Started',
|
|
}}
|
|
/>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default DestinationRow
|