| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | 'use client' | 
					
						
							|  |  |  | import type { FC } from 'react' | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  | import React, { useCallback, useState } from 'react' | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | import { useTranslation } from 'react-i18next' | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   RiArrowRightSLine, | 
					
						
							|  |  |  |   RiCloseLine, | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  |   RiErrorWarningLine, | 
					
						
							| 
									
										
										
										
											2024-11-11 14:47:52 +08:00
										 |  |  |   RiLoader2Line, | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  | } from '@remixicon/react' | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | import { ArrowNarrowLeft } from '../../base/icons/src/vender/line/arrows' | 
					
						
							| 
									
										
										
										
											2024-11-11 14:47:52 +08:00
										 |  |  | import { NodeRunningStatus } from '../types' | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  | import TracingPanel from './tracing-panel' | 
					
						
							| 
									
										
										
										
											2024-12-23 15:55:50 +08:00
										 |  |  | import RetryResultPanel from './retry-result-panel' | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  | import { Iteration } from '@/app/components/base/icons/src/vender/workflow' | 
					
						
							| 
									
										
										
										
											2024-07-09 15:05:40 +08:00
										 |  |  | import cn from '@/utils/classnames' | 
					
						
							| 
									
										
										
										
											2024-11-11 14:47:52 +08:00
										 |  |  | import type { IterationDurationMap, NodeTracing } from '@/types/workflow' | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | const i18nPrefix = 'workflow.singleRun' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Props = { | 
					
						
							|  |  |  |   list: NodeTracing[][] | 
					
						
							|  |  |  |   onHide: () => void | 
					
						
							|  |  |  |   onBack: () => void | 
					
						
							|  |  |  |   noWrap?: boolean | 
					
						
							| 
									
										
										
										
											2024-11-11 14:47:52 +08:00
										 |  |  |   iterDurationMap?: IterationDurationMap | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const IterationResultPanel: FC<Props> = ({ | 
					
						
							|  |  |  |   list, | 
					
						
							|  |  |  |   onHide, | 
					
						
							|  |  |  |   onBack, | 
					
						
							|  |  |  |   noWrap, | 
					
						
							| 
									
										
										
										
											2024-11-11 14:47:52 +08:00
										 |  |  |   iterDurationMap, | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | }) => { | 
					
						
							|  |  |  |   const { t } = useTranslation() | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  |   const [expandedIterations, setExpandedIterations] = useState<Record<number, boolean>>({}) | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const toggleIteration = useCallback((index: number) => { | 
					
						
							|  |  |  |     setExpandedIterations(prev => ({ | 
					
						
							|  |  |  |       ...prev, | 
					
						
							|  |  |  |       [index]: !prev[index], | 
					
						
							|  |  |  |     })) | 
					
						
							|  |  |  |   }, []) | 
					
						
							| 
									
										
										
										
											2024-11-11 14:47:52 +08:00
										 |  |  |   const countIterDuration = (iteration: NodeTracing[], iterDurationMap: IterationDurationMap): string => { | 
					
						
							| 
									
										
										
										
											2024-12-23 15:55:50 +08:00
										 |  |  |     const IterRunIndex = iteration[0]?.execution_metadata?.iteration_index as number | 
					
						
							|  |  |  |     const iterRunId = iteration[0]?.execution_metadata?.parallel_mode_run_id | 
					
						
							| 
									
										
										
										
											2024-11-11 14:47:52 +08:00
										 |  |  |     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', | 
					
						
							|  |  |  |           )} | 
					
						
							|  |  |  |         /> | 
					
						
							|  |  |  |       </> | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-12-23 15:55:50 +08:00
										 |  |  |   const [retryRunResult, setRetryRunResult] = useState<Record<string, NodeTracing[]> | undefined>() | 
					
						
							|  |  |  |   const handleRetryDetail = (v: number, detail?: NodeTracing[]) => { | 
					
						
							|  |  |  |     setRetryRunResult({ ...retryRunResult, [v]: detail }) | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const main = ( | 
					
						
							|  |  |  |     <> | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |       <div className={cn(!noWrap && 'shrink-0 ', 'px-4 pt-3')}> | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         <div className='shrink-0 flex justify-between items-center h-8'> | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |           <div className='system-xl-semibold text-text-primary truncate'> | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |             {t(`${i18nPrefix}.testRunIteration`)} | 
					
						
							|  |  |  |           </div> | 
					
						
							|  |  |  |           <div className='ml-2 shrink-0 p-1 cursor-pointer' onClick={onHide}> | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |             <RiCloseLine className='w-4 h-4 text-text-tertiary' /> | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |           </div> | 
					
						
							|  |  |  |         </div> | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |         <div className='flex items-center py-2 space-x-1 text-text-accent-secondary cursor-pointer' onClick={onBack}> | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |           <ArrowNarrowLeft className='w-4 h-4' /> | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |           <div className='system-sm-medium'>{t(`${i18nPrefix}.back`)}</div> | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         </div> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |       {/* List */} | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |       <div className={cn(!noWrap ? 'flex-grow overflow-auto' : 'max-h-full', 'p-2 bg-components-panel-bg')}> | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         {list.map((iteration, index) => ( | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |           <div key={index} className={cn('mb-1 overflow-hidden rounded-xl bg-background-section-burn border-none')}> | 
					
						
							|  |  |  |             <div | 
					
						
							|  |  |  |               className={cn( | 
					
						
							|  |  |  |                 'flex items-center justify-between w-full px-3 cursor-pointer', | 
					
						
							|  |  |  |                 expandedIterations[index] ? 'pt-3 pb-2' : 'py-3', | 
					
						
							|  |  |  |                 'rounded-xl text-left', | 
					
						
							|  |  |  |               )} | 
					
						
							|  |  |  |               onClick={() => toggleIteration(index)} | 
					
						
							|  |  |  |             > | 
					
						
							|  |  |  |               <div className={cn('flex items-center gap-2 flex-grow')}> | 
					
						
							|  |  |  |                 <div className='flex items-center justify-center w-4 h-4 rounded-[5px] border-divider-subtle bg-util-colors-cyan-cyan-500 flex-shrink-0'> | 
					
						
							|  |  |  |                   <Iteration className='w-3 h-3 text-text-primary-on-surface' /> | 
					
						
							|  |  |  |                 </div> | 
					
						
							|  |  |  |                 <span className='system-sm-semibold-uppercase text-text-primary flex-grow'> | 
					
						
							|  |  |  |                   {t(`${i18nPrefix}.iteration`)} {index + 1} | 
					
						
							|  |  |  |                 </span> | 
					
						
							| 
									
										
										
										
											2024-11-11 14:47:52 +08:00
										 |  |  |                 {iterationStatusShow(index, iteration, iterDurationMap)} | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |               </div> | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |             </div> | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |             {expandedIterations[index] && <div | 
					
						
							|  |  |  |               className="flex-grow h-px bg-divider-subtle" | 
					
						
							|  |  |  |             ></div>} | 
					
						
							| 
									
										
										
										
											2024-12-23 15:55:50 +08:00
										 |  |  |             { | 
					
						
							|  |  |  |               !retryRunResult?.[index] && ( | 
					
						
							|  |  |  |                 <div className={cn( | 
					
						
							|  |  |  |                   'overflow-hidden transition-all duration-200', | 
					
						
							|  |  |  |                   expandedIterations[index] ? 'max-h-[1000px] opacity-100' : 'max-h-0 opacity-0', | 
					
						
							|  |  |  |                 )}> | 
					
						
							|  |  |  |                   <TracingPanel | 
					
						
							|  |  |  |                     list={iteration} | 
					
						
							|  |  |  |                     className='bg-background-section-burn' | 
					
						
							|  |  |  |                     onShowRetryDetail={v => handleRetryDetail(index, v)} | 
					
						
							|  |  |  |                   /> | 
					
						
							|  |  |  |                 </div> | 
					
						
							|  |  |  |               ) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               retryRunResult?.[index] && ( | 
					
						
							|  |  |  |                 <RetryResultPanel | 
					
						
							|  |  |  |                   list={retryRunResult[index]} | 
					
						
							|  |  |  |                   onBack={() => handleRetryDetail(index, undefined)} | 
					
						
							|  |  |  |                 /> | 
					
						
							|  |  |  |               ) | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |           </div> | 
					
						
							|  |  |  |         ))} | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     </> | 
					
						
							|  |  |  |   ) | 
					
						
							|  |  |  |   const handleNotBubble = useCallback((e: React.MouseEvent) => { | 
					
						
							|  |  |  |     // if not do this, it will trigger the message log modal disappear(useClickAway)
 | 
					
						
							|  |  |  |     e.stopPropagation() | 
					
						
							|  |  |  |     e.nativeEvent.stopImmediatePropagation() | 
					
						
							|  |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (noWrap) | 
					
						
							|  |  |  |     return main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <div | 
					
						
							|  |  |  |       className='absolute inset-0 z-10 rounded-2xl pt-10' | 
					
						
							|  |  |  |       style={{ | 
					
						
							|  |  |  |         backgroundColor: 'rgba(16, 24, 40, 0.20)', | 
					
						
							|  |  |  |       }} | 
					
						
							|  |  |  |       onClick={handleNotBubble} | 
					
						
							|  |  |  |     > | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       <div className='h-full rounded-2xl bg-components-panel-bg flex flex-col'> | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         {main} | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     </div > | 
					
						
							|  |  |  |   ) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | export default React.memo(IterationResultPanel) |