Feat/iteration single run time (#10512)

This commit is contained in:
Novice 2024-11-11 14:47:52 +08:00 committed by GitHub
parent 0c1307b083
commit f414d241c1
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
16 changed files with 101 additions and 29 deletions

View File

@ -361,6 +361,7 @@ class WorkflowBasedAppRunner(AppRunner):
node_run_index=workflow_entry.graph_engine.graph_runtime_state.node_run_steps, node_run_index=workflow_entry.graph_engine.graph_runtime_state.node_run_steps,
output=event.pre_iteration_output, output=event.pre_iteration_output,
parallel_mode_run_id=event.parallel_mode_run_id, parallel_mode_run_id=event.parallel_mode_run_id,
duration=event.duration,
) )
) )
elif isinstance(event, (IterationRunSucceededEvent | IterationRunFailedEvent)): elif isinstance(event, (IterationRunSucceededEvent | IterationRunFailedEvent)):

View File

@ -111,6 +111,7 @@ class QueueIterationNextEvent(AppQueueEvent):
"""iteratoin run in parallel mode run id""" """iteratoin run in parallel mode run id"""
node_run_index: int node_run_index: int
output: Optional[Any] = None # output for the current iteration output: Optional[Any] = None # output for the current iteration
duration: Optional[float] = None
@field_validator("output", mode="before") @field_validator("output", mode="before")
@classmethod @classmethod
@ -307,6 +308,8 @@ class QueueNodeSucceededEvent(AppQueueEvent):
execution_metadata: Optional[dict[NodeRunMetadataKey, Any]] = None execution_metadata: Optional[dict[NodeRunMetadataKey, Any]] = None
error: Optional[str] = None error: Optional[str] = None
"""single iteration duration map"""
iteration_duration_map: Optional[dict[str, float]] = None
class QueueNodeInIterationFailedEvent(AppQueueEvent): class QueueNodeInIterationFailedEvent(AppQueueEvent):

View File

@ -434,6 +434,7 @@ class IterationNodeNextStreamResponse(StreamResponse):
parallel_id: Optional[str] = None parallel_id: Optional[str] = None
parallel_start_node_id: Optional[str] = None parallel_start_node_id: Optional[str] = None
parallel_mode_run_id: Optional[str] = None parallel_mode_run_id: Optional[str] = None
duration: Optional[float] = None
event: StreamEvent = StreamEvent.ITERATION_NEXT event: StreamEvent = StreamEvent.ITERATION_NEXT
workflow_run_id: str workflow_run_id: str

View File

@ -624,6 +624,7 @@ class WorkflowCycleManage:
parallel_id=event.parallel_id, parallel_id=event.parallel_id,
parallel_start_node_id=event.parallel_start_node_id, parallel_start_node_id=event.parallel_start_node_id,
parallel_mode_run_id=event.parallel_mode_run_id, parallel_mode_run_id=event.parallel_mode_run_id,
duration=event.duration,
), ),
) )

View File

@ -24,6 +24,7 @@ class NodeRunMetadataKey(str, Enum):
PARENT_PARALLEL_ID = "parent_parallel_id" PARENT_PARALLEL_ID = "parent_parallel_id"
PARENT_PARALLEL_START_NODE_ID = "parent_parallel_start_node_id" PARENT_PARALLEL_START_NODE_ID = "parent_parallel_start_node_id"
PARALLEL_MODE_RUN_ID = "parallel_mode_run_id" PARALLEL_MODE_RUN_ID = "parallel_mode_run_id"
ITERATION_DURATION_MAP = "iteration_duration_map" # single iteration duration if iteration node runs
class NodeRunResult(BaseModel): class NodeRunResult(BaseModel):

View File

