Files
common_agent/frontend/src/pages/studio/WorkflowBuilderPage.vue

164 lines
5.4 KiB
Vue

<script setup lang="ts">
import { computed, onMounted, ref } from 'vue';
import { Connection, Cpu, VideoPlay } from '@element-plus/icons-vue';
import type { WorkflowWorkspace } from '@/api/workflow';
import { getWorkflowWorkspace } from '@/api/workflow';
const loading = ref(false);
const projectId = ref('101');
const workflowId = ref('201');
const workspace = ref<WorkflowWorkspace | null>(null);
const nodeLibrary = ['Start', 'LLM', 'Knowledge Retrieval', 'MCP Tool', 'Skill', 'Condition', 'Answer'];
const workflowNodes = computed(() => {
const graphJson = workspace.value?.versions?.[0]?.graphJson;
if (!graphJson) {
return [];
}
try {
const parsed = JSON.parse(graphJson) as { nodes?: Array<Record<string, unknown>> };
return (parsed.nodes ?? []).map((node, index) => ({
id: String(node.id ?? `node-${index}`),
type: String(node.type ?? 'NODE'),
label: String(node.label ?? node.id ?? `Node ${index + 1}`),
description: String(node.description ?? node.prompt ?? ''),
x: Number(node.x ?? 10 + index * 18),
y: Number(node.y ?? 42),
}));
} catch {
return [];
}
});
const workflowEdges = computed(() => {
const graphJson = workspace.value?.versions?.[0]?.graphJson;
if (!graphJson) {
return [];
}
try {
const parsed = JSON.parse(graphJson) as { edges?: Array<Record<string, unknown>> };
return (parsed.edges ?? []).map((edge, index) => ({
id: `edge-${index}`,
from: String(edge.from ?? edge.source ?? ''),
to: String(edge.to ?? edge.target ?? ''),
}));
} catch {
return [];
}
});
const canvasEdges = computed(() => {
const nodeById = Object.fromEntries(workflowNodes.value.map((node) => [node.id, node]));
return workflowEdges.value.flatMap((edge) => {
const from = nodeById[edge.from];
const to = nodeById[edge.to];
if (!from || !to) {
return [];
}
return [{
id: edge.id,
x1: from.x + 5,
y1: from.y + 4,
x2: to.x,
y2: to.y + 4,
}];
});
});
async function loadWorkspace() {
loading.value = true;
try {
const response = await getWorkflowWorkspace(projectId.value, workflowId.value);
workspace.value = response.data;
} finally {
loading.value = false;
}
}
onMounted(loadWorkspace);
</script>
<template>
<section class="studio-page workflow-page">
<header class="page-title-row">
<div>
<p class="studio-kicker">WorkflowBuilderView · Draft / Published</p>
<h1>Workflow 图形化编排</h1>
</div>
<div class="toolbar-actions">
<el-button>保存草稿</el-button>
<el-button type="primary"><el-icon><VideoPlay /></el-icon> 运行测试</el-button>
</div>
</header>
<div class="workflow-layout" v-loading="loading">
<aside class="studio-panel node-library">
<div class="panel-heading compact">
<h2>节点库</h2>
<span>JSON Graph</span>
</div>
<button v-for="node in nodeLibrary" :key="node">{{ node }}</button>
</aside>
<main class="studio-panel workflow-canvas">
<div class="canvas-toolbar">
<span><el-icon><Connection /></el-icon> {{ workspace?.workflowCode || 'workflow' }}</span>
<span>版本快照 v{{ workspace?.currentPublishedVersionNo || workspace?.versions?.[0]?.versionNo || '-' }}</span>
<span>环境: {{ workspace?.environment || 'Dev' }}</span>
</div>
<div class="canvas-surface">
<svg class="edge-layer" viewBox="0 0 100 100" preserveAspectRatio="none">
<line
v-for="edge in canvasEdges"
:key="edge.id"
:x1="edge.x1"
:y1="edge.y1"
:x2="edge.x2"
:y2="edge.y2"
/>
</svg>
<article
v-for="node in workflowNodes"
:key="node.id"
class="workflow-node"
:class="{ selected: node.type === 'LLM' }"
:style="{ left: `${node.x}%`, top: `${node.y}%` }"
:data-test="`workflow-node-${node.id}`"
>
<span>{{ node.type }}</span>
<strong>{{ node.label }}</strong>
<em>{{ node.description }}</em>
</article>
</div>
<div class="run-trace-drawer">
<strong>Run Trace</strong>
<div v-for="run in workspace?.recentRuns ?? []" :key="run.requestId" :data-test="`workflow-run-${run.requestId}`">
<span>{{ run.requestId }}</span>
<em>{{ run.status }} · {{ run.durationMs || 0 }}ms · {{ run.outputJson || '-' }}</em>
</div>
</div>
</main>
<aside class="studio-panel inspector-panel">
<div class="panel-heading compact">
<h2>节点 Inspector</h2>
<span>{{ workflowNodes[0]?.type || 'LLM' }}</span>
</div>
<dl class="inspector-list">
<dt>工作流</dt>
<dd>{{ workspace?.workflowName || '-' }}</dd>
<dt>发布状态</dt>
<dd>{{ workspace?.publishStatus || '-' }}</dd>
<dt>当前版本</dt>
<dd>v{{ workspace?.currentPublishedVersionNo || workspace?.versions?.[0]?.versionNo || '-' }}</dd>
<dt>最近请求</dt>
<dd>{{ workspace?.latestRequestId || '-' }}</dd>
</dl>
<button class="blue-command"><el-icon><Cpu /></el-icon> 打开模型路由</button>
</aside>
</div>
</section>
</template>