@humanclaw/humanclaw 1.1.2 → 1.1.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -985,9 +985,24 @@ main{padding:24px 32px;max-width:1200px}
985
985
  .lane{min-height:40px}
986
986
  .lane-hd{font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:1px;margin-bottom:6px;padding-bottom:4px;border-bottom:2px solid var(--border)}
987
987
  .lane-hd.y{color:var(--yellow);border-color:var(--yellow)}.lane-hd.r{color:var(--red);border-color:var(--red)}.lane-hd.g{color:var(--green);border-color:var(--green)}
988
- .tcard{background:var(--bg);border:1px solid var(--border);border-radius:6px;padding:8px 10px;margin-bottom:6px;font-size:12px}
988
+ .tcard{background:var(--bg);border:1px solid var(--border);border-radius:6px;padding:8px 10px;margin-bottom:6px;font-size:12px;cursor:pointer;transition:all .15s}
989
+ .tcard:hover{border-color:var(--accent);background:var(--surface-hover)}
989
990
  .tcard-trace{font-family:var(--font-mono);font-size:10px;color:var(--accent);margin-bottom:3px}
990
991
  .tcard-meta{font-size:10px;color:var(--text-dim);margin-top:3px}
992
+ /* Task detail modal */
993
+ .task-detail-hd{display:flex;align-items:center;gap:10px;margin-bottom:16px}
994
+ .task-detail-hd .dot{width:12px;height:12px}
995
+ .task-detail-status{font-size:11px;padding:3px 10px;border-radius:12px;font-weight:600;text-transform:uppercase}
996
+ .task-detail-status.DISPATCHED,.task-detail-status.PENDING{background:rgba(234,179,8,.15);color:var(--yellow)}
997
+ .task-detail-status.OVERDUE{background:rgba(239,68,68,.15);color:var(--red)}
998
+ .task-detail-status.RESOLVED{background:rgba(34,197,94,.15);color:var(--green)}
999
+ .trace-copy-row{display:flex;align-items:center;gap:8px;background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:8px 12px;margin-bottom:14px;font-family:var(--font-mono);font-size:13px;color:var(--accent)}
1000
+ .trace-copy-row button{background:var(--surface-hover);border:1px solid var(--border);color:var(--text-dim);border-radius:4px;padding:3px 10px;font-size:11px;cursor:pointer;margin-left:auto;flex-shrink:0}
1001
+ .trace-copy-row button:hover{color:var(--accent);border-color:var(--accent)}
1002
+ .detail-row{margin-bottom:12px}
1003
+ .detail-label{font-size:11px;color:var(--text-dim);margin-bottom:3px;font-weight:500}
1004
+ .detail-value{font-size:13px}
1005
+ .result-display{background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:12px;font-family:var(--font-mono);font-size:12px;white-space:pre-wrap;max-height:200px;overflow-y:auto}
991
1006
  /* Toast */
992
1007
  .toast{position:fixed;bottom:24px;right:24px;padding:12px 20px;border-radius:8px;font-size:13px;z-index:9999;animation:slide-up .25s ease;font-weight:500;box-shadow:0 8px 24px rgba(0,0,0,.4)}
