'use client'
|
import type { FC } from 'react'
|
import React, { useCallback } from 'react'
|
import { useTranslation } from 'react-i18next'
|
import {
|
RiCloseLine,
|
RiLoader2Line,
|
} from '@remixicon/react'
|
import type { Props as FormProps } from './form'
|
import Form from './form'
|
import cn from '@/utils/classnames'
|
import Button from '@/app/components/base/button'
|
import { StopCircle } from '@/app/components/base/icons/src/vender/solid/mediaAndDevices'
|
import Split from '@/app/components/workflow/nodes/_base/components/split'
|
import { InputVarType, NodeRunningStatus } from '@/app/components/workflow/types'
|
import ResultPanel from '@/app/components/workflow/run/result-panel'
|
import Toast from '@/app/components/base/toast'
|
import { TransferMethod } from '@/types/app'
|
import { getProcessedFiles } from '@/app/components/base/file-uploader/utils'
|
import type { BlockEnum } from '@/app/components/workflow/types'
|
import type { Emoji } from '@/app/components/tools/types'
|
import type { SpecialResultPanelProps } from '@/app/components/workflow/run/special-result-panel'
|
import SpecialResultPanel from '@/app/components/workflow/run/special-result-panel'
|
|
const i18nPrefix = 'workflow.singleRun'
|
|
type BeforeRunFormProps = {
|
nodeName: string
|
nodeType?: BlockEnum
|
toolIcon?: string | Emoji
|
onHide: () => void
|
onRun: (submitData: Record<string, any>) => void
|
onStop: () => void
|
runningStatus: NodeRunningStatus
|
result?: React.JSX.Element
|
forms: FormProps[]
|
showSpecialResultPanel?: boolean
|
} & Partial<SpecialResultPanelProps>
|
|
function formatValue(value: string | any, type: InputVarType) {
|
if (type === InputVarType.number)
|
return Number.parseFloat(value)
|
if (type === InputVarType.json)
|
return JSON.parse(value)
|
if (type === InputVarType.contexts) {
|
return value.map((item: any) => {
|
return JSON.parse(item)
|
})
|
}
|
if (type === InputVarType.multiFiles)
|
return getProcessedFiles(value)
|
|
if (type === InputVarType.singleFile) {
|
if (Array.isArray(value))
|
return getProcessedFiles(value)
|
return getProcessedFiles([value])[0]
|
}
|
|
return value
|
}
|
const BeforeRunForm: FC<BeforeRunFormProps> = ({
|
nodeName,
|
nodeType,
|
toolIcon,
|
onHide,
|
onRun,
|
onStop,
|
runningStatus,
|
result,
|
forms,
|
showSpecialResultPanel,
|
...restResultPanelParams
|
}) => {
|
const { t } = useTranslation()
|
|
const isFinished = runningStatus === NodeRunningStatus.Succeeded || runningStatus === NodeRunningStatus.Failed || runningStatus === NodeRunningStatus.Exception
|
const isRunning = runningStatus === NodeRunningStatus.Running
|
const isFileLoaded = (() => {
|
// system files
|
const filesForm = forms.find(item => !!item.values['#files#'])
|
if (!filesForm)
|
return true
|
|
const files = filesForm.values['#files#'] as any
|
if (files?.some((item: any) => item.transfer_method === TransferMethod.local_file && !item.upload_file_id))
|
return false
|
|
return true
|
})()
|
const handleRun = useCallback(() => {
|
let errMsg = ''
|
forms.forEach((form) => {
|
form.inputs.forEach((input) => {
|
const value = form.values[input.variable] as any
|
if (!errMsg && input.required && (value === '' || value === undefined || value === null || (input.type === InputVarType.files && value.length === 0)))
|
errMsg = t('workflow.errorMsg.fieldRequired', { field: typeof input.label === 'object' ? input.label.variable : input.label })
|
|
if (!errMsg && (input.type === InputVarType.singleFile || input.type === InputVarType.multiFiles) && value) {
|
let fileIsUploading = false
|
if (Array.isArray(value))
|
fileIsUploading = value.find(item => item.transferMethod === TransferMethod.local_file && !item.uploadedId)
|
else
|
fileIsUploading = value.transferMethod === TransferMethod.local_file && !value.uploadedId
|
|
if (fileIsUploading)
|
errMsg = t('appDebug.errorMessage.waitForFileUpload')
|
}
|
})
|
})
|
if (errMsg) {
|
Toast.notify({
|
message: errMsg,
|
type: 'error',
|
})
|
return
|
}
|
|
const submitData: Record<string, any> = {}
|
let parseErrorJsonField = ''
|
forms.forEach((form) => {
|
form.inputs.forEach((input) => {
|
try {
|
const value = formatValue(form.values[input.variable], input.type)
|
submitData[input.variable] = value
|
}
|
catch {
|
parseErrorJsonField = input.variable
|
}
|
})
|
})
|
if (parseErrorJsonField) {
|
Toast.notify({
|
message: t('workflow.errorMsg.invalidJson', { field: parseErrorJsonField }),
|
type: 'error',
|
})
|
return
|
}
|
|
onRun(submitData)
|
}, [forms, onRun, t])
|
return (
|
<div className='absolute inset-0 z-10 rounded-2xl bg-background-overlay-alt pt-10'>
|
<div className='flex h-full flex-col rounded-2xl bg-components-panel-bg'>
|
<div className='flex h-8 shrink-0 items-center justify-between pl-4 pr-3 pt-3'>
|
<div className='truncate text-base font-semibold text-text-primary'>
|
{t(`${i18nPrefix}.testRun`)} {nodeName}
|
</div>
|
<div className='ml-2 shrink-0 cursor-pointer p-1' onClick={() => {
|
onHide()
|
}}>
|
<RiCloseLine className='h-4 w-4 text-text-tertiary ' />
|
</div>
|
</div>
|
{
|
showSpecialResultPanel && (
|
<div className='h-0 grow overflow-y-auto pb-4'>
|
<SpecialResultPanel {...restResultPanelParams} />
|
</div>
|
)
|
}
|
{
|
!showSpecialResultPanel && (
|
<div className='h-0 grow overflow-y-auto pb-4'>
|
<div className='mt-3 space-y-4 px-4'>
|
{forms.map((form, index) => (
|
<div key={index}>
|
<Form
|
key={index}
|
className={cn(index < forms.length - 1 && 'mb-4')}
|
{...form}
|
/>
|
{index < forms.length - 1 && <Split />}
|
</div>
|
))}
|
</div>
|
<div className='mt-4 flex justify-between space-x-2 px-4' >
|
{isRunning && (
|
<div
|
className='cursor-pointer rounded-lg border border-divider-regular bg-components-button-secondary-bg p-2 shadow-xs'
|
onClick={onStop}
|
>
|
<StopCircle className='h-4 w-4 text-text-tertiary' />
|
</div>
|
)}
|
<Button disabled={!isFileLoaded || isRunning} variant='primary' className='w-0 grow space-x-2' onClick={handleRun}>
|
{isRunning && <RiLoader2Line className='h-4 w-4 animate-spin' />}
|
<div>{t(`${i18nPrefix}.${isRunning ? 'running' : 'startRun'}`)}</div>
|
</Button>
|
</div>
|
{isRunning && (
|
<ResultPanel status='running' showSteps={false} />
|
)}
|
{isFinished && (
|
<>
|
{result}
|
</>
|
)}
|
</div>
|
)
|
}
|
</div>
|
</div>
|
)
|
}
|
export default React.memo(BeforeRunForm)
|