dify/web/features/deployments/components/rollback-modal.tsx
Stephen Zhou 60d5187bcf
tweaks
2026-04-30 17:32:36 +08:00

141 lines
5.3 KiB
TypeScript

'use client'
import type { FC } from 'react'
import {
AlertDialog,
AlertDialogActions,
AlertDialogCancelButton,
AlertDialogConfirmButton,
AlertDialogContent,
AlertDialogDescription,
AlertDialogTitle,
} from '@langgenius/dify-ui/alert-dialog'
import { useQuery } from '@tanstack/react-query'
import { useMemo } from 'react'
import { useTranslation } from 'react-i18next'
import { consoleQuery } from '@/service/client'
import { useStartDeployment } from '../hooks/use-deployment-mutations'
import {
deploymentEnvironmentDeploymentsQueryOptions,
deploymentOverviewQueryOptions,
deploymentReleaseHistoryQueryOptions,
} from '../queries'
import { useDeploymentsStore } from '../store'
import {
activeRelease,
deployedRows,
environmentId,
environmentName,
environmentOptionsFromOptionsReply,
releaseCommit,
releaseLabel,
toAppInfoFromOverview,
} from '../utils'
const InfoRow: FC<{ label: string, value: string }> = ({ label, value }) => {
return (
<div className="flex items-start justify-between gap-4">
<span className="system-xs-medium-uppercase text-text-tertiary">{label}</span>
<span className="system-sm-medium text-text-primary">{value}</span>
</div>
)
}
const RollbackModal: FC = () => {
const { t } = useTranslation('deployments')
const modal = useDeploymentsStore(state => state.rollbackModal)
const closeRollbackModal = useDeploymentsStore(state => state.closeRollbackModal)
const rollbackDeployment = useStartDeployment()
const { data: overview } = useQuery({
...deploymentOverviewQueryOptions(modal.appInstanceId),
enabled: modal.open && Boolean(modal.appInstanceId),
})
const { data: environmentDeployments } = useQuery({
...deploymentEnvironmentDeploymentsQueryOptions(modal.appInstanceId),
enabled: modal.open && Boolean(modal.appInstanceId),
})
const { data: environmentOptionsReply } = useQuery({
...consoleQuery.enterprise.appDeploy.listDeploymentEnvironmentOptions.queryOptions(),
enabled: modal.open,
})
const { data: releaseHistory } = useQuery({
...deploymentReleaseHistoryQueryOptions(modal.appInstanceId),
enabled: modal.open && Boolean(modal.appInstanceId),
})
const environmentOptions = useMemo(
() => environmentOptionsFromOptionsReply(environmentOptionsReply),
[environmentOptionsReply],
)
const currentRow = deployedRows(environmentDeployments?.data)
.find(row => environmentId(row.environment) === modal.environmentId)
const targetRelease = [
...(releaseHistory?.data?.filter(release => !!release.id) ?? []),
].find(release => release.id === modal.targetReleaseId)
const currentRelease = activeRelease(currentRow)
const environment = currentRow?.environment
?? environmentOptions.find(env => env.id === modal.environmentId)
const app = toAppInfoFromOverview(overview?.instance)
const confirm = () => {
if (!modal.appInstanceId || !modal.environmentId || !modal.targetReleaseId)
return
closeRollbackModal()
rollbackDeployment.mutate({
appInstanceId: modal.appInstanceId,
environmentId: modal.environmentId,
releaseId: modal.targetReleaseId,
})
}
return (
<AlertDialog
open={modal.open}
onOpenChange={next => !next && closeRollbackModal()}
>
<AlertDialogContent className="w-[520px]">
<div className="flex flex-col gap-3 px-6 pt-6 pb-2">
<AlertDialogTitle className="title-2xl-semi-bold text-text-primary">
{t('rollback.title', { release: releaseLabel(targetRelease) })}
</AlertDialogTitle>
<AlertDialogDescription className="system-md-regular text-text-tertiary">
{t('rollback.description')}
</AlertDialogDescription>
<div className="mt-2 flex flex-col gap-2 rounded-lg border border-components-panel-border bg-components-panel-bg-blur p-3">
<InfoRow label={t('rollback.instance')} value={app?.name ?? '-'} />
<InfoRow label={t('rollback.sourceApp')} value={app?.name ?? '-'} />
<InfoRow label={t('rollback.environment')} value={environmentName(environment)} />
<InfoRow
label={t('rollback.currentRelease')}
value={currentRelease ? `${releaseLabel(currentRelease)} / ${releaseCommit(currentRelease)}` : '-'}
/>
<InfoRow
label={t('rollback.rollbackTo')}
value={targetRelease ? `${releaseLabel(targetRelease)} / ${releaseCommit(targetRelease)}` : '-'}
/>
</div>
<div className="rounded-lg border border-dashed border-util-colors-red-red-200 bg-util-colors-red-red-50 p-3">
<div className="title-sm-semi-bold text-util-colors-red-red-700">
{t('rollback.impactingTitle')}
</div>
<p className="mt-1 system-xs-regular text-util-colors-red-red-600">
{t('rollback.impactingBody')}
</p>
</div>
</div>
<AlertDialogActions>
<AlertDialogCancelButton variant="secondary">
{t('rollback.cancel')}
</AlertDialogCancelButton>
<AlertDialogConfirmButton onClick={confirm}>
{t('rollback.confirm')}
</AlertDialogConfirmButton>
</AlertDialogActions>
</AlertDialogContent>
</AlertDialog>
)
}
export default RollbackModal