993
1008
  .toast.ok{background:var(--green);color:#fff}.toast.err{background:var(--red);color:#fff}
@@ -1147,11 +1162,13 @@ window.deleteAgent=async function(id){
1147
1162
  // \u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550
1148
1163
  // PIPELINE
1149
1164
  // \u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550
1165
+ let allTasks=[];
1150
1166
  function tcard(t){
1151
- return '<div class="tcard"><div class="tcard-trace">'+t.trace_id+'</div><div>'+esc(t.todo_description)+'</div><div class="tcard-meta">'+t.assignee_id+' | '+new Date(t.deadline).toLocaleString()+'</div></div>';
1167
+ return '<div class="tcard" onclick="showTaskDetail(\\''+t.trace_id+'\\')"><div class="tcard-trace">'+t.trace_id+'</div><div>'+esc(t.todo_description)+'</div><div class="tcard-meta">'+t.assignee_id+' | '+new Date(t.deadline).toLocaleString()+'</div></div>';
1152
1168
  }
1153
1169
  async function loadPipeline(el){
1154
1170
  el.innerHTML='<div class="empty-state"><p>\u52A0\u8F7D\u4E2D...</p></div>';
1171
+ allTasks=[];
1155
1172
  try{
1156
1173
  try{const ar=await fetch(API+'/nodes/status');const ad=await ar.json();cachedAgents=ad.agents||[]}catch{}
1157
1174
  const r=await fetch(API+'/jobs/active');
@@ -1170,6 +1187,7 @@ async function loadPipeline(el){
1170
1187
  const dispatched=j.tasks.filter(t=>t.status==='DISPATCHED'||t.status==='PENDING');
1171
1188
  const overdue=j.tasks.filter(t=>t.status==='OVERDUE');
1172
1189
  const done=j.tasks.filter(t=>t.status==='RESOLVED');
1190
+ allTasks.push(...j.tasks);
1173
1191
  h+='<div class="job-card"><div class="job-hd"><span class="job-title">'+esc(j.original_prompt)+'</span><span class="job-id">'+j.job_id+'</span></div>';
1174
1192
  h+='<div class="pbar-wrap"><div class="pbar-label">'+res+'/'+tot+' \u5DF2\u5B8C\u6210 ('+pct+'%)</div><div class="pbar"><div class="pbar-fill" style="width:'+pct+'%"></div></div></div>';
1175
1193
  if(pct===100)h+='<div style="margin-bottom:12px"><button class="btn btn-green btn-sm" onclick="syncJob(\\''+j.job_id+'\\')">\u805A\u5408\u5E76\u540C\u6B65\u5230 OpenClaw</button></div>';
@@ -1360,6 +1378,70 @@ window.syncJob=async function(jobId){
1360
1378
  }catch{toast('\u7F51\u7EDC\u9519\u8BEF',false)}
1361
1379
  };
1362
1380
 
1381
+ // \u2500\u2500\u2500 Task Detail Modal \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
1382
+ window.showTaskDetail=function(traceId){
1383
+ const t=allTasks.find(x=>x.trace_id===traceId);
1384
+ if(!t)return;
1385
+ const agent=cachedAgents.find(a=>a.agent_id===t.assignee_id);
1386
+ const agentName=agent?agent.name:t.assignee_id;
1387
+ const ov=document.createElement('div');ov.className='overlay';ov.id='overlay';
1388
+ ov.addEventListener('click',e=>{if(e.target===ov)ov.remove()});
1389
+
1390
+ let h='<div class="form-card"><div class="task-detail-hd"><span class="task-detail-status '+t.status+'">'+t.status+'</span><span style="font-size:15px;font-weight:600">\u4EFB\u52A1\u8BE6\u60C5</span></div>';
1391
+
1392
+ // Trace ID with copy
1393
+ h+='<div class="trace-copy-row"><span>'+t.trace_id+'</span><button onclick="navigator.clipboard.writeText(\\''+t.trace_id+'\\').then(()=>toast(\\'\u5DF2\u590D\u5236\\',true))">\u590D\u5236</button></div>';
1394
+
1395
+ // Info rows
1396
+ h+='<div class="detail-row"><div class="detail-label">\u6307\u6D3E\u8282\u70B9</div><div class="detail-value">'+esc(agentName)+' <span style="color:var(--text-dim);font-family:var(--font-mono);font-size:11px">'+t.assignee_id+'</span></div></div>';
1397
+ h+='<div class="detail-row"><div class="detail-label">\u4EFB\u52A1\u63CF\u8FF0</div><div class="detail-value">'+esc(t.todo_description)+'</div></div>';
1398
+ h+='<div class="detail-row"><div class="detail-label">\u622A\u6B62\u65F6\u95F4</div><div class="detail-value" style="font-family:var(--font-mono)">'+new Date(t.deadline).toLocaleString()+'</div></div>';
1399
+
1400
+ if(t.status==='RESOLVED'&&t.result_data){
1401
+ // Show result
1402
+ let resultText='';
1403
+ try{const rd=JSON.parse(t.result_data);resultText=rd.text||JSON.stringify(rd,null,2)}catch{resultText=t.result_data}
1404
+ h+='<div class="detail-row"><div class="detail-label">\u4EA4\u4ED8\u7ED3\u679C</div><div class="result-display">'+esc(resultText)+'</div></div>';
1405
+ h+='<div class="btn-group"><button class="btn btn-ghost" onclick="document.getElementById(\\'overlay\\').remove()">\u5173\u95ED</button></div>';
1406
+ }else{
1407
+ // Input for resume/reject
1408
+ h+='<div class="fg" style="margin-top:16px"><label>\u63D0\u4EA4 Human \u4EA4\u4ED8\u7269</label><textarea id="td-payload" rows="4" placeholder="\u7C98\u8D34\u4EA4\u4ED8\u7269\u5185\u5BB9\u3001\u5DE5\u4F5C\u6C47\u62A5\u6216\u4EE3\u7801..."></textarea></div>';
1409
+ h+='<div class="btn-group">';
1410
+ h+='<button class="btn btn-green" onclick="taskResume(\\''+t.trace_id+'\\')">\u63D0\u4EA4\u4EA4\u4ED8 (Resume)</button>';
1411
+ h+='<button class="btn btn-danger" onclick="taskReject(\\''+t.trace_id+'\\')">\u6253\u56DE\u91CD\u505A (Reject)</button>';
1412
+ h+='<button class="btn btn-ghost" onclick="document.getElementById(\\'overlay\\').remove()">\u53D6\u6D88</button>';
1413
+ h+='</div>';
1414
+ }
1415
+ h+='</div>';
1416
+ ov.innerHTML=h;
1417
+ document.body.appendChild(ov);
1418
+ };
1419
+
1420
+ window.taskResume=async function(traceId){
1421
+ const payload=document.getElementById('td-payload').value.trim();
1422
+ if(!payload){toast('\u8BF7\u8F93\u5165\u4EA4\u4ED8\u5185\u5BB9',false);return}
1423
+ try{
1424
+ const r=await fetch(API+'/tasks/resume',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({trace_id:traceId,result_data:{text:payload}})});
1425
+ const d=await r.json();
1426
+ if(!r.ok){toast(d.error||'\u63D0\u4EA4\u5931\u8D25',false);return}
1427
+ toast(d.job_complete?'\u4EFB\u52A1\u5DF2\u4EA4\u4ED8\uFF01Job \u5DF2\u5168\u90E8\u5B8C\u6210\uFF0C\u53EF\u4EE5\u805A\u5408\u540C\u6B65\u3002':'\u4EFB\u52A1 '+traceId+' \u5DF2\u4EA4\u4ED8\u3002',true);
1428
+ document.getElementById('overlay').remove();
1429
+ load('pipeline');
1430
+ }catch{toast('\u7F51\u7EDC\u9519\u8BEF',false)}
1431
+ };
1432
+
1433
+ window.taskReject=async function(traceId){
1434
+ if(!confirm('\u786E\u5B9A\u6253\u56DE\u4EFB\u52A1 '+traceId+'\uFF1F\u622A\u6B62\u65F6\u95F4\u5C06\u5EF6\u957F 24 \u5C0F\u65F6\u3002'))return;
1435
+ try{
1436
+ const r=await fetch(API+'/tasks/reject',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({trace_id:traceId})});
1437
+ const d=await r.json();
1438
+ if(!r.ok){toast(d.error||'\u64CD\u4F5C\u5931\u8D25',false);return}
1439
+ toast('\u4EFB\u52A1 '+traceId+' \u5DF2\u6253\u56DE\uFF0C\u622A\u6B62\u65F6\u95F4\u5DF2\u5EF6\u957F 24 \u5C0F\u65F6\u3002',true);
1440
+ document.getElementById('overlay').remove();
1441
+ load('pipeline');
1442
+ }catch{toast('\u7F51\u7EDC\u9519\u8BEF',false)}
1443
+ };
1444
+
1363
1445
  // \u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550
1364
1446
  // TERMINAL
1365
1447
  // \u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/server.ts","../src/db/connection.ts","../src/db/schema.ts","../src/routes/nodes.ts","../src/models/agent.ts","../src/utils/trace-id.ts","../src/routes/jobs.ts","../src/models/task.ts","../src/models/job.ts","../src/services/dispatch.ts","../src/llm/claude.ts","../src/llm/openai.ts","../src/models/config.ts","../src/llm/index.ts","../src/services/planner.ts","../src/routes/tasks.ts","../src/services/resume.ts","../src/routes/sync.ts","../src/services/aggregation.ts","../src/routes/config.ts","../src/dashboard.ts"],"sourcesContent":["import { Command } from 'commander';\nimport * as p from '@clack/prompts';\nimport chalk from 'chalk';\nimport { startServer } from './server.js';\nimport { getDb } from './db/connection.js';\nimport { initSchema } from './db/schema.js';\nimport { createAgent, listAgents } from './models/agent.js';\nimport { listActiveJobs } from './models/job.js';\nimport { markOverdueTasks } from './models/task.js';\nimport { dispatchJob } from './services/dispatch.js';\nimport { planJob } from './services/planner.js';\nimport { generateId } from './utils/trace-id.js';\nimport type { AgentStatus } from './models/types.js';\n\nconst program = new Command();\n\nprogram\n .name('humanclaw')\n .description(\n 'Async physical node orchestration framework - treating humans as distributed worker nodes'\n )\n .version('1.0.0');\n\n// ─── serve ───────────────────────────────────────────────────────────────────\n\nprogram\n .command('serve')\n .description('Start the HumanClaw server')\n .option('-p, --port <port>', 'Server port', '2026')\n .action(opts => {\n const port = parseInt(opts.port, 10);\n startServer(port);\n });\n\n// ─── agent ───────────────────────────────────────────────────────────────────\n\nconst agentCmd = program\n .command('agent')\n .description('Manage physical nodes (HumanAgent)');\n\nagentCmd\n .command('add')\n .description('Register a new physical node')\n .action(async () => {\n const db = getDb();\n initSchema(db);\n\n p.intro(chalk.bgCyan(' Register New Physical Node '));\n\n const name = await p.text({\n message: 'Node alias (name):',\n placeholder: 'e.g. Frontend Lao Li',\n validate: v => (!v ? 'Name is required' : undefined),\n });\n\n if (p.isCancel(name)) {\n p.cancel('Cancelled.');\n process.exit(0);\n }\n\n const capInput = await p.text({\n message: 'Capabilities (comma-separated):',\n placeholder: 'e.g. UI/UX, Frontend Dev, Stress Resistant',\n validate: v => (!v ? 'At least one capability required' : undefined),\n });\n\n if (p.isCancel(capInput)) {\n p.cancel('Cancelled.');\n process.exit(0);\n }\n\n const capabilities = (capInput as string)\n .split(',')\n .map(s => s.trim())\n .filter(Boolean);\n\n const agent = createAgent({\n agent_id: generateId('emp'),\n name: name as string,\n capabilities,\n status: 'IDLE',\n });\n\n p.outro(\n `${chalk.green('Node registered!')} ID: ${chalk.bold(agent.agent_id)}`\n );\n });\n\nagentCmd\n .command('list')\n .description('Show fleet status')\n .action(() => {\n const db = getDb();\n initSchema(db);\n\n const agents = listAgents();\n if (agents.length === 0) {\n console.log(chalk.dim(' No physical nodes registered.'));\n return;\n }\n\n const statusIcon: Record<AgentStatus, string> = {\n IDLE: '🟢',\n BUSY: '🟡',\n OFFLINE: '🔴',\n OOM: '🟣',\n };\n\n console.log(chalk.bold('\\n Carbon Compute Pool\\n'));\n\n for (const agent of agents) {\n const icon = statusIcon[agent.status];\n const caps = agent.capabilities.join(', ');\n console.log(\n ` ${icon} ${chalk.bold(agent.name)} (${chalk.dim(agent.agent_id)})`\n );\n console.log(` Capabilities: ${chalk.cyan(caps)}`);\n console.log(` Status: ${agent.status}\\n`);\n }\n });\n\n// ─── status ──────────────────────────────────────────────────────────────────\n\nprogram\n .command('status')\n .description('Show active jobs overview')\n .action(() => {\n const db = getDb();\n initSchema(db);\n\n markOverdueTasks();\n const jobs = listActiveJobs();\n\n if (jobs.length === 0) {\n console.log(chalk.dim('\\n No active jobs.\\n'));\n return;\n }\n\n console.log(chalk.bold('\\n Async Orchestration Dashboard\\n'));\n\n for (const job of jobs) {\n const resolved = job.tasks.filter(t => t.status === 'RESOLVED').length;\n const total = job.tasks.length;\n const pct = Math.round((resolved / total) * 100);\n const bar = '█'.repeat(Math.round(pct / 5)) + '░'.repeat(20 - Math.round(pct / 5));\n\n console.log(` ${chalk.bold(job.job_id)} - ${job.original_prompt}`);\n console.log(` Progress: [${bar}] ${resolved}/${total} (${pct}%)`);\n\n for (const task of job.tasks) {\n const statusColor =\n task.status === 'RESOLVED'\n ? chalk.green\n : task.status === 'OVERDUE'\n ? chalk.red\n : chalk.yellow;\n console.log(\n ` ${statusColor(task.status.padEnd(10))} ${task.trace_id} -> ${chalk.dim(task.assignee_id)}`\n );\n console.log(` ${task.todo_description}`);\n }\n console.log();\n }\n });\n\n// ─── plan ───────────────────────────────────────────────────────────────────\n\nprogram\n .command('plan')\n .description('AI-powered task planning from natural language')\n .argument('[prompt]', 'What you want to get done')\n .option('--agents <ids>', 'Comma-separated agent IDs (default: all IDLE)')\n .option('--dispatch', 'Automatically dispatch after planning')\n .action(async (promptArg?: string, opts?: { agents?: string; dispatch?: boolean }) => {\n const db = getDb();\n initSchema(db);\n\n p.intro(chalk.bgCyan(' AI 智能规划 '));\n\n let prompt = promptArg;\n if (!prompt) {\n const input = await p.text({\n message: '输入你的需求:',\n placeholder: '例: 完成首页重构,包括导航栏和内容区的响应式改版',\n validate: v => (!v ? '需求描述不能为空' : undefined),\n });\n if (p.isCancel(input)) {\n p.cancel('已取消');\n process.exit(0);\n }\n prompt = input as string;\n }\n\n const agentIds = opts?.agents?.split(',').map(s => s.trim()).filter(Boolean);\n\n const spin = p.spinner();\n spin.start('AI 正在分析需求、匹配节点、生成话术...');\n\n try {\n const plan = await planJob({ prompt, agent_ids: agentIds }, undefined, db);\n spin.stop('规划完成!');\n\n console.log(chalk.bold(`\\n 需求: ${chalk.white(plan.original_prompt)}\\n`));\n\n for (const task of plan.planned_tasks) {\n console.log(chalk.cyan(` ┌─ ${chalk.bold(task.assignee_name)} (${chalk.dim(task.assignee_id)})`));\n console.log(chalk.cyan(' │') + ` 任务: ${task.todo_description}`);\n console.log(chalk.cyan(' │') + ` 话术: ${chalk.italic(task.briefing)}`);\n console.log(chalk.cyan(' │') + ` 截止: ${new Date(task.deadline).toLocaleString()}`);\n console.log(chalk.cyan(' └─'));\n console.log();\n }\n\n if (opts?.dispatch) {\n const tasks = plan.planned_tasks.map(t => ({\n assignee_id: t.assignee_id,\n todo_description: t.todo_description,\n deadline: t.deadline,\n }));\n const job = dispatchJob({\n original_prompt: plan.original_prompt,\n openclaw_callback: '',\n tasks,\n }, db);\n p.outro(`${chalk.green('已分发!')} Job: ${chalk.bold(job.job_id)}`);\n } else {\n const confirm = await p.confirm({\n message: '确认分发这些任务?',\n });\n if (p.isCancel(confirm) || !confirm) {\n p.outro(chalk.dim('已取消分发'));\n process.exit(0);\n }\n const tasks = plan.planned_tasks.map(t => ({\n assignee_id: t.assignee_id,\n todo_description: t.todo_description,\n deadline: t.deadline,\n }));\n const job = dispatchJob({\n original_prompt: plan.original_prompt,\n openclaw_callback: '',\n tasks,\n }, db);\n p.outro(`${chalk.green('已分发!')} Job: ${chalk.bold(job.job_id)}`);\n }\n } catch (error) {\n spin.stop('规划失败');\n const message = error instanceof Error ? error.message : 'Unknown error';\n p.outro(chalk.red(message));\n process.exit(1);\n }\n });\n\nprogram.parse();\n","import express from 'express';\nimport cors from 'cors';\nimport { getDb } from './db/connection.js';\nimport { initSchema } from './db/schema.js';\nimport nodesRouter from './routes/nodes.js';\nimport jobsRouter from './routes/jobs.js';\nimport tasksRouter from './routes/tasks.js';\nimport syncRouter from './routes/sync.js';\nimport configRouter from './routes/config.js';\nimport { getDashboardHtml } from './dashboard.js';\n\nexport function createServer(port = 2026) {\n // Initialize database\n const db = getDb();\n initSchema(db);\n\n const app = express();\n\n // Middleware\n app.use(cors());\n app.use(express.json({ limit: '10mb' }));\n\n // API routes\n app.use('/api/v1/nodes', nodesRouter);\n app.use('/api/v1/jobs', jobsRouter);\n app.use('/api/v1/tasks', tasksRouter);\n app.use('/api/v1/jobs', syncRouter);\n app.use('/api/v1/config', configRouter);\n\n // Serve dashboard as inline HTML (no build step needed)\n const dashboardHtml = getDashboardHtml();\n app.get('/', (_req, res) => {\n res.type('html').send(dashboardHtml);\n });\n\n // Error handler\n app.use(\n (\n err: Error,\n _req: express.Request,\n res: express.Response,\n _next: express.NextFunction\n ) => {\n console.error('Server error:', err.message);\n res.status(500).json({ error: 'Internal server error' });\n }\n );\n\n return { app, port };\n}\n\nexport function startServer(port = 2026) {\n const { app } = createServer(port);\n\n app.listen(port, () => {\n console.log(`\\n HumanClaw server running at http://localhost:${port}`);\n console.log(` Dashboard: http://localhost:${port}`);\n console.log(` API base: http://localhost:${port}/api/v1\\n`);\n });\n}\n","import Database from 'better-sqlite3';\nimport path from 'node:path';\nimport fs from 'node:fs';\n\nlet db: Database.Database | null = null;\n\nconst DEFAULT_DB_PATH = path.join(\n process.env.HUMANCLAW_DATA_DIR ?? process.cwd(),\n 'humanclaw.db'\n);\n\nexport function getDb(dbPath?: string): Database.Database {\n if (db) return db;\n\n const resolvedPath = dbPath ?? DEFAULT_DB_PATH;\n const dir = path.dirname(resolvedPath);\n if (!fs.existsSync(dir)) {\n fs.mkdirSync(dir, { recursive: true });\n }\n\n db = new Database(resolvedPath);\n db.pragma('journal_mode = WAL');\n db.pragma('foreign_keys = ON');\n\n return db;\n}\n\nexport function createInMemoryDb(): Database.Database {\n const memDb = new Database(':memory:');\n memDb.pragma('foreign_keys = ON');\n return memDb;\n}\n\nexport function closeDb(): void {\n if (db) {\n db.close();\n db = null;\n }\n}\n\nexport function setDb(newDb: Database.Database): void {\n db = newDb;\n}\n","import type Database from 'better-sqlite3';\n\nexport function initSchema(db: Database.Database): void {\n db.exec(`\n CREATE TABLE IF NOT EXISTS agents (\n agent_id TEXT PRIMARY KEY,\n name TEXT NOT NULL,\n capabilities TEXT NOT NULL DEFAULT '[]',\n status TEXT NOT NULL DEFAULT 'IDLE'\n CHECK (status IN ('IDLE', 'BUSY', 'OFFLINE', 'OOM')),\n created_at TEXT NOT NULL DEFAULT (datetime('now'))\n );\n\n CREATE TABLE IF NOT EXISTS jobs (\n job_id TEXT PRIMARY KEY,\n original_prompt TEXT NOT NULL,\n openclaw_callback TEXT NOT NULL DEFAULT '',\n created_at TEXT NOT NULL DEFAULT (datetime('now'))\n );\n\n CREATE TABLE IF NOT EXISTS tasks (\n trace_id TEXT PRIMARY KEY,\n job_id TEXT NOT NULL REFERENCES jobs(job_id) ON DELETE CASCADE,\n assignee_id TEXT NOT NULL REFERENCES agents(agent_id),\n todo_description TEXT NOT NULL,\n deadline TEXT NOT NULL,\n payload TEXT NOT NULL DEFAULT '{}',\n status TEXT NOT NULL DEFAULT 'PENDING'\n CHECK (status IN ('PENDING', 'DISPATCHED', 'RESOLVED', 'OVERDUE')),\n result_data TEXT,\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n updated_at TEXT NOT NULL DEFAULT (datetime('now'))\n );\n\n CREATE INDEX IF NOT EXISTS idx_tasks_job_id ON tasks(job_id);\n CREATE INDEX IF NOT EXISTS idx_tasks_assignee_id ON tasks(assignee_id);\n CREATE INDEX IF NOT EXISTS idx_tasks_status ON tasks(status);\n\n CREATE TABLE IF NOT EXISTS config (\n key TEXT PRIMARY KEY,\n value TEXT NOT NULL\n );\n `);\n}\n","import { Router } from 'express';\nimport {\n listAgentsWithMetrics,\n createAgent,\n updateAgentStatus,\n deleteAgent,\n} from '../models/agent.js';\nimport { generateId } from '../utils/trace-id.js';\nimport type { AgentStatus } from '../models/types.js';\n\nconst router = Router();\n\n// GET /api/v1/nodes/status - Fleet status with metrics\nrouter.get('/status', (_req, res) => {\n const agents = listAgentsWithMetrics();\n res.json({\n total: agents.length,\n idle: agents.filter(a => a.status === 'IDLE').length,\n busy: agents.filter(a => a.status === 'BUSY').length,\n offline: agents.filter(a => a.status === 'OFFLINE').length,\n oom: agents.filter(a => a.status === 'OOM').length,\n agents,\n });\n});\n\n// POST /api/v1/nodes - Register a new agent\nrouter.post('/', (req, res) => {\n const { name, capabilities, status } = req.body;\n\n if (!name || !Array.isArray(capabilities)) {\n res.status(400).json({ error: 'name and capabilities[] are required' });\n return;\n }\n\n const agent = createAgent({\n agent_id: generateId('emp'),\n name,\n capabilities,\n status: (status as AgentStatus) ?? 'IDLE',\n });\n\n res.status(201).json(agent);\n});\n\n// PATCH /api/v1/nodes/:agent_id/status - Update agent status\nrouter.patch('/:agent_id/status', (req, res) => {\n const { agent_id } = req.params;\n const { status } = req.body;\n\n const validStatuses: AgentStatus[] = ['IDLE', 'BUSY', 'OFFLINE', 'OOM'];\n if (!validStatuses.includes(status)) {\n res.status(400).json({\n error: `Invalid status. Must be one of: ${validStatuses.join(', ')}`,\n });\n return;\n }\n\n const updated = updateAgentStatus(agent_id, status);\n if (!updated) {\n res.status(404).json({ error: `Agent not found: ${agent_id}` });\n return;\n }\n\n res.json({ agent_id, status });\n});\n\n// DELETE /api/v1/nodes/:agent_id\nrouter.delete('/:agent_id', (req, res) => {\n const { agent_id } = req.params;\n const deleted = deleteAgent(agent_id);\n if (!deleted) {\n res.status(404).json({ error: `Agent not found: ${agent_id}` });\n return;\n }\n res.json({ deleted: agent_id });\n});\n\nexport default router;\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport type {\n HumanAgent,\n AgentRow,\n AgentStatus,\n AgentWithMetrics,\n} from './types.js';\n\nfunction rowToAgent(row: AgentRow): HumanAgent {\n return {\n ...row,\n capabilities: JSON.parse(row.capabilities) as string[],\n };\n}\n\nexport function createAgent(\n agent: Omit<HumanAgent, 'created_at'>,\n db?: Database.Database\n): HumanAgent {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n conn\n .prepare(\n `INSERT INTO agents (agent_id, name, capabilities, status, created_at)\n VALUES (?, ?, ?, ?, ?)`\n )\n .run(\n agent.agent_id,\n agent.name,\n JSON.stringify(agent.capabilities),\n agent.status,\n now\n );\n return { ...agent, created_at: now };\n}\n\nexport function getAgent(\n agentId: string,\n db?: Database.Database\n): HumanAgent | undefined {\n const conn = db ?? getDb();\n const row = conn\n .prepare('SELECT * FROM agents WHERE agent_id = ?')\n .get(agentId) as AgentRow | undefined;\n return row ? rowToAgent(row) : undefined;\n}\n\nexport function listAgents(db?: Database.Database): HumanAgent[] {\n const conn = db ?? getDb();\n const rows = conn.prepare('SELECT * FROM agents ORDER BY created_at').all() as AgentRow[];\n return rows.map(rowToAgent);\n}\n\nexport function updateAgentStatus(\n agentId: string,\n status: AgentStatus,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const result = conn\n .prepare('UPDATE agents SET status = ? WHERE agent_id = ?')\n .run(status, agentId);\n return result.changes > 0;\n}\n\nexport function deleteAgent(\n agentId: string,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const result = conn\n .prepare('DELETE FROM agents WHERE agent_id = ?')\n .run(agentId);\n return result.changes > 0;\n}\n\nexport function listAgentsWithMetrics(\n db?: Database.Database\n): AgentWithMetrics[] {\n const conn = db ?? getDb();\n const rows = conn\n .prepare(\n `SELECT\n a.*,\n COUNT(CASE WHEN t.status IN ('DISPATCHED', 'PENDING') THEN 1 END) AS active_task_count,\n AVG(\n CASE WHEN t.status = 'RESOLVED'\n THEN (julianday(t.updated_at) - julianday(t.created_at)) * 24\n END\n ) AS avg_delivery_hours\n FROM agents a\n LEFT JOIN tasks t ON a.agent_id = t.assignee_id\n GROUP BY a.agent_id\n ORDER BY a.created_at`\n )\n .all() as (AgentRow & { active_task_count: number; avg_delivery_hours: number | null })[];\n\n return rows.map(row => ({\n ...rowToAgent(row),\n active_task_count: row.active_task_count,\n avg_delivery_hours: row.avg_delivery_hours\n ? Math.round(row.avg_delivery_hours * 100) / 100\n : null,\n }));\n}\n","import { nanoid } from 'nanoid';\n\nexport function generateTraceId(): string {\n const num = Math.floor(Math.random() * 10000)\n .toString()\n .padStart(4, '0');\n return `TK-${num}`;\n}\n\nexport function generateId(prefix: string): string {\n return `${prefix}_${nanoid(8)}`;\n}\n","import { Router } from 'express';\nimport { dispatchJob } from '../services/dispatch.js';\nimport { planJob } from '../services/planner.js';\nimport { listActiveJobs, getJobWithTasks } from '../models/job.js';\nimport { markOverdueTasks } from '../models/task.js';\nimport type { CreateJobRequest, PlanRequest } from '../models/types.js';\n\nconst router = Router();\n\n// POST /api/v1/jobs/create - Create and dispatch a new job\nrouter.post('/create', (req, res) => {\n const body = req.body as CreateJobRequest;\n\n if (!body.original_prompt || !Array.isArray(body.tasks) || body.tasks.length === 0) {\n res.status(400).json({\n error: 'original_prompt and non-empty tasks[] are required',\n });\n return;\n }\n\n for (const task of body.tasks) {\n if (!task.assignee_id || !task.todo_description || !task.deadline) {\n res.status(400).json({\n error: 'Each task requires assignee_id, todo_description, and deadline',\n });\n return;\n }\n }\n\n try {\n const job = dispatchJob(body);\n res.status(201).json(job);\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n res.status(400).json({ error: message });\n }\n});\n\n// GET /api/v1/jobs/active - List active jobs for kanban\nrouter.get('/active', (_req, res) => {\n // Mark overdue tasks before returning\n markOverdueTasks();\n const jobs = listActiveJobs();\n res.json({ jobs });\n});\n\n// POST /api/v1/jobs/plan - AI-powered task planning (does NOT dispatch)\nrouter.post('/plan', async (req, res) => {\n const body = req.body as PlanRequest;\n\n if (!body.prompt || typeof body.prompt !== 'string') {\n res.status(400).json({ error: 'prompt is required' });\n return;\n }\n\n try {\n const plan = await planJob(body);\n res.json(plan);\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n const status = message.includes('API Key') || message.includes('API key') ? 503 : 400;\n res.status(status).json({ error: message });\n }\n});\n\n// GET /api/v1/jobs/:job_id - Get a single job with tasks\nrouter.get('/:job_id', (req, res) => {\n const { job_id } = req.params;\n const job = getJobWithTasks(job_id);\n if (!job) {\n res.status(404).json({ error: `Job not found: ${job_id}` });\n return;\n }\n res.json(job);\n});\n\nexport default router;\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport type { HumanTask, TaskRow, TaskStatus } from './types.js';\n\nfunction rowToTask(row: TaskRow): HumanTask {\n return {\n ...row,\n payload: JSON.parse(row.payload) as Record<string, unknown>,\n result_data: row.result_data ? JSON.parse(row.result_data) : null,\n };\n}\n\nexport function createTask(\n task: Omit<HumanTask, 'created_at' | 'updated_at' | 'result_data'>,\n db?: Database.Database\n): HumanTask {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n conn\n .prepare(\n `INSERT INTO tasks (trace_id, job_id, assignee_id, todo_description, deadline, payload, status, created_at, updated_at)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`\n )\n .run(\n task.trace_id,\n task.job_id,\n task.assignee_id,\n task.todo_description,\n task.deadline,\n JSON.stringify(task.payload),\n task.status,\n now,\n now\n );\n return { ...task, result_data: null, created_at: now, updated_at: now };\n}\n\nexport function getTask(\n traceId: string,\n db?: Database.Database\n): HumanTask | undefined {\n const conn = db ?? getDb();\n const row = conn\n .prepare('SELECT * FROM tasks WHERE trace_id = ?')\n .get(traceId) as TaskRow | undefined;\n return row ? rowToTask(row) : undefined;\n}\n\nexport function listTasksByJob(\n jobId: string,\n db?: Database.Database\n): HumanTask[] {\n const conn = db ?? getDb();\n const rows = conn\n .prepare('SELECT * FROM tasks WHERE job_id = ? ORDER BY created_at')\n .all(jobId) as TaskRow[];\n return rows.map(rowToTask);\n}\n\nexport function listTasksByAssignee(\n assigneeId: string,\n db?: Database.Database\n): HumanTask[] {\n const conn = db ?? getDb();\n const rows = conn\n .prepare('SELECT * FROM tasks WHERE assignee_id = ? ORDER BY created_at')\n .all(assigneeId) as TaskRow[];\n return rows.map(rowToTask);\n}\n\nexport function updateTaskStatus(\n traceId: string,\n status: TaskStatus,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n const result = conn\n .prepare('UPDATE tasks SET status = ?, updated_at = ? WHERE trace_id = ?')\n .run(status, now, traceId);\n return result.changes > 0;\n}\n\nexport function resolveTask(\n traceId: string,\n resultData: unknown,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n const result = conn\n .prepare(\n `UPDATE tasks\n SET status = 'RESOLVED', result_data = ?, updated_at = ?\n WHERE trace_id = ? AND status IN ('DISPATCHED', 'OVERDUE')`\n )\n .run(JSON.stringify(resultData), now, traceId);\n return result.changes > 0;\n}\n\nexport function markOverdueTasks(db?: Database.Database): number {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n const result = conn\n .prepare(\n `UPDATE tasks\n SET status = 'OVERDUE', updated_at = ?\n WHERE status = 'DISPATCHED' AND deadline < ?`\n )\n .run(now, now);\n return result.changes;\n}\n\nexport function resetTaskDeadline(\n traceId: string,\n newDeadline: string,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n const result = conn\n .prepare(\n `UPDATE tasks\n SET status = 'DISPATCHED', deadline = ?, result_data = NULL, updated_at = ?\n WHERE trace_id = ?`\n )\n .run(newDeadline, now, traceId);\n return result.changes > 0;\n}\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport type { OrchestrationJob, JobRow, JobWithTasks } from './types.js';\nimport { listTasksByJob } from './task.js';\n\nexport function createJob(\n job: OrchestrationJob,\n db?: Database.Database\n): OrchestrationJob {\n const conn = db ?? getDb();\n conn\n .prepare(\n `INSERT INTO jobs (job_id, original_prompt, openclaw_callback, created_at)\n VALUES (?, ?, ?, ?)`\n )\n .run(job.job_id, job.original_prompt, job.openclaw_callback, job.created_at);\n return job;\n}\n\nexport function getJob(\n jobId: string,\n db?: Database.Database\n): OrchestrationJob | undefined {\n const conn = db ?? getDb();\n return conn\n .prepare('SELECT * FROM jobs WHERE job_id = ?')\n .get(jobId) as JobRow | undefined;\n}\n\nexport function getJobWithTasks(\n jobId: string,\n db?: Database.Database\n): JobWithTasks | undefined {\n const conn = db ?? getDb();\n const job = conn\n .prepare('SELECT * FROM jobs WHERE job_id = ?')\n .get(jobId) as JobRow | undefined;\n if (!job) return undefined;\n\n const tasks = listTasksByJob(jobId, conn);\n return { ...job, tasks };\n}\n\nexport function listActiveJobs(db?: Database.Database): JobWithTasks[] {\n const conn = db ?? getDb();\n const jobs = conn\n .prepare(\n `SELECT DISTINCT j.*\n FROM jobs j\n INNER JOIN tasks t ON j.job_id = t.job_id\n WHERE t.status != 'RESOLVED'\n ORDER BY j.created_at DESC`\n )\n .all() as JobRow[];\n\n // Also include jobs where all tasks are resolved but not yet synced\n const allJobs = conn\n .prepare('SELECT * FROM jobs ORDER BY created_at DESC')\n .all() as JobRow[];\n\n const activeJobIds = new Set(jobs.map(j => j.job_id));\n const result: JobWithTasks[] = [];\n\n for (const job of allJobs) {\n const tasks = listTasksByJob(job.job_id, conn);\n if (tasks.length > 0) {\n result.push({ ...job, tasks });\n }\n }\n\n return result;\n}\n\nexport function isJobComplete(\n jobId: string,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const row = conn\n .prepare(\n `SELECT COUNT(*) as total,\n SUM(CASE WHEN status = 'RESOLVED' THEN 1 ELSE 0 END) as resolved\n FROM tasks WHERE job_id = ?`\n )\n .get(jobId) as { total: number; resolved: number };\n return row.total > 0 && row.total === row.resolved;\n}\n\nexport function deleteJob(\n jobId: string,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const result = conn\n .prepare('DELETE FROM jobs WHERE job_id = ?')\n .run(jobId);\n return result.changes > 0;\n}\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport { createJob } from '../models/job.js';\nimport { createTask } from '../models/task.js';\nimport { getAgent, updateAgentStatus } from '../models/agent.js';\nimport { generateTraceId, generateId } from '../utils/trace-id.js';\nimport type { CreateJobRequest, JobWithTasks } from '../models/types.js';\n\nexport function dispatchJob(\n request: CreateJobRequest,\n db?: Database.Database\n): JobWithTasks {\n const conn = db ?? getDb();\n const jobId = generateId('job');\n const now = new Date().toISOString();\n\n // Validate all assignees exist\n for (const taskReq of request.tasks) {\n const agent = getAgent(taskReq.assignee_id, conn);\n if (!agent) {\n throw new Error(`Agent not found: ${taskReq.assignee_id}`);\n }\n if (agent.status === 'OFFLINE') {\n throw new Error(`Agent is offline: ${taskReq.assignee_id} (${agent.name})`);\n }\n }\n\n // Create the job\n const job = createJob(\n {\n job_id: jobId,\n original_prompt: request.original_prompt,\n openclaw_callback: request.openclaw_callback,\n created_at: now,\n },\n conn\n );\n\n // Create and dispatch all tasks\n const tasks = request.tasks.map(taskReq => {\n const traceId = generateTraceId();\n const task = createTask(\n {\n trace_id: traceId,\n job_id: jobId,\n assignee_id: taskReq.assignee_id,\n todo_description: taskReq.todo_description,\n deadline: taskReq.deadline,\n payload: taskReq.payload ?? {},\n status: 'DISPATCHED',\n },\n conn\n );\n\n // Mark the agent as busy\n updateAgentStatus(taskReq.assignee_id, 'BUSY', conn);\n\n return task;\n });\n\n return { ...job, tasks };\n}\n","import type { LlmProvider, LlmCompletionRequest, LlmCompletionResponse } from './types.js';\n\nconst DEFAULT_BASE_URL = 'https://api.anthropic.com';\n\nexport class ClaudeProvider implements LlmProvider {\n private apiKey: string;\n private model: string;\n private baseUrl: string;\n\n constructor(apiKey: string, model?: string, baseUrl?: string) {\n this.apiKey = apiKey;\n this.model = model || 'claude-sonnet-4-20250514';\n this.baseUrl = (baseUrl || DEFAULT_BASE_URL).replace(/\\/+$/, '');\n }\n\n async complete(request: LlmCompletionRequest): Promise<LlmCompletionResponse> {\n const systemMessages = request.messages.filter(m => m.role === 'system');\n const nonSystemMessages = request.messages.filter(m => m.role !== 'system');\n\n const body: Record<string, unknown> = {\n model: this.model,\n max_tokens: request.max_tokens || 4096,\n messages: nonSystemMessages.map(m => ({ role: m.role, content: m.content })),\n };\n\n if (request.temperature !== undefined) {\n body.temperature = request.temperature;\n }\n\n if (systemMessages.length > 0) {\n body.system = systemMessages.map(m => m.content).join('\\n\\n');\n }\n\n const response = await fetch(`${this.baseUrl}/v1/messages`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'x-api-key': this.apiKey,\n 'anthropic-version': '2023-06-01',\n },\n body: JSON.stringify(body),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n throw new Error(`Claude API error (${response.status}): ${errorText}`);\n }\n\n const data = await response.json() as { content: Array<{ type: string; text: string }> };\n const textBlock = data.content.find(c => c.type === 'text');\n\n if (!textBlock) {\n throw new Error('Claude API returned no text content');\n }\n\n return { content: textBlock.text };\n }\n}\n","import type { LlmProvider, LlmCompletionRequest, LlmCompletionResponse } from './types.js';\n\nconst DEFAULT_BASE_URL = 'https://api.openai.com';\n\nexport class OpenAIProvider implements LlmProvider {\n private apiKey: string;\n private model: string;\n private baseUrl: string;\n\n constructor(apiKey: string, model?: string, baseUrl?: string) {\n this.apiKey = apiKey;\n this.model = model || 'gpt-4o';\n this.baseUrl = (baseUrl || DEFAULT_BASE_URL).replace(/\\/+$/, '');\n }\n\n async complete(request: LlmCompletionRequest): Promise<LlmCompletionResponse> {\n const body: Record<string, unknown> = {\n model: this.model,\n max_tokens: request.max_tokens || 4096,\n messages: request.messages.map(m => ({ role: m.role, content: m.content })),\n };\n\n if (request.temperature !== undefined) {\n body.temperature = request.temperature;\n }\n\n const response = await fetch(`${this.baseUrl}/v1/chat/completions`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': `Bearer ${this.apiKey}`,\n },\n body: JSON.stringify(body),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n throw new Error(`OpenAI API error (${response.status}): ${errorText}`);\n }\n\n const data = await response.json() as {\n choices: Array<{ message: { content: string } }>;\n };\n\n const content = data.choices[0]?.message?.content;\n if (!content) {\n throw new Error('OpenAI API returned no content');\n }\n\n return { content };\n }\n}\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\n\nexport function getConfig(key: string, db?: Database.Database): string | undefined {\n const conn = db ?? getDb();\n const row = conn.prepare('SELECT value FROM config WHERE key = ?').get(key) as { value: string } | undefined;\n return row?.value;\n}\n\nexport function setConfig(key: string, value: string, db?: Database.Database): void {\n const conn = db ?? getDb();\n conn.prepare('INSERT OR REPLACE INTO config (key, value) VALUES (?, ?)').run(key, value);\n}\n\nexport function deleteConfig(key: string, db?: Database.Database): void {\n const conn = db ?? getDb();\n conn.prepare('DELETE FROM config WHERE key = ?').run(key);\n}\n\nexport function getAllConfig(db?: Database.Database): Record<string, string> {\n const conn = db ?? getDb();\n const rows = conn.prepare('SELECT key, value FROM config').all() as Array<{ key: string; value: string }>;\n const result: Record<string, string> = {};\n for (const row of rows) {\n result[row.key] = row.value;\n }\n return result;\n}\n","import type { LlmProvider, LlmProviderName } from './types.js';\nimport { ClaudeProvider } from './claude.js';\nimport { OpenAIProvider } from './openai.js';\nimport { getConfig } from '../models/config.js';\n\nexport interface LlmConfig {\n provider: LlmProviderName;\n apiKey: string;\n model?: string;\n baseUrl?: string;\n}\n\nexport function getLlmConfig(): LlmConfig {\n // DB config takes priority over env vars\n const dbProvider = getConfig('llm_provider');\n const dbApiKey = getConfig('llm_api_key');\n const dbModel = getConfig('llm_model');\n const dbBaseUrl = getConfig('llm_base_url');\n\n const provider = (dbProvider || process.env.HUMANCLAW_LLM_PROVIDER || 'claude') as LlmProviderName;\n const apiKey = dbApiKey || process.env.HUMANCLAW_LLM_API_KEY || '';\n const model = dbModel || process.env.HUMANCLAW_LLM_MODEL;\n const baseUrl = dbBaseUrl || process.env.HUMANCLAW_LLM_BASE_URL;\n\n if (!apiKey) {\n throw new Error(\n '未配置 LLM API Key。请在 Dashboard 设置中配置,或设置环境变量 HUMANCLAW_LLM_API_KEY。'\n );\n }\n\n if (provider !== 'claude' && provider !== 'openai') {\n throw new Error(`不支持的 LLM 提供商: ${provider}。支持: claude, openai`);\n }\n\n return { provider, apiKey, model: model || undefined, baseUrl: baseUrl || undefined };\n}\n\nexport function createLlmProvider(config?: LlmConfig): LlmProvider {\n const cfg = config || getLlmConfig();\n\n switch (cfg.provider) {\n case 'claude':\n return new ClaudeProvider(cfg.apiKey, cfg.model, cfg.baseUrl);\n case 'openai':\n return new OpenAIProvider(cfg.apiKey, cfg.model, cfg.baseUrl);\n }\n}\n\nexport type { LlmProvider, LlmProviderName } from './types.js';\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport { listAgentsWithMetrics, getAgent } from '../models/agent.js';\nimport { createLlmProvider } from '../llm/index.js';\nimport type { PlanRequest, PlanResponse, PlannedTask, AgentWithMetrics } from '../models/types.js';\nimport type { LlmProvider } from '../llm/types.js';\n\nfunction buildSystemPrompt(): string {\n return `你是 HumanClaw 任务编排规划器。你的工作是将用户的需求拆解为可以分发给物理节点(真实人类)执行的独立子任务。\n\n规则:\n1. 将需求拆解为扁平的、无依赖的子任务列表(每个任务可以独立执行)\n2. 根据每个 Agent 的技能(capabilities)和当前负载来匹配分配\n3. 为每个任务生成一段「话术」—— 这是直接发给该人类执行者的任务说明,语气自然、清晰、专业,包含具体的交付物要求\n4. 根据任务复杂度设置合理的截止时间(相对于当前时间)\n5. 每个 Agent 最多分配一个任务(除非人手不够)\n\n你必须严格输出以下 JSON 格式(不要输出任何其他内容):\n\n\\`\\`\\`json\n[\n {\n \"assignee_id\": \"agent的ID\",\n \"assignee_name\": \"agent的名字\",\n \"todo_description\": \"简短的任务标题/描述\",\n \"briefing\": \"话术:详细的任务说明,包括目标、交付物要求、注意事项。语气像一个靠谱的项目经理在给组员分配任务。\",\n \"deadline\": \"ISO 8601 时间\"\n }\n]\n\\`\\`\\``;\n}\n\nfunction buildUserPrompt(prompt: string, agents: AgentWithMetrics[]): string {\n const now = new Date();\n const agentList = agents.map(a => {\n const load = a.active_task_count > 0\n ? `当前有 ${a.active_task_count} 个进行中的任务`\n : '当前空闲';\n const speed = a.avg_delivery_hours !== null\n ? `平均交付时间 ${a.avg_delivery_hours}h`\n : '暂无历史数据';\n return `- ${a.name} (ID: ${a.agent_id}) 技能: [${a.capabilities.join(', ')}] ${load} ${speed}`;\n }).join('\\n');\n\n return `当前时间: ${now.toISOString()}\n\n可用的物理节点:\n${agentList}\n\n需求:\n${prompt}\n\n请根据以上信息拆解任务并分配。输出 JSON 数组。`;\n}\n\nfunction extractJson(raw: string): string {\n // Try to extract JSON from markdown code block\n const codeBlockMatch = raw.match(/```(?:json)?\\s*\\n?([\\s\\S]*?)\\n?\\s*```/);\n if (codeBlockMatch) {\n return codeBlockMatch[1].trim();\n }\n // Try to find a JSON array directly\n const arrayMatch = raw.match(/\\[[\\s\\S]*\\]/);\n if (arrayMatch) {\n return arrayMatch[0];\n }\n return raw.trim();\n}\n\nfunction validatePlannedTasks(\n parsed: unknown,\n validAgentIds: Set<string>,\n agents: AgentWithMetrics[]\n): PlannedTask[] {\n if (!Array.isArray(parsed)) {\n throw new Error('LLM 返回的不是有效的任务数组');\n }\n\n if (parsed.length === 0) {\n throw new Error('LLM 未生成任何任务');\n }\n\n return parsed.map((item: Record<string, unknown>, i: number) => {\n if (!item.todo_description || typeof item.todo_description !== 'string') {\n throw new Error(`任务 ${i + 1} 缺少 todo_description`);\n }\n if (!item.briefing || typeof item.briefing !== 'string') {\n throw new Error(`任务 ${i + 1} 缺少 briefing (话术)`);\n }\n\n // Validate or fallback assignee_id\n let assigneeId = String(item.assignee_id || '');\n let assigneeName = String(item.assignee_name || '');\n\n if (!validAgentIds.has(assigneeId)) {\n // Fallback to first available agent\n const fallback = agents[0];\n if (fallback) {\n assigneeId = fallback.agent_id;\n assigneeName = fallback.name;\n }\n }\n\n // Validate deadline or generate a default\n let deadline = String(item.deadline || '');\n if (!deadline || isNaN(Date.parse(deadline))) {\n // Default: 24 hours from now\n deadline = new Date(Date.now() + 24 * 60 * 60 * 1000).toISOString();\n }\n\n return {\n assignee_id: assigneeId,\n assignee_name: assigneeName,\n todo_description: String(item.todo_description),\n briefing: String(item.briefing),\n deadline,\n };\n });\n}\n\nexport async function planJob(\n request: PlanRequest,\n provider?: LlmProvider,\n db?: Database.Database\n): Promise<PlanResponse> {\n const conn = db ?? getDb();\n\n // Get available agents\n const allAgents = listAgentsWithMetrics(conn);\n let agents: AgentWithMetrics[];\n\n if (request.agent_ids && request.agent_ids.length > 0) {\n agents = allAgents.filter(a => request.agent_ids!.includes(a.agent_id));\n if (agents.length === 0) {\n throw new Error('指定的 Agent 均不存在');\n }\n } else {\n // Default: only IDLE agents\n agents = allAgents.filter(a => a.status === 'IDLE');\n if (agents.length === 0) {\n // Fallback: include BUSY agents too (but not OFFLINE/OOM)\n agents = allAgents.filter(a => a.status !== 'OFFLINE' && a.status !== 'OOM');\n }\n if (agents.length === 0) {\n throw new Error('没有可用的物理节点。请先在碳基算力池中添加节点。');\n }\n }\n\n // Get LLM provider\n const llm = provider ?? createLlmProvider();\n\n const systemPrompt = buildSystemPrompt();\n const userPrompt = buildUserPrompt(request.prompt, agents);\n\n const response = await llm.complete({\n messages: [\n { role: 'system', content: systemPrompt },\n { role: 'user', content: userPrompt },\n ],\n temperature: 0.3,\n max_tokens: 4096,\n });\n\n // Parse LLM response\n const jsonStr = extractJson(response.content);\n let parsed: unknown;\n try {\n parsed = JSON.parse(jsonStr);\n } catch {\n throw new Error('AI 返回的内容无法解析为 JSON,请重试');\n }\n\n const validIds = new Set(agents.map(a => a.agent_id));\n const plannedTasks = validatePlannedTasks(parsed, validIds, agents);\n\n return {\n original_prompt: request.prompt,\n planned_tasks: plannedTasks,\n };\n}\n","import { Router } from 'express';\nimport { resumeTask, rejectTask } from '../services/resume.js';\n\nconst router = Router();\n\n// POST /api/v1/tasks/resume - Submit result for a task\nrouter.post('/resume', (req, res) => {\n const { trace_id, result_data } = req.body;\n\n if (!trace_id) {\n res.status(400).json({ error: 'trace_id is required' });\n return;\n }\n\n if (result_data === undefined) {\n res.status(400).json({ error: 'result_data is required' });\n return;\n }\n\n try {\n const result = resumeTask(trace_id, result_data);\n res.json({\n task: result.task,\n job_complete: result.jobComplete,\n job: result.job ?? null,\n });\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n res.status(400).json({ error: message });\n }\n});\n\n// POST /api/v1/tasks/reject - Reject and retry a task\nrouter.post('/reject', (req, res) => {\n const { trace_id, new_deadline } = req.body;\n\n if (!trace_id) {\n res.status(400).json({ error: 'trace_id is required' });\n return;\n }\n\n try {\n const task = rejectTask(trace_id, new_deadline);\n res.json({ task });\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n res.status(400).json({ error: message });\n }\n});\n\nexport default router;\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport { getTask, resolveTask, resetTaskDeadline } from '../models/task.js';\nimport { isJobComplete, getJobWithTasks } from '../models/job.js';\nimport { updateAgentStatus } from '../models/agent.js';\nimport { listTasksByAssignee } from '../models/task.js';\nimport type { HumanTask, JobWithTasks } from '../models/types.js';\n\nexport interface ResumeResult {\n task: HumanTask;\n jobComplete: boolean;\n job: JobWithTasks | undefined;\n}\n\nexport function resumeTask(\n traceId: string,\n resultData: unknown,\n db?: Database.Database\n): ResumeResult {\n const conn = db ?? getDb();\n\n // Validate trace_id exists\n const task = getTask(traceId, conn);\n if (!task) {\n throw new Error(`Task not found: ${traceId}`);\n }\n\n if (task.status === 'RESOLVED') {\n throw new Error(`Task already resolved: ${traceId}`);\n }\n\n if (task.status === 'PENDING') {\n throw new Error(`Task not yet dispatched: ${traceId}`);\n }\n\n // Resolve the task\n const updated = resolveTask(traceId, resultData, conn);\n if (!updated) {\n throw new Error(`Failed to resolve task: ${traceId}`);\n }\n\n // Check if the agent has other active tasks; if not, mark IDLE\n const activeTasks = listTasksByAssignee(task.assignee_id, conn).filter(\n t => t.status === 'DISPATCHED' || t.status === 'PENDING'\n );\n if (activeTasks.length === 0) {\n updateAgentStatus(task.assignee_id, 'IDLE', conn);\n }\n\n // Check if the whole job is now complete\n const jobComplete = isJobComplete(task.job_id, conn);\n const job = jobComplete ? getJobWithTasks(task.job_id, conn) : undefined;\n\n const resolvedTask = getTask(traceId, conn)!;\n\n return { task: resolvedTask, jobComplete, job };\n}\n\nexport function rejectTask(\n traceId: string,\n newDeadline?: string,\n db?: Database.Database\n): HumanTask {\n const conn = db ?? getDb();\n\n const task = getTask(traceId, conn);\n if (!task) {\n throw new Error(`Task not found: ${traceId}`);\n }\n\n // Default: extend deadline by 24h from now\n const deadline =\n newDeadline ?? new Date(Date.now() + 24 * 60 * 60 * 1000).toISOString();\n\n resetTaskDeadline(traceId, deadline, conn);\n\n return getTask(traceId, conn)!;\n}\n","import { Router } from 'express';\nimport { aggregateJob, syncToOpenClaw } from '../services/aggregation.js';\n\nconst router = Router();\n\n// POST /api/v1/jobs/:job_id/sync - Aggregate and sync to OpenClaw\nrouter.post('/:job_id/sync', async (req, res) => {\n const { job_id } = req.params;\n\n try {\n const aggregation = aggregateJob(job_id);\n const syncResult = await syncToOpenClaw(aggregation);\n\n res.json({\n aggregation,\n sync: syncResult,\n });\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n res.status(400).json({ error: message });\n }\n});\n\nexport default router;\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport { getJobWithTasks, isJobComplete } from '../models/job.js';\nimport type { JobWithTasks } from '../models/types.js';\n\nexport interface AggregationResult {\n job_id: string;\n original_prompt: string;\n openclaw_callback: string;\n results: Array<{\n trace_id: string;\n assignee_id: string;\n todo_description: string;\n result_data: unknown;\n }>;\n aggregated_at: string;\n}\n\nexport function aggregateJob(\n jobId: string,\n db?: Database.Database\n): AggregationResult {\n const conn = db ?? getDb();\n\n const job = getJobWithTasks(jobId, conn);\n if (!job) {\n throw new Error(`Job not found: ${jobId}`);\n }\n\n if (!isJobComplete(jobId, conn)) {\n const resolved = job.tasks.filter(t => t.status === 'RESOLVED').length;\n throw new Error(\n `Job not complete: ${resolved}/${job.tasks.length} tasks resolved`\n );\n }\n\n return {\n job_id: job.job_id,\n original_prompt: job.original_prompt,\n openclaw_callback: job.openclaw_callback,\n results: job.tasks.map(t => ({\n trace_id: t.trace_id,\n assignee_id: t.assignee_id,\n todo_description: t.todo_description,\n result_data: t.result_data,\n })),\n aggregated_at: new Date().toISOString(),\n };\n}\n\nexport async function syncToOpenClaw(\n aggregation: AggregationResult\n): Promise<{ success: boolean; message: string }> {\n if (!aggregation.openclaw_callback) {\n return {\n success: true,\n message: 'No OpenClaw callback configured, skipping sync',\n };\n }\n\n try {\n const response = await fetch(aggregation.openclaw_callback, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(aggregation),\n });\n\n if (!response.ok) {\n return {\n success: false,\n message: `OpenClaw sync failed: ${response.status} ${response.statusText}`,\n };\n }\n\n return { success: true, message: 'Synced to OpenClaw successfully' };\n } catch (error) {\n const message =\n error instanceof Error ? error.message : 'Unknown error';\n return { success: false, message: `OpenClaw sync error: ${message}` };\n }\n}\n","import { Router } from 'express';\nimport { getConfig, setConfig, deleteConfig } from '../models/config.js';\n\nconst router = Router();\n\n// GET /api/v1/config - Get LLM config (masks API key)\nrouter.get('/', (_req, res) => {\n const provider = getConfig('llm_provider') || process.env.HUMANCLAW_LLM_PROVIDER || 'claude';\n const apiKey = getConfig('llm_api_key') || process.env.HUMANCLAW_LLM_API_KEY || '';\n const model = getConfig('llm_model') || process.env.HUMANCLAW_LLM_MODEL || '';\n const baseUrl = getConfig('llm_base_url') || process.env.HUMANCLAW_LLM_BASE_URL || '';\n\n const keySource = getConfig('llm_api_key') ? 'dashboard' : (process.env.HUMANCLAW_LLM_API_KEY ? 'env' : 'none');\n\n res.json({\n provider,\n api_key_set: apiKey.length > 0,\n api_key_masked: apiKey ? apiKey.slice(0, 8) + '...' + apiKey.slice(-4) : '',\n api_key_source: keySource,\n model,\n base_url: baseUrl,\n });\n});\n\n// PUT /api/v1/config - Update LLM config\nrouter.put('/', (req, res) => {\n const { provider, api_key, model, base_url } = req.body as {\n provider?: string;\n api_key?: string;\n model?: string;\n base_url?: string;\n };\n\n if (provider !== undefined) {\n if (provider !== 'claude' && provider !== 'openai') {\n res.status(400).json({ error: '不支持的提供商,支持: claude, openai' });\n return;\n }\n setConfig('llm_provider', provider);\n }\n\n if (api_key !== undefined) {\n if (api_key === '') {\n deleteConfig('llm_api_key');\n } else {\n setConfig('llm_api_key', api_key);\n }\n }\n\n if (model !== undefined) {\n if (model === '') {\n deleteConfig('llm_model');\n } else {\n setConfig('llm_model', model);\n }\n }\n\n if (base_url !== undefined) {\n if (base_url === '') {\n deleteConfig('llm_base_url');\n } else {\n setConfig('llm_base_url', base_url);\n }\n }\n\n res.json({ ok: true });\n});\n\nexport default router;\n","export function getDashboardHtml(): string {\n return `<!DOCTYPE html>\n<html lang=\"zh-CN\">\n<head>\n<meta charset=\"UTF-8\"/>\n<meta name=\"viewport\" content=\"width=device-width,initial-scale=1.0\"/>\n<title>HumanClaw - 异步物理节点编排</title>\n<style>\n:root{--bg:#0f1117;--surface:#1a1d27;--surface-hover:#242836;--border:#2e3346;--text:#e2e4ed;--text-dim:#7b8196;--accent:#00d4ff;--accent-dim:rgba(0,212,255,.12);--green:#22c55e;--yellow:#eab308;--red:#ef4444;--purple:#a855f7;--font-mono:'SF Mono','JetBrains Mono','Fira Code',monospace;--font-sans:-apple-system,'Inter','Segoe UI',sans-serif;--radius:10px}\n*{margin:0;padding:0;box-sizing:border-box}\nbody{background:var(--bg);color:var(--text);font-family:var(--font-sans);min-height:100vh;line-height:1.5}\nheader{padding:20px 32px 12px;border-bottom:1px solid var(--border);display:flex;align-items:center;gap:14px}\nheader h1{font-family:var(--font-mono);font-size:22px;color:var(--accent);letter-spacing:1.5px}\n.subtitle{color:var(--text-dim);font-size:12px;border-left:1px solid var(--border);padding-left:14px}\n.hdr-spacer{flex:1}\n.gear-btn{background:none;border:1px solid var(--border);color:var(--text-dim);width:34px;height:34px;border-radius:8px;cursor:pointer;font-size:18px;display:flex;align-items:center;justify-content:center;transition:all .15s}\n.gear-btn:hover{color:var(--accent);border-color:var(--accent);background:var(--accent-dim)}\nnav{display:flex;border-bottom:1px solid var(--border);padding:0 32px;gap:4px}\n.tab{background:none;border:none;color:var(--text-dim);padding:12px 18px;font-size:13px;cursor:pointer;border-bottom:2px solid transparent;transition:all .15s;font-family:var(--font-sans);font-weight:500}\n.tab:hover{color:var(--text);background:var(--surface)}\n.tab.active{color:var(--accent);border-bottom-color:var(--accent)}\nmain{padding:24px 32px;max-width:1200px}\n.panel{display:none}.panel.active{display:block}\n/* Cards */\n.card{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:20px;transition:border-color .15s}\n.card:hover{border-color:color-mix(in srgb,var(--accent) 40%,transparent)}\n.grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(260px,1fr));gap:14px;margin-top:16px}\n/* Agent Card */\n.agent-header{display:flex;align-items:center;gap:10px;margin-bottom:10px}\n.dot{width:10px;height:10px;border-radius:50%;flex-shrink:0}\n.dot.IDLE{background:var(--green);box-shadow:0 0 8px var(--green)}.dot.BUSY{background:var(--yellow);box-shadow:0 0 8px var(--yellow)}.dot.OFFLINE{background:var(--red);box-shadow:0 0 6px var(--red)}.dot.OOM{background:var(--purple);box-shadow:0 0 8px var(--purple)}\n.agent-name{font-weight:600;font-size:15px}\n.agent-id{color:var(--text-dim);font-family:var(--font-mono);font-size:11px;margin-bottom:8px}\n.caps{display:flex;flex-wrap:wrap;gap:5px;margin-top:6px}\n.cap{background:var(--accent-dim);border:1px solid color-mix(in srgb,var(--accent) 25%,transparent);border-radius:4px;padding:1px 8px;font-size:11px;color:var(--accent)}\n.agent-meta{margin-top:10px;font-size:11px;color:var(--text-dim);display:flex;gap:12px}\n.agent-actions{margin-top:12px;display:flex;gap:6px}\n.agent-actions select{background:var(--surface-hover);color:var(--text);border:1px solid var(--border);border-radius:6px;padding:4px 8px;font-size:11px;cursor:pointer;outline:none}\n.agent-actions button{background:var(--red);color:#fff;border:none;border-radius:6px;padding:4px 10px;font-size:11px;cursor:pointer}\n.agent-actions button:hover{opacity:.8}\n/* Stats Bar */\n.stats{display:flex;gap:12px;flex-wrap:wrap}\n.stat{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:14px 22px;text-align:center;min-width:90px}\n.stat-val{font-size:28px;font-weight:700;font-family:var(--font-mono)}\n.stat-lbl{font-size:10px;color:var(--text-dim);text-transform:uppercase;letter-spacing:1px;margin-top:2px}\n/* Forms */\n.form-card{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:24px;margin-top:16px;max-width:620px}\n.form-card h3{font-size:15px;color:var(--accent);font-family:var(--font-mono);margin-bottom:16px}\n.fg{margin-bottom:14px}\n.fg label{display:block;font-size:12px;color:var(--text-dim);margin-bottom:5px;font-weight:500}\n.fg input,.fg textarea,.fg select{width:100%;background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:10px 12px;color:var(--text);font-size:13px;font-family:var(--font-sans);outline:none;transition:border-color .15s}\n.fg input:focus,.fg textarea:focus,.fg select:focus{border-color:var(--accent)}\n.fg textarea{min-height:80px;resize:vertical;font-family:var(--font-mono);font-size:12px}\n.fg .hint{font-size:11px;color:var(--text-dim);margin-top:4px}\n.btn{padding:10px 22px;border:none;border-radius:8px;font-size:13px;font-weight:600;cursor:pointer;transition:all .15s;display:inline-flex;align-items:center;gap:6px}\n.btn:hover{opacity:.85;transform:translateY(-1px)}\n.btn:active{transform:translateY(0)}\n.btn:disabled{opacity:.5;cursor:not-allowed;transform:none}\n.btn-primary{background:var(--accent);color:var(--bg)}\n.btn-green{background:var(--green);color:#fff}\n.btn-danger{background:var(--red);color:#fff}\n.btn-ghost{background:transparent;color:var(--accent);border:1px solid var(--border)}\n.btn-ghost:hover{background:var(--accent-dim)}\n.btn-sm{padding:6px 14px;font-size:12px}\n.btn-group{display:flex;gap:10px;margin-top:16px;flex-wrap:wrap}\n/* Section header */\n.section-hd{display:flex;justify-content:space-between;align-items:center;margin-bottom:4px}\n.section-hd h2{font-size:16px;font-weight:600}\n/* Job card */\n.job-card{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:20px;margin-top:14px}\n.job-hd{display:flex;justify-content:space-between;align-items:center;margin-bottom:10px}\n.job-title{font-weight:600;font-size:14px}\n.job-id{font-family:var(--font-mono);font-size:11px;color:var(--text-dim)}\n.pbar-wrap{margin:6px 0 14px}\n.pbar-label{font-size:11px;color:var(--text-dim);margin-bottom:4px}\n.pbar{background:var(--surface-hover);border-radius:4px;height:6px;overflow:hidden}\n.pbar-fill{background:var(--accent);height:100%;border-radius:4px;transition:width .3s}\n.kanban{display:grid;grid-template-columns:1fr 1fr 1fr;gap:10px}\n.lane{min-height:40px}\n.lane-hd{font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:1px;margin-bottom:6px;padding-bottom:4px;border-bottom:2px solid var(--border)}\n.lane-hd.y{color:var(--yellow);border-color:var(--yellow)}.lane-hd.r{color:var(--red);border-color:var(--red)}.lane-hd.g{color:var(--green);border-color:var(--green)}\n.tcard{background:var(--bg);border:1px solid var(--border);border-radius:6px;padding:8px 10px;margin-bottom:6px;font-size:12px}\n.tcard-trace{font-family:var(--font-mono);font-size:10px;color:var(--accent);margin-bottom:3px}\n.tcard-meta{font-size:10px;color:var(--text-dim);margin-top:3px}\n/* Toast */\n.toast{position:fixed;bottom:24px;right:24px;padding:12px 20px;border-radius:8px;font-size:13px;z-index:9999;animation:slide-up .25s ease;font-weight:500;box-shadow:0 8px 24px rgba(0,0,0,.4)}\n.toast.ok{background:var(--green);color:#fff}.toast.err{background:var(--red);color:#fff}\n@keyframes slide-up{from{transform:translateY(16px);opacity:0}to{transform:translateY(0);opacity:1}}\n/* Empty state */\n.empty-state{text-align:center;padding:48px 20px;color:var(--text-dim)}\n.empty-state .icon{font-size:48px;margin-bottom:12px;opacity:.6}\n.empty-state p{font-size:14px;margin-bottom:16px;max-width:360px;margin-left:auto;margin-right:auto;line-height:1.6}\n/* Modal/Overlay */\n.overlay{position:fixed;inset:0;background:rgba(0,0,0,.6);z-index:100;display:flex;align-items:flex-start;justify-content:center;padding-top:60px;animation:fade-in .15s ease}\n.overlay .form-card{max-width:620px;width:100%;max-height:85vh;overflow-y:auto;margin:0}\n@keyframes fade-in{from{opacity:0}to{opacity:1}}\n/* Agent chips for AI planning */\n.chip-bar{display:flex;gap:6px;margin-bottom:10px;flex-wrap:wrap}\n.chip-filter{background:var(--surface-hover);color:var(--text-dim);border:1px solid var(--border);border-radius:14px;padding:3px 12px;font-size:11px;cursor:pointer;transition:all .15s}\n.chip-filter.active{background:var(--accent-dim);color:var(--accent);border-color:var(--accent)}\n.agent-chips{display:flex;flex-wrap:wrap;gap:6px;margin-top:8px}\n.achip{background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:6px 12px;font-size:12px;cursor:pointer;transition:all .15s;display:flex;align-items:center;gap:6px;user-select:none}\n.achip:hover{border-color:var(--text-dim)}\n.achip.selected{border-color:var(--accent);background:var(--accent-dim)}\n.achip .adot{width:8px;height:8px;border-radius:50%;flex-shrink:0}\n.achip .adot.IDLE{background:var(--green)}.achip .adot.BUSY{background:var(--yellow)}.achip .adot.OFFLINE{background:var(--red)}.achip .adot.OOM{background:var(--purple)}\n/* Plan preview cards */\n.plan-card{background:var(--bg);border:1px solid var(--border);border-radius:var(--radius);padding:16px;margin-bottom:12px}\n.plan-card-hd{display:flex;align-items:center;gap:8px;margin-bottom:8px}\n.plan-card-hd .adot{width:8px;height:8px;border-radius:50%}\n.plan-card-agent{font-weight:600;font-size:13px}\n.plan-card-agentid{font-family:var(--font-mono);font-size:10px;color:var(--text-dim)}\n.plan-card-desc{font-size:13px;margin-bottom:10px;font-weight:500}\n.briefing-box{background:var(--surface);border-left:3px solid var(--accent);border-radius:0 6px 6px 0;padding:10px 14px;font-size:12px;line-height:1.6;color:var(--text);margin-bottom:8px;position:relative}\n.briefing-label{font-size:10px;color:var(--accent);font-weight:600;margin-bottom:4px;font-family:var(--font-mono)}\n.briefing-copy{position:absolute;top:8px;right:8px;background:var(--surface-hover);border:1px solid var(--border);color:var(--text-dim);border-radius:4px;padding:2px 8px;font-size:10px;cursor:pointer}\n.briefing-copy:hover{color:var(--accent);border-color:var(--accent)}\n.plan-card-dl{font-size:11px;color:var(--text-dim);font-family:var(--font-mono)}\n/* Spinner */\n.spinner-wrap{text-align:center;padding:40px 20px}\n.spinner{display:inline-block;width:28px;height:28px;border:3px solid var(--border);border-top-color:var(--accent);border-radius:50%;animation:spin .6s linear infinite}\n@keyframes spin{to{transform:rotate(360deg)}}\n.spinner-text{color:var(--text-dim);font-size:13px;margin-top:12px}\n/* Manual task row */\n.task-row{background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:14px;margin-bottom:10px;position:relative}\n.task-row .remove-task{position:absolute;top:8px;right:10px;background:none;border:none;color:var(--red);cursor:pointer;font-size:16px;line-height:1}\n.task-row .fg{margin-bottom:10px}\n.task-row .fg:last-child{margin-bottom:0}\n</style>\n</head>\n<body>\n<header>\n <h1>HumanClaw</h1>\n <span class=\"subtitle\">异步物理节点编排框架</span>\n <span class=\"hdr-spacer\"></span>\n <button class=\"gear-btn\" onclick=\"showSettings()\" title=\"设置\">&#9881;</button>\n</header>\n<nav>\n <button class=\"tab active\" data-panel=\"fleet\">碳基算力池</button>\n <button class=\"tab\" data-panel=\"pipeline\">编排大盘</button>\n <button class=\"tab\" data-panel=\"terminal\">I/O 终端</button>\n</nav>\n<main>\n <section id=\"fleet\" class=\"panel active\"></section>\n <section id=\"pipeline\" class=\"panel\"></section>\n <section id=\"terminal\" class=\"panel\"></section>\n</main>\n\n<script>\nconst API='/api/v1';\nfunction esc(s){const d=document.createElement('div');d.textContent=s;return d.innerHTML}\nfunction toast(msg,ok){\n document.querySelectorAll('.toast').forEach(t=>t.remove());\n const t=document.createElement('div');t.className='toast '+(ok?'ok':'err');t.textContent=msg;document.body.appendChild(t);\n setTimeout(()=>t.remove(),3500);\n}\nlet cachedAgents=[];\nlet currentPlan=null;\nlet selectedAgentIds=new Set();\n\n// ═══════════════════════════════════════════════\n// FLEET\n// ═══════════════════════════════════════════════\nasync function loadFleet(el){\n el.innerHTML='<div class=\"empty-state\"><p>加载中...</p></div>';\n try{\n const r=await fetch(API+'/nodes/status');\n const d=await r.json();\n cachedAgents=d.agents||[];\n let h='<div class=\"section-hd\"><h2>碳基算力池</h2><button class=\"btn btn-primary btn-sm\" onclick=\"showAddAgent()\">+ 添加节点</button></div>';\n h+='<div class=\"stats\">';\n h+='<div class=\"stat\"><div class=\"stat-val\">'+d.total+'</div><div class=\"stat-lbl\">总计</div></div>';\n h+='<div class=\"stat\"><div class=\"stat-val\" style=\"color:var(--green)\">'+d.idle+'</div><div class=\"stat-lbl\">空闲</div></div>';\n h+='<div class=\"stat\"><div class=\"stat-val\" style=\"color:var(--yellow)\">'+d.busy+'</div><div class=\"stat-lbl\">忙碌</div></div>';\n h+='<div class=\"stat\"><div class=\"stat-val\" style=\"color:var(--red)\">'+d.offline+'</div><div class=\"stat-lbl\">离线</div></div>';\n h+='<div class=\"stat\"><div class=\"stat-val\" style=\"color:var(--purple)\">'+d.oom+'</div><div class=\"stat-lbl\">崩溃</div></div>';\n h+='</div>';\n if(!d.agents.length){\n h+='<div class=\"empty-state\" style=\"margin-top:32px\"><div class=\"icon\">👤</div><p>还没有物理节点。点击上方「+ 添加节点」注册你的第一个碳基算力单元。</p></div>';\n el.innerHTML=h;return;\n }\n h+='<div class=\"grid\">';\n for(const a of d.agents){\n h+='<div class=\"card\"><div class=\"agent-header\"><span class=\"dot '+a.status+'\"></span><span class=\"agent-name\">'+esc(a.name)+'</span></div>';\n h+='<div class=\"agent-id\">'+a.agent_id+'</div>';\n h+='<div class=\"caps\">';for(const c of a.capabilities)h+='<span class=\"cap\">'+esc(c)+'</span>';h+='</div>';\n h+='<div class=\"agent-meta\"><span>任务: '+a.active_task_count+'</span>';\n if(a.avg_delivery_hours!==null)h+='<span>平均交付: '+a.avg_delivery_hours+'h</span>';\n h+='</div>';\n h+='<div class=\"agent-actions\">';\n h+='<select onchange=\"changeStatus(\\\\''+a.agent_id+'\\\\',this.value)\">';\n for(const s of ['IDLE','BUSY','OFFLINE','OOM'])h+='<option'+(s===a.status?' selected':'')+'>'+s+'</option>';\n h+='</select>';\n h+='<button onclick=\"deleteAgent(\\\\''+a.agent_id+'\\\\')\">删除</button>';\n h+='</div></div>';\n }\n h+='</div>';\n el.innerHTML=h;\n }catch(e){el.innerHTML='<div class=\"empty-state\"><p>加载失败: '+e.message+'</p></div>'}\n}\nwindow.showAddAgent=function(){\n const ov=document.createElement('div');ov.className='overlay';ov.id='overlay';\n ov.addEventListener('click',e=>{if(e.target===ov)ov.remove()});\n ov.innerHTML='<div class=\"form-card\"><h3>+ 添加物理节点</h3>'\n +'<div class=\"fg\"><label>节点名称</label><input id=\"aa-name\" placeholder=\"例: 前端老李\"/></div>'\n +'<div class=\"fg\"><label>技能标签</label><input id=\"aa-caps\" placeholder=\"例: UI/UX, 前端开发, 抗压能力强\"/><div class=\"hint\">多个标签用逗号分隔</div></div>'\n +'<div class=\"btn-group\"><button class=\"btn btn-primary\" onclick=\"submitAgent()\">注册节点</button><button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button></div></div>';\n document.body.appendChild(ov);\n document.getElementById('aa-name').focus();\n};\nwindow.submitAgent=async function(){\n const name=document.getElementById('aa-name').value.trim();\n const caps=document.getElementById('aa-caps').value.trim();\n if(!name){toast('请输入节点名称',false);return}\n if(!caps){toast('请输入至少一个技能标签',false);return}\n try{\n const r=await fetch(API+'/nodes',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({name,capabilities:caps.split(',').map(s=>s.trim()).filter(Boolean)})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'注册失败',false);return}\n toast('节点 '+d.agent_id+' 注册成功!',true);\n document.getElementById('overlay').remove();\n load('fleet');\n }catch{toast('网络错误',false)}\n};\nwindow.changeStatus=async function(id,status){\n try{\n const r=await fetch(API+'/nodes/'+id+'/status',{method:'PATCH',headers:{'Content-Type':'application/json'},body:JSON.stringify({status})});\n if(!r.ok){const d=await r.json();toast(d.error||'更新失败',false);return}\n toast('状态已更新',true);\n }catch{toast('网络错误',false)}\n};\nwindow.deleteAgent=async function(id){\n if(!confirm('确定删除节点 '+id+'?'))return;\n try{\n const r=await fetch(API+'/nodes/'+id,{method:'DELETE'});\n if(!r.ok){const d=await r.json();toast(d.error||'删除失败',false);return}\n toast('节点已删除',true);load('fleet');\n }catch{toast('网络错误',false)}\n};\n\n// ═══════════════════════════════════════════════\n// PIPELINE\n// ═══════════════════════════════════════════════\nfunction tcard(t){\n return '<div class=\"tcard\"><div class=\"tcard-trace\">'+t.trace_id+'</div><div>'+esc(t.todo_description)+'</div><div class=\"tcard-meta\">'+t.assignee_id+' | '+new Date(t.deadline).toLocaleString()+'</div></div>';\n}\nasync function loadPipeline(el){\n el.innerHTML='<div class=\"empty-state\"><p>加载中...</p></div>';\n try{\n try{const ar=await fetch(API+'/nodes/status');const ad=await ar.json();cachedAgents=ad.agents||[]}catch{}\n const r=await fetch(API+'/jobs/active');\n const d=await r.json();\n let h='<div class=\"section-hd\"><h2>异步编排大盘</h2><button class=\"btn btn-primary btn-sm\" onclick=\"showCreateJob()\">+ 创建任务</button></div>';\n if(!d.jobs.length){\n h+='<div class=\"empty-state\" style=\"margin-top:32px\"><div class=\"icon\">📋</div><p>暂无进行中的任务。点击上方「+ 创建任务」,输入需求后 AI 自动规划分发。';\n if(!cachedAgents.length)h+='<br/><br/>⚠️ 需要先在「碳基算力池」中添加物理节点。';\n h+='</p></div>';\n el.innerHTML=h;return;\n }\n for(const j of d.jobs){\n const res=j.tasks.filter(t=>t.status==='RESOLVED').length;\n const tot=j.tasks.length;\n const pct=tot?Math.round(res/tot*100):0;\n const dispatched=j.tasks.filter(t=>t.status==='DISPATCHED'||t.status==='PENDING');\n const overdue=j.tasks.filter(t=>t.status==='OVERDUE');\n const done=j.tasks.filter(t=>t.status==='RESOLVED');\n h+='<div class=\"job-card\"><div class=\"job-hd\"><span class=\"job-title\">'+esc(j.original_prompt)+'</span><span class=\"job-id\">'+j.job_id+'</span></div>';\n h+='<div class=\"pbar-wrap\"><div class=\"pbar-label\">'+res+'/'+tot+' 已完成 ('+pct+'%)</div><div class=\"pbar\"><div class=\"pbar-fill\" style=\"width:'+pct+'%\"></div></div></div>';\n if(pct===100)h+='<div style=\"margin-bottom:12px\"><button class=\"btn btn-green btn-sm\" onclick=\"syncJob(\\\\''+j.job_id+'\\\\')\">聚合并同步到 OpenClaw</button></div>';\n h+='<div class=\"kanban\"><div class=\"lane\"><div class=\"lane-hd y\">已分发 ('+dispatched.length+')</div>'+dispatched.map(tcard).join('')+'</div>';\n h+='<div class=\"lane\"><div class=\"lane-hd r\">已超时 ('+overdue.length+')</div>'+overdue.map(tcard).join('')+'</div>';\n h+='<div class=\"lane\"><div class=\"lane-hd g\">已交付 ('+done.length+')</div>'+done.map(tcard).join('')+'</div></div></div>';\n }\n el.innerHTML=h;\n }catch(e){el.innerHTML='<div class=\"empty-state\"><p>加载失败: '+e.message+'</p></div>'}\n}\n\n// ─── AI Planning Flow ────────────────────────\nwindow.showCreateJob=function(){\n if(!cachedAgents.length){toast('请先在「碳基算力池」中添加至少一个物理节点',false);return}\n currentPlan=null;\n // Pre-select all IDLE agents\n selectedAgentIds=new Set(cachedAgents.filter(a=>a.status==='IDLE').map(a=>a.agent_id));\n const ov=document.createElement('div');ov.className='overlay';ov.id='overlay';\n ov.addEventListener('click',e=>{if(e.target===ov)ov.remove()});\n renderPlanStep1(ov);\n document.body.appendChild(ov);\n};\n\nfunction renderPlanStep1(ov){\n let chipFilter='all';\n function renderChips(filter){\n chipFilter=filter;\n const filtered=filter==='idle'?cachedAgents.filter(a=>a.status==='IDLE'):cachedAgents;\n let ch='<div class=\"chip-bar\">';\n ch+='<span class=\"chip-filter'+(filter==='all'?' active':'')+'\" onclick=\"window._filterChips(\\\\'all\\\\')\">全部</span>';\n ch+='<span class=\"chip-filter'+(filter==='idle'?' active':'')+'\" onclick=\"window._filterChips(\\\\'idle\\\\')\">仅空闲</span>';\n ch+='</div>';\n ch+='<div class=\"agent-chips\">';\n for(const a of filtered){\n const sel=selectedAgentIds.has(a.agent_id);\n ch+='<div class=\"achip'+(sel?' selected':'')+'\" onclick=\"window._toggleChip(\\\\''+a.agent_id+'\\\\')\">';\n ch+='<span class=\"adot '+a.status+'\"></span>';\n ch+=esc(a.name);\n ch+='</div>';\n }\n ch+='</div>';\n return ch;\n }\n window._filterChips=function(f){\n chipFilter=f;\n const el=document.getElementById('agent-chip-area');\n if(el)el.innerHTML=renderChips(f);\n };\n window._toggleChip=function(id){\n if(selectedAgentIds.has(id))selectedAgentIds.delete(id);\n else selectedAgentIds.add(id);\n const el=document.getElementById('agent-chip-area');\n if(el)el.innerHTML=renderChips(chipFilter);\n };\n\n ov.innerHTML='<div class=\"form-card\"><h3>AI 智能规划</h3>'\n +'<div class=\"fg\"><label>输入你的需求</label><textarea id=\"plan-prompt\" rows=\"3\" placeholder=\"例: 完成首页重构,包括导航栏、内容区和页脚的响应式改版\" style=\"font-family:var(--font-sans);font-size:13px\"></textarea></div>'\n +'<div class=\"fg\"><label>选择参与的物理节点 <span style=\"color:var(--text-dim);font-weight:400\">(默认选中空闲节点)</span></label>'\n +'<div id=\"agent-chip-area\">'+renderChips('all')+'</div></div>'\n +'<div class=\"fg\"><label>OpenClaw 回调地址 <span style=\"color:var(--text-dim);font-weight:400\">(可选)</span></label><input id=\"plan-callback\" placeholder=\"https://...\"/></div>'\n +'<div class=\"btn-group\">'\n +'<button class=\"btn btn-primary\" onclick=\"planWithAI()\">AI 规划</button>'\n +'<button class=\"btn btn-ghost\" onclick=\"showManualCreate()\">手动创建</button>'\n +'<button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button>'\n +'</div></div>';\n setTimeout(()=>{const ta=document.getElementById('plan-prompt');if(ta)ta.focus()},50);\n};\n\nwindow.planWithAI=async function(){\n const prompt=document.getElementById('plan-prompt').value.trim();\n const callback=document.getElementById('plan-callback')?.value.trim()||'';\n if(!prompt){toast('请输入需求描述',false);return}\n if(selectedAgentIds.size===0){toast('请至少选择一个物理节点',false);return}\n\n const ov=document.getElementById('overlay');\n const fc=ov.querySelector('.form-card');\n fc.innerHTML='<h3>AI 智能规划</h3><div class=\"spinner-wrap\"><div class=\"spinner\"></div><div class=\"spinner-text\">AI 正在分析需求、匹配节点、生成话术...</div></div>';\n\n try{\n const r=await fetch(API+'/jobs/plan',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({prompt,agent_ids:Array.from(selectedAgentIds)})});\n const d=await r.json();\n if(!r.ok){\n toast(d.error||'规划失败',false);\n renderPlanStep1(ov);\n return;\n }\n currentPlan={...d,openclaw_callback:callback};\n renderPlanStep2(ov);\n }catch(e){\n toast('网络错误: '+e.message,false);\n renderPlanStep1(ov);\n }\n};\n\nfunction renderPlanStep2(ov){\n const p=currentPlan;\n let h='<h3>规划预览</h3>';\n h+='<div style=\"font-size:12px;color:var(--text-dim);margin-bottom:14px\">需求: '+esc(p.original_prompt)+'</div>';\n for(let i=0;i<p.planned_tasks.length;i++){\n const t=p.planned_tasks[i];\n const agent=cachedAgents.find(a=>a.agent_id===t.assignee_id);\n const status=agent?agent.status:'IDLE';\n h+='<div class=\"plan-card\">';\n h+='<div class=\"plan-card-hd\"><span class=\"adot '+status+'\"></span><span class=\"plan-card-agent\">'+esc(t.assignee_name)+'</span><span class=\"plan-card-agentid\">'+t.assignee_id+'</span></div>';\n h+='<div class=\"plan-card-desc\">'+esc(t.todo_description)+'</div>';\n h+='<div class=\"briefing-box\"><div class=\"briefing-label\">话术 (可直接发给 TA)</div><button class=\"briefing-copy\" onclick=\"window._copyBriefing('+i+')\">复制</button>'+esc(t.briefing)+'</div>';\n h+='<div class=\"plan-card-dl\">截止: '+new Date(t.deadline).toLocaleString()+'</div>';\n h+='</div>';\n }\n h+='<div class=\"btn-group\">';\n h+='<button class=\"btn btn-green\" onclick=\"dispatchPlan()\">确认分发</button>';\n h+='<button class=\"btn btn-ghost\" onclick=\"renderPlanStep1(document.getElementById(\\\\'overlay\\\\'))\">重新规划</button>';\n h+='<button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button>';\n h+='</div>';\n ov.querySelector('.form-card').innerHTML=h;\n}\nwindow.renderPlanStep1=renderPlanStep1;\n\nwindow._copyBriefing=function(i){\n const text=currentPlan.planned_tasks[i].briefing;\n navigator.clipboard.writeText(text).then(()=>toast('话术已复制',true)).catch(()=>toast('复制失败',false));\n};\n\nwindow.dispatchPlan=async function(){\n const p=currentPlan;\n const tasks=p.planned_tasks.map(t=>({assignee_id:t.assignee_id,todo_description:t.todo_description,deadline:t.deadline}));\n try{\n const r=await fetch(API+'/jobs/create',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({original_prompt:p.original_prompt,openclaw_callback:p.openclaw_callback||'',tasks})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'分发失败',false);return}\n toast('任务已分发!Job: '+d.job_id,true);\n document.getElementById('overlay').remove();\n load('pipeline');\n }catch{toast('网络错误',false)}\n};\n\n// ─── Manual creation (fallback) ──────────────\nwindow.showManualCreate=function(){\n const ov=document.getElementById('overlay');\n const optionsHtml=cachedAgents.map(a=>'<option value=\"'+a.agent_id+'\">'+esc(a.name)+' ('+a.agent_id+')</option>').join('');\n const tomorrow=new Date(Date.now()+86400000).toISOString().slice(0,16);\n ov.querySelector('.form-card').innerHTML='<h3>手动创建任务</h3>'\n +'<div class=\"fg\"><label>任务描述 (原始需求)</label><input id=\"cj-prompt\" placeholder=\"例: 完成首页改版\"/></div>'\n +'<div class=\"fg\"><label>OpenClaw 回调地址 (可选)</label><input id=\"cj-callback\" placeholder=\"https://...\"/></div>'\n +'<div style=\"margin-bottom:10px;display:flex;justify-content:space-between;align-items:center\"><label style=\"font-size:13px;font-weight:600;color:var(--text)\">子任务列表</label><button class=\"btn btn-ghost btn-sm\" onclick=\"addTaskRow()\">+ 添加子任务</button></div>'\n +'<div id=\"task-rows\"><div class=\"task-row\"><div class=\"fg\"><label>指派节点</label><select class=\"tr-agent\">'+optionsHtml+'</select></div><div class=\"fg\"><label>任务描述</label><input class=\"tr-desc\" placeholder=\"具体要做什么...\"/></div><div class=\"fg\"><label>截止时间</label><input class=\"tr-deadline\" type=\"datetime-local\" value=\"'+tomorrow+'\"/></div></div></div>'\n +'<div class=\"btn-group\"><button class=\"btn btn-primary\" onclick=\"submitJob()\">创建并分发</button><button class=\"btn btn-ghost\" onclick=\"renderPlanStep1(document.getElementById(\\\\'overlay\\\\'))\">返回 AI 规划</button><button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button></div>';\n};\nwindow.addTaskRow=function(){\n const optionsHtml=cachedAgents.map(a=>'<option value=\"'+a.agent_id+'\">'+esc(a.name)+' ('+a.agent_id+')</option>').join('');\n const tomorrow=new Date(Date.now()+86400000).toISOString().slice(0,16);\n const row=document.createElement('div');row.className='task-row';\n row.innerHTML='<button class=\"remove-task\" onclick=\"this.parentElement.remove()\">&times;</button>'\n +'<div class=\"fg\"><label>指派节点</label><select class=\"tr-agent\">'+optionsHtml+'</select></div>'\n +'<div class=\"fg\"><label>任务描述</label><input class=\"tr-desc\" placeholder=\"具体要做什么...\"/></div>'\n +'<div class=\"fg\"><label>截止时间</label><input class=\"tr-deadline\" type=\"datetime-local\" value=\"'+tomorrow+'\"/></div>';\n document.getElementById('task-rows').appendChild(row);\n};\nwindow.submitJob=async function(){\n const prompt=document.getElementById('cj-prompt').value.trim();\n const callback=document.getElementById('cj-callback').value.trim();\n if(!prompt){toast('请输入任务描述',false);return}\n const rows=document.querySelectorAll('.task-row');\n const tasks=[];\n for(const row of rows){\n const aid=row.querySelector('.tr-agent').value;\n const desc=row.querySelector('.tr-desc').value.trim();\n const dl=row.querySelector('.tr-deadline').value;\n if(!desc){toast('每个子任务都需要填写描述',false);return}\n tasks.push({assignee_id:aid,todo_description:desc,deadline:new Date(dl).toISOString()});\n }\n if(!tasks.length){toast('至少添加一个子任务',false);return}\n try{\n const r=await fetch(API+'/jobs/create',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({original_prompt:prompt,openclaw_callback:callback,tasks})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'创建失败',false);return}\n toast('任务已创建并分发!Job: '+d.job_id,true);\n document.getElementById('overlay').remove();\n load('pipeline');\n }catch{toast('网络错误',false)}\n};\nwindow.syncJob=async function(jobId){\n try{\n const r=await fetch(API+'/jobs/'+jobId+'/sync',{method:'POST'});\n const d=await r.json();\n if(!r.ok){toast(d.error||'同步失败',false);return}\n toast('聚合完成!'+d.sync.message,true);\n }catch{toast('网络错误',false)}\n};\n\n// ═══════════════════════════════════════════════\n// TERMINAL\n// ═══════════════════════════════════════════════\nfunction loadTerminal(el){\n el.innerHTML='<div class=\"section-hd\"><h2>I/O 交付终端</h2></div>'\n +'<div class=\"form-card\" style=\"margin-top:12px\"><h3>> 提交物理节点产出</h3>'\n +'<div class=\"fg\"><label>Trace ID (追踪码)</label><input id=\"t-tid\" placeholder=\"例: TK-9527\"/></div>'\n +'<div class=\"fg\"><label>交付载荷</label><textarea id=\"t-payload\" placeholder=\"粘贴交付物内容、工作汇报或代码...\"></textarea></div>'\n +'<div class=\"btn-group\"><button class=\"btn btn-primary\" onclick=\"doResume()\">提交并恢复 (Resume)</button><button class=\"btn btn-danger\" onclick=\"doReject()\">打回重做 (Reject)</button></div></div>';\n}\nwindow.doResume=async function(){\n const tid=document.getElementById('t-tid').value.trim();\n const payload=document.getElementById('t-payload').value.trim();\n if(!tid){toast('请输入 Trace ID',false);return}\n if(!payload){toast('请输入交付载荷',false);return}\n try{\n const r=await fetch(API+'/tasks/resume',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({trace_id:tid,result_data:{text:payload}})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'提交失败',false);return}\n toast(d.job_complete?'任务已交付!Job 已全部完成,可以聚合同步。':'任务 '+tid+' 已交付。',true);\n document.getElementById('t-tid').value='';document.getElementById('t-payload').value='';\n }catch{toast('网络错误',false)}\n};\nwindow.doReject=async function(){\n const tid=document.getElementById('t-tid').value.trim();\n if(!tid){toast('请输入 Trace ID',false);return}\n try{\n const r=await fetch(API+'/tasks/reject',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({trace_id:tid})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'操作失败',false);return}\n toast('任务 '+tid+' 已打回,截止时间已延长 24 小时。',true);\n document.getElementById('t-tid').value='';document.getElementById('t-payload').value='';\n }catch{toast('网络错误',false)}\n};\n\n// ═══════════════════════════════════════════════\n// SETTINGS\n// ═══════════════════════════════════════════════\nwindow.showSettings=async function(){\n const ov=document.createElement('div');ov.className='overlay';ov.id='overlay';\n ov.addEventListener('click',e=>{if(e.target===ov)ov.remove()});\n ov.innerHTML='<div class=\"form-card\"><h3>LLM 设置</h3><div class=\"spinner-wrap\"><div class=\"spinner\"></div></div></div>';\n document.body.appendChild(ov);\n\n try{\n const r=await fetch(API+'/config');\n const cfg=await r.json();\n const fc=ov.querySelector('.form-card');\n let statusHtml='';\n if(cfg.api_key_set){\n const src=cfg.api_key_source==='dashboard'?'Dashboard 配置':'环境变量';\n statusHtml='<div style=\"background:var(--accent-dim);border:1px solid color-mix(in srgb,var(--accent) 25%,transparent);border-radius:8px;padding:10px 14px;margin-bottom:16px;font-size:12px\"><span style=\"color:var(--green)\">&#10003;</span> API Key 已配置 <span style=\"color:var(--text-dim)\">('+esc(cfg.api_key_masked)+' | 来源: '+src+')</span></div>';\n }else{\n statusHtml='<div style=\"background:rgba(239,68,68,.1);border:1px solid rgba(239,68,68,.25);border-radius:8px;padding:10px 14px;margin-bottom:16px;font-size:12px;color:var(--red)\">&#9888; 未配置 API Key,AI 规划功能不可用</div>';\n }\n fc.innerHTML='<h3>LLM 设置</h3>'+statusHtml\n +'<div class=\"fg\"><label>提供商</label><select id=\"cfg-provider\"><option value=\"claude\"'+(cfg.provider==='claude'?' selected':'')+'>Claude (Anthropic)</option><option value=\"openai\"'+(cfg.provider==='openai'?' selected':'')+'>OpenAI</option></select></div>'\n +'<div class=\"fg\"><label>API Key</label><input id=\"cfg-key\" type=\"password\" placeholder=\"'+(cfg.api_key_set?'已配置,留空则不修改':'输入你的 API Key...')+'\"/><div class=\"hint\">Claude: sk-ant-... | OpenAI: sk-...</div></div>'\n +'<div class=\"fg\"><label>模型 <span style=\"color:var(--text-dim);font-weight:400\">(可选,留空用默认)</span></label><input id=\"cfg-model\" value=\"'+esc(cfg.model||'')+'\" placeholder=\"例: claude-sonnet-4-20250514 / gpt-4o\"/></div>'\n +'<div class=\"fg\"><label>API Base URL <span style=\"color:var(--text-dim);font-weight:400\">(可选,留空用官方地址)</span></label><input id=\"cfg-baseurl\" value=\"'+esc(cfg.base_url||'')+'\" placeholder=\"例: https://your-proxy.com\"/><div class=\"hint\">私有部署: 填写你的模型服务地址,如 vLLM / Ollama / Azure 等</div></div>'\n +'<div class=\"btn-group\"><button class=\"btn btn-primary\" onclick=\"saveSettings()\">保存</button><button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button></div>';\n }catch{\n ov.querySelector('.form-card').innerHTML='<h3>LLM 设置</h3><p style=\"color:var(--red)\">加载配置失败</p>';\n }\n};\nwindow.saveSettings=async function(){\n const provider=document.getElementById('cfg-provider').value;\n const apiKey=document.getElementById('cfg-key').value.trim();\n const model=document.getElementById('cfg-model').value.trim();\n const baseUrl=document.getElementById('cfg-baseurl').value.trim();\n const body={provider};\n if(apiKey)body.api_key=apiKey;\n body.model=model;\n body.base_url=baseUrl;\n try{\n const r=await fetch(API+'/config',{method:'PUT',headers:{'Content-Type':'application/json'},body:JSON.stringify(body)});\n if(!r.ok){const d=await r.json();toast(d.error||'保存失败',false);return}\n toast('设置已保存',true);\n document.getElementById('overlay').remove();\n }catch{toast('网络错误',false)}\n};\n\n// ═══════════════════════════════════════════════\n// TABS & INIT\n// ═══════════════════════════════════════════════\nconst tabs=document.querySelectorAll('.tab');\nconst panels=document.querySelectorAll('.panel');\nfunction load(id){\n const el=document.getElementById(id);\n if(id==='fleet')loadFleet(el);\n else if(id==='pipeline')loadPipeline(el);\n else if(id==='terminal')loadTerminal(el);\n}\ntabs.forEach(t=>t.addEventListener('click',()=>{\n tabs.forEach(x=>x.classList.remove('active'));\n panels.forEach(x=>x.classList.remove('active'));\n t.classList.add('active');\n const id=t.dataset.panel;\n document.getElementById(id).classList.add('active');\n load(id);\n}));\nload('fleet');\nsetInterval(()=>{\n const a=document.querySelector('.tab.active');\n if(a&&a.dataset.panel!=='terminal')load(a.dataset.panel);\n},15000);\n</script>\n</body>\n</html>`;\n}\n"],"mappings":"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AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAojBT;;;ApB1iBO,SAAS,aAAa,OAAO,MAAM;AAExC,QA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","names":["db","db","Router","db","db","db","db","db","DEFAULT_BASE_URL","db","db","router","Router","Router","db","router","Router","Router","db","router","Router","Router","router","Router","db","db","confirm"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/server.ts","../src/db/connection.ts","../src/db/schema.ts","../src/routes/nodes.ts","../src/models/agent.ts","../src/utils/trace-id.ts","../src/routes/jobs.ts","../src/models/task.ts","../src/models/job.ts","../src/services/dispatch.ts","../src/llm/claude.ts","../src/llm/openai.ts","../src/models/config.ts","../src/llm/index.ts","../src/services/planner.ts","../src/routes/tasks.ts","../src/services/resume.ts","../src/routes/sync.ts","../src/services/aggregation.ts","../src/routes/config.ts","../src/dashboard.ts"],"sourcesContent":["import { Command } from 'commander';\nimport * as p from '@clack/prompts';\nimport chalk from 'chalk';\nimport { startServer } from './server.js';\nimport { getDb } from './db/connection.js';\nimport { initSchema } from './db/schema.js';\nimport { createAgent, listAgents } from './models/agent.js';\nimport { listActiveJobs } from './models/job.js';\nimport { markOverdueTasks } from './models/task.js';\nimport { dispatchJob } from './services/dispatch.js';\nimport { planJob } from './services/planner.js';\nimport { generateId } from './utils/trace-id.js';\nimport type { AgentStatus } from './models/types.js';\n\nconst program = new Command();\n\nprogram\n .name('humanclaw')\n .description(\n 'Async physical node orchestration framework - treating humans as distributed worker nodes'\n )\n .version('1.0.0');\n\n// ─── serve ───────────────────────────────────────────────────────────────────\n\nprogram\n .command('serve')\n .description('Start the HumanClaw server')\n .option('-p, --port <port>', 'Server port', '2026')\n .action(opts => {\n const port = parseInt(opts.port, 10);\n startServer(port);\n });\n\n// ─── agent ───────────────────────────────────────────────────────────────────\n\nconst agentCmd = program\n .command('agent')\n .description('Manage physical nodes (HumanAgent)');\n\nagentCmd\n .command('add')\n .description('Register a new physical node')\n .action(async () => {\n const db = getDb();\n initSchema(db);\n\n p.intro(chalk.bgCyan(' Register New Physical Node '));\n\n const name = await p.text({\n message: 'Node alias (name):',\n placeholder: 'e.g. Frontend Lao Li',\n validate: v => (!v ? 'Name is required' : undefined),\n });\n\n if (p.isCancel(name)) {\n p.cancel('Cancelled.');\n process.exit(0);\n }\n\n const capInput = await p.text({\n message: 'Capabilities (comma-separated):',\n placeholder: 'e.g. UI/UX, Frontend Dev, Stress Resistant',\n validate: v => (!v ? 'At least one capability required' : undefined),\n });\n\n if (p.isCancel(capInput)) {\n p.cancel('Cancelled.');\n process.exit(0);\n }\n\n const capabilities = (capInput as string)\n .split(',')\n .map(s => s.trim())\n .filter(Boolean);\n\n const agent = createAgent({\n agent_id: generateId('emp'),\n name: name as string,\n capabilities,\n status: 'IDLE',\n });\n\n p.outro(\n `${chalk.green('Node registered!')} ID: ${chalk.bold(agent.agent_id)}`\n );\n });\n\nagentCmd\n .command('list')\n .description('Show fleet status')\n .action(() => {\n const db = getDb();\n initSchema(db);\n\n const agents = listAgents();\n if (agents.length === 0) {\n console.log(chalk.dim(' No physical nodes registered.'));\n return;\n }\n\n const statusIcon: Record<AgentStatus, string> = {\n IDLE: '🟢',\n BUSY: '🟡',\n OFFLINE: '🔴',\n OOM: '🟣',\n };\n\n console.log(chalk.bold('\\n Carbon Compute Pool\\n'));\n\n for (const agent of agents) {\n const icon = statusIcon[agent.status];\n const caps = agent.capabilities.join(', ');\n console.log(\n ` ${icon} ${chalk.bold(agent.name)} (${chalk.dim(agent.agent_id)})`\n );\n console.log(` Capabilities: ${chalk.cyan(caps)}`);\n console.log(` Status: ${agent.status}\\n`);\n }\n });\n\n// ─── status ──────────────────────────────────────────────────────────────────\n\nprogram\n .command('status')\n .description('Show active jobs overview')\n .action(() => {\n const db = getDb();\n initSchema(db);\n\n markOverdueTasks();\n const jobs = listActiveJobs();\n\n if (jobs.length === 0) {\n console.log(chalk.dim('\\n No active jobs.\\n'));\n return;\n }\n\n console.log(chalk.bold('\\n Async Orchestration Dashboard\\n'));\n\n for (const job of jobs) {\n const resolved = job.tasks.filter(t => t.status === 'RESOLVED').length;\n const total = job.tasks.length;\n const pct = Math.round((resolved / total) * 100);\n const bar = '█'.repeat(Math.round(pct / 5)) + '░'.repeat(20 - Math.round(pct / 5));\n\n console.log(` ${chalk.bold(job.job_id)} - ${job.original_prompt}`);\n console.log(` Progress: [${bar}] ${resolved}/${total} (${pct}%)`);\n\n for (const task of job.tasks) {\n const statusColor =\n task.status === 'RESOLVED'\n ? chalk.green\n : task.status === 'OVERDUE'\n ? chalk.red\n : chalk.yellow;\n console.log(\n ` ${statusColor(task.status.padEnd(10))} ${task.trace_id} -> ${chalk.dim(task.assignee_id)}`\n );\n console.log(` ${task.todo_description}`);\n }\n console.log();\n }\n });\n\n// ─── plan ───────────────────────────────────────────────────────────────────\n\nprogram\n .command('plan')\n .description('AI-powered task planning from natural language')\n .argument('[prompt]', 'What you want to get done')\n .option('--agents <ids>', 'Comma-separated agent IDs (default: all IDLE)')\n .option('--dispatch', 'Automatically dispatch after planning')\n .action(async (promptArg?: string, opts?: { agents?: string; dispatch?: boolean }) => {\n const db = getDb();\n initSchema(db);\n\n p.intro(chalk.bgCyan(' AI 智能规划 '));\n\n let prompt = promptArg;\n if (!prompt) {\n const input = await p.text({\n message: '输入你的需求:',\n placeholder: '例: 完成首页重构,包括导航栏和内容区的响应式改版',\n validate: v => (!v ? '需求描述不能为空' : undefined),\n });\n if (p.isCancel(input)) {\n p.cancel('已取消');\n process.exit(0);\n }\n prompt = input as string;\n }\n\n const agentIds = opts?.agents?.split(',').map(s => s.trim()).filter(Boolean);\n\n const spin = p.spinner();\n spin.start('AI 正在分析需求、匹配节点、生成话术...');\n\n try {\n const plan = await planJob({ prompt, agent_ids: agentIds }, undefined, db);\n spin.stop('规划完成!');\n\n console.log(chalk.bold(`\\n 需求: ${chalk.white(plan.original_prompt)}\\n`));\n\n for (const task of plan.planned_tasks) {\n console.log(chalk.cyan(` ┌─ ${chalk.bold(task.assignee_name)} (${chalk.dim(task.assignee_id)})`));\n console.log(chalk.cyan(' │') + ` 任务: ${task.todo_description}`);\n console.log(chalk.cyan(' │') + ` 话术: ${chalk.italic(task.briefing)}`);\n console.log(chalk.cyan(' │') + ` 截止: ${new Date(task.deadline).toLocaleString()}`);\n console.log(chalk.cyan(' └─'));\n console.log();\n }\n\n if (opts?.dispatch) {\n const tasks = plan.planned_tasks.map(t => ({\n assignee_id: t.assignee_id,\n todo_description: t.todo_description,\n deadline: t.deadline,\n }));\n const job = dispatchJob({\n original_prompt: plan.original_prompt,\n openclaw_callback: '',\n tasks,\n }, db);\n p.outro(`${chalk.green('已分发!')} Job: ${chalk.bold(job.job_id)}`);\n } else {\n const confirm = await p.confirm({\n message: '确认分发这些任务?',\n });\n if (p.isCancel(confirm) || !confirm) {\n p.outro(chalk.dim('已取消分发'));\n process.exit(0);\n }\n const tasks = plan.planned_tasks.map(t => ({\n assignee_id: t.assignee_id,\n todo_description: t.todo_description,\n deadline: t.deadline,\n }));\n const job = dispatchJob({\n original_prompt: plan.original_prompt,\n openclaw_callback: '',\n tasks,\n }, db);\n p.outro(`${chalk.green('已分发!')} Job: ${chalk.bold(job.job_id)}`);\n }\n } catch (error) {\n spin.stop('规划失败');\n const message = error instanceof Error ? error.message : 'Unknown error';\n p.outro(chalk.red(message));\n process.exit(1);\n }\n });\n\nprogram.parse();\n","import express from 'express';\nimport cors from 'cors';\nimport { getDb } from './db/connection.js';\nimport { initSchema } from './db/schema.js';\nimport nodesRouter from './routes/nodes.js';\nimport jobsRouter from './routes/jobs.js';\nimport tasksRouter from './routes/tasks.js';\nimport syncRouter from './routes/sync.js';\nimport configRouter from './routes/config.js';\nimport { getDashboardHtml } from './dashboard.js';\n\nexport function createServer(port = 2026) {\n // Initialize database\n const db = getDb();\n initSchema(db);\n\n const app = express();\n\n // Middleware\n app.use(cors());\n app.use(express.json({ limit: '10mb' }));\n\n // API routes\n app.use('/api/v1/nodes', nodesRouter);\n app.use('/api/v1/jobs', jobsRouter);\n app.use('/api/v1/tasks', tasksRouter);\n app.use('/api/v1/jobs', syncRouter);\n app.use('/api/v1/config', configRouter);\n\n // Serve dashboard as inline HTML (no build step needed)\n const dashboardHtml = getDashboardHtml();\n app.get('/', (_req, res) => {\n res.type('html').send(dashboardHtml);\n });\n\n // Error handler\n app.use(\n (\n err: Error,\n _req: express.Request,\n res: express.Response,\n _next: express.NextFunction\n ) => {\n console.error('Server error:', err.message);\n res.status(500).json({ error: 'Internal server error' });\n }\n );\n\n return { app, port };\n}\n\nexport function startServer(port = 2026) {\n const { app } = createServer(port);\n\n app.listen(port, () => {\n console.log(`\\n HumanClaw server running at http://localhost:${port}`);\n console.log(` Dashboard: http://localhost:${port}`);\n console.log(` API base: http://localhost:${port}/api/v1\\n`);\n });\n}\n","import Database from 'better-sqlite3';\nimport path from 'node:path';\nimport fs from 'node:fs';\n\nlet db: Database.Database | null = null;\n\nconst DEFAULT_DB_PATH = path.join(\n process.env.HUMANCLAW_DATA_DIR ?? process.cwd(),\n 'humanclaw.db'\n);\n\nexport function getDb(dbPath?: string): Database.Database {\n if (db) return db;\n\n const resolvedPath = dbPath ?? DEFAULT_DB_PATH;\n const dir = path.dirname(resolvedPath);\n if (!fs.existsSync(dir)) {\n fs.mkdirSync(dir, { recursive: true });\n }\n\n db = new Database(resolvedPath);\n db.pragma('journal_mode = WAL');\n db.pragma('foreign_keys = ON');\n\n return db;\n}\n\nexport function createInMemoryDb(): Database.Database {\n const memDb = new Database(':memory:');\n memDb.pragma('foreign_keys = ON');\n return memDb;\n}\n\nexport function closeDb(): void {\n if (db) {\n db.close();\n db = null;\n }\n}\n\nexport function setDb(newDb: Database.Database): void {\n db = newDb;\n}\n","import type Database from 'better-sqlite3';\n\nexport function initSchema(db: Database.Database): void {\n db.exec(`\n CREATE TABLE IF NOT EXISTS agents (\n agent_id TEXT PRIMARY KEY,\n name TEXT NOT NULL,\n capabilities TEXT NOT NULL DEFAULT '[]',\n status TEXT NOT NULL DEFAULT 'IDLE'\n CHECK (status IN ('IDLE', 'BUSY', 'OFFLINE', 'OOM')),\n created_at TEXT NOT NULL DEFAULT (datetime('now'))\n );\n\n CREATE TABLE IF NOT EXISTS jobs (\n job_id TEXT PRIMARY KEY,\n original_prompt TEXT NOT NULL,\n openclaw_callback TEXT NOT NULL DEFAULT '',\n created_at TEXT NOT NULL DEFAULT (datetime('now'))\n );\n\n CREATE TABLE IF NOT EXISTS tasks (\n trace_id TEXT PRIMARY KEY,\n job_id TEXT NOT NULL REFERENCES jobs(job_id) ON DELETE CASCADE,\n assignee_id TEXT NOT NULL REFERENCES agents(agent_id),\n todo_description TEXT NOT NULL,\n deadline TEXT NOT NULL,\n payload TEXT NOT NULL DEFAULT '{}',\n status TEXT NOT NULL DEFAULT 'PENDING'\n CHECK (status IN ('PENDING', 'DISPATCHED', 'RESOLVED', 'OVERDUE')),\n result_data TEXT,\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n updated_at TEXT NOT NULL DEFAULT (datetime('now'))\n );\n\n CREATE INDEX IF NOT EXISTS idx_tasks_job_id ON tasks(job_id);\n CREATE INDEX IF NOT EXISTS idx_tasks_assignee_id ON tasks(assignee_id);\n CREATE INDEX IF NOT EXISTS idx_tasks_status ON tasks(status);\n\n CREATE TABLE IF NOT EXISTS config (\n key TEXT PRIMARY KEY,\n value TEXT NOT NULL\n );\n `);\n}\n","import { Router } from 'express';\nimport {\n listAgentsWithMetrics,\n createAgent,\n updateAgentStatus,\n deleteAgent,\n} from '../models/agent.js';\nimport { generateId } from '../utils/trace-id.js';\nimport type { AgentStatus } from '../models/types.js';\n\nconst router = Router();\n\n// GET /api/v1/nodes/status - Fleet status with metrics\nrouter.get('/status', (_req, res) => {\n const agents = listAgentsWithMetrics();\n res.json({\n total: agents.length,\n idle: agents.filter(a => a.status === 'IDLE').length,\n busy: agents.filter(a => a.status === 'BUSY').length,\n offline: agents.filter(a => a.status === 'OFFLINE').length,\n oom: agents.filter(a => a.status === 'OOM').length,\n agents,\n });\n});\n\n// POST /api/v1/nodes - Register a new agent\nrouter.post('/', (req, res) => {\n const { name, capabilities, status } = req.body;\n\n if (!name || !Array.isArray(capabilities)) {\n res.status(400).json({ error: 'name and capabilities[] are required' });\n return;\n }\n\n const agent = createAgent({\n agent_id: generateId('emp'),\n name,\n capabilities,\n status: (status as AgentStatus) ?? 'IDLE',\n });\n\n res.status(201).json(agent);\n});\n\n// PATCH /api/v1/nodes/:agent_id/status - Update agent status\nrouter.patch('/:agent_id/status', (req, res) => {\n const { agent_id } = req.params;\n const { status } = req.body;\n\n const validStatuses: AgentStatus[] = ['IDLE', 'BUSY', 'OFFLINE', 'OOM'];\n if (!validStatuses.includes(status)) {\n res.status(400).json({\n error: `Invalid status. Must be one of: ${validStatuses.join(', ')}`,\n });\n return;\n }\n\n const updated = updateAgentStatus(agent_id, status);\n if (!updated) {\n res.status(404).json({ error: `Agent not found: ${agent_id}` });\n return;\n }\n\n res.json({ agent_id, status });\n});\n\n// DELETE /api/v1/nodes/:agent_id\nrouter.delete('/:agent_id', (req, res) => {\n const { agent_id } = req.params;\n const deleted = deleteAgent(agent_id);\n if (!deleted) {\n res.status(404).json({ error: `Agent not found: ${agent_id}` });\n return;\n }\n res.json({ deleted: agent_id });\n});\n\nexport default router;\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport type {\n HumanAgent,\n AgentRow,\n AgentStatus,\n AgentWithMetrics,\n} from './types.js';\n\nfunction rowToAgent(row: AgentRow): HumanAgent {\n return {\n ...row,\n capabilities: JSON.parse(row.capabilities) as string[],\n };\n}\n\nexport function createAgent(\n agent: Omit<HumanAgent, 'created_at'>,\n db?: Database.Database\n): HumanAgent {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n conn\n .prepare(\n `INSERT INTO agents (agent_id, name, capabilities, status, created_at)\n VALUES (?, ?, ?, ?, ?)`\n )\n .run(\n agent.agent_id,\n agent.name,\n JSON.stringify(agent.capabilities),\n agent.status,\n now\n );\n return { ...agent, created_at: now };\n}\n\nexport function getAgent(\n agentId: string,\n db?: Database.Database\n): HumanAgent | undefined {\n const conn = db ?? getDb();\n const row = conn\n .prepare('SELECT * FROM agents WHERE agent_id = ?')\n .get(agentId) as AgentRow | undefined;\n return row ? rowToAgent(row) : undefined;\n}\n\nexport function listAgents(db?: Database.Database): HumanAgent[] {\n const conn = db ?? getDb();\n const rows = conn.prepare('SELECT * FROM agents ORDER BY created_at').all() as AgentRow[];\n return rows.map(rowToAgent);\n}\n\nexport function updateAgentStatus(\n agentId: string,\n status: AgentStatus,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const result = conn\n .prepare('UPDATE agents SET status = ? WHERE agent_id = ?')\n .run(status, agentId);\n return result.changes > 0;\n}\n\nexport function deleteAgent(\n agentId: string,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const result = conn\n .prepare('DELETE FROM agents WHERE agent_id = ?')\n .run(agentId);\n return result.changes > 0;\n}\n\nexport function listAgentsWithMetrics(\n db?: Database.Database\n): AgentWithMetrics[] {\n const conn = db ?? getDb();\n const rows = conn\n .prepare(\n `SELECT\n a.*,\n COUNT(CASE WHEN t.status IN ('DISPATCHED', 'PENDING') THEN 1 END) AS active_task_count,\n AVG(\n CASE WHEN t.status = 'RESOLVED'\n THEN (julianday(t.updated_at) - julianday(t.created_at)) * 24\n END\n ) AS avg_delivery_hours\n FROM agents a\n LEFT JOIN tasks t ON a.agent_id = t.assignee_id\n GROUP BY a.agent_id\n ORDER BY a.created_at`\n )\n .all() as (AgentRow & { active_task_count: number; avg_delivery_hours: number | null })[];\n\n return rows.map(row => ({\n ...rowToAgent(row),\n active_task_count: row.active_task_count,\n avg_delivery_hours: row.avg_delivery_hours\n ? Math.round(row.avg_delivery_hours * 100) / 100\n : null,\n }));\n}\n","import { nanoid } from 'nanoid';\n\nexport function generateTraceId(): string {\n const num = Math.floor(Math.random() * 10000)\n .toString()\n .padStart(4, '0');\n return `TK-${num}`;\n}\n\nexport function generateId(prefix: string): string {\n return `${prefix}_${nanoid(8)}`;\n}\n","import { Router } from 'express';\nimport { dispatchJob } from '../services/dispatch.js';\nimport { planJob } from '../services/planner.js';\nimport { listActiveJobs, getJobWithTasks } from '../models/job.js';\nimport { markOverdueTasks } from '../models/task.js';\nimport type { CreateJobRequest, PlanRequest } from '../models/types.js';\n\nconst router = Router();\n\n// POST /api/v1/jobs/create - Create and dispatch a new job\nrouter.post('/create', (req, res) => {\n const body = req.body as CreateJobRequest;\n\n if (!body.original_prompt || !Array.isArray(body.tasks) || body.tasks.length === 0) {\n res.status(400).json({\n error: 'original_prompt and non-empty tasks[] are required',\n });\n return;\n }\n\n for (const task of body.tasks) {\n if (!task.assignee_id || !task.todo_description || !task.deadline) {\n res.status(400).json({\n error: 'Each task requires assignee_id, todo_description, and deadline',\n });\n return;\n }\n }\n\n try {\n const job = dispatchJob(body);\n res.status(201).json(job);\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n res.status(400).json({ error: message });\n }\n});\n\n// GET /api/v1/jobs/active - List active jobs for kanban\nrouter.get('/active', (_req, res) => {\n // Mark overdue tasks before returning\n markOverdueTasks();\n const jobs = listActiveJobs();\n res.json({ jobs });\n});\n\n// POST /api/v1/jobs/plan - AI-powered task planning (does NOT dispatch)\nrouter.post('/plan', async (req, res) => {\n const body = req.body as PlanRequest;\n\n if (!body.prompt || typeof body.prompt !== 'string') {\n res.status(400).json({ error: 'prompt is required' });\n return;\n }\n\n try {\n const plan = await planJob(body);\n res.json(plan);\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n const status = message.includes('API Key') || message.includes('API key') ? 503 : 400;\n res.status(status).json({ error: message });\n }\n});\n\n// GET /api/v1/jobs/:job_id - Get a single job with tasks\nrouter.get('/:job_id', (req, res) => {\n const { job_id } = req.params;\n const job = getJobWithTasks(job_id);\n if (!job) {\n res.status(404).json({ error: `Job not found: ${job_id}` });\n return;\n }\n res.json(job);\n});\n\nexport default router;\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport type { HumanTask, TaskRow, TaskStatus } from './types.js';\n\nfunction rowToTask(row: TaskRow): HumanTask {\n return {\n ...row,\n payload: JSON.parse(row.payload) as Record<string, unknown>,\n result_data: row.result_data ? JSON.parse(row.result_data) : null,\n };\n}\n\nexport function createTask(\n task: Omit<HumanTask, 'created_at' | 'updated_at' | 'result_data'>,\n db?: Database.Database\n): HumanTask {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n conn\n .prepare(\n `INSERT INTO tasks (trace_id, job_id, assignee_id, todo_description, deadline, payload, status, created_at, updated_at)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`\n )\n .run(\n task.trace_id,\n task.job_id,\n task.assignee_id,\n task.todo_description,\n task.deadline,\n JSON.stringify(task.payload),\n task.status,\n now,\n now\n );\n return { ...task, result_data: null, created_at: now, updated_at: now };\n}\n\nexport function getTask(\n traceId: string,\n db?: Database.Database\n): HumanTask | undefined {\n const conn = db ?? getDb();\n const row = conn\n .prepare('SELECT * FROM tasks WHERE trace_id = ?')\n .get(traceId) as TaskRow | undefined;\n return row ? rowToTask(row) : undefined;\n}\n\nexport function listTasksByJob(\n jobId: string,\n db?: Database.Database\n): HumanTask[] {\n const conn = db ?? getDb();\n const rows = conn\n .prepare('SELECT * FROM tasks WHERE job_id = ? ORDER BY created_at')\n .all(jobId) as TaskRow[];\n return rows.map(rowToTask);\n}\n\nexport function listTasksByAssignee(\n assigneeId: string,\n db?: Database.Database\n): HumanTask[] {\n const conn = db ?? getDb();\n const rows = conn\n .prepare('SELECT * FROM tasks WHERE assignee_id = ? ORDER BY created_at')\n .all(assigneeId) as TaskRow[];\n return rows.map(rowToTask);\n}\n\nexport function updateTaskStatus(\n traceId: string,\n status: TaskStatus,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n const result = conn\n .prepare('UPDATE tasks SET status = ?, updated_at = ? WHERE trace_id = ?')\n .run(status, now, traceId);\n return result.changes > 0;\n}\n\nexport function resolveTask(\n traceId: string,\n resultData: unknown,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n const result = conn\n .prepare(\n `UPDATE tasks\n SET status = 'RESOLVED', result_data = ?, updated_at = ?\n WHERE trace_id = ? AND status IN ('DISPATCHED', 'OVERDUE')`\n )\n .run(JSON.stringify(resultData), now, traceId);\n return result.changes > 0;\n}\n\nexport function markOverdueTasks(db?: Database.Database): number {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n const result = conn\n .prepare(\n `UPDATE tasks\n SET status = 'OVERDUE', updated_at = ?\n WHERE status = 'DISPATCHED' AND deadline < ?`\n )\n .run(now, now);\n return result.changes;\n}\n\nexport function resetTaskDeadline(\n traceId: string,\n newDeadline: string,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const now = new Date().toISOString();\n const result = conn\n .prepare(\n `UPDATE tasks\n SET status = 'DISPATCHED', deadline = ?, result_data = NULL, updated_at = ?\n WHERE trace_id = ?`\n )\n .run(newDeadline, now, traceId);\n return result.changes > 0;\n}\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport type { OrchestrationJob, JobRow, JobWithTasks } from './types.js';\nimport { listTasksByJob } from './task.js';\n\nexport function createJob(\n job: OrchestrationJob,\n db?: Database.Database\n): OrchestrationJob {\n const conn = db ?? getDb();\n conn\n .prepare(\n `INSERT INTO jobs (job_id, original_prompt, openclaw_callback, created_at)\n VALUES (?, ?, ?, ?)`\n )\n .run(job.job_id, job.original_prompt, job.openclaw_callback, job.created_at);\n return job;\n}\n\nexport function getJob(\n jobId: string,\n db?: Database.Database\n): OrchestrationJob | undefined {\n const conn = db ?? getDb();\n return conn\n .prepare('SELECT * FROM jobs WHERE job_id = ?')\n .get(jobId) as JobRow | undefined;\n}\n\nexport function getJobWithTasks(\n jobId: string,\n db?: Database.Database\n): JobWithTasks | undefined {\n const conn = db ?? getDb();\n const job = conn\n .prepare('SELECT * FROM jobs WHERE job_id = ?')\n .get(jobId) as JobRow | undefined;\n if (!job) return undefined;\n\n const tasks = listTasksByJob(jobId, conn);\n return { ...job, tasks };\n}\n\nexport function listActiveJobs(db?: Database.Database): JobWithTasks[] {\n const conn = db ?? getDb();\n const jobs = conn\n .prepare(\n `SELECT DISTINCT j.*\n FROM jobs j\n INNER JOIN tasks t ON j.job_id = t.job_id\n WHERE t.status != 'RESOLVED'\n ORDER BY j.created_at DESC`\n )\n .all() as JobRow[];\n\n // Also include jobs where all tasks are resolved but not yet synced\n const allJobs = conn\n .prepare('SELECT * FROM jobs ORDER BY created_at DESC')\n .all() as JobRow[];\n\n const activeJobIds = new Set(jobs.map(j => j.job_id));\n const result: JobWithTasks[] = [];\n\n for (const job of allJobs) {\n const tasks = listTasksByJob(job.job_id, conn);\n if (tasks.length > 0) {\n result.push({ ...job, tasks });\n }\n }\n\n return result;\n}\n\nexport function isJobComplete(\n jobId: string,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const row = conn\n .prepare(\n `SELECT COUNT(*) as total,\n SUM(CASE WHEN status = 'RESOLVED' THEN 1 ELSE 0 END) as resolved\n FROM tasks WHERE job_id = ?`\n )\n .get(jobId) as { total: number; resolved: number };\n return row.total > 0 && row.total === row.resolved;\n}\n\nexport function deleteJob(\n jobId: string,\n db?: Database.Database\n): boolean {\n const conn = db ?? getDb();\n const result = conn\n .prepare('DELETE FROM jobs WHERE job_id = ?')\n .run(jobId);\n return result.changes > 0;\n}\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport { createJob } from '../models/job.js';\nimport { createTask } from '../models/task.js';\nimport { getAgent, updateAgentStatus } from '../models/agent.js';\nimport { generateTraceId, generateId } from '../utils/trace-id.js';\nimport type { CreateJobRequest, JobWithTasks } from '../models/types.js';\n\nexport function dispatchJob(\n request: CreateJobRequest,\n db?: Database.Database\n): JobWithTasks {\n const conn = db ?? getDb();\n const jobId = generateId('job');\n const now = new Date().toISOString();\n\n // Validate all assignees exist\n for (const taskReq of request.tasks) {\n const agent = getAgent(taskReq.assignee_id, conn);\n if (!agent) {\n throw new Error(`Agent not found: ${taskReq.assignee_id}`);\n }\n if (agent.status === 'OFFLINE') {\n throw new Error(`Agent is offline: ${taskReq.assignee_id} (${agent.name})`);\n }\n }\n\n // Create the job\n const job = createJob(\n {\n job_id: jobId,\n original_prompt: request.original_prompt,\n openclaw_callback: request.openclaw_callback,\n created_at: now,\n },\n conn\n );\n\n // Create and dispatch all tasks\n const tasks = request.tasks.map(taskReq => {\n const traceId = generateTraceId();\n const task = createTask(\n {\n trace_id: traceId,\n job_id: jobId,\n assignee_id: taskReq.assignee_id,\n todo_description: taskReq.todo_description,\n deadline: taskReq.deadline,\n payload: taskReq.payload ?? {},\n status: 'DISPATCHED',\n },\n conn\n );\n\n // Mark the agent as busy\n updateAgentStatus(taskReq.assignee_id, 'BUSY', conn);\n\n return task;\n });\n\n return { ...job, tasks };\n}\n","import type { LlmProvider, LlmCompletionRequest, LlmCompletionResponse } from './types.js';\n\nconst DEFAULT_BASE_URL = 'https://api.anthropic.com';\n\nexport class ClaudeProvider implements LlmProvider {\n private apiKey: string;\n private model: string;\n private baseUrl: string;\n\n constructor(apiKey: string, model?: string, baseUrl?: string) {\n this.apiKey = apiKey;\n this.model = model || 'claude-sonnet-4-20250514';\n this.baseUrl = (baseUrl || DEFAULT_BASE_URL).replace(/\\/+$/, '');\n }\n\n async complete(request: LlmCompletionRequest): Promise<LlmCompletionResponse> {\n const systemMessages = request.messages.filter(m => m.role === 'system');\n const nonSystemMessages = request.messages.filter(m => m.role !== 'system');\n\n const body: Record<string, unknown> = {\n model: this.model,\n max_tokens: request.max_tokens || 4096,\n messages: nonSystemMessages.map(m => ({ role: m.role, content: m.content })),\n };\n\n if (request.temperature !== undefined) {\n body.temperature = request.temperature;\n }\n\n if (systemMessages.length > 0) {\n body.system = systemMessages.map(m => m.content).join('\\n\\n');\n }\n\n const response = await fetch(`${this.baseUrl}/v1/messages`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'x-api-key': this.apiKey,\n 'anthropic-version': '2023-06-01',\n },\n body: JSON.stringify(body),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n throw new Error(`Claude API error (${response.status}): ${errorText}`);\n }\n\n const data = await response.json() as { content: Array<{ type: string; text: string }> };\n const textBlock = data.content.find(c => c.type === 'text');\n\n if (!textBlock) {\n throw new Error('Claude API returned no text content');\n }\n\n return { content: textBlock.text };\n }\n}\n","import type { LlmProvider, LlmCompletionRequest, LlmCompletionResponse } from './types.js';\n\nconst DEFAULT_BASE_URL = 'https://api.openai.com';\n\nexport class OpenAIProvider implements LlmProvider {\n private apiKey: string;\n private model: string;\n private baseUrl: string;\n\n constructor(apiKey: string, model?: string, baseUrl?: string) {\n this.apiKey = apiKey;\n this.model = model || 'gpt-4o';\n this.baseUrl = (baseUrl || DEFAULT_BASE_URL).replace(/\\/+$/, '');\n }\n\n async complete(request: LlmCompletionRequest): Promise<LlmCompletionResponse> {\n const body: Record<string, unknown> = {\n model: this.model,\n max_tokens: request.max_tokens || 4096,\n messages: request.messages.map(m => ({ role: m.role, content: m.content })),\n };\n\n if (request.temperature !== undefined) {\n body.temperature = request.temperature;\n }\n\n const response = await fetch(`${this.baseUrl}/v1/chat/completions`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': `Bearer ${this.apiKey}`,\n },\n body: JSON.stringify(body),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n throw new Error(`OpenAI API error (${response.status}): ${errorText}`);\n }\n\n const data = await response.json() as {\n choices: Array<{ message: { content: string } }>;\n };\n\n const content = data.choices[0]?.message?.content;\n if (!content) {\n throw new Error('OpenAI API returned no content');\n }\n\n return { content };\n }\n}\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\n\nexport function getConfig(key: string, db?: Database.Database): string | undefined {\n const conn = db ?? getDb();\n const row = conn.prepare('SELECT value FROM config WHERE key = ?').get(key) as { value: string } | undefined;\n return row?.value;\n}\n\nexport function setConfig(key: string, value: string, db?: Database.Database): void {\n const conn = db ?? getDb();\n conn.prepare('INSERT OR REPLACE INTO config (key, value) VALUES (?, ?)').run(key, value);\n}\n\nexport function deleteConfig(key: string, db?: Database.Database): void {\n const conn = db ?? getDb();\n conn.prepare('DELETE FROM config WHERE key = ?').run(key);\n}\n\nexport function getAllConfig(db?: Database.Database): Record<string, string> {\n const conn = db ?? getDb();\n const rows = conn.prepare('SELECT key, value FROM config').all() as Array<{ key: string; value: string }>;\n const result: Record<string, string> = {};\n for (const row of rows) {\n result[row.key] = row.value;\n }\n return result;\n}\n","import type { LlmProvider, LlmProviderName } from './types.js';\nimport { ClaudeProvider } from './claude.js';\nimport { OpenAIProvider } from './openai.js';\nimport { getConfig } from '../models/config.js';\n\nexport interface LlmConfig {\n provider: LlmProviderName;\n apiKey: string;\n model?: string;\n baseUrl?: string;\n}\n\nexport function getLlmConfig(): LlmConfig {\n // DB config takes priority over env vars\n const dbProvider = getConfig('llm_provider');\n const dbApiKey = getConfig('llm_api_key');\n const dbModel = getConfig('llm_model');\n const dbBaseUrl = getConfig('llm_base_url');\n\n const provider = (dbProvider || process.env.HUMANCLAW_LLM_PROVIDER || 'claude') as LlmProviderName;\n const apiKey = dbApiKey || process.env.HUMANCLAW_LLM_API_KEY || '';\n const model = dbModel || process.env.HUMANCLAW_LLM_MODEL;\n const baseUrl = dbBaseUrl || process.env.HUMANCLAW_LLM_BASE_URL;\n\n if (!apiKey) {\n throw new Error(\n '未配置 LLM API Key。请在 Dashboard 设置中配置,或设置环境变量 HUMANCLAW_LLM_API_KEY。'\n );\n }\n\n if (provider !== 'claude' && provider !== 'openai') {\n throw new Error(`不支持的 LLM 提供商: ${provider}。支持: claude, openai`);\n }\n\n return { provider, apiKey, model: model || undefined, baseUrl: baseUrl || undefined };\n}\n\nexport function createLlmProvider(config?: LlmConfig): LlmProvider {\n const cfg = config || getLlmConfig();\n\n switch (cfg.provider) {\n case 'claude':\n return new ClaudeProvider(cfg.apiKey, cfg.model, cfg.baseUrl);\n case 'openai':\n return new OpenAIProvider(cfg.apiKey, cfg.model, cfg.baseUrl);\n }\n}\n\nexport type { LlmProvider, LlmProviderName } from './types.js';\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport { listAgentsWithMetrics, getAgent } from '../models/agent.js';\nimport { createLlmProvider } from '../llm/index.js';\nimport type { PlanRequest, PlanResponse, PlannedTask, AgentWithMetrics } from '../models/types.js';\nimport type { LlmProvider } from '../llm/types.js';\n\nfunction buildSystemPrompt(): string {\n return `你是 HumanClaw 任务编排规划器。你的工作是将用户的需求拆解为可以分发给物理节点(真实人类)执行的独立子任务。\n\n规则:\n1. 将需求拆解为扁平的、无依赖的子任务列表(每个任务可以独立执行)\n2. 根据每个 Agent 的技能(capabilities)和当前负载来匹配分配\n3. 为每个任务生成一段「话术」—— 这是直接发给该人类执行者的任务说明,语气自然、清晰、专业,包含具体的交付物要求\n4. 根据任务复杂度设置合理的截止时间(相对于当前时间)\n5. 每个 Agent 最多分配一个任务(除非人手不够)\n\n你必须严格输出以下 JSON 格式(不要输出任何其他内容):\n\n\\`\\`\\`json\n[\n {\n \"assignee_id\": \"agent的ID\",\n \"assignee_name\": \"agent的名字\",\n \"todo_description\": \"简短的任务标题/描述\",\n \"briefing\": \"话术:详细的任务说明,包括目标、交付物要求、注意事项。语气像一个靠谱的项目经理在给组员分配任务。\",\n \"deadline\": \"ISO 8601 时间\"\n }\n]\n\\`\\`\\``;\n}\n\nfunction buildUserPrompt(prompt: string, agents: AgentWithMetrics[]): string {\n const now = new Date();\n const agentList = agents.map(a => {\n const load = a.active_task_count > 0\n ? `当前有 ${a.active_task_count} 个进行中的任务`\n : '当前空闲';\n const speed = a.avg_delivery_hours !== null\n ? `平均交付时间 ${a.avg_delivery_hours}h`\n : '暂无历史数据';\n return `- ${a.name} (ID: ${a.agent_id}) 技能: [${a.capabilities.join(', ')}] ${load} ${speed}`;\n }).join('\\n');\n\n return `当前时间: ${now.toISOString()}\n\n可用的物理节点:\n${agentList}\n\n需求:\n${prompt}\n\n请根据以上信息拆解任务并分配。输出 JSON 数组。`;\n}\n\nfunction extractJson(raw: string): string {\n // Try to extract JSON from markdown code block\n const codeBlockMatch = raw.match(/```(?:json)?\\s*\\n?([\\s\\S]*?)\\n?\\s*```/);\n if (codeBlockMatch) {\n return codeBlockMatch[1].trim();\n }\n // Try to find a JSON array directly\n const arrayMatch = raw.match(/\\[[\\s\\S]*\\]/);\n if (arrayMatch) {\n return arrayMatch[0];\n }\n return raw.trim();\n}\n\nfunction validatePlannedTasks(\n parsed: unknown,\n validAgentIds: Set<string>,\n agents: AgentWithMetrics[]\n): PlannedTask[] {\n if (!Array.isArray(parsed)) {\n throw new Error('LLM 返回的不是有效的任务数组');\n }\n\n if (parsed.length === 0) {\n throw new Error('LLM 未生成任何任务');\n }\n\n return parsed.map((item: Record<string, unknown>, i: number) => {\n if (!item.todo_description || typeof item.todo_description !== 'string') {\n throw new Error(`任务 ${i + 1} 缺少 todo_description`);\n }\n if (!item.briefing || typeof item.briefing !== 'string') {\n throw new Error(`任务 ${i + 1} 缺少 briefing (话术)`);\n }\n\n // Validate or fallback assignee_id\n let assigneeId = String(item.assignee_id || '');\n let assigneeName = String(item.assignee_name || '');\n\n if (!validAgentIds.has(assigneeId)) {\n // Fallback to first available agent\n const fallback = agents[0];\n if (fallback) {\n assigneeId = fallback.agent_id;\n assigneeName = fallback.name;\n }\n }\n\n // Validate deadline or generate a default\n let deadline = String(item.deadline || '');\n if (!deadline || isNaN(Date.parse(deadline))) {\n // Default: 24 hours from now\n deadline = new Date(Date.now() + 24 * 60 * 60 * 1000).toISOString();\n }\n\n return {\n assignee_id: assigneeId,\n assignee_name: assigneeName,\n todo_description: String(item.todo_description),\n briefing: String(item.briefing),\n deadline,\n };\n });\n}\n\nexport async function planJob(\n request: PlanRequest,\n provider?: LlmProvider,\n db?: Database.Database\n): Promise<PlanResponse> {\n const conn = db ?? getDb();\n\n // Get available agents\n const allAgents = listAgentsWithMetrics(conn);\n let agents: AgentWithMetrics[];\n\n if (request.agent_ids && request.agent_ids.length > 0) {\n agents = allAgents.filter(a => request.agent_ids!.includes(a.agent_id));\n if (agents.length === 0) {\n throw new Error('指定的 Agent 均不存在');\n }\n } else {\n // Default: only IDLE agents\n agents = allAgents.filter(a => a.status === 'IDLE');\n if (agents.length === 0) {\n // Fallback: include BUSY agents too (but not OFFLINE/OOM)\n agents = allAgents.filter(a => a.status !== 'OFFLINE' && a.status !== 'OOM');\n }\n if (agents.length === 0) {\n throw new Error('没有可用的物理节点。请先在碳基算力池中添加节点。');\n }\n }\n\n // Get LLM provider\n const llm = provider ?? createLlmProvider();\n\n const systemPrompt = buildSystemPrompt();\n const userPrompt = buildUserPrompt(request.prompt, agents);\n\n const response = await llm.complete({\n messages: [\n { role: 'system', content: systemPrompt },\n { role: 'user', content: userPrompt },\n ],\n temperature: 0.3,\n max_tokens: 4096,\n });\n\n // Parse LLM response\n const jsonStr = extractJson(response.content);\n let parsed: unknown;\n try {\n parsed = JSON.parse(jsonStr);\n } catch {\n throw new Error('AI 返回的内容无法解析为 JSON,请重试');\n }\n\n const validIds = new Set(agents.map(a => a.agent_id));\n const plannedTasks = validatePlannedTasks(parsed, validIds, agents);\n\n return {\n original_prompt: request.prompt,\n planned_tasks: plannedTasks,\n };\n}\n","import { Router } from 'express';\nimport { resumeTask, rejectTask } from '../services/resume.js';\n\nconst router = Router();\n\n// POST /api/v1/tasks/resume - Submit result for a task\nrouter.post('/resume', (req, res) => {\n const { trace_id, result_data } = req.body;\n\n if (!trace_id) {\n res.status(400).json({ error: 'trace_id is required' });\n return;\n }\n\n if (result_data === undefined) {\n res.status(400).json({ error: 'result_data is required' });\n return;\n }\n\n try {\n const result = resumeTask(trace_id, result_data);\n res.json({\n task: result.task,\n job_complete: result.jobComplete,\n job: result.job ?? null,\n });\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n res.status(400).json({ error: message });\n }\n});\n\n// POST /api/v1/tasks/reject - Reject and retry a task\nrouter.post('/reject', (req, res) => {\n const { trace_id, new_deadline } = req.body;\n\n if (!trace_id) {\n res.status(400).json({ error: 'trace_id is required' });\n return;\n }\n\n try {\n const task = rejectTask(trace_id, new_deadline);\n res.json({ task });\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n res.status(400).json({ error: message });\n }\n});\n\nexport default router;\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport { getTask, resolveTask, resetTaskDeadline } from '../models/task.js';\nimport { isJobComplete, getJobWithTasks } from '../models/job.js';\nimport { updateAgentStatus } from '../models/agent.js';\nimport { listTasksByAssignee } from '../models/task.js';\nimport type { HumanTask, JobWithTasks } from '../models/types.js';\n\nexport interface ResumeResult {\n task: HumanTask;\n jobComplete: boolean;\n job: JobWithTasks | undefined;\n}\n\nexport function resumeTask(\n traceId: string,\n resultData: unknown,\n db?: Database.Database\n): ResumeResult {\n const conn = db ?? getDb();\n\n // Validate trace_id exists\n const task = getTask(traceId, conn);\n if (!task) {\n throw new Error(`Task not found: ${traceId}`);\n }\n\n if (task.status === 'RESOLVED') {\n throw new Error(`Task already resolved: ${traceId}`);\n }\n\n if (task.status === 'PENDING') {\n throw new Error(`Task not yet dispatched: ${traceId}`);\n }\n\n // Resolve the task\n const updated = resolveTask(traceId, resultData, conn);\n if (!updated) {\n throw new Error(`Failed to resolve task: ${traceId}`);\n }\n\n // Check if the agent has other active tasks; if not, mark IDLE\n const activeTasks = listTasksByAssignee(task.assignee_id, conn).filter(\n t => t.status === 'DISPATCHED' || t.status === 'PENDING'\n );\n if (activeTasks.length === 0) {\n updateAgentStatus(task.assignee_id, 'IDLE', conn);\n }\n\n // Check if the whole job is now complete\n const jobComplete = isJobComplete(task.job_id, conn);\n const job = jobComplete ? getJobWithTasks(task.job_id, conn) : undefined;\n\n const resolvedTask = getTask(traceId, conn)!;\n\n return { task: resolvedTask, jobComplete, job };\n}\n\nexport function rejectTask(\n traceId: string,\n newDeadline?: string,\n db?: Database.Database\n): HumanTask {\n const conn = db ?? getDb();\n\n const task = getTask(traceId, conn);\n if (!task) {\n throw new Error(`Task not found: ${traceId}`);\n }\n\n // Default: extend deadline by 24h from now\n const deadline =\n newDeadline ?? new Date(Date.now() + 24 * 60 * 60 * 1000).toISOString();\n\n resetTaskDeadline(traceId, deadline, conn);\n\n return getTask(traceId, conn)!;\n}\n","import { Router } from 'express';\nimport { aggregateJob, syncToOpenClaw } from '../services/aggregation.js';\n\nconst router = Router();\n\n// POST /api/v1/jobs/:job_id/sync - Aggregate and sync to OpenClaw\nrouter.post('/:job_id/sync', async (req, res) => {\n const { job_id } = req.params;\n\n try {\n const aggregation = aggregateJob(job_id);\n const syncResult = await syncToOpenClaw(aggregation);\n\n res.json({\n aggregation,\n sync: syncResult,\n });\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Unknown error';\n res.status(400).json({ error: message });\n }\n});\n\nexport default router;\n","import type Database from 'better-sqlite3';\nimport { getDb } from '../db/connection.js';\nimport { getJobWithTasks, isJobComplete } from '../models/job.js';\nimport type { JobWithTasks } from '../models/types.js';\n\nexport interface AggregationResult {\n job_id: string;\n original_prompt: string;\n openclaw_callback: string;\n results: Array<{\n trace_id: string;\n assignee_id: string;\n todo_description: string;\n result_data: unknown;\n }>;\n aggregated_at: string;\n}\n\nexport function aggregateJob(\n jobId: string,\n db?: Database.Database\n): AggregationResult {\n const conn = db ?? getDb();\n\n const job = getJobWithTasks(jobId, conn);\n if (!job) {\n throw new Error(`Job not found: ${jobId}`);\n }\n\n if (!isJobComplete(jobId, conn)) {\n const resolved = job.tasks.filter(t => t.status === 'RESOLVED').length;\n throw new Error(\n `Job not complete: ${resolved}/${job.tasks.length} tasks resolved`\n );\n }\n\n return {\n job_id: job.job_id,\n original_prompt: job.original_prompt,\n openclaw_callback: job.openclaw_callback,\n results: job.tasks.map(t => ({\n trace_id: t.trace_id,\n assignee_id: t.assignee_id,\n todo_description: t.todo_description,\n result_data: t.result_data,\n })),\n aggregated_at: new Date().toISOString(),\n };\n}\n\nexport async function syncToOpenClaw(\n aggregation: AggregationResult\n): Promise<{ success: boolean; message: string }> {\n if (!aggregation.openclaw_callback) {\n return {\n success: true,\n message: 'No OpenClaw callback configured, skipping sync',\n };\n }\n\n try {\n const response = await fetch(aggregation.openclaw_callback, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(aggregation),\n });\n\n if (!response.ok) {\n return {\n success: false,\n message: `OpenClaw sync failed: ${response.status} ${response.statusText}`,\n };\n }\n\n return { success: true, message: 'Synced to OpenClaw successfully' };\n } catch (error) {\n const message =\n error instanceof Error ? error.message : 'Unknown error';\n return { success: false, message: `OpenClaw sync error: ${message}` };\n }\n}\n","import { Router } from 'express';\nimport { getConfig, setConfig, deleteConfig } from '../models/config.js';\n\nconst router = Router();\n\n// GET /api/v1/config - Get LLM config (masks API key)\nrouter.get('/', (_req, res) => {\n const provider = getConfig('llm_provider') || process.env.HUMANCLAW_LLM_PROVIDER || 'claude';\n const apiKey = getConfig('llm_api_key') || process.env.HUMANCLAW_LLM_API_KEY || '';\n const model = getConfig('llm_model') || process.env.HUMANCLAW_LLM_MODEL || '';\n const baseUrl = getConfig('llm_base_url') || process.env.HUMANCLAW_LLM_BASE_URL || '';\n\n const keySource = getConfig('llm_api_key') ? 'dashboard' : (process.env.HUMANCLAW_LLM_API_KEY ? 'env' : 'none');\n\n res.json({\n provider,\n api_key_set: apiKey.length > 0,\n api_key_masked: apiKey ? apiKey.slice(0, 8) + '...' + apiKey.slice(-4) : '',\n api_key_source: keySource,\n model,\n base_url: baseUrl,\n });\n});\n\n// PUT /api/v1/config - Update LLM config\nrouter.put('/', (req, res) => {\n const { provider, api_key, model, base_url } = req.body as {\n provider?: string;\n api_key?: string;\n model?: string;\n base_url?: string;\n };\n\n if (provider !== undefined) {\n if (provider !== 'claude' && provider !== 'openai') {\n res.status(400).json({ error: '不支持的提供商,支持: claude, openai' });\n return;\n }\n setConfig('llm_provider', provider);\n }\n\n if (api_key !== undefined) {\n if (api_key === '') {\n deleteConfig('llm_api_key');\n } else {\n setConfig('llm_api_key', api_key);\n }\n }\n\n if (model !== undefined) {\n if (model === '') {\n deleteConfig('llm_model');\n } else {\n setConfig('llm_model', model);\n }\n }\n\n if (base_url !== undefined) {\n if (base_url === '') {\n deleteConfig('llm_base_url');\n } else {\n setConfig('llm_base_url', base_url);\n }\n }\n\n res.json({ ok: true });\n});\n\nexport default router;\n","export function getDashboardHtml(): string {\n return `<!DOCTYPE html>\n<html lang=\"zh-CN\">\n<head>\n<meta charset=\"UTF-8\"/>\n<meta name=\"viewport\" content=\"width=device-width,initial-scale=1.0\"/>\n<title>HumanClaw - 异步物理节点编排</title>\n<style>\n:root{--bg:#0f1117;--surface:#1a1d27;--surface-hover:#242836;--border:#2e3346;--text:#e2e4ed;--text-dim:#7b8196;--accent:#00d4ff;--accent-dim:rgba(0,212,255,.12);--green:#22c55e;--yellow:#eab308;--red:#ef4444;--purple:#a855f7;--font-mono:'SF Mono','JetBrains Mono','Fira Code',monospace;--font-sans:-apple-system,'Inter','Segoe UI',sans-serif;--radius:10px}\n*{margin:0;padding:0;box-sizing:border-box}\nbody{background:var(--bg);color:var(--text);font-family:var(--font-sans);min-height:100vh;line-height:1.5}\nheader{padding:20px 32px 12px;border-bottom:1px solid var(--border);display:flex;align-items:center;gap:14px}\nheader h1{font-family:var(--font-mono);font-size:22px;color:var(--accent);letter-spacing:1.5px}\n.subtitle{color:var(--text-dim);font-size:12px;border-left:1px solid var(--border);padding-left:14px}\n.hdr-spacer{flex:1}\n.gear-btn{background:none;border:1px solid var(--border);color:var(--text-dim);width:34px;height:34px;border-radius:8px;cursor:pointer;font-size:18px;display:flex;align-items:center;justify-content:center;transition:all .15s}\n.gear-btn:hover{color:var(--accent);border-color:var(--accent);background:var(--accent-dim)}\nnav{display:flex;border-bottom:1px solid var(--border);padding:0 32px;gap:4px}\n.tab{background:none;border:none;color:var(--text-dim);padding:12px 18px;font-size:13px;cursor:pointer;border-bottom:2px solid transparent;transition:all .15s;font-family:var(--font-sans);font-weight:500}\n.tab:hover{color:var(--text);background:var(--surface)}\n.tab.active{color:var(--accent);border-bottom-color:var(--accent)}\nmain{padding:24px 32px;max-width:1200px}\n.panel{display:none}.panel.active{display:block}\n/* Cards */\n.card{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:20px;transition:border-color .15s}\n.card:hover{border-color:color-mix(in srgb,var(--accent) 40%,transparent)}\n.grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(260px,1fr));gap:14px;margin-top:16px}\n/* Agent Card */\n.agent-header{display:flex;align-items:center;gap:10px;margin-bottom:10px}\n.dot{width:10px;height:10px;border-radius:50%;flex-shrink:0}\n.dot.IDLE{background:var(--green);box-shadow:0 0 8px var(--green)}.dot.BUSY{background:var(--yellow);box-shadow:0 0 8px var(--yellow)}.dot.OFFLINE{background:var(--red);box-shadow:0 0 6px var(--red)}.dot.OOM{background:var(--purple);box-shadow:0 0 8px var(--purple)}\n.agent-name{font-weight:600;font-size:15px}\n.agent-id{color:var(--text-dim);font-family:var(--font-mono);font-size:11px;margin-bottom:8px}\n.caps{display:flex;flex-wrap:wrap;gap:5px;margin-top:6px}\n.cap{background:var(--accent-dim);border:1px solid color-mix(in srgb,var(--accent) 25%,transparent);border-radius:4px;padding:1px 8px;font-size:11px;color:var(--accent)}\n.agent-meta{margin-top:10px;font-size:11px;color:var(--text-dim);display:flex;gap:12px}\n.agent-actions{margin-top:12px;display:flex;gap:6px}\n.agent-actions select{background:var(--surface-hover);color:var(--text);border:1px solid var(--border);border-radius:6px;padding:4px 8px;font-size:11px;cursor:pointer;outline:none}\n.agent-actions button{background:var(--red);color:#fff;border:none;border-radius:6px;padding:4px 10px;font-size:11px;cursor:pointer}\n.agent-actions button:hover{opacity:.8}\n/* Stats Bar */\n.stats{display:flex;gap:12px;flex-wrap:wrap}\n.stat{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:14px 22px;text-align:center;min-width:90px}\n.stat-val{font-size:28px;font-weight:700;font-family:var(--font-mono)}\n.stat-lbl{font-size:10px;color:var(--text-dim);text-transform:uppercase;letter-spacing:1px;margin-top:2px}\n/* Forms */\n.form-card{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:24px;margin-top:16px;max-width:620px}\n.form-card h3{font-size:15px;color:var(--accent);font-family:var(--font-mono);margin-bottom:16px}\n.fg{margin-bottom:14px}\n.fg label{display:block;font-size:12px;color:var(--text-dim);margin-bottom:5px;font-weight:500}\n.fg input,.fg textarea,.fg select{width:100%;background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:10px 12px;color:var(--text);font-size:13px;font-family:var(--font-sans);outline:none;transition:border-color .15s}\n.fg input:focus,.fg textarea:focus,.fg select:focus{border-color:var(--accent)}\n.fg textarea{min-height:80px;resize:vertical;font-family:var(--font-mono);font-size:12px}\n.fg .hint{font-size:11px;color:var(--text-dim);margin-top:4px}\n.btn{padding:10px 22px;border:none;border-radius:8px;font-size:13px;font-weight:600;cursor:pointer;transition:all .15s;display:inline-flex;align-items:center;gap:6px}\n.btn:hover{opacity:.85;transform:translateY(-1px)}\n.btn:active{transform:translateY(0)}\n.btn:disabled{opacity:.5;cursor:not-allowed;transform:none}\n.btn-primary{background:var(--accent);color:var(--bg)}\n.btn-green{background:var(--green);color:#fff}\n.btn-danger{background:var(--red);color:#fff}\n.btn-ghost{background:transparent;color:var(--accent);border:1px solid var(--border)}\n.btn-ghost:hover{background:var(--accent-dim)}\n.btn-sm{padding:6px 14px;font-size:12px}\n.btn-group{display:flex;gap:10px;margin-top:16px;flex-wrap:wrap}\n/* Section header */\n.section-hd{display:flex;justify-content:space-between;align-items:center;margin-bottom:4px}\n.section-hd h2{font-size:16px;font-weight:600}\n/* Job card */\n.job-card{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:20px;margin-top:14px}\n.job-hd{display:flex;justify-content:space-between;align-items:center;margin-bottom:10px}\n.job-title{font-weight:600;font-size:14px}\n.job-id{font-family:var(--font-mono);font-size:11px;color:var(--text-dim)}\n.pbar-wrap{margin:6px 0 14px}\n.pbar-label{font-size:11px;color:var(--text-dim);margin-bottom:4px}\n.pbar{background:var(--surface-hover);border-radius:4px;height:6px;overflow:hidden}\n.pbar-fill{background:var(--accent);height:100%;border-radius:4px;transition:width .3s}\n.kanban{display:grid;grid-template-columns:1fr 1fr 1fr;gap:10px}\n.lane{min-height:40px}\n.lane-hd{font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:1px;margin-bottom:6px;padding-bottom:4px;border-bottom:2px solid var(--border)}\n.lane-hd.y{color:var(--yellow);border-color:var(--yellow)}.lane-hd.r{color:var(--red);border-color:var(--red)}.lane-hd.g{color:var(--green);border-color:var(--green)}\n.tcard{background:var(--bg);border:1px solid var(--border);border-radius:6px;padding:8px 10px;margin-bottom:6px;font-size:12px;cursor:pointer;transition:all .15s}\n.tcard:hover{border-color:var(--accent);background:var(--surface-hover)}\n.tcard-trace{font-family:var(--font-mono);font-size:10px;color:var(--accent);margin-bottom:3px}\n.tcard-meta{font-size:10px;color:var(--text-dim);margin-top:3px}\n/* Task detail modal */\n.task-detail-hd{display:flex;align-items:center;gap:10px;margin-bottom:16px}\n.task-detail-hd .dot{width:12px;height:12px}\n.task-detail-status{font-size:11px;padding:3px 10px;border-radius:12px;font-weight:600;text-transform:uppercase}\n.task-detail-status.DISPATCHED,.task-detail-status.PENDING{background:rgba(234,179,8,.15);color:var(--yellow)}\n.task-detail-status.OVERDUE{background:rgba(239,68,68,.15);color:var(--red)}\n.task-detail-status.RESOLVED{background:rgba(34,197,94,.15);color:var(--green)}\n.trace-copy-row{display:flex;align-items:center;gap:8px;background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:8px 12px;margin-bottom:14px;font-family:var(--font-mono);font-size:13px;color:var(--accent)}\n.trace-copy-row button{background:var(--surface-hover);border:1px solid var(--border);color:var(--text-dim);border-radius:4px;padding:3px 10px;font-size:11px;cursor:pointer;margin-left:auto;flex-shrink:0}\n.trace-copy-row button:hover{color:var(--accent);border-color:var(--accent)}\n.detail-row{margin-bottom:12px}\n.detail-label{font-size:11px;color:var(--text-dim);margin-bottom:3px;font-weight:500}\n.detail-value{font-size:13px}\n.result-display{background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:12px;font-family:var(--font-mono);font-size:12px;white-space:pre-wrap;max-height:200px;overflow-y:auto}\n/* Toast */\n.toast{position:fixed;bottom:24px;right:24px;padding:12px 20px;border-radius:8px;font-size:13px;z-index:9999;animation:slide-up .25s ease;font-weight:500;box-shadow:0 8px 24px rgba(0,0,0,.4)}\n.toast.ok{background:var(--green);color:#fff}.toast.err{background:var(--red);color:#fff}\n@keyframes slide-up{from{transform:translateY(16px);opacity:0}to{transform:translateY(0);opacity:1}}\n/* Empty state */\n.empty-state{text-align:center;padding:48px 20px;color:var(--text-dim)}\n.empty-state .icon{font-size:48px;margin-bottom:12px;opacity:.6}\n.empty-state p{font-size:14px;margin-bottom:16px;max-width:360px;margin-left:auto;margin-right:auto;line-height:1.6}\n/* Modal/Overlay */\n.overlay{position:fixed;inset:0;background:rgba(0,0,0,.6);z-index:100;display:flex;align-items:flex-start;justify-content:center;padding-top:60px;animation:fade-in .15s ease}\n.overlay .form-card{max-width:620px;width:100%;max-height:85vh;overflow-y:auto;margin:0}\n@keyframes fade-in{from{opacity:0}to{opacity:1}}\n/* Agent chips for AI planning */\n.chip-bar{display:flex;gap:6px;margin-bottom:10px;flex-wrap:wrap}\n.chip-filter{background:var(--surface-hover);color:var(--text-dim);border:1px solid var(--border);border-radius:14px;padding:3px 12px;font-size:11px;cursor:pointer;transition:all .15s}\n.chip-filter.active{background:var(--accent-dim);color:var(--accent);border-color:var(--accent)}\n.agent-chips{display:flex;flex-wrap:wrap;gap:6px;margin-top:8px}\n.achip{background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:6px 12px;font-size:12px;cursor:pointer;transition:all .15s;display:flex;align-items:center;gap:6px;user-select:none}\n.achip:hover{border-color:var(--text-dim)}\n.achip.selected{border-color:var(--accent);background:var(--accent-dim)}\n.achip .adot{width:8px;height:8px;border-radius:50%;flex-shrink:0}\n.achip .adot.IDLE{background:var(--green)}.achip .adot.BUSY{background:var(--yellow)}.achip .adot.OFFLINE{background:var(--red)}.achip .adot.OOM{background:var(--purple)}\n/* Plan preview cards */\n.plan-card{background:var(--bg);border:1px solid var(--border);border-radius:var(--radius);padding:16px;margin-bottom:12px}\n.plan-card-hd{display:flex;align-items:center;gap:8px;margin-bottom:8px}\n.plan-card-hd .adot{width:8px;height:8px;border-radius:50%}\n.plan-card-agent{font-weight:600;font-size:13px}\n.plan-card-agentid{font-family:var(--font-mono);font-size:10px;color:var(--text-dim)}\n.plan-card-desc{font-size:13px;margin-bottom:10px;font-weight:500}\n.briefing-box{background:var(--surface);border-left:3px solid var(--accent);border-radius:0 6px 6px 0;padding:10px 14px;font-size:12px;line-height:1.6;color:var(--text);margin-bottom:8px;position:relative}\n.briefing-label{font-size:10px;color:var(--accent);font-weight:600;margin-bottom:4px;font-family:var(--font-mono)}\n.briefing-copy{position:absolute;top:8px;right:8px;background:var(--surface-hover);border:1px solid var(--border);color:var(--text-dim);border-radius:4px;padding:2px 8px;font-size:10px;cursor:pointer}\n.briefing-copy:hover{color:var(--accent);border-color:var(--accent)}\n.plan-card-dl{font-size:11px;color:var(--text-dim);font-family:var(--font-mono)}\n/* Spinner */\n.spinner-wrap{text-align:center;padding:40px 20px}\n.spinner{display:inline-block;width:28px;height:28px;border:3px solid var(--border);border-top-color:var(--accent);border-radius:50%;animation:spin .6s linear infinite}\n@keyframes spin{to{transform:rotate(360deg)}}\n.spinner-text{color:var(--text-dim);font-size:13px;margin-top:12px}\n/* Manual task row */\n.task-row{background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:14px;margin-bottom:10px;position:relative}\n.task-row .remove-task{position:absolute;top:8px;right:10px;background:none;border:none;color:var(--red);cursor:pointer;font-size:16px;line-height:1}\n.task-row .fg{margin-bottom:10px}\n.task-row .fg:last-child{margin-bottom:0}\n</style>\n</head>\n<body>\n<header>\n <h1>HumanClaw</h1>\n <span class=\"subtitle\">异步物理节点编排框架</span>\n <span class=\"hdr-spacer\"></span>\n <button class=\"gear-btn\" onclick=\"showSettings()\" title=\"设置\">&#9881;</button>\n</header>\n<nav>\n <button class=\"tab active\" data-panel=\"fleet\">碳基算力池</button>\n <button class=\"tab\" data-panel=\"pipeline\">编排大盘</button>\n <button class=\"tab\" data-panel=\"terminal\">I/O 终端</button>\n</nav>\n<main>\n <section id=\"fleet\" class=\"panel active\"></section>\n <section id=\"pipeline\" class=\"panel\"></section>\n <section id=\"terminal\" class=\"panel\"></section>\n</main>\n\n<script>\nconst API='/api/v1';\nfunction esc(s){const d=document.createElement('div');d.textContent=s;return d.innerHTML}\nfunction toast(msg,ok){\n document.querySelectorAll('.toast').forEach(t=>t.remove());\n const t=document.createElement('div');t.className='toast '+(ok?'ok':'err');t.textContent=msg;document.body.appendChild(t);\n setTimeout(()=>t.remove(),3500);\n}\nlet cachedAgents=[];\nlet currentPlan=null;\nlet selectedAgentIds=new Set();\n\n// ═══════════════════════════════════════════════\n// FLEET\n// ═══════════════════════════════════════════════\nasync function loadFleet(el){\n el.innerHTML='<div class=\"empty-state\"><p>加载中...</p></div>';\n try{\n const r=await fetch(API+'/nodes/status');\n const d=await r.json();\n cachedAgents=d.agents||[];\n let h='<div class=\"section-hd\"><h2>碳基算力池</h2><button class=\"btn btn-primary btn-sm\" onclick=\"showAddAgent()\">+ 添加节点</button></div>';\n h+='<div class=\"stats\">';\n h+='<div class=\"stat\"><div class=\"stat-val\">'+d.total+'</div><div class=\"stat-lbl\">总计</div></div>';\n h+='<div class=\"stat\"><div class=\"stat-val\" style=\"color:var(--green)\">'+d.idle+'</div><div class=\"stat-lbl\">空闲</div></div>';\n h+='<div class=\"stat\"><div class=\"stat-val\" style=\"color:var(--yellow)\">'+d.busy+'</div><div class=\"stat-lbl\">忙碌</div></div>';\n h+='<div class=\"stat\"><div class=\"stat-val\" style=\"color:var(--red)\">'+d.offline+'</div><div class=\"stat-lbl\">离线</div></div>';\n h+='<div class=\"stat\"><div class=\"stat-val\" style=\"color:var(--purple)\">'+d.oom+'</div><div class=\"stat-lbl\">崩溃</div></div>';\n h+='</div>';\n if(!d.agents.length){\n h+='<div class=\"empty-state\" style=\"margin-top:32px\"><div class=\"icon\">👤</div><p>还没有物理节点。点击上方「+ 添加节点」注册你的第一个碳基算力单元。</p></div>';\n el.innerHTML=h;return;\n }\n h+='<div class=\"grid\">';\n for(const a of d.agents){\n h+='<div class=\"card\"><div class=\"agent-header\"><span class=\"dot '+a.status+'\"></span><span class=\"agent-name\">'+esc(a.name)+'</span></div>';\n h+='<div class=\"agent-id\">'+a.agent_id+'</div>';\n h+='<div class=\"caps\">';for(const c of a.capabilities)h+='<span class=\"cap\">'+esc(c)+'</span>';h+='</div>';\n h+='<div class=\"agent-meta\"><span>任务: '+a.active_task_count+'</span>';\n if(a.avg_delivery_hours!==null)h+='<span>平均交付: '+a.avg_delivery_hours+'h</span>';\n h+='</div>';\n h+='<div class=\"agent-actions\">';\n h+='<select onchange=\"changeStatus(\\\\''+a.agent_id+'\\\\',this.value)\">';\n for(const s of ['IDLE','BUSY','OFFLINE','OOM'])h+='<option'+(s===a.status?' selected':'')+'>'+s+'</option>';\n h+='</select>';\n h+='<button onclick=\"deleteAgent(\\\\''+a.agent_id+'\\\\')\">删除</button>';\n h+='</div></div>';\n }\n h+='</div>';\n el.innerHTML=h;\n }catch(e){el.innerHTML='<div class=\"empty-state\"><p>加载失败: '+e.message+'</p></div>'}\n}\nwindow.showAddAgent=function(){\n const ov=document.createElement('div');ov.className='overlay';ov.id='overlay';\n ov.addEventListener('click',e=>{if(e.target===ov)ov.remove()});\n ov.innerHTML='<div class=\"form-card\"><h3>+ 添加物理节点</h3>'\n +'<div class=\"fg\"><label>节点名称</label><input id=\"aa-name\" placeholder=\"例: 前端老李\"/></div>'\n +'<div class=\"fg\"><label>技能标签</label><input id=\"aa-caps\" placeholder=\"例: UI/UX, 前端开发, 抗压能力强\"/><div class=\"hint\">多个标签用逗号分隔</div></div>'\n +'<div class=\"btn-group\"><button class=\"btn btn-primary\" onclick=\"submitAgent()\">注册节点</button><button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button></div></div>';\n document.body.appendChild(ov);\n document.getElementById('aa-name').focus();\n};\nwindow.submitAgent=async function(){\n const name=document.getElementById('aa-name').value.trim();\n const caps=document.getElementById('aa-caps').value.trim();\n if(!name){toast('请输入节点名称',false);return}\n if(!caps){toast('请输入至少一个技能标签',false);return}\n try{\n const r=await fetch(API+'/nodes',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({name,capabilities:caps.split(',').map(s=>s.trim()).filter(Boolean)})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'注册失败',false);return}\n toast('节点 '+d.agent_id+' 注册成功!',true);\n document.getElementById('overlay').remove();\n load('fleet');\n }catch{toast('网络错误',false)}\n};\nwindow.changeStatus=async function(id,status){\n try{\n const r=await fetch(API+'/nodes/'+id+'/status',{method:'PATCH',headers:{'Content-Type':'application/json'},body:JSON.stringify({status})});\n if(!r.ok){const d=await r.json();toast(d.error||'更新失败',false);return}\n toast('状态已更新',true);\n }catch{toast('网络错误',false)}\n};\nwindow.deleteAgent=async function(id){\n if(!confirm('确定删除节点 '+id+'?'))return;\n try{\n const r=await fetch(API+'/nodes/'+id,{method:'DELETE'});\n if(!r.ok){const d=await r.json();toast(d.error||'删除失败',false);return}\n toast('节点已删除',true);load('fleet');\n }catch{toast('网络错误',false)}\n};\n\n// ═══════════════════════════════════════════════\n// PIPELINE\n// ═══════════════════════════════════════════════\nlet allTasks=[];\nfunction tcard(t){\n return '<div class=\"tcard\" onclick=\"showTaskDetail(\\\\''+t.trace_id+'\\\\')\"><div class=\"tcard-trace\">'+t.trace_id+'</div><div>'+esc(t.todo_description)+'</div><div class=\"tcard-meta\">'+t.assignee_id+' | '+new Date(t.deadline).toLocaleString()+'</div></div>';\n}\nasync function loadPipeline(el){\n el.innerHTML='<div class=\"empty-state\"><p>加载中...</p></div>';\n allTasks=[];\n try{\n try{const ar=await fetch(API+'/nodes/status');const ad=await ar.json();cachedAgents=ad.agents||[]}catch{}\n const r=await fetch(API+'/jobs/active');\n const d=await r.json();\n let h='<div class=\"section-hd\"><h2>异步编排大盘</h2><button class=\"btn btn-primary btn-sm\" onclick=\"showCreateJob()\">+ 创建任务</button></div>';\n if(!d.jobs.length){\n h+='<div class=\"empty-state\" style=\"margin-top:32px\"><div class=\"icon\">📋</div><p>暂无进行中的任务。点击上方「+ 创建任务」,输入需求后 AI 自动规划分发。';\n if(!cachedAgents.length)h+='<br/><br/>⚠️ 需要先在「碳基算力池」中添加物理节点。';\n h+='</p></div>';\n el.innerHTML=h;return;\n }\n for(const j of d.jobs){\n const res=j.tasks.filter(t=>t.status==='RESOLVED').length;\n const tot=j.tasks.length;\n const pct=tot?Math.round(res/tot*100):0;\n const dispatched=j.tasks.filter(t=>t.status==='DISPATCHED'||t.status==='PENDING');\n const overdue=j.tasks.filter(t=>t.status==='OVERDUE');\n const done=j.tasks.filter(t=>t.status==='RESOLVED');\n allTasks.push(...j.tasks);\n h+='<div class=\"job-card\"><div class=\"job-hd\"><span class=\"job-title\">'+esc(j.original_prompt)+'</span><span class=\"job-id\">'+j.job_id+'</span></div>';\n h+='<div class=\"pbar-wrap\"><div class=\"pbar-label\">'+res+'/'+tot+' 已完成 ('+pct+'%)</div><div class=\"pbar\"><div class=\"pbar-fill\" style=\"width:'+pct+'%\"></div></div></div>';\n if(pct===100)h+='<div style=\"margin-bottom:12px\"><button class=\"btn btn-green btn-sm\" onclick=\"syncJob(\\\\''+j.job_id+'\\\\')\">聚合并同步到 OpenClaw</button></div>';\n h+='<div class=\"kanban\"><div class=\"lane\"><div class=\"lane-hd y\">已分发 ('+dispatched.length+')</div>'+dispatched.map(tcard).join('')+'</div>';\n h+='<div class=\"lane\"><div class=\"lane-hd r\">已超时 ('+overdue.length+')</div>'+overdue.map(tcard).join('')+'</div>';\n h+='<div class=\"lane\"><div class=\"lane-hd g\">已交付 ('+done.length+')</div>'+done.map(tcard).join('')+'</div></div></div>';\n }\n el.innerHTML=h;\n }catch(e){el.innerHTML='<div class=\"empty-state\"><p>加载失败: '+e.message+'</p></div>'}\n}\n\n// ─── AI Planning Flow ────────────────────────\nwindow.showCreateJob=function(){\n if(!cachedAgents.length){toast('请先在「碳基算力池」中添加至少一个物理节点',false);return}\n currentPlan=null;\n // Pre-select all IDLE agents\n selectedAgentIds=new Set(cachedAgents.filter(a=>a.status==='IDLE').map(a=>a.agent_id));\n const ov=document.createElement('div');ov.className='overlay';ov.id='overlay';\n ov.addEventListener('click',e=>{if(e.target===ov)ov.remove()});\n renderPlanStep1(ov);\n document.body.appendChild(ov);\n};\n\nfunction renderPlanStep1(ov){\n let chipFilter='all';\n function renderChips(filter){\n chipFilter=filter;\n const filtered=filter==='idle'?cachedAgents.filter(a=>a.status==='IDLE'):cachedAgents;\n let ch='<div class=\"chip-bar\">';\n ch+='<span class=\"chip-filter'+(filter==='all'?' active':'')+'\" onclick=\"window._filterChips(\\\\'all\\\\')\">全部</span>';\n ch+='<span class=\"chip-filter'+(filter==='idle'?' active':'')+'\" onclick=\"window._filterChips(\\\\'idle\\\\')\">仅空闲</span>';\n ch+='</div>';\n ch+='<div class=\"agent-chips\">';\n for(const a of filtered){\n const sel=selectedAgentIds.has(a.agent_id);\n ch+='<div class=\"achip'+(sel?' selected':'')+'\" onclick=\"window._toggleChip(\\\\''+a.agent_id+'\\\\')\">';\n ch+='<span class=\"adot '+a.status+'\"></span>';\n ch+=esc(a.name);\n ch+='</div>';\n }\n ch+='</div>';\n return ch;\n }\n window._filterChips=function(f){\n chipFilter=f;\n const el=document.getElementById('agent-chip-area');\n if(el)el.innerHTML=renderChips(f);\n };\n window._toggleChip=function(id){\n if(selectedAgentIds.has(id))selectedAgentIds.delete(id);\n else selectedAgentIds.add(id);\n const el=document.getElementById('agent-chip-area');\n if(el)el.innerHTML=renderChips(chipFilter);\n };\n\n ov.innerHTML='<div class=\"form-card\"><h3>AI 智能规划</h3>'\n +'<div class=\"fg\"><label>输入你的需求</label><textarea id=\"plan-prompt\" rows=\"3\" placeholder=\"例: 完成首页重构,包括导航栏、内容区和页脚的响应式改版\" style=\"font-family:var(--font-sans);font-size:13px\"></textarea></div>'\n +'<div class=\"fg\"><label>选择参与的物理节点 <span style=\"color:var(--text-dim);font-weight:400\">(默认选中空闲节点)</span></label>'\n +'<div id=\"agent-chip-area\">'+renderChips('all')+'</div></div>'\n +'<div class=\"fg\"><label>OpenClaw 回调地址 <span style=\"color:var(--text-dim);font-weight:400\">(可选)</span></label><input id=\"plan-callback\" placeholder=\"https://...\"/></div>'\n +'<div class=\"btn-group\">'\n +'<button class=\"btn btn-primary\" onclick=\"planWithAI()\">AI 规划</button>'\n +'<button class=\"btn btn-ghost\" onclick=\"showManualCreate()\">手动创建</button>'\n +'<button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button>'\n +'</div></div>';\n setTimeout(()=>{const ta=document.getElementById('plan-prompt');if(ta)ta.focus()},50);\n};\n\nwindow.planWithAI=async function(){\n const prompt=document.getElementById('plan-prompt').value.trim();\n const callback=document.getElementById('plan-callback')?.value.trim()||'';\n if(!prompt){toast('请输入需求描述',false);return}\n if(selectedAgentIds.size===0){toast('请至少选择一个物理节点',false);return}\n\n const ov=document.getElementById('overlay');\n const fc=ov.querySelector('.form-card');\n fc.innerHTML='<h3>AI 智能规划</h3><div class=\"spinner-wrap\"><div class=\"spinner\"></div><div class=\"spinner-text\">AI 正在分析需求、匹配节点、生成话术...</div></div>';\n\n try{\n const r=await fetch(API+'/jobs/plan',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({prompt,agent_ids:Array.from(selectedAgentIds)})});\n const d=await r.json();\n if(!r.ok){\n toast(d.error||'规划失败',false);\n renderPlanStep1(ov);\n return;\n }\n currentPlan={...d,openclaw_callback:callback};\n renderPlanStep2(ov);\n }catch(e){\n toast('网络错误: '+e.message,false);\n renderPlanStep1(ov);\n }\n};\n\nfunction renderPlanStep2(ov){\n const p=currentPlan;\n let h='<h3>规划预览</h3>';\n h+='<div style=\"font-size:12px;color:var(--text-dim);margin-bottom:14px\">需求: '+esc(p.original_prompt)+'</div>';\n for(let i=0;i<p.planned_tasks.length;i++){\n const t=p.planned_tasks[i];\n const agent=cachedAgents.find(a=>a.agent_id===t.assignee_id);\n const status=agent?agent.status:'IDLE';\n h+='<div class=\"plan-card\">';\n h+='<div class=\"plan-card-hd\"><span class=\"adot '+status+'\"></span><span class=\"plan-card-agent\">'+esc(t.assignee_name)+'</span><span class=\"plan-card-agentid\">'+t.assignee_id+'</span></div>';\n h+='<div class=\"plan-card-desc\">'+esc(t.todo_description)+'</div>';\n h+='<div class=\"briefing-box\"><div class=\"briefing-label\">话术 (可直接发给 TA)</div><button class=\"briefing-copy\" onclick=\"window._copyBriefing('+i+')\">复制</button>'+esc(t.briefing)+'</div>';\n h+='<div class=\"plan-card-dl\">截止: '+new Date(t.deadline).toLocaleString()+'</div>';\n h+='</div>';\n }\n h+='<div class=\"btn-group\">';\n h+='<button class=\"btn btn-green\" onclick=\"dispatchPlan()\">确认分发</button>';\n h+='<button class=\"btn btn-ghost\" onclick=\"renderPlanStep1(document.getElementById(\\\\'overlay\\\\'))\">重新规划</button>';\n h+='<button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button>';\n h+='</div>';\n ov.querySelector('.form-card').innerHTML=h;\n}\nwindow.renderPlanStep1=renderPlanStep1;\n\nwindow._copyBriefing=function(i){\n const text=currentPlan.planned_tasks[i].briefing;\n navigator.clipboard.writeText(text).then(()=>toast('话术已复制',true)).catch(()=>toast('复制失败',false));\n};\n\nwindow.dispatchPlan=async function(){\n const p=currentPlan;\n const tasks=p.planned_tasks.map(t=>({assignee_id:t.assignee_id,todo_description:t.todo_description,deadline:t.deadline}));\n try{\n const r=await fetch(API+'/jobs/create',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({original_prompt:p.original_prompt,openclaw_callback:p.openclaw_callback||'',tasks})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'分发失败',false);return}\n toast('任务已分发!Job: '+d.job_id,true);\n document.getElementById('overlay').remove();\n load('pipeline');\n }catch{toast('网络错误',false)}\n};\n\n// ─── Manual creation (fallback) ──────────────\nwindow.showManualCreate=function(){\n const ov=document.getElementById('overlay');\n const optionsHtml=cachedAgents.map(a=>'<option value=\"'+a.agent_id+'\">'+esc(a.name)+' ('+a.agent_id+')</option>').join('');\n const tomorrow=new Date(Date.now()+86400000).toISOString().slice(0,16);\n ov.querySelector('.form-card').innerHTML='<h3>手动创建任务</h3>'\n +'<div class=\"fg\"><label>任务描述 (原始需求)</label><input id=\"cj-prompt\" placeholder=\"例: 完成首页改版\"/></div>'\n +'<div class=\"fg\"><label>OpenClaw 回调地址 (可选)</label><input id=\"cj-callback\" placeholder=\"https://...\"/></div>'\n +'<div style=\"margin-bottom:10px;display:flex;justify-content:space-between;align-items:center\"><label style=\"font-size:13px;font-weight:600;color:var(--text)\">子任务列表</label><button class=\"btn btn-ghost btn-sm\" onclick=\"addTaskRow()\">+ 添加子任务</button></div>'\n +'<div id=\"task-rows\"><div class=\"task-row\"><div class=\"fg\"><label>指派节点</label><select class=\"tr-agent\">'+optionsHtml+'</select></div><div class=\"fg\"><label>任务描述</label><input class=\"tr-desc\" placeholder=\"具体要做什么...\"/></div><div class=\"fg\"><label>截止时间</label><input class=\"tr-deadline\" type=\"datetime-local\" value=\"'+tomorrow+'\"/></div></div></div>'\n +'<div class=\"btn-group\"><button class=\"btn btn-primary\" onclick=\"submitJob()\">创建并分发</button><button class=\"btn btn-ghost\" onclick=\"renderPlanStep1(document.getElementById(\\\\'overlay\\\\'))\">返回 AI 规划</button><button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button></div>';\n};\nwindow.addTaskRow=function(){\n const optionsHtml=cachedAgents.map(a=>'<option value=\"'+a.agent_id+'\">'+esc(a.name)+' ('+a.agent_id+')</option>').join('');\n const tomorrow=new Date(Date.now()+86400000).toISOString().slice(0,16);\n const row=document.createElement('div');row.className='task-row';\n row.innerHTML='<button class=\"remove-task\" onclick=\"this.parentElement.remove()\">&times;</button>'\n +'<div class=\"fg\"><label>指派节点</label><select class=\"tr-agent\">'+optionsHtml+'</select></div>'\n +'<div class=\"fg\"><label>任务描述</label><input class=\"tr-desc\" placeholder=\"具体要做什么...\"/></div>'\n +'<div class=\"fg\"><label>截止时间</label><input class=\"tr-deadline\" type=\"datetime-local\" value=\"'+tomorrow+'\"/></div>';\n document.getElementById('task-rows').appendChild(row);\n};\nwindow.submitJob=async function(){\n const prompt=document.getElementById('cj-prompt').value.trim();\n const callback=document.getElementById('cj-callback').value.trim();\n if(!prompt){toast('请输入任务描述',false);return}\n const rows=document.querySelectorAll('.task-row');\n const tasks=[];\n for(const row of rows){\n const aid=row.querySelector('.tr-agent').value;\n const desc=row.querySelector('.tr-desc').value.trim();\n const dl=row.querySelector('.tr-deadline').value;\n if(!desc){toast('每个子任务都需要填写描述',false);return}\n tasks.push({assignee_id:aid,todo_description:desc,deadline:new Date(dl).toISOString()});\n }\n if(!tasks.length){toast('至少添加一个子任务',false);return}\n try{\n const r=await fetch(API+'/jobs/create',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({original_prompt:prompt,openclaw_callback:callback,tasks})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'创建失败',false);return}\n toast('任务已创建并分发!Job: '+d.job_id,true);\n document.getElementById('overlay').remove();\n load('pipeline');\n }catch{toast('网络错误',false)}\n};\nwindow.syncJob=async function(jobId){\n try{\n const r=await fetch(API+'/jobs/'+jobId+'/sync',{method:'POST'});\n const d=await r.json();\n if(!r.ok){toast(d.error||'同步失败',false);return}\n toast('聚合完成!'+d.sync.message,true);\n }catch{toast('网络错误',false)}\n};\n\n// ─── Task Detail Modal ──────────────────────\nwindow.showTaskDetail=function(traceId){\n const t=allTasks.find(x=>x.trace_id===traceId);\n if(!t)return;\n const agent=cachedAgents.find(a=>a.agent_id===t.assignee_id);\n const agentName=agent?agent.name:t.assignee_id;\n const ov=document.createElement('div');ov.className='overlay';ov.id='overlay';\n ov.addEventListener('click',e=>{if(e.target===ov)ov.remove()});\n\n let h='<div class=\"form-card\"><div class=\"task-detail-hd\"><span class=\"task-detail-status '+t.status+'\">'+t.status+'</span><span style=\"font-size:15px;font-weight:600\">任务详情</span></div>';\n\n // Trace ID with copy\n h+='<div class=\"trace-copy-row\"><span>'+t.trace_id+'</span><button onclick=\"navigator.clipboard.writeText(\\\\''+t.trace_id+'\\\\').then(()=>toast(\\\\'已复制\\\\',true))\">复制</button></div>';\n\n // Info rows\n h+='<div class=\"detail-row\"><div class=\"detail-label\">指派节点</div><div class=\"detail-value\">'+esc(agentName)+' <span style=\"color:var(--text-dim);font-family:var(--font-mono);font-size:11px\">'+t.assignee_id+'</span></div></div>';\n h+='<div class=\"detail-row\"><div class=\"detail-label\">任务描述</div><div class=\"detail-value\">'+esc(t.todo_description)+'</div></div>';\n h+='<div class=\"detail-row\"><div class=\"detail-label\">截止时间</div><div class=\"detail-value\" style=\"font-family:var(--font-mono)\">'+new Date(t.deadline).toLocaleString()+'</div></div>';\n\n if(t.status==='RESOLVED'&&t.result_data){\n // Show result\n let resultText='';\n try{const rd=JSON.parse(t.result_data);resultText=rd.text||JSON.stringify(rd,null,2)}catch{resultText=t.result_data}\n h+='<div class=\"detail-row\"><div class=\"detail-label\">交付结果</div><div class=\"result-display\">'+esc(resultText)+'</div></div>';\n h+='<div class=\"btn-group\"><button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">关闭</button></div>';\n }else{\n // Input for resume/reject\n h+='<div class=\"fg\" style=\"margin-top:16px\"><label>提交 Human 交付物</label><textarea id=\"td-payload\" rows=\"4\" placeholder=\"粘贴交付物内容、工作汇报或代码...\"></textarea></div>';\n h+='<div class=\"btn-group\">';\n h+='<button class=\"btn btn-green\" onclick=\"taskResume(\\\\''+t.trace_id+'\\\\')\">提交交付 (Resume)</button>';\n h+='<button class=\"btn btn-danger\" onclick=\"taskReject(\\\\''+t.trace_id+'\\\\')\">打回重做 (Reject)</button>';\n h+='<button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button>';\n h+='</div>';\n }\n h+='</div>';\n ov.innerHTML=h;\n document.body.appendChild(ov);\n};\n\nwindow.taskResume=async function(traceId){\n const payload=document.getElementById('td-payload').value.trim();\n if(!payload){toast('请输入交付内容',false);return}\n try{\n const r=await fetch(API+'/tasks/resume',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({trace_id:traceId,result_data:{text:payload}})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'提交失败',false);return}\n toast(d.job_complete?'任务已交付!Job 已全部完成,可以聚合同步。':'任务 '+traceId+' 已交付。',true);\n document.getElementById('overlay').remove();\n load('pipeline');\n }catch{toast('网络错误',false)}\n};\n\nwindow.taskReject=async function(traceId){\n if(!confirm('确定打回任务 '+traceId+'?截止时间将延长 24 小时。'))return;\n try{\n const r=await fetch(API+'/tasks/reject',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({trace_id:traceId})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'操作失败',false);return}\n toast('任务 '+traceId+' 已打回,截止时间已延长 24 小时。',true);\n document.getElementById('overlay').remove();\n load('pipeline');\n }catch{toast('网络错误',false)}\n};\n\n// ═══════════════════════════════════════════════\n// TERMINAL\n// ═══════════════════════════════════════════════\nfunction loadTerminal(el){\n el.innerHTML='<div class=\"section-hd\"><h2>I/O 交付终端</h2></div>'\n +'<div class=\"form-card\" style=\"margin-top:12px\"><h3>> 提交物理节点产出</h3>'\n +'<div class=\"fg\"><label>Trace ID (追踪码)</label><input id=\"t-tid\" placeholder=\"例: TK-9527\"/></div>'\n +'<div class=\"fg\"><label>交付载荷</label><textarea id=\"t-payload\" placeholder=\"粘贴交付物内容、工作汇报或代码...\"></textarea></div>'\n +'<div class=\"btn-group\"><button class=\"btn btn-primary\" onclick=\"doResume()\">提交并恢复 (Resume)</button><button class=\"btn btn-danger\" onclick=\"doReject()\">打回重做 (Reject)</button></div></div>';\n}\nwindow.doResume=async function(){\n const tid=document.getElementById('t-tid').value.trim();\n const payload=document.getElementById('t-payload').value.trim();\n if(!tid){toast('请输入 Trace ID',false);return}\n if(!payload){toast('请输入交付载荷',false);return}\n try{\n const r=await fetch(API+'/tasks/resume',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({trace_id:tid,result_data:{text:payload}})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'提交失败',false);return}\n toast(d.job_complete?'任务已交付!Job 已全部完成,可以聚合同步。':'任务 '+tid+' 已交付。',true);\n document.getElementById('t-tid').value='';document.getElementById('t-payload').value='';\n }catch{toast('网络错误',false)}\n};\nwindow.doReject=async function(){\n const tid=document.getElementById('t-tid').value.trim();\n if(!tid){toast('请输入 Trace ID',false);return}\n try{\n const r=await fetch(API+'/tasks/reject',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({trace_id:tid})});\n const d=await r.json();\n if(!r.ok){toast(d.error||'操作失败',false);return}\n toast('任务 '+tid+' 已打回,截止时间已延长 24 小时。',true);\n document.getElementById('t-tid').value='';document.getElementById('t-payload').value='';\n }catch{toast('网络错误',false)}\n};\n\n// ═══════════════════════════════════════════════\n// SETTINGS\n// ═══════════════════════════════════════════════\nwindow.showSettings=async function(){\n const ov=document.createElement('div');ov.className='overlay';ov.id='overlay';\n ov.addEventListener('click',e=>{if(e.target===ov)ov.remove()});\n ov.innerHTML='<div class=\"form-card\"><h3>LLM 设置</h3><div class=\"spinner-wrap\"><div class=\"spinner\"></div></div></div>';\n document.body.appendChild(ov);\n\n try{\n const r=await fetch(API+'/config');\n const cfg=await r.json();\n const fc=ov.querySelector('.form-card');\n let statusHtml='';\n if(cfg.api_key_set){\n const src=cfg.api_key_source==='dashboard'?'Dashboard 配置':'环境变量';\n statusHtml='<div style=\"background:var(--accent-dim);border:1px solid color-mix(in srgb,var(--accent) 25%,transparent);border-radius:8px;padding:10px 14px;margin-bottom:16px;font-size:12px\"><span style=\"color:var(--green)\">&#10003;</span> API Key 已配置 <span style=\"color:var(--text-dim)\">('+esc(cfg.api_key_masked)+' | 来源: '+src+')</span></div>';\n }else{\n statusHtml='<div style=\"background:rgba(239,68,68,.1);border:1px solid rgba(239,68,68,.25);border-radius:8px;padding:10px 14px;margin-bottom:16px;font-size:12px;color:var(--red)\">&#9888; 未配置 API Key,AI 规划功能不可用</div>';\n }\n fc.innerHTML='<h3>LLM 设置</h3>'+statusHtml\n +'<div class=\"fg\"><label>提供商</label><select id=\"cfg-provider\"><option value=\"claude\"'+(cfg.provider==='claude'?' selected':'')+'>Claude (Anthropic)</option><option value=\"openai\"'+(cfg.provider==='openai'?' selected':'')+'>OpenAI</option></select></div>'\n +'<div class=\"fg\"><label>API Key</label><input id=\"cfg-key\" type=\"password\" placeholder=\"'+(cfg.api_key_set?'已配置,留空则不修改':'输入你的 API Key...')+'\"/><div class=\"hint\">Claude: sk-ant-... | OpenAI: sk-...</div></div>'\n +'<div class=\"fg\"><label>模型 <span style=\"color:var(--text-dim);font-weight:400\">(可选,留空用默认)</span></label><input id=\"cfg-model\" value=\"'+esc(cfg.model||'')+'\" placeholder=\"例: claude-sonnet-4-20250514 / gpt-4o\"/></div>'\n +'<div class=\"fg\"><label>API Base URL <span style=\"color:var(--text-dim);font-weight:400\">(可选,留空用官方地址)</span></label><input id=\"cfg-baseurl\" value=\"'+esc(cfg.base_url||'')+'\" placeholder=\"例: https://your-proxy.com\"/><div class=\"hint\">私有部署: 填写你的模型服务地址,如 vLLM / Ollama / Azure 等</div></div>'\n +'<div class=\"btn-group\"><button class=\"btn btn-primary\" onclick=\"saveSettings()\">保存</button><button class=\"btn btn-ghost\" onclick=\"document.getElementById(\\\\'overlay\\\\').remove()\">取消</button></div>';\n }catch{\n ov.querySelector('.form-card').innerHTML='<h3>LLM 设置</h3><p style=\"color:var(--red)\">加载配置失败</p>';\n }\n};\nwindow.saveSettings=async function(){\n const provider=document.getElementById('cfg-provider').value;\n const apiKey=document.getElementById('cfg-key').value.trim();\n const model=document.getElementById('cfg-model').value.trim();\n const baseUrl=document.getElementById('cfg-baseurl').value.trim();\n const body={provider};\n if(apiKey)body.api_key=apiKey;\n body.model=model;\n body.base_url=baseUrl;\n try{\n const r=await fetch(API+'/config',{method:'PUT',headers:{'Content-Type':'application/json'},body:JSON.stringify(body)});\n if(!r.ok){const d=await r.json();toast(d.error||'保存失败',false);return}\n toast('设置已保存',true);\n document.getElementById('overlay').remove();\n }catch{toast('网络错误',false)}\n};\n\n// ═══════════════════════════════════════════════\n// TABS & INIT\n// ═══════════════════════════════════════════════\nconst tabs=document.querySelectorAll('.tab');\nconst panels=document.querySelectorAll('.panel');\nfunction load(id){\n const el=document.getElementById(id);\n if(id==='fleet')loadFleet(el);\n else if(id==='pipeline')loadPipeline(el);\n else if(id==='terminal')loadTerminal(el);\n}\ntabs.forEach(t=>t.addEventListener('click',()=>{\n tabs.forEach(x=>x.classList.remove('active'));\n panels.forEach(x=>x.classList.remove('active'));\n t.classList.add('active');\n const id=t.dataset.panel;\n document.getElementById(id).classList.add('active');\n load(id);\n}));\nload('fleet');\nsetInterval(()=>{\n const a=document.querySelector('.tab.active');\n if(a&&a.dataset.panel!=='terminal')load(a.dataset.panel);\n},15000);\n</script>\n</body>\n</html>`;\n}\n"],"mappings":"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AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;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","names":["db","db","Router","db","db","db","db","db","DEFAULT_BASE_URL","db","db","router","Router","Router","db","router","Router","Router","db","router","Router","Router","router","Router","db","db","confirm"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@humanclaw/humanclaw",
3
- "version": "1.1.2",
3
+ "version": "1.1.3",
4
4
  "description": "Async physical node orchestration framework - treating humans as distributed worker nodes",
5
5
  "type": "module",
6
6
  "license": "MIT",