@ -148,6 +148,7 @@ class IterationRunStartedEvent(BaseIterationEvent):
class IterationRunNextEvent(BaseIterationEvent): class IterationRunNextEvent(BaseIterationEvent):
index: int = Field(..., description="index") index: int = Field(..., description="index")
pre_iteration_output: Optional[Any] = Field(None, description="pre iteration output") pre_iteration_output: Optional[Any] = Field(None, description="pre iteration output")
duration: Optional[float] = Field(None, description="duration")
class IterationRunSucceededEvent(BaseIterationEvent): class IterationRunSucceededEvent(BaseIterationEvent):
@ -156,6 +157,7 @@ class IterationRunSucceededEvent(BaseIterationEvent):
outputs: Optional[dict[str, Any]] = None outputs: Optional[dict[str, Any]] = None
metadata: Optional[dict[str, Any]] = None metadata: Optional[dict[str, Any]] = None
steps: int = 0 steps: int = 0
iteration_duration_map: Optional[dict[str, float]] = None
class IterationRunFailedEvent(BaseIterationEvent): class IterationRunFailedEvent(BaseIterationEvent):

View File

@ -156,6 +156,7 @@ class IterationNode(BaseNode[IterationNodeData]):
index=0, index=0,
pre_iteration_output=None, pre_iteration_output=None,
) )
iter_run_map: dict[str, float] = {}
outputs: list[Any] = [None] * len(iterator_list_value) outputs: list[Any] = [None] * len(iterator_list_value)
try: try:
if self.node_data.is_parallel: if self.node_data.is_parallel:
@ -175,6 +176,7 @@ class IterationNode(BaseNode[IterationNodeData]):
iteration_graph, iteration_graph,
index, index,
item, item,
iter_run_map,
) )
future.add_done_callback(thread_pool.task_done_callback) future.add_done_callback(thread_pool.task_done_callback)
futures.append(future) futures.append(future)
@ -213,6 +215,7 @@ class IterationNode(BaseNode[IterationNodeData]):
start_at, start_at,
graph_engine, graph_engine,
iteration_graph, iteration_graph,
iter_run_map,
) )
if self.node_data.error_handle_mode == ErrorHandleMode.REMOVE_ABNORMAL_OUTPUT: if self.node_data.error_handle_mode == ErrorHandleMode.REMOVE_ABNORMAL_OUTPUT:
outputs = [output for output in outputs if output is not None] outputs = [output for output in outputs if output is not None]
@ -230,7 +233,9 @@ class IterationNode(BaseNode[IterationNodeData]):
yield RunCompletedEvent( yield RunCompletedEvent(
run_result=NodeRunResult( run_result=NodeRunResult(
status=WorkflowNodeExecutionStatus.SUCCEEDED, outputs={"output": jsonable_encoder(outputs)} status=WorkflowNodeExecutionStatus.SUCCEEDED,
outputs={"output": jsonable_encoder(outputs)},
metadata={NodeRunMetadataKey.ITERATION_DURATION_MAP: iter_run_map},
) )
) )
except IterationNodeError as e: except IterationNodeError as e:
@ -356,15 +361,19 @@ class IterationNode(BaseNode[IterationNodeData]):
start_at: datetime, start_at: datetime,
graph_engine: "GraphEngine", graph_engine: "GraphEngine",
iteration_graph: Graph, iteration_graph: Graph,
iter_run_map: dict[str, float],
parallel_mode_run_id: Optional[str] = None, parallel_mode_run_id: Optional[str] = None,
) -> Generator[NodeEvent | InNodeEvent, None, None]: ) -> Generator[NodeEvent | InNodeEvent, None, None]:
""" """
run single iteration run single iteration
""" """
iter_start_at = datetime.now(timezone.utc).replace(tzinfo=None)
try: try:
rst = graph_engine.run() rst = graph_engine.run()
# get current iteration index # get current iteration index
current_index = variable_pool.get([self.node_id, "index"]).value current_index = variable_pool.get([self.node_id, "index"]).value
iteration_run_id = parallel_mode_run_id if parallel_mode_run_id is not None else f"{current_index}"
next_index = int(current_index) + 1 next_index = int(current_index) + 1
if current_index is None: if current_index is None:
@ -431,6 +440,8 @@ class IterationNode(BaseNode[IterationNodeData]):
variable_pool.add([self.node_id, "index"], next_index) variable_pool.add([self.node_id, "index"], next_index)
if next_index < len(iterator_list_value): if next_index < len(iterator_list_value):
variable_pool.add([self.node_id, "item"], iterator_list_value[next_index]) variable_pool.add([self.node_id, "item"], iterator_list_value[next_index])
duration = (datetime.now(timezone.utc).replace(tzinfo=None) - iter_start_at).total_seconds()
iter_run_map[iteration_run_id] = duration
yield IterationRunNextEvent( yield IterationRunNextEvent(
iteration_id=self.id, iteration_id=self.id,
iteration_node_id=self.node_id, iteration_node_id=self.node_id,
@ -439,6 +450,7 @@ class IterationNode(BaseNode[IterationNodeData]):
index=next_index, index=next_index,
parallel_mode_run_id=parallel_mode_run_id, parallel_mode_run_id=parallel_mode_run_id,
pre_iteration_output=None, pre_iteration_output=None,
duration=duration,
) )
return return
elif self.node_data.error_handle_mode == ErrorHandleMode.REMOVE_ABNORMAL_OUTPUT: elif self.node_data.error_handle_mode == ErrorHandleMode.REMOVE_ABNORMAL_OUTPUT:
@ -449,6 +461,8 @@ class IterationNode(BaseNode[IterationNodeData]):
if next_index < len(iterator_list_value): if next_index < len(iterator_list_value):
variable_pool.add([self.node_id, "item"], iterator_list_value[next_index]) variable_pool.add([self.node_id, "item"], iterator_list_value[next_index])
duration = (datetime.now(timezone.utc).replace(tzinfo=None) - iter_start_at).total_seconds()
iter_run_map[iteration_run_id] = duration
yield IterationRunNextEvent( yield IterationRunNextEvent(
iteration_id=self.id, iteration_id=self.id,
iteration_node_id=self.node_id, iteration_node_id=self.node_id,
@ -457,6 +471,7 @@ class IterationNode(BaseNode[IterationNodeData]):
index=next_index, index=next_index,
parallel_mode_run_id=parallel_mode_run_id, parallel_mode_run_id=parallel_mode_run_id,
pre_iteration_output=None, pre_iteration_output=None,
duration=duration,
) )
return return
elif self.node_data.error_handle_mode == ErrorHandleMode.TERMINATED: elif self.node_data.error_handle_mode == ErrorHandleMode.TERMINATED:
@ -485,6 +500,8 @@ class IterationNode(BaseNode[IterationNodeData]):
if next_index < len(iterator_list_value): if next_index < len(iterator_list_value):
variable_pool.add([self.node_id, "item"], iterator_list_value[next_index]) variable_pool.add([self.node_id, "item"], iterator_list_value[next_index])
duration = (datetime.now(timezone.utc).replace(tzinfo=None) - iter_start_at).total_seconds()
iter_run_map[iteration_run_id] = duration
yield IterationRunNextEvent( yield IterationRunNextEvent(
iteration_id=self.id, iteration_id=self.id,
iteration_node_id=self.node_id, iteration_node_id=self.node_id,
@ -493,6 +510,7 @@ class IterationNode(BaseNode[IterationNodeData]):
index=next_index, index=next_index,
parallel_mode_run_id=parallel_mode_run_id, parallel_mode_run_id=parallel_mode_run_id,
pre_iteration_output=jsonable_encoder(current_iteration_output) if current_iteration_output else None, pre_iteration_output=jsonable_encoder(current_iteration_output) if current_iteration_output else None,
duration=duration,
) )
except IterationNodeError as e: except IterationNodeError as e:
@ -528,6 +546,7 @@ class IterationNode(BaseNode[IterationNodeData]):
iteration_graph: Graph, iteration_graph: Graph,
index: int, index: int,
item: Any, item: Any,
iter_run_map: dict[str, float],
) -> Generator[NodeEvent | InNodeEvent, None, None]: ) -> Generator[NodeEvent | InNodeEvent, None, None]:
""" """
run single iteration in parallel mode run single iteration in parallel mode
@ -546,6 +565,7 @@ class IterationNode(BaseNode[IterationNodeData]):
start_at=start_at, start_at=start_at,
graph_engine=graph_engine_copy, graph_engine=graph_engine_copy,
iteration_graph=iteration_graph, iteration_graph=iteration_graph,
iter_run_map=iter_run_map,
parallel_mode_run_id=parallel_mode_run_id, parallel_mode_run_id=parallel_mode_run_id,
): ):
q.put(event) q.put(event)

