pi-crew 0.2.3 → 0.2.4
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/AGENTS.md +57 -32
- package/CHANGELOG.md +466 -448
- package/LICENSE +21 -21
- package/NOTICE.md +16 -16
- package/README.md +323 -323
- package/docs/FEATURE_INTAKE.md +126 -0
- package/docs/HARNESS.md +86 -0
- package/docs/HARNESS_BACKLOG.md +41 -0
- package/docs/TEST_MATRIX.md +49 -0
- package/docs/actions-reference.md +595 -595
- package/docs/architecture.md +180 -180
- package/docs/code-review-2026-05-11.md +592 -592
- package/docs/commands-reference.md +347 -347
- package/docs/comparison-pi-subagents-vs-pi-crew.md +303 -0
- package/docs/decisions/0001-durable-state.md +41 -0
- package/docs/decisions/0002-child-process-for-async.md +42 -0
- package/docs/decisions/0003-depth-guard.md +36 -0
- package/docs/decisions/0004-execfile-over-exec.md +34 -0
- package/docs/decisions/0005-no-parameter-properties.md +49 -0
- package/docs/decisions/0006-publish-bundled-esm.md +63 -0
- package/docs/decisions/0007-active-run-binary-index.md +54 -0
- package/docs/decisions/0008-child-pi-warm-pool.md +61 -0
- package/docs/decisions/README.md +23 -0
- package/docs/followup-review-round4-2026-05-13.md +107 -0
- package/docs/implementation-plan-top3.md +333 -0
- package/docs/live-mailbox-runtime.md +36 -36
- package/docs/next-upgrade-roadmap.md +808 -808
- package/docs/oh-my-pi-research.md +509 -0
- package/docs/perf/baseline-2026-05.md +113 -0
- package/docs/perf/final-report-2026-05.md +206 -0
- package/docs/perf/sprint-1-report.md +71 -0
- package/docs/perf/sprint-2-report.md +81 -0
- package/docs/perf/sprint-2.5-report.md +53 -0
- package/docs/perf/sprint-3-report.md +36 -0
- package/docs/perf/sprint-4-report.md +47 -0
- package/docs/perf/sprint-5-report.md +51 -0
- package/docs/perf/sprint-6-report.md +94 -0
- package/docs/perf/sprint-7-report.md +74 -0
- package/docs/perf/upgrade-plan-2026-05.md +147 -0
- package/docs/pi-subagents3-deep-analysis.md +508 -0
- package/docs/product/README.md +31 -0
- package/docs/product/platform.md +27 -0
- package/docs/product/runtime-safety.md +37 -0
- package/docs/product/team-run.md +39 -0
- package/docs/product/team-tool.md +37 -0
- package/docs/publishing.md +65 -65
- package/docs/resource-formats.md +134 -134
- package/docs/runtime-analysis-child-vs-live.md +171 -0
- package/docs/runtime-flow.md +148 -148
- package/docs/runtime-migration-in-process-analysis.md +250 -0
- package/docs/stories/README.md +30 -0
- package/docs/stories/backlog.md +36 -0
- package/docs/templates/decision.md +27 -0
- package/docs/templates/story.md +44 -0
- package/docs/templates/validation-report.md +32 -0
- package/docs/usage.md +238 -238
- package/index.ts +7 -6
- package/install.mjs +65 -65
- package/package.json +107 -100
- package/schema.json +222 -222
- package/skills/child-pi-spawning/SKILL.md +213 -0
- package/skills/context-artifact-hygiene/SKILL.md +32 -0
- package/skills/event-log-tracing/SKILL.md +299 -0
- package/skills/git-master/SKILL.md +225 -24
- package/skills/live-agent-lifecycle/SKILL.md +192 -0
- package/skills/mailbox-interactive/SKILL.md +300 -19
- package/skills/model-routing-context/SKILL.md +94 -0
- package/skills/multi-perspective-review/SKILL.md +88 -0
- package/skills/read-only-explorer/SKILL.md +250 -26
- package/skills/safe-bash/SKILL.md +307 -21
- package/skills/verification-before-done/SKILL.md +11 -2
- package/skills/widget-rendering/SKILL.md +258 -0
- package/skills/workspace-isolation/SKILL.md +202 -0
- package/skills/worktree-isolation/SKILL.md +202 -18
- package/src/adapters/claude-adapter.ts +25 -25
- package/src/adapters/codex-adapter.ts +21 -21
- package/src/adapters/cursor-adapter.ts +17 -17
- package/src/adapters/export-util.ts +137 -137
- package/src/adapters/index.ts +15 -15
- package/src/adapters/registry.ts +18 -18
- package/src/adapters/types.ts +23 -23
- package/src/agents/agent-config.ts +38 -38
- package/src/agents/agent-serializer.ts +38 -38
- package/src/agents/discover-agents.ts +121 -118
- package/src/config/config.ts +740 -858
- package/src/config/defaults.ts +96 -96
- package/src/config/drift-detector.ts +211 -211
- package/src/config/markers.ts +327 -327
- package/src/config/resilient-parser.ts +109 -108
- package/src/config/suggestions.ts +74 -74
- package/src/config/types.ts +199 -0
- package/src/extension/async-notifier.ts +123 -89
- package/src/extension/autonomous-policy.ts +169 -169
- package/src/extension/cross-extension-rpc.ts +104 -104
- package/src/extension/help.ts +47 -47
- package/src/extension/import-index.ts +69 -69
- package/src/extension/management.ts +395 -382
- package/src/extension/notification-router.ts +116 -116
- package/src/extension/notification-sink.ts +51 -51
- package/src/extension/project-init.ts +168 -168
- package/src/extension/register.ts +859 -668
- package/src/extension/registration/artifact-cleanup.ts +15 -15
- package/src/extension/registration/command-utils.ts +54 -54
- package/src/extension/registration/commands.ts +559 -452
- package/src/extension/registration/compaction-guard.ts +125 -125
- package/src/extension/registration/subagent-helpers.ts +102 -102
- package/src/extension/registration/subagent-tools.ts +220 -159
- package/src/extension/registration/team-tool.ts +159 -99
- package/src/extension/registration/viewers.ts +29 -0
- package/src/extension/result-watcher.ts +128 -128
- package/src/extension/run-bundle-schema.ts +89 -89
- package/src/extension/run-export.ts +73 -73
- package/src/extension/run-import.ts +84 -84
- package/src/extension/run-index.ts +94 -94
- package/src/extension/run-maintenance.ts +142 -142
- package/src/extension/session-summary.ts +8 -8
- package/src/extension/team-manager-command.ts +96 -96
- package/src/extension/team-recommendation.ts +188 -188
- package/src/extension/team-tool/api.ts +5 -2
- package/src/extension/team-tool/cancel.ts +224 -209
- package/src/extension/team-tool/config-patch.ts +36 -36
- package/src/extension/team-tool/context.ts +60 -60
- package/src/extension/team-tool/doctor.ts +242 -242
- package/src/extension/team-tool/handle-settings.ts +421 -195
- package/src/extension/team-tool/inspect.ts +41 -41
- package/src/extension/team-tool/lifecycle-actions.ts +139 -139
- package/src/extension/team-tool/parallel-dispatch.ts +156 -156
- package/src/extension/team-tool/plan.ts +19 -19
- package/src/extension/team-tool/respond.ts +112 -111
- package/src/extension/team-tool/run.ts +246 -229
- package/src/extension/team-tool/status.ts +110 -110
- package/src/extension/team-tool-types.ts +13 -13
- package/src/extension/team-tool.ts +344 -344
- package/src/extension/tool-result.ts +16 -16
- package/src/extension/validate-resources.ts +77 -77
- package/src/hooks/registry.ts +61 -61
- package/src/hooks/types.ts +40 -40
- package/src/i18n.ts +184 -184
- package/src/observability/correlation.ts +35 -35
- package/src/observability/event-to-metric.ts +68 -68
- package/src/observability/exporters/adapter.ts +30 -30
- package/src/observability/exporters/otlp-exporter.ts +106 -92
- package/src/observability/exporters/prometheus-exporter.ts +54 -54
- package/src/observability/metric-registry.ts +87 -87
- package/src/observability/metric-retention.ts +54 -54
- package/src/observability/metric-sink.ts +81 -56
- package/src/observability/metrics-primitives.ts +167 -167
- package/src/prompt/prompt-runtime.ts +72 -72
- package/src/runtime/adaptive-plan.ts +338 -0
- package/src/runtime/agent-control.ts +169 -169
- package/src/runtime/agent-memory.ts +72 -72
- package/src/runtime/agent-observability.ts +114 -114
- package/src/runtime/async-marker.ts +26 -26
- package/src/runtime/async-runner.ts +153 -153
- package/src/runtime/attention-events.ts +28 -28
- package/src/runtime/auto-resume.ts +100 -100
- package/src/runtime/background-runner.ts +122 -89
- package/src/runtime/cancellation.ts +61 -61
- package/src/runtime/capability-inventory.ts +116 -116
- package/src/runtime/child-pi-pool.ts +68 -0
- package/src/runtime/child-pi.ts +541 -461
- package/src/runtime/code-summary.ts +247 -247
- package/src/runtime/compaction-summary.ts +271 -271
- package/src/runtime/concurrency.ts +58 -58
- package/src/runtime/crash-recovery.ts +317 -301
- package/src/runtime/crew-agent-records.ts +379 -281
- package/src/runtime/crew-agent-runtime.ts +60 -60
- package/src/runtime/cross-extension-rpc.ts +72 -0
- package/src/runtime/custom-tools/irc-tool.ts +201 -201
- package/src/runtime/custom-tools/submit-result-tool.ts +90 -90
- package/src/runtime/deadletter.ts +47 -47
- package/src/runtime/delivery-coordinator.ts +176 -176
- package/src/runtime/delta-conflict.ts +360 -360
- package/src/runtime/diagnostic-export.ts +102 -102
- package/src/runtime/direct-run.ts +35 -35
- package/src/runtime/effectiveness.ts +82 -81
- package/src/runtime/errors/crew-errors.ts +166 -0
- package/src/runtime/event-stream-bridge.ts +92 -92
- package/src/runtime/foreground-control.ts +82 -82
- package/src/runtime/green-contract.ts +46 -46
- package/src/runtime/group-join.ts +234 -106
- package/src/runtime/heartbeat-watcher.ts +145 -124
- package/src/runtime/iteration-hooks.ts +267 -267
- package/src/runtime/live-agent-control.ts +88 -88
- package/src/runtime/live-agent-manager.ts +377 -179
- package/src/runtime/live-control-realtime.ts +36 -36
- package/src/runtime/live-session-runtime.ts +676 -600
- package/src/runtime/loop-gates.ts +129 -129
- package/src/runtime/manifest-cache.ts +263 -263
- package/src/runtime/mcp-proxy.ts +113 -113
- package/src/runtime/metric-parser.ts +40 -40
- package/src/runtime/model-fallback.ts +282 -274
- package/src/runtime/model-resolver.ts +118 -0
- package/src/runtime/output-validator.ts +187 -187
- package/src/runtime/overflow-recovery.ts +175 -175
- package/src/runtime/parallel-research.ts +44 -44
- package/src/runtime/parallel-utils.ts +156 -156
- package/src/runtime/parent-guard.ts +80 -80
- package/src/runtime/phase-progress.ts +217 -217
- package/src/runtime/pi-args.ts +165 -165
- package/src/runtime/pi-json-output.ts +111 -111
- package/src/runtime/pi-spawn.ts +167 -167
- package/src/runtime/policy-engine.ts +79 -79
- package/src/runtime/post-checks.ts +125 -125
- package/src/runtime/post-exit-stdio-guard.ts +86 -86
- package/src/runtime/process-status.ts +97 -73
- package/src/runtime/progress-event-coalescer.ts +43 -43
- package/src/runtime/recovery-recipes.ts +74 -74
- package/src/runtime/retry-executor.ts +81 -81
- package/src/runtime/role-permission.ts +39 -39
- package/src/runtime/run-tracker.ts +99 -0
- package/src/runtime/runtime-policy.ts +21 -0
- package/src/runtime/runtime-resolver.ts +94 -91
- package/src/runtime/scheduler.ts +294 -0
- package/src/runtime/semaphore.ts +131 -131
- package/src/runtime/sensitive-paths.ts +92 -92
- package/src/runtime/session-usage.ts +79 -79
- package/src/runtime/settings-store.ts +103 -0
- package/src/runtime/sidechain-output.ts +29 -29
- package/src/runtime/skill-instructions.ts +222 -222
- package/src/runtime/stale-reconciler.ts +198 -189
- package/src/runtime/streaming-output.ts +47 -0
- package/src/runtime/subagent-manager.ts +404 -400
- package/src/runtime/subprocess-tool-registry.ts +67 -67
- package/src/runtime/task-display.ts +38 -38
- package/src/runtime/task-graph-scheduler.ts +122 -122
- package/src/runtime/task-graph.ts +207 -207
- package/src/runtime/task-output-context.ts +177 -177
- package/src/runtime/task-packet.ts +93 -93
- package/src/runtime/task-quality.ts +207 -207
- package/src/runtime/task-runner/capabilities.ts +78 -78
- package/src/runtime/task-runner/live-executor.ts +131 -113
- package/src/runtime/task-runner/progress.ts +119 -119
- package/src/runtime/task-runner/prompt-builder.ts +139 -139
- package/src/runtime/task-runner/prompt-pipeline.ts +64 -64
- package/src/runtime/task-runner/result-utils.ts +14 -14
- package/src/runtime/task-runner/run-projection.ts +103 -103
- package/src/runtime/task-runner/state-helpers.ts +22 -22
- package/src/runtime/task-runner.ts +469 -459
- package/src/runtime/team-runner.ts +693 -945
- package/src/runtime/usage-tracker.ts +71 -0
- package/src/runtime/worker-heartbeat.ts +21 -21
- package/src/runtime/worker-startup.ts +57 -57
- package/src/runtime/workflow-state.ts +187 -187
- package/src/runtime/yield-handler.ts +190 -190
- package/src/schema/config-schema.ts +172 -168
- package/src/schema/team-tool-schema.ts +126 -126
- package/src/schema/validation-types.ts +151 -148
- package/src/skills/discover-skills.ts +67 -67
- package/src/skills/skill-templates.ts +374 -374
- package/src/state/active-run-registry.ts +227 -191
- package/src/state/artifact-store.ts +130 -129
- package/src/state/atomic-write.ts +262 -195
- package/src/state/blob-store.ts +116 -116
- package/src/state/contracts.ts +111 -111
- package/src/state/event-log-rotation.ts +161 -158
- package/src/state/event-log.ts +383 -303
- package/src/state/event-reconstructor.ts +217 -217
- package/src/state/jsonl-writer.ts +82 -82
- package/src/state/locks.ts +146 -146
- package/src/state/mailbox.ts +446 -405
- package/src/state/state-store.ts +364 -351
- package/src/state/task-claims.ts +44 -44
- package/src/state/types.ts +285 -285
- package/src/state/usage.ts +29 -29
- package/src/subagents/async-entry.ts +1 -1
- package/src/subagents/index.ts +3 -3
- package/src/subagents/live/control.ts +1 -1
- package/src/subagents/live/manager.ts +1 -1
- package/src/subagents/live/realtime.ts +1 -1
- package/src/subagents/live/session-runtime.ts +1 -1
- package/src/subagents/manager.ts +1 -1
- package/src/subagents/spawn.ts +1 -1
- package/src/teams/discover-teams.ts +116 -116
- package/src/teams/team-config.ts +27 -27
- package/src/teams/team-serializer.ts +38 -38
- package/src/types/diff.d.ts +18 -18
- package/src/ui/agent-management-overlay.ts +144 -144
- package/src/ui/crew-widget.ts +487 -370
- package/src/ui/dashboard-panes/agents-pane.ts +109 -28
- package/src/ui/dashboard-panes/cancellation-pane.ts +42 -42
- package/src/ui/dashboard-panes/capability-pane.ts +59 -59
- package/src/ui/dashboard-panes/health-pane.ts +30 -30
- package/src/ui/dashboard-panes/mailbox-pane.ts +35 -35
- package/src/ui/dashboard-panes/progress-pane.ts +30 -30
- package/src/ui/dashboard-panes/transcript-pane.ts +10 -10
- package/src/ui/heartbeat-aggregator.ts +63 -63
- package/src/ui/keybinding-map.ts +97 -94
- package/src/ui/live-conversation-overlay.ts +152 -0
- package/src/ui/live-run-sidebar.ts +180 -180
- package/src/ui/mascot.ts +442 -442
- package/src/ui/overlays/agent-picker-overlay.ts +57 -57
- package/src/ui/overlays/confirm-overlay.ts +58 -58
- package/src/ui/overlays/mailbox-compose-overlay.ts +144 -144
- package/src/ui/overlays/mailbox-compose-preview.ts +63 -63
- package/src/ui/overlays/mailbox-detail-overlay.ts +122 -122
- package/src/ui/pi-ui-compat.ts +57 -57
- package/src/ui/powerbar-publisher.ts +221 -197
- package/src/ui/render-scheduler.ts +216 -143
- package/src/ui/run-action-dispatcher.ts +118 -118
- package/src/ui/run-dashboard.ts +526 -464
- package/src/ui/run-event-bus.ts +208 -208
- package/src/ui/run-snapshot-cache.ts +826 -777
- package/src/ui/settings-overlay.ts +721 -0
- package/src/ui/snapshot-types.ts +86 -70
- package/src/ui/theme-adapter.ts +190 -190
- package/src/ui/tool-progress-formatter.ts +89 -0
- package/src/ui/transcript-cache.ts +94 -94
- package/src/ui/transcript-viewer.ts +335 -335
- package/src/utils/conflict-detect.ts +662 -0
- package/src/utils/file-coalescer.ts +86 -86
- package/src/utils/frontmatter.ts +68 -68
- package/src/utils/fs-watch.ts +88 -31
- package/src/utils/gh-protocol.ts +479 -0
- package/src/utils/ids.ts +17 -17
- package/src/utils/incremental-reader.ts +104 -104
- package/src/utils/internal-error.ts +6 -6
- package/src/utils/names.ts +27 -27
- package/src/utils/paths.ts +102 -63
- package/src/utils/redaction.ts +44 -44
- package/src/utils/safe-paths.ts +47 -47
- package/src/utils/scan-cache.ts +136 -136
- package/src/utils/sse-parser.ts +134 -134
- package/src/utils/task-name-generator.ts +337 -337
- package/src/utils/timings.ts +33 -33
- package/src/utils/visual.ts +243 -198
- package/src/workflows/discover-workflows.ts +139 -139
- package/src/workflows/validate-workflow.ts +40 -40
- package/src/workflows/workflow-config.ts +26 -26
- package/src/workflows/workflow-serializer.ts +32 -32
- package/src/worktree/branch-freshness.ts +45 -45
- package/src/worktree/cleanup.ts +75 -75
- package/src/worktree/worktree-manager.ts +188 -188
- package/teams/default.team.md +12 -12
- package/teams/fast-fix.team.md +11 -11
- package/teams/implementation.team.md +18 -18
- package/teams/parallel-research.team.md +14 -14
- package/teams/research.team.md +11 -11
- package/teams/review.team.md +12 -12
- package/tsconfig.json +19 -19
- package/workflows/default.workflow.md +30 -30
- package/workflows/fast-fix.workflow.md +23 -23
- package/workflows/implementation.workflow.md +43 -43
- package/workflows/parallel-research.workflow.md +46 -46
- package/workflows/research.workflow.md +22 -22
- package/workflows/review.workflow.md +30 -30
- package/skills/task-packet/SKILL.md +0 -28
- package/skills/verify-evidence/SKILL.md +0 -27
|
@@ -1,177 +1,177 @@
|
|
|
1
|
-
import * as fs from "node:fs";
|
|
2
|
-
import * as path from "node:path";
|
|
3
|
-
import type { ArtifactDescriptor, TeamRunManifest, TeamTaskState } from "../state/types.ts";
|
|
4
|
-
import { writeArtifact } from "../state/artifact-store.ts";
|
|
5
|
-
import { resolveRealContainedPath } from "../utils/safe-paths.ts";
|
|
6
|
-
import type { WorkflowStep } from "../workflows/workflow-config.ts";
|
|
7
|
-
|
|
8
|
-
export interface DependencyContextEntry {
|
|
9
|
-
taskId: string;
|
|
10
|
-
role: string;
|
|
11
|
-
status: string;
|
|
12
|
-
resultSummary: string;
|
|
13
|
-
resultPath?: string;
|
|
14
|
-
structuredResults?: Record<string, unknown>;
|
|
15
|
-
artifactsProduced?: string[];
|
|
16
|
-
usage?: { inputTokens: number; outputTokens: number; durationMs: number };
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
export interface DependencyOutputContext {
|
|
20
|
-
dependencies: DependencyContextEntry[];
|
|
21
|
-
sharedReads: Array<{ name: string; path: string; content: string }>;
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
function containedExists(filePath: string, baseDir?: string): boolean {
|
|
25
|
-
try {
|
|
26
|
-
const safePath = baseDir ? resolveRealContainedPath(baseDir, filePath) : filePath;
|
|
27
|
-
return fs.existsSync(safePath);
|
|
28
|
-
} catch {
|
|
29
|
-
return false;
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
function readIfSmall(filePath: string, maxBytes = 24_000, baseDir?: string): string | undefined {
|
|
34
|
-
try {
|
|
35
|
-
const safePath = baseDir ? resolveRealContainedPath(baseDir, filePath) : filePath;
|
|
36
|
-
const stat = fs.statSync(safePath);
|
|
37
|
-
if (stat.size > maxBytes) return `${fs.readFileSync(safePath, "utf-8").slice(0, maxBytes)}\n\n...(truncated ${stat.size - maxBytes} bytes)`;
|
|
38
|
-
return fs.readFileSync(safePath, "utf-8");
|
|
39
|
-
} catch {
|
|
40
|
-
return undefined;
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
function safeSharedName(name: string): string {
|
|
45
|
-
const normalized = name.replaceAll("\\", "/").replace(/^\.\/+/, "");
|
|
46
|
-
if (!normalized || normalized.split("/").some((segment) => segment === "..") || path.isAbsolute(normalized)) throw new Error(`Invalid shared artifact name: ${name}`);
|
|
47
|
-
return normalized;
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
export function sharedPath(manifest: TeamRunManifest, name: string): string {
|
|
51
|
-
const sharedRoot = path.resolve(manifest.artifactsRoot, "shared");
|
|
52
|
-
const resolved = path.resolve(sharedRoot, safeSharedName(name));
|
|
53
|
-
const relative = path.relative(sharedRoot, resolved);
|
|
54
|
-
if (relative.startsWith("..") || path.isAbsolute(relative)) throw new Error(`Invalid shared artifact name: ${name}`);
|
|
55
|
-
return resolved;
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
function tryParseJson(text: string): Record<string, unknown> | undefined {
|
|
59
|
-
try {
|
|
60
|
-
const parsed = JSON.parse(text);
|
|
61
|
-
if (typeof parsed === "object" && parsed !== null && !Array.isArray(parsed)) return parsed as Record<string, unknown>;
|
|
62
|
-
} catch {
|
|
63
|
-
// Not valid JSON object — return undefined.
|
|
64
|
-
}
|
|
65
|
-
return undefined;
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
function listTaskArtifacts(manifest: TeamRunManifest, taskId: string): string[] | undefined {
|
|
69
|
-
const produced = manifest.artifacts.filter((a) => a.producer === taskId);
|
|
70
|
-
if (produced.length === 0) return undefined;
|
|
71
|
-
return produced.map((a) => {
|
|
72
|
-
const relative = path.relative(manifest.artifactsRoot, a.path);
|
|
73
|
-
return relative.startsWith("..") ? a.path : relative;
|
|
74
|
-
});
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
function aggregateUsage(task: TeamTaskState): DependencyContextEntry["usage"] {
|
|
78
|
-
if (!task.usage) return undefined;
|
|
79
|
-
const inputTokens = task.usage.input ?? 0;
|
|
80
|
-
const outputTokens = task.usage.output ?? 0;
|
|
81
|
-
const started = task.startedAt ? new Date(task.startedAt).getTime() : 0;
|
|
82
|
-
const finished = task.finishedAt ? new Date(task.finishedAt).getTime() : 0;
|
|
83
|
-
const durationMs = started && finished ? finished - started : 0;
|
|
84
|
-
if (inputTokens === 0 && outputTokens === 0 && durationMs === 0) return undefined;
|
|
85
|
-
return { inputTokens, outputTokens, durationMs };
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
export function collectDependencyOutputContext(manifest: TeamRunManifest, tasks: TeamTaskState[], task: TeamTaskState, step: WorkflowStep): DependencyOutputContext {
|
|
89
|
-
const byStep = new Map(tasks.map((item) => [item.stepId, item]).filter((entry): entry is [string, TeamTaskState] => Boolean(entry[0])));
|
|
90
|
-
const byId = new Map(tasks.map((item) => [item.id, item]));
|
|
91
|
-
const dependencies = task.dependsOn.map((dep) => byStep.get(dep) ?? byId.get(dep)).filter((item): item is TeamTaskState => Boolean(item)).map((item) => {
|
|
92
|
-
const resultText = item.resultArtifact ? readIfSmall(item.resultArtifact.path, 24_000, manifest.artifactsRoot) : undefined;
|
|
93
|
-
return {
|
|
94
|
-
taskId: item.id,
|
|
95
|
-
role: item.role,
|
|
96
|
-
status: item.status,
|
|
97
|
-
resultSummary: resultText ?? "",
|
|
98
|
-
resultPath: item.resultArtifact?.path,
|
|
99
|
-
structuredResults: resultText ? tryParseJson(resultText) : undefined,
|
|
100
|
-
artifactsProduced: listTaskArtifacts(manifest, item.id),
|
|
101
|
-
usage: aggregateUsage(item),
|
|
102
|
-
};
|
|
103
|
-
});
|
|
104
|
-
const sharedReads = (step.reads === false ? [] : step.reads ?? []).map((name) => {
|
|
105
|
-
const filePath = sharedPath(manifest, name);
|
|
106
|
-
return { name, path: filePath, content: readIfSmall(filePath, 24_000, path.resolve(manifest.artifactsRoot, "shared")) ?? "" };
|
|
107
|
-
}).filter((item) => item.content.trim().length > 0);
|
|
108
|
-
return { dependencies, sharedReads };
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
export function renderDependencyOutputContext(context: DependencyOutputContext): string {
|
|
112
|
-
const parts: string[] = [];
|
|
113
|
-
if (context.dependencies.length) {
|
|
114
|
-
parts.push("# Dependency Outputs", "");
|
|
115
|
-
for (const dep of context.dependencies) {
|
|
116
|
-
parts.push(`## ${dep.taskId} (${dep.role})`, `Status: ${dep.status}`, dep.resultPath ? `Result artifact: ${dep.resultPath}` : "", "", dep.resultSummary?.trim() || "(no result output)", "");
|
|
117
|
-
if (dep.structuredResults) parts.push("Structured results:", JSON.stringify(dep.structuredResults, null, 2), "");
|
|
118
|
-
if (dep.artifactsProduced?.length) parts.push(`Artifacts produced: ${dep.artifactsProduced.join(", ")}`, "");
|
|
119
|
-
if (dep.usage) parts.push(`Usage: ${dep.usage.inputTokens} input tokens, ${dep.usage.outputTokens} output tokens, ${dep.usage.durationMs}ms`, "");
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
if (context.sharedReads.length) {
|
|
123
|
-
parts.push("# Shared Run Context Reads", "");
|
|
124
|
-
for (const read of context.sharedReads) parts.push(`## shared/${read.name}`, `Path: ${read.path}`, "", read.content.trim(), "");
|
|
125
|
-
}
|
|
126
|
-
return parts.join("\n").trim();
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
export function writeTaskSharedOutput(manifest: TeamRunManifest, step: WorkflowStep, task: TeamTaskState): ArtifactDescriptor | undefined {
|
|
130
|
-
if (step.output === false) return undefined;
|
|
131
|
-
const name = safeSharedName(step.output || `${task.id}.md`);
|
|
132
|
-
const source = task.resultArtifact ? readIfSmall(task.resultArtifact.path, 80_000, manifest.artifactsRoot) : undefined;
|
|
133
|
-
if (!source) return undefined;
|
|
134
|
-
return writeArtifact(manifest.artifactsRoot, {
|
|
135
|
-
kind: "metadata",
|
|
136
|
-
relativePath: `shared/${name}`,
|
|
137
|
-
producer: task.id,
|
|
138
|
-
content: source.endsWith("\n") ? source : `${source}\n`,
|
|
139
|
-
});
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
export function writeTaskInputsArtifact(manifest: TeamRunManifest, task: TeamTaskState, context: DependencyOutputContext): ArtifactDescriptor {
|
|
143
|
-
return writeArtifact(manifest.artifactsRoot, {
|
|
144
|
-
kind: "metadata",
|
|
145
|
-
relativePath: `metadata/${task.id}.inputs.json`,
|
|
146
|
-
producer: task.id,
|
|
147
|
-
content: `${JSON.stringify(context, null, 2)}\n`,
|
|
148
|
-
});
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
export function aggregateTaskOutputs(tasks: TeamTaskState[], manifest?: TeamRunManifest): string {
|
|
152
|
-
return tasks.map((task, index) => {
|
|
153
|
-
const body = task.resultArtifact ? readIfSmall(task.resultArtifact.path, 40_000, manifest?.artifactsRoot) : undefined;
|
|
154
|
-
const hasBody = Boolean(body?.trim());
|
|
155
|
-
const expectedMissing = task.resultArtifact && !containedExists(task.resultArtifact.path, manifest?.artifactsRoot);
|
|
156
|
-
const status = task.status === "skipped"
|
|
157
|
-
? "SKIPPED"
|
|
158
|
-
: task.status === "failed"
|
|
159
|
-
? `FAILED${task.exitCode !== undefined ? ` (exit code ${task.exitCode ?? "null"})` : ""}${task.error ? `: ${task.error}` : ""}`
|
|
160
|
-
: expectedMissing
|
|
161
|
-
? `EMPTY OUTPUT (expected result artifact missing: ${task.resultArtifact?.path})`
|
|
162
|
-
: !hasBody
|
|
163
|
-
? "EMPTY OUTPUT (no textual response returned)"
|
|
164
|
-
: task.status.toUpperCase();
|
|
165
|
-
return [
|
|
166
|
-
`=== Task ${index + 1}: ${task.id} (${task.agent}) ===`,
|
|
167
|
-
`Status: ${status}`,
|
|
168
|
-
task.role ? `Role: ${task.role}` : "",
|
|
169
|
-
task.resultArtifact?.path ? `Result artifact: ${task.resultArtifact.path}` : "",
|
|
170
|
-
task.logArtifact?.path ? `Log artifact: ${task.logArtifact.path}` : "",
|
|
171
|
-
task.transcriptArtifact?.path ? `Transcript: ${task.transcriptArtifact.path}` : "",
|
|
172
|
-
task.usage ? `Usage: ${JSON.stringify(task.usage)}` : "",
|
|
173
|
-
"",
|
|
174
|
-
hasBody ? body!.trim() : status,
|
|
175
|
-
].filter(Boolean).join("\n");
|
|
176
|
-
}).join("\n\n");
|
|
177
|
-
}
|
|
1
|
+
import * as fs from "node:fs";
|
|
2
|
+
import * as path from "node:path";
|
|
3
|
+
import type { ArtifactDescriptor, TeamRunManifest, TeamTaskState } from "../state/types.ts";
|
|
4
|
+
import { writeArtifact } from "../state/artifact-store.ts";
|
|
5
|
+
import { resolveRealContainedPath } from "../utils/safe-paths.ts";
|
|
6
|
+
import type { WorkflowStep } from "../workflows/workflow-config.ts";
|
|
7
|
+
|
|
8
|
+
export interface DependencyContextEntry {
|
|
9
|
+
taskId: string;
|
|
10
|
+
role: string;
|
|
11
|
+
status: string;
|
|
12
|
+
resultSummary: string;
|
|
13
|
+
resultPath?: string;
|
|
14
|
+
structuredResults?: Record<string, unknown>;
|
|
15
|
+
artifactsProduced?: string[];
|
|
16
|
+
usage?: { inputTokens: number; outputTokens: number; durationMs: number };
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export interface DependencyOutputContext {
|
|
20
|
+
dependencies: DependencyContextEntry[];
|
|
21
|
+
sharedReads: Array<{ name: string; path: string; content: string }>;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
function containedExists(filePath: string, baseDir?: string): boolean {
|
|
25
|
+
try {
|
|
26
|
+
const safePath = baseDir ? resolveRealContainedPath(baseDir, filePath) : filePath;
|
|
27
|
+
return fs.existsSync(safePath);
|
|
28
|
+
} catch {
|
|
29
|
+
return false;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function readIfSmall(filePath: string, maxBytes = 24_000, baseDir?: string): string | undefined {
|
|
34
|
+
try {
|
|
35
|
+
const safePath = baseDir ? resolveRealContainedPath(baseDir, filePath) : filePath;
|
|
36
|
+
const stat = fs.statSync(safePath);
|
|
37
|
+
if (stat.size > maxBytes) return `${fs.readFileSync(safePath, "utf-8").slice(0, maxBytes)}\n\n...(truncated ${stat.size - maxBytes} bytes)`;
|
|
38
|
+
return fs.readFileSync(safePath, "utf-8");
|
|
39
|
+
} catch {
|
|
40
|
+
return undefined;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function safeSharedName(name: string): string {
|
|
45
|
+
const normalized = name.replaceAll("\\", "/").replace(/^\.\/+/, "");
|
|
46
|
+
if (!normalized || normalized.split("/").some((segment) => segment === "..") || path.isAbsolute(normalized)) throw new Error(`Invalid shared artifact name: ${name}`);
|
|
47
|
+
return normalized;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export function sharedPath(manifest: TeamRunManifest, name: string): string {
|
|
51
|
+
const sharedRoot = path.resolve(manifest.artifactsRoot, "shared");
|
|
52
|
+
const resolved = path.resolve(sharedRoot, safeSharedName(name));
|
|
53
|
+
const relative = path.relative(sharedRoot, resolved);
|
|
54
|
+
if (relative.startsWith("..") || path.isAbsolute(relative)) throw new Error(`Invalid shared artifact name: ${name}`);
|
|
55
|
+
return resolved;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
function tryParseJson(text: string): Record<string, unknown> | undefined {
|
|
59
|
+
try {
|
|
60
|
+
const parsed = JSON.parse(text);
|
|
61
|
+
if (typeof parsed === "object" && parsed !== null && !Array.isArray(parsed)) return parsed as Record<string, unknown>;
|
|
62
|
+
} catch {
|
|
63
|
+
// Not valid JSON object — return undefined.
|
|
64
|
+
}
|
|
65
|
+
return undefined;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
function listTaskArtifacts(manifest: TeamRunManifest, taskId: string): string[] | undefined {
|
|
69
|
+
const produced = manifest.artifacts.filter((a) => a.producer === taskId);
|
|
70
|
+
if (produced.length === 0) return undefined;
|
|
71
|
+
return produced.map((a) => {
|
|
72
|
+
const relative = path.relative(manifest.artifactsRoot, a.path);
|
|
73
|
+
return relative.startsWith("..") ? a.path : relative;
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
function aggregateUsage(task: TeamTaskState): DependencyContextEntry["usage"] {
|
|
78
|
+
if (!task.usage) return undefined;
|
|
79
|
+
const inputTokens = task.usage.input ?? 0;
|
|
80
|
+
const outputTokens = task.usage.output ?? 0;
|
|
81
|
+
const started = task.startedAt ? new Date(task.startedAt).getTime() : 0;
|
|
82
|
+
const finished = task.finishedAt ? new Date(task.finishedAt).getTime() : 0;
|
|
83
|
+
const durationMs = started && finished ? finished - started : 0;
|
|
84
|
+
if (inputTokens === 0 && outputTokens === 0 && durationMs === 0) return undefined;
|
|
85
|
+
return { inputTokens, outputTokens, durationMs };
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
export function collectDependencyOutputContext(manifest: TeamRunManifest, tasks: TeamTaskState[], task: TeamTaskState, step: WorkflowStep): DependencyOutputContext {
|
|
89
|
+
const byStep = new Map(tasks.map((item) => [item.stepId, item]).filter((entry): entry is [string, TeamTaskState] => Boolean(entry[0])));
|
|
90
|
+
const byId = new Map(tasks.map((item) => [item.id, item]));
|
|
91
|
+
const dependencies = task.dependsOn.map((dep) => byStep.get(dep) ?? byId.get(dep)).filter((item): item is TeamTaskState => Boolean(item)).map((item) => {
|
|
92
|
+
const resultText = item.resultArtifact ? readIfSmall(item.resultArtifact.path, 24_000, manifest.artifactsRoot) : undefined;
|
|
93
|
+
return {
|
|
94
|
+
taskId: item.id,
|
|
95
|
+
role: item.role,
|
|
96
|
+
status: item.status,
|
|
97
|
+
resultSummary: resultText ?? "",
|
|
98
|
+
resultPath: item.resultArtifact?.path,
|
|
99
|
+
structuredResults: resultText ? tryParseJson(resultText) : undefined,
|
|
100
|
+
artifactsProduced: listTaskArtifacts(manifest, item.id),
|
|
101
|
+
usage: aggregateUsage(item),
|
|
102
|
+
};
|
|
103
|
+
});
|
|
104
|
+
const sharedReads = (step.reads === false ? [] : step.reads ?? []).map((name) => {
|
|
105
|
+
const filePath = sharedPath(manifest, name);
|
|
106
|
+
return { name, path: filePath, content: readIfSmall(filePath, 24_000, path.resolve(manifest.artifactsRoot, "shared")) ?? "" };
|
|
107
|
+
}).filter((item) => item.content.trim().length > 0);
|
|
108
|
+
return { dependencies, sharedReads };
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
export function renderDependencyOutputContext(context: DependencyOutputContext): string {
|
|
112
|
+
const parts: string[] = [];
|
|
113
|
+
if (context.dependencies.length) {
|
|
114
|
+
parts.push("# Dependency Outputs", "");
|
|
115
|
+
for (const dep of context.dependencies) {
|
|
116
|
+
parts.push(`## ${dep.taskId} (${dep.role})`, `Status: ${dep.status}`, dep.resultPath ? `Result artifact: ${dep.resultPath}` : "", "", dep.resultSummary?.trim() || "(no result output)", "");
|
|
117
|
+
if (dep.structuredResults) parts.push("Structured results:", JSON.stringify(dep.structuredResults, null, 2), "");
|
|
118
|
+
if (dep.artifactsProduced?.length) parts.push(`Artifacts produced: ${dep.artifactsProduced.join(", ")}`, "");
|
|
119
|
+
if (dep.usage) parts.push(`Usage: ${dep.usage.inputTokens} input tokens, ${dep.usage.outputTokens} output tokens, ${dep.usage.durationMs}ms`, "");
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
if (context.sharedReads.length) {
|
|
123
|
+
parts.push("# Shared Run Context Reads", "");
|
|
124
|
+
for (const read of context.sharedReads) parts.push(`## shared/${read.name}`, `Path: ${read.path}`, "", read.content.trim(), "");
|
|
125
|
+
}
|
|
126
|
+
return parts.join("\n").trim();
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
export function writeTaskSharedOutput(manifest: TeamRunManifest, step: WorkflowStep, task: TeamTaskState): ArtifactDescriptor | undefined {
|
|
130
|
+
if (step.output === false) return undefined;
|
|
131
|
+
const name = safeSharedName(step.output || `${task.id}.md`);
|
|
132
|
+
const source = task.resultArtifact ? readIfSmall(task.resultArtifact.path, 80_000, manifest.artifactsRoot) : undefined;
|
|
133
|
+
if (!source) return undefined;
|
|
134
|
+
return writeArtifact(manifest.artifactsRoot, {
|
|
135
|
+
kind: "metadata",
|
|
136
|
+
relativePath: `shared/${name}`,
|
|
137
|
+
producer: task.id,
|
|
138
|
+
content: source.endsWith("\n") ? source : `${source}\n`,
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
export function writeTaskInputsArtifact(manifest: TeamRunManifest, task: TeamTaskState, context: DependencyOutputContext): ArtifactDescriptor {
|
|
143
|
+
return writeArtifact(manifest.artifactsRoot, {
|
|
144
|
+
kind: "metadata",
|
|
145
|
+
relativePath: `metadata/${task.id}.inputs.json`,
|
|
146
|
+
producer: task.id,
|
|
147
|
+
content: `${JSON.stringify(context, null, 2)}\n`,
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
export function aggregateTaskOutputs(tasks: TeamTaskState[], manifest?: TeamRunManifest): string {
|
|
152
|
+
return tasks.map((task, index) => {
|
|
153
|
+
const body = task.resultArtifact ? readIfSmall(task.resultArtifact.path, 40_000, manifest?.artifactsRoot) : undefined;
|
|
154
|
+
const hasBody = Boolean(body?.trim());
|
|
155
|
+
const expectedMissing = task.resultArtifact && !containedExists(task.resultArtifact.path, manifest?.artifactsRoot);
|
|
156
|
+
const status = task.status === "skipped"
|
|
157
|
+
? "SKIPPED"
|
|
158
|
+
: task.status === "failed"
|
|
159
|
+
? `FAILED${task.exitCode !== undefined ? ` (exit code ${task.exitCode ?? "null"})` : ""}${task.error ? `: ${task.error}` : ""}`
|
|
160
|
+
: expectedMissing
|
|
161
|
+
? `EMPTY OUTPUT (expected result artifact missing: ${task.resultArtifact?.path})`
|
|
162
|
+
: !hasBody
|
|
163
|
+
? "EMPTY OUTPUT (no textual response returned)"
|
|
164
|
+
: task.status.toUpperCase();
|
|
165
|
+
return [
|
|
166
|
+
`=== Task ${index + 1}: ${task.id} (${task.agent}) ===`,
|
|
167
|
+
`Status: ${status}`,
|
|
168
|
+
task.role ? `Role: ${task.role}` : "",
|
|
169
|
+
task.resultArtifact?.path ? `Result artifact: ${task.resultArtifact.path}` : "",
|
|
170
|
+
task.logArtifact?.path ? `Log artifact: ${task.logArtifact.path}` : "",
|
|
171
|
+
task.transcriptArtifact?.path ? `Transcript: ${task.transcriptArtifact.path}` : "",
|
|
172
|
+
task.usage ? `Usage: ${JSON.stringify(task.usage)}` : "",
|
|
173
|
+
"",
|
|
174
|
+
hasBody ? body!.trim() : status,
|
|
175
|
+
].filter(Boolean).join("\n");
|
|
176
|
+
}).join("\n\n");
|
|
177
|
+
}
|
|
@@ -1,93 +1,93 @@
|
|
|
1
|
-
import * as path from "node:path";
|
|
2
|
-
import type { TeamRunManifest, TaskPacket, TaskScope, VerificationContract } from "../state/types.ts";
|
|
3
|
-
import type { WorkflowStep } from "../workflows/workflow-config.ts";
|
|
4
|
-
|
|
5
|
-
export interface BuildTaskPacketInput {
|
|
6
|
-
manifest: TeamRunManifest;
|
|
7
|
-
step: WorkflowStep;
|
|
8
|
-
taskId: string;
|
|
9
|
-
cwd: string;
|
|
10
|
-
worktreePath?: string;
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
export interface TaskPacketValidationResult {
|
|
14
|
-
valid: boolean;
|
|
15
|
-
errors: string[];
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
export function inferTaskScope(step: WorkflowStep): TaskScope {
|
|
19
|
-
const reads = step.reads === false ? [] : step.reads ?? [];
|
|
20
|
-
if (reads.length === 1) return "single_file";
|
|
21
|
-
if (reads.length > 1) return "module";
|
|
22
|
-
return "workspace";
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
export function defaultVerificationContract(step: WorkflowStep): VerificationContract {
|
|
26
|
-
return {
|
|
27
|
-
requiredGreenLevel: step.verify ? "targeted" : "none",
|
|
28
|
-
commands: [],
|
|
29
|
-
allowManualEvidence: true,
|
|
30
|
-
};
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
export function buildTaskPacket(input: BuildTaskPacketInput): TaskPacket {
|
|
34
|
-
const scope = inferTaskScope(input.step);
|
|
35
|
-
const reads = input.step.reads === false ? [] : input.step.reads ?? [];
|
|
36
|
-
const scopePath = reads.length === 1 ? reads[0] : reads.length > 1 ? reads.join(", ") : undefined;
|
|
37
|
-
return {
|
|
38
|
-
objective: input.step.task.replaceAll("{goal}", input.manifest.goal),
|
|
39
|
-
scope,
|
|
40
|
-
scopePath,
|
|
41
|
-
repo: path.basename(input.manifest.cwd) || input.manifest.cwd,
|
|
42
|
-
worktree: input.worktreePath,
|
|
43
|
-
branchPolicy: input.manifest.workspaceMode === "worktree" ? "Use the assigned task worktree and avoid modifying the leader checkout." : "Use the current checkout; do not create branches unless explicitly requested.",
|
|
44
|
-
acceptanceTests: [],
|
|
45
|
-
commitPolicy: "Do not commit unless explicitly requested by the user or workflow.",
|
|
46
|
-
reportingContract: "Report intended/changed files, verification evidence, blockers, conflict risks, and next recommended action.",
|
|
47
|
-
escalationPolicy: "Stop and report if scope is ambiguous, destructive action is needed, permissions are missing, verification cannot be completed, or edits may overlap with another worker/task.",
|
|
48
|
-
constraints: [
|
|
49
|
-
"Stay within the assigned task scope.",
|
|
50
|
-
"Do not claim completion without verification evidence.",
|
|
51
|
-
"Use mailbox/API state for coordination when available.",
|
|
52
|
-
"Do not make overlapping edits to the same file/symbol without explicit leader sequencing or ownership guidance.",
|
|
53
|
-
],
|
|
54
|
-
expectedArtifacts: ["prompt", "result", "verification"],
|
|
55
|
-
verification: defaultVerificationContract(input.step),
|
|
56
|
-
};
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
export function validateTaskPacket(packet: TaskPacket): TaskPacketValidationResult {
|
|
60
|
-
const errors: string[] = [];
|
|
61
|
-
if (!packet.objective.trim()) errors.push("objective must not be empty");
|
|
62
|
-
if (!packet.repo.trim()) errors.push("repo must not be empty");
|
|
63
|
-
if (!packet.branchPolicy.trim()) errors.push("branchPolicy must not be empty");
|
|
64
|
-
if (!packet.commitPolicy.trim()) errors.push("commitPolicy must not be empty");
|
|
65
|
-
if (!packet.reportingContract.trim()) errors.push("reportingContract must not be empty");
|
|
66
|
-
if (!packet.escalationPolicy.trim()) errors.push("escalationPolicy must not be empty");
|
|
67
|
-
if ((packet.scope === "module" || packet.scope === "single_file" || packet.scope === "custom") && !packet.scopePath?.trim()) {
|
|
68
|
-
errors.push(`scopePath is required for scope '${packet.scope}'`);
|
|
69
|
-
}
|
|
70
|
-
if (packet.constraints.length === 0) errors.push("constraints must contain at least one entry");
|
|
71
|
-
for (const [index, constraint] of packet.constraints.entries()) {
|
|
72
|
-
if (!constraint.trim()) errors.push(`constraints contains an empty value at index ${index}`);
|
|
73
|
-
}
|
|
74
|
-
if (packet.expectedArtifacts.length === 0) errors.push("expectedArtifacts must contain at least one entry");
|
|
75
|
-
for (const [index, artifact] of packet.expectedArtifacts.entries()) {
|
|
76
|
-
if (!artifact.trim()) errors.push(`expectedArtifacts contains an empty value at index ${index}`);
|
|
77
|
-
}
|
|
78
|
-
for (const [index, test] of packet.acceptanceTests.entries()) {
|
|
79
|
-
if (!test.trim()) errors.push(`acceptanceTests contains an empty value at index ${index}`);
|
|
80
|
-
}
|
|
81
|
-
return { valid: errors.length === 0, errors };
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
export function renderTaskPacket(packet: TaskPacket): string {
|
|
85
|
-
return [
|
|
86
|
-
"# Task Packet",
|
|
87
|
-
"",
|
|
88
|
-
"```json",
|
|
89
|
-
JSON.stringify(packet, null, 2),
|
|
90
|
-
"```",
|
|
91
|
-
"",
|
|
92
|
-
].join("\n");
|
|
93
|
-
}
|
|
1
|
+
import * as path from "node:path";
|
|
2
|
+
import type { TeamRunManifest, TaskPacket, TaskScope, VerificationContract } from "../state/types.ts";
|
|
3
|
+
import type { WorkflowStep } from "../workflows/workflow-config.ts";
|
|
4
|
+
|
|
5
|
+
export interface BuildTaskPacketInput {
|
|
6
|
+
manifest: TeamRunManifest;
|
|
7
|
+
step: WorkflowStep;
|
|
8
|
+
taskId: string;
|
|
9
|
+
cwd: string;
|
|
10
|
+
worktreePath?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export interface TaskPacketValidationResult {
|
|
14
|
+
valid: boolean;
|
|
15
|
+
errors: string[];
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export function inferTaskScope(step: WorkflowStep): TaskScope {
|
|
19
|
+
const reads = step.reads === false ? [] : step.reads ?? [];
|
|
20
|
+
if (reads.length === 1) return "single_file";
|
|
21
|
+
if (reads.length > 1) return "module";
|
|
22
|
+
return "workspace";
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export function defaultVerificationContract(step: WorkflowStep): VerificationContract {
|
|
26
|
+
return {
|
|
27
|
+
requiredGreenLevel: step.verify ? "targeted" : "none",
|
|
28
|
+
commands: [],
|
|
29
|
+
allowManualEvidence: true,
|
|
30
|
+
};
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export function buildTaskPacket(input: BuildTaskPacketInput): TaskPacket {
|
|
34
|
+
const scope = inferTaskScope(input.step);
|
|
35
|
+
const reads = input.step.reads === false ? [] : input.step.reads ?? [];
|
|
36
|
+
const scopePath = reads.length === 1 ? reads[0] : reads.length > 1 ? reads.join(", ") : undefined;
|
|
37
|
+
return {
|
|
38
|
+
objective: input.step.task.replaceAll("{goal}", input.manifest.goal),
|
|
39
|
+
scope,
|
|
40
|
+
scopePath,
|
|
41
|
+
repo: path.basename(input.manifest.cwd) || input.manifest.cwd,
|
|
42
|
+
worktree: input.worktreePath,
|
|
43
|
+
branchPolicy: input.manifest.workspaceMode === "worktree" ? "Use the assigned task worktree and avoid modifying the leader checkout." : "Use the current checkout; do not create branches unless explicitly requested.",
|
|
44
|
+
acceptanceTests: [],
|
|
45
|
+
commitPolicy: "Do not commit unless explicitly requested by the user or workflow.",
|
|
46
|
+
reportingContract: "Report intended/changed files, verification evidence, blockers, conflict risks, and next recommended action.",
|
|
47
|
+
escalationPolicy: "Stop and report if scope is ambiguous, destructive action is needed, permissions are missing, verification cannot be completed, or edits may overlap with another worker/task.",
|
|
48
|
+
constraints: [
|
|
49
|
+
"Stay within the assigned task scope.",
|
|
50
|
+
"Do not claim completion without verification evidence.",
|
|
51
|
+
"Use mailbox/API state for coordination when available.",
|
|
52
|
+
"Do not make overlapping edits to the same file/symbol without explicit leader sequencing or ownership guidance.",
|
|
53
|
+
],
|
|
54
|
+
expectedArtifacts: ["prompt", "result", "verification"],
|
|
55
|
+
verification: defaultVerificationContract(input.step),
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
export function validateTaskPacket(packet: TaskPacket): TaskPacketValidationResult {
|
|
60
|
+
const errors: string[] = [];
|
|
61
|
+
if (!packet.objective.trim()) errors.push("objective must not be empty");
|
|
62
|
+
if (!packet.repo.trim()) errors.push("repo must not be empty");
|
|
63
|
+
if (!packet.branchPolicy.trim()) errors.push("branchPolicy must not be empty");
|
|
64
|
+
if (!packet.commitPolicy.trim()) errors.push("commitPolicy must not be empty");
|
|
65
|
+
if (!packet.reportingContract.trim()) errors.push("reportingContract must not be empty");
|
|
66
|
+
if (!packet.escalationPolicy.trim()) errors.push("escalationPolicy must not be empty");
|
|
67
|
+
if ((packet.scope === "module" || packet.scope === "single_file" || packet.scope === "custom") && !packet.scopePath?.trim()) {
|
|
68
|
+
errors.push(`scopePath is required for scope '${packet.scope}'`);
|
|
69
|
+
}
|
|
70
|
+
if (packet.constraints.length === 0) errors.push("constraints must contain at least one entry");
|
|
71
|
+
for (const [index, constraint] of packet.constraints.entries()) {
|
|
72
|
+
if (!constraint.trim()) errors.push(`constraints contains an empty value at index ${index}`);
|
|
73
|
+
}
|
|
74
|
+
if (packet.expectedArtifacts.length === 0) errors.push("expectedArtifacts must contain at least one entry");
|
|
75
|
+
for (const [index, artifact] of packet.expectedArtifacts.entries()) {
|
|
76
|
+
if (!artifact.trim()) errors.push(`expectedArtifacts contains an empty value at index ${index}`);
|
|
77
|
+
}
|
|
78
|
+
for (const [index, test] of packet.acceptanceTests.entries()) {
|
|
79
|
+
if (!test.trim()) errors.push(`acceptanceTests contains an empty value at index ${index}`);
|
|
80
|
+
}
|
|
81
|
+
return { valid: errors.length === 0, errors };
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
export function renderTaskPacket(packet: TaskPacket): string {
|
|
85
|
+
return [
|
|
86
|
+
"# Task Packet",
|
|
87
|
+
"",
|
|
88
|
+
"```json",
|
|
89
|
+
JSON.stringify(packet, null, 2),
|
|
90
|
+
"```",
|
|
91
|
+
"",
|
|
92
|
+
].join("\n");
|
|
93
|
+
}
|