View File

@ -445,6 +445,7 @@ export const useWorkflowRun = () => {
...data, ...data,
status: NodeRunningStatus.Running, status: NodeRunningStatus.Running,
details: [], details: [],
iterDurationMap: {},
} as any) } as any)
})) }))
@ -496,6 +497,8 @@ export const useWorkflowRun = () => {
setWorkflowRunningData(produce(workflowRunningData!, (draft) => { setWorkflowRunningData(produce(workflowRunningData!, (draft) => {
const iteration = draft.tracing!.find(trace => trace.node_id === data.node_id) const iteration = draft.tracing!.find(trace => trace.node_id === data.node_id)
if (iteration) { if (iteration) {
if (iteration.iterDurationMap && data.duration)
iteration.iterDurationMap[data.parallel_mode_run_id ?? `${data.index - 1}`] = data.duration
if (iteration.details!.length >= iteration.metadata.iterator_length!) if (iteration.details!.length >= iteration.metadata.iterator_length!)
return return
} }

View File

@ -193,7 +193,7 @@ const BaseNode: FC<BaseNodeProps> = ({
{ {
data._iterationLength && data._iterationIndex && data._runningStatus === NodeRunningStatus.Running && ( data._iterationLength && data._iterationIndex && data._runningStatus === NodeRunningStatus.Running && (
<div className='mr-1.5 text-xs font-medium text-primary-600'> <div className='mr-1.5 text-xs font-medium text-primary-600'>
{data._iterationIndex}/{data._iterationLength} {data._iterationIndex > data._iterationLength ? data._iterationLength : data._iterationIndex}/{data._iterationLength}
</div> </div>
) )
} }

View File

@ -28,7 +28,7 @@ import IterationResultPanel from '../run/iteration-result-panel'
import InputsPanel from './inputs-panel' import InputsPanel from './inputs-panel'
import cn from '@/utils/classnames' import cn from '@/utils/classnames'
import Loading from '@/app/components/base/loading' import Loading from '@/app/components/base/loading'
import type { NodeTracing } from '@/types/workflow' import type { IterationDurationMap, NodeTracing } from '@/types/workflow'
const WorkflowPreview = () => { const WorkflowPreview = () => {
const { t } = useTranslation() const { t } = useTranslation()
@ -53,12 +53,14 @@ const WorkflowPreview = () => {
}, [workflowRunningData]) }, [workflowRunningData])
const [iterationRunResult, setIterationRunResult] = useState<NodeTracing[][]>([]) const [iterationRunResult, setIterationRunResult] = useState<NodeTracing[][]>([])
const [iterDurationMap, setIterDurationMap] = useState<IterationDurationMap>({})
const [isShowIterationDetail, { const [isShowIterationDetail, {
setTrue: doShowIterationDetail, setTrue: doShowIterationDetail,
setFalse: doHideIterationDetail, setFalse: doHideIterationDetail,
}] = useBoolean(false) }] = useBoolean(false)
const handleShowIterationDetail = useCallback((detail: NodeTracing[][]) => { const handleShowIterationDetail = useCallback((detail: NodeTracing[][], iterationDurationMap: IterationDurationMap) => {
setIterDurationMap(iterationDurationMap)
setIterationRunResult(detail) setIterationRunResult(detail)
doShowIterationDetail() doShowIterationDetail()
}, [doShowIterationDetail]) }, [doShowIterationDetail])
@ -72,6 +74,7 @@ const WorkflowPreview = () => {
list={iterationRunResult} list={iterationRunResult}
onHide={doHideIterationDetail} onHide={doHideIterationDetail}
onBack={doHideIterationDetail} onBack={doHideIterationDetail}
iterDurationMap={iterDurationMap}
/> />
</div> </div>
) )
@ -94,6 +97,7 @@ const WorkflowPreview = () => {
list={iterationRunResult} list={iterationRunResult}
onHide={doHideIterationDetail} onHide={doHideIterationDetail}
onBack={doHideIterationDetail} onBack={doHideIterationDetail}
iterDurationMap={iterDurationMap}
/> />
) )
: ( : (

View File

@ -13,7 +13,7 @@ import cn from '@/utils/classnames'
import { ToastContext } from '@/app/components/base/toast' import { ToastContext } from '@/app/components/base/toast'
import Loading from '@/app/components/base/loading' import Loading from '@/app/components/base/loading'
import { fetchRunDetail, fetchTracingList } from '@/service/log' import { fetchRunDetail, fetchTracingList } from '@/service/log'
import type { NodeTracing } from '@/types/workflow' import type { IterationDurationMap, NodeTracing } from '@/types/workflow'
import type { WorkflowRunDetailResponse } from '@/models/log' import type { WorkflowRunDetailResponse } from '@/models/log'
import { useStore as useAppStore } from '@/app/components/app/store' import { useStore as useAppStore } from '@/app/components/app/store'
@ -172,15 +172,17 @@ const RunPanel: FC<RunProps> = ({ hideResult, activeTab = 'RESULT', runID, getRe
}, [loading]) }, [loading])
const [iterationRunResult, setIterationRunResult] = useState<NodeTracing[][]>([]) const [iterationRunResult, setIterationRunResult] = useState<NodeTracing[][]>([])
const [iterDurationMap, setIterDurationMap] = useState<IterationDurationMap>({})
const [isShowIterationDetail, { const [isShowIterationDetail, {
setTrue: doShowIterationDetail, setTrue: doShowIterationDetail,
setFalse: doHideIterationDetail, setFalse: doHideIterationDetail,
}] = useBoolean(false) }] = useBoolean(false)
const handleShowIterationDetail = useCallback((detail: NodeTracing[][]) => { const handleShowIterationDetail = useCallback((detail: NodeTracing[][], iterDurationMap: IterationDurationMap) => {
setIterationRunResult(detail) setIterationRunResult(detail)
doShowIterationDetail() doShowIterationDetail()
}, [doShowIterationDetail]) setIterDurationMap(iterDurationMap)
}, [doShowIterationDetail, setIterationRunResult, setIterDurationMap])
if (isShowIterationDetail) { if (isShowIterationDetail) {
return ( return (
@ -189,6 +191,7 @@ const RunPanel: FC<RunProps> = ({ hideResult, activeTab = 'RESULT', runID, getRe
list={iterationRunResult} list={iterationRunResult}
onHide={doHideIterationDetail} onHide={doHideIterationDetail}
onBack={doHideIterationDetail} onBack={doHideIterationDetail}
iterDurationMap={iterDurationMap}
/> />
</div> </div>
) )

View File

@ -6,12 +6,14 @@ import {
RiArrowRightSLine, RiArrowRightSLine,
RiCloseLine, RiCloseLine,
RiErrorWarningLine, RiErrorWarningLine,
RiLoader2Line,
} from '@remixicon/react' } from '@remixicon/react'
import { ArrowNarrowLeft } from '../../base/icons/src/vender/line/arrows' import { ArrowNarrowLeft } from '../../base/icons/src/vender/line/arrows'
import { NodeRunningStatus } from '../types'
import TracingPanel from './tracing-panel' import TracingPanel from './tracing-panel'
import { Iteration } from '@/app/components/base/icons/src/vender/workflow' import { Iteration } from '@/app/components/base/icons/src/vender/workflow'
import cn from '@/utils/classnames' import cn from '@/utils/classnames'
import type { NodeTracing } from '@/types/workflow' import type { IterationDurationMap, NodeTracing } from '@/types/workflow'
const i18nPrefix = 'workflow.singleRun' const i18nPrefix = 'workflow.singleRun'
type Props = { type Props = {
@ -19,6 +21,7 @@ type Props = {
onHide: () => void onHide: () => void
onBack: () => void onBack: () => void
noWrap?: boolean noWrap?: boolean
iterDurationMap?: IterationDurationMap
} }
const IterationResultPanel: FC<Props> = ({ const IterationResultPanel: FC<Props> = ({
@ -26,6 +29,7 @@ const IterationResultPanel: FC<Props> = ({
onHide, onHide,
onBack, onBack,
noWrap, noWrap,
iterDurationMap,
}) => { }) => {
const { t } = useTranslation() const { t } = useTranslation()
const [expandedIterations, setExpandedIterations] = useState<Record<number, boolean>>({}) const [expandedIterations, setExpandedIterations] = useState<Record<number, boolean>>({})
@ -36,6 +40,40 @@ const IterationResultPanel: FC<Props> = ({
[index]: !prev[index], [index]: !prev[index],
})) }))
}, []) }, [])
const countIterDuration = (iteration: NodeTracing[], iterDurationMap: IterationDurationMap): string => {
const IterRunIndex = iteration[0].execution_metadata.iteration_index as number
const iterRunId = iteration[0].execution_metadata.parallel_mode_run_id
const iterItem = iterDurationMap[iterRunId || IterRunIndex]
const duration = iterItem
return `${(duration && duration > 0.01) ? duration.toFixed(2) : 0.01}s`
}
const iterationStatusShow = (index: number, iteration: NodeTracing[], iterDurationMap?: IterationDurationMap) => {
const hasFailed = iteration.some(item => item.status === NodeRunningStatus.Failed)
const isRunning = iteration.some(item => item.status === NodeRunningStatus.Running)
const hasDurationMap = iterDurationMap && Object.keys(iterDurationMap).length !== 0
if (hasFailed)
return <RiErrorWarningLine className='w-4 h-4 text-text-destructive' />
if (isRunning)
return <RiLoader2Line className='w-3.5 h-3.5 text-primary-600 animate-spin' />
return (
<>
{hasDurationMap && (
<div className='system-xs-regular text-text-tertiary'>
{countIterDuration(iteration, iterDurationMap)}
</div>
)}
<RiArrowRightSLine
className={cn(
'w-4 h-4 text-text-tertiary transition-transform duration-200 flex-shrink-0',
expandedIterations[index] && 'transform rotate-90',
)}
/>
</>
)
}
const main = ( const main = (
<> <>
@ -72,19 +110,7 @@ const IterationResultPanel: FC<Props> = ({
<span className='system-sm-semibold-uppercase text-text-primary flex-grow'> <span className='system-sm-semibold-uppercase text-text-primary flex-grow'>
{t(`${i18nPrefix}.iteration`)} {index + 1} {t(`${i18nPrefix}.iteration`)} {index + 1}
</span> </span>
{ {iterationStatusShow(index, iteration, iterDurationMap)}
iteration.some(item => item.status === 'failed')
? (
<RiErrorWarningLine className='w-4 h-4 text-text-destructive' />
)
: (< RiArrowRightSLine className={
cn(
'w-4 h-4 text-text-tertiary transition-transform duration-200 flex-shrink-0',
expandedIterations[index] && 'transform rotate-90',
)} />
)
}
</div> </div>
</div> </div>
{expandedIterations[index] && <div {expandedIterations[index] && <div

View File

@ -18,7 +18,7 @@ import StatusContainer from '@/app/components/workflow/run/status-container'
import CodeEditor from '@/app/components/workflow/nodes/_base/components/editor/code-editor' import CodeEditor from '@/app/components/workflow/nodes/_base/components/editor/code-editor'
import Button from '@/app/components/base/button' import Button from '@/app/components/base/button'
import { CodeLanguage } from '@/app/components/workflow/nodes/code/types' import { CodeLanguage } from '@/app/components/workflow/nodes/code/types'
import type { NodeTracing } from '@/types/workflow' import type { IterationDurationMap, NodeTracing } from '@/types/workflow'
type Props = { type Props = {
className?: string className?: string
@ -26,7 +26,7 @@ type Props = {
inMessage?: boolean inMessage?: boolean
hideInfo?: boolean hideInfo?: boolean
hideProcessDetail?: boolean hideProcessDetail?: boolean
onShowIterationDetail?: (detail: NodeTracing[][]) => void onShowIterationDetail?: (detail: NodeTracing[][], iterDurationMap: IterationDurationMap) => void
notShowIterationNav?: boolean notShowIterationNav?: boolean
justShowIterationNavArrow?: boolean justShowIterationNavArrow?: boolean
} }
@ -90,7 +90,7 @@ const NodePanel: FC<Props> = ({
const handleOnShowIterationDetail = (e: React.MouseEvent<HTMLButtonElement>) => { const handleOnShowIterationDetail = (e: React.MouseEvent<HTMLButtonElement>) => {
e.stopPropagation() e.stopPropagation()
e.nativeEvent.stopImmediatePropagation() e.nativeEvent.stopImmediatePropagation()
onShowIterationDetail?.(nodeInfo.details || []) onShowIterationDetail?.(nodeInfo.details || [], nodeInfo?.iterDurationMap || nodeInfo.execution_metadata?.iteration_duration_map || {})
} }
return ( return (
<div className={cn('px-2 py-1', className)}> <div className={cn('px-2 py-1', className)}>

View File

@ -16,11 +16,11 @@ import NodePanel from './node'
import { import {
BlockEnum, BlockEnum,
} from '@/app/components/workflow/types' } from '@/app/components/workflow/types'
import type { NodeTracing } from '@/types/workflow' import type { IterationDurationMap, NodeTracing } from '@/types/workflow'
type TracingPanelProps = { type TracingPanelProps = {
list: NodeTracing[] list: NodeTracing[]
onShowIterationDetail?: (detail: NodeTracing[][]) => void onShowIterationDetail?: (detail: NodeTracing[][], iterDurationMap: IterationDurationMap) => void
className?: string className?: string
hideNodeInfo?: boolean hideNodeInfo?: boolean
hideNodeProcessDetail?: boolean hideNodeProcessDetail?: boolean

View File

@ -569,9 +569,9 @@ const translation = {
MaxParallelismDesc: 'The maximum parallelism is used to control the number of tasks executed simultaneously in a single iteration.', MaxParallelismDesc: 'The maximum parallelism is used to control the number of tasks executed simultaneously in a single iteration.',
errorResponseMethod: 'Error response method', errorResponseMethod: 'Error response method',
ErrorMethod: { ErrorMethod: {
operationTerminated: 'terminated', operationTerminated: 'Terminated',
continueOnError: 'continue on error', continueOnError: 'Continue on Error',
removeAbnormalOutput: 'remove abnormal output', removeAbnormalOutput: 'Remove Abnormal Output',
}, },
answerNodeWarningDesc: 'Parallel mode warning: Answer nodes, conversation variable assignments, and persistent read/write operations within iterations may cause exceptions.', answerNodeWarningDesc: 'Parallel mode warning: Answer nodes, conversation variable assignments, and persistent read/write operations within iterations may cause exceptions.',
}, },

View File

@ -33,6 +33,7 @@ export type NodeTracing = {
parent_parallel_id?: string parent_parallel_id?: string
parent_parallel_start_node_id?: string parent_parallel_start_node_id?: string
parallel_mode_run_id?: string parallel_mode_run_id?: string
iteration_duration_map?: IterationDurationMap
} }
metadata: { metadata: {
iterator_length: number iterator_length: number
@ -44,6 +45,7 @@ export type NodeTracing = {
name: string name: string
email: string email: string
} }
iterDurationMap?: IterationDurationMap
finished_at: number finished_at: number
extras?: any extras?: any
expand?: boolean // for UI expand?: boolean // for UI
@ -207,7 +209,10 @@ export type IterationNextResponse = {
parallel_mode_run_id: string parallel_mode_run_id: string
execution_metadata: { execution_metadata: {
parallel_id?: string parallel_id?: string
iteration_index: number
parallel_mode_run_id?: string
} }
duration?: number
} }
} }
@ -323,3 +328,5 @@ export type ConversationVariableResponse = {
total: number total: number
page: number page: number
} }
export type IterationDurationMap = Record<string, number>