pi-crew 0.1.46 → 0.1.49
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/CHANGELOG.md +97 -0
- package/agents/analyst.md +11 -11
- package/agents/critic.md +11 -11
- package/agents/executor.md +11 -11
- package/agents/explorer.md +11 -11
- package/agents/planner.md +11 -11
- package/agents/reviewer.md +11 -11
- package/agents/security-reviewer.md +11 -11
- package/agents/test-engineer.md +11 -11
- package/agents/verifier.md +11 -11
- package/agents/writer.md +11 -11
- package/docs/next-upgrade-roadmap.md +117 -42
- package/docs/refactor-tasks-phase3.md +394 -394
- package/docs/refactor-tasks-phase4.md +564 -564
- package/docs/refactor-tasks-phase5.md +402 -402
- package/docs/refactor-tasks-phase6.md +662 -662
- package/docs/research/AGENT-EXECUTION-ARCHITECTURE.md +261 -0
- package/docs/research/AGENT-LIFECYCLE-COMPARISON.md +111 -0
- package/docs/research/AUDIT_OH_MY_PI.md +261 -0
- package/docs/research/AUDIT_PI_CREW.md +457 -0
- package/docs/research/CAVEMAN-DEEP-RESEARCH.md +281 -0
- package/docs/research/COMPARISON_OH_MY_PI_VS_PI_CREW.md +264 -0
- package/docs/research/DEEP-RESEARCH-PI-POWERBAR.md +343 -0
- package/docs/research/DEEP_RESEARCH_SUBAGENT_ARCHITECTURE.md +480 -0
- package/docs/research/GAP_CLOSURE_IMPLEMENTATION_PLAN.md +354 -0
- package/docs/research/IMPLEMENTATION_PLAN.md +385 -0
- package/docs/research/LIVE-SESSION-PRODUCTION-READY-PLAN.md +502 -0
- package/docs/research/OH-MY-PI-DEEP-RESEARCH-v14.7.6.md +266 -0
- package/docs/research/REMAINING-GAPS-PLAN.md +363 -0
- package/docs/research/SESSION-SUMMARY-2026-05-08.md +146 -0
- package/docs/research/UI-RESPONSIVENESS-AUDIT.md +173 -0
- package/docs/research-awesome-agent-skills-distillation.md +100 -100
- package/docs/research-extension-examples.md +297 -297
- package/docs/research-extension-system.md +324 -324
- package/docs/research-oh-my-pi-distillation.md +56 -9
- package/docs/research-optimization-plan.md +548 -548
- package/docs/research-phase10-distillation.md +198 -198
- package/docs/research-phase11-distillation.md +201 -201
- package/docs/research-pi-coding-agent.md +357 -357
- package/docs/research-source-pi-crew-reference.md +174 -174
- package/docs/runtime-flow.md +148 -148
- package/docs/source-runtime-refactor-map.md +107 -107
- package/index.ts +6 -6
- package/package.json +99 -98
- package/schema.json +8 -0
- package/skills/async-worker-recovery/SKILL.md +42 -42
- package/skills/context-artifact-hygiene/SKILL.md +52 -52
- package/skills/delegation-patterns/SKILL.md +54 -54
- package/skills/mailbox-interactive/SKILL.md +40 -40
- package/skills/model-routing-context/SKILL.md +39 -39
- package/skills/multi-perspective-review/SKILL.md +58 -58
- package/skills/observability-reliability/SKILL.md +41 -41
- package/skills/orchestration/SKILL.md +157 -0
- package/skills/ownership-session-security/SKILL.md +41 -41
- package/skills/pi-extension-lifecycle/SKILL.md +39 -39
- package/skills/requirements-to-task-packet/SKILL.md +63 -63
- package/skills/resource-discovery-config/SKILL.md +41 -41
- package/skills/runtime-state-reader/SKILL.md +44 -44
- package/skills/secure-agent-orchestration-review/SKILL.md +45 -45
- package/skills/state-mutation-locking/SKILL.md +42 -42
- package/skills/systematic-debugging/SKILL.md +67 -67
- package/skills/ui-render-performance/SKILL.md +39 -39
- package/skills/verification-before-done/SKILL.md +57 -57
- package/skills/worktree-isolation/SKILL.md +39 -39
- package/src/agents/agent-config.ts +6 -0
- package/src/agents/agent-search.ts +98 -0
- package/src/agents/agent-serializer.ts +4 -0
- package/src/agents/discover-agents.ts +17 -4
- package/src/config/config.ts +24 -0
- package/src/config/defaults.ts +11 -0
- package/src/extension/autonomous-policy.ts +26 -33
- package/src/extension/cross-extension-rpc.ts +82 -82
- package/src/extension/help.ts +1 -0
- package/src/extension/management.ts +5 -0
- package/src/extension/register.ts +58 -13
- package/src/extension/registration/commands.ts +33 -1
- package/src/extension/registration/compaction-guard.ts +125 -125
- package/src/extension/registration/team-tool.ts +6 -4
- package/src/extension/run-bundle-schema.ts +89 -89
- package/src/extension/run-index.ts +24 -18
- package/src/extension/run-maintenance.ts +68 -62
- package/src/extension/team-tool/api.ts +23 -2
- package/src/extension/team-tool/cancel.ts +86 -11
- package/src/extension/team-tool/context.ts +3 -0
- package/src/extension/team-tool/handle-settings.ts +188 -188
- package/src/extension/team-tool/inspect.ts +41 -41
- package/src/extension/team-tool/intent-policy.ts +42 -0
- package/src/extension/team-tool/lifecycle-actions.ts +47 -18
- package/src/extension/team-tool/parallel-dispatch.ts +156 -0
- package/src/extension/team-tool/plan.ts +19 -19
- package/src/extension/team-tool/respond.ts +10 -2
- package/src/extension/team-tool/run.ts +3 -2
- package/src/extension/team-tool/status.ts +1 -1
- package/src/extension/team-tool-types.ts +1 -0
- package/src/extension/team-tool.ts +13 -3
- package/src/hooks/registry.ts +61 -0
- package/src/hooks/types.ts +41 -0
- package/src/i18n.ts +184 -184
- package/src/observability/exporters/otlp-exporter.ts +77 -77
- package/src/prompt/prompt-runtime.ts +72 -72
- package/src/runtime/agent-control.ts +108 -2
- 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 +3 -1
- package/src/runtime/attention-events.ts +28 -28
- package/src/runtime/background-runner.ts +19 -0
- package/src/runtime/cancellation-token.ts +89 -0
- package/src/runtime/cancellation.ts +61 -51
- package/src/runtime/capability-inventory.ts +116 -0
- package/src/runtime/child-pi.ts +2 -1
- package/src/runtime/code-summary.ts +247 -0
- package/src/runtime/completion-guard.ts +190 -190
- package/src/runtime/crash-recovery.ts +181 -0
- package/src/runtime/crew-agent-records.ts +35 -7
- package/src/runtime/crew-agent-runtime.ts +1 -0
- package/src/runtime/custom-tools/irc-tool.ts +201 -0
- package/src/runtime/custom-tools/submit-result-tool.ts +90 -0
- package/src/runtime/delivery-coordinator.ts +3 -1
- package/src/runtime/direct-run.ts +35 -35
- package/src/runtime/effectiveness.ts +81 -76
- package/src/runtime/event-stream-bridge.ts +90 -0
- package/src/runtime/foreground-control.ts +82 -82
- package/src/runtime/green-contract.ts +46 -46
- package/src/runtime/group-join.ts +106 -106
- package/src/runtime/heartbeat-gradient.ts +28 -28
- package/src/runtime/heartbeat-watcher.ts +124 -124
- package/src/runtime/live-agent-control.ts +88 -88
- package/src/runtime/live-agent-manager.ts +78 -2
- package/src/runtime/live-control-realtime.ts +36 -36
- package/src/runtime/live-extension-bridge.ts +150 -0
- package/src/runtime/live-irc.ts +92 -0
- package/src/runtime/live-session-health.ts +100 -0
- package/src/runtime/live-session-runtime.ts +297 -7
- package/src/runtime/mcp-proxy.ts +113 -0
- package/src/runtime/notebook-helpers.ts +90 -0
- package/src/runtime/orphan-sentinel.ts +7 -0
- package/src/runtime/output-validator.ts +187 -0
- package/src/runtime/parallel-research.ts +44 -44
- package/src/runtime/parallel-utils.ts +57 -0
- package/src/runtime/parent-guard.ts +80 -0
- package/src/runtime/pi-json-output.ts +111 -111
- package/src/runtime/policy-engine.ts +79 -79
- package/src/runtime/progress-event-coalescer.ts +43 -43
- package/src/runtime/prose-compressor.ts +164 -0
- package/src/runtime/recovery-recipes.ts +74 -74
- package/src/runtime/result-extractor.ts +121 -0
- package/src/runtime/role-permission.ts +39 -39
- package/src/runtime/runtime-resolver.ts +1 -4
- package/src/runtime/semaphore.ts +131 -0
- package/src/runtime/sensitive-paths.ts +92 -0
- package/src/runtime/session-resources.ts +25 -25
- package/src/runtime/session-snapshot.ts +59 -59
- package/src/runtime/session-usage.ts +79 -79
- package/src/runtime/sidechain-output.ts +29 -29
- package/src/runtime/stream-preview.ts +177 -0
- package/src/runtime/subagent-manager.ts +3 -2
- package/src/runtime/subprocess-tool-registry.ts +67 -0
- package/src/runtime/supervisor-contact.ts +59 -59
- package/src/runtime/task-display.ts +38 -38
- package/src/runtime/task-output-context.ts +59 -9
- package/src/runtime/task-runner/capabilities.ts +78 -78
- package/src/runtime/task-runner/live-executor.ts +2 -0
- package/src/runtime/task-runner/progress.ts +119 -119
- package/src/runtime/task-runner/prompt-builder.ts +70 -8
- 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 +104 -0
- package/src/runtime/task-runner/state-helpers.ts +22 -22
- package/src/runtime/task-runner.ts +75 -4
- package/src/runtime/team-runner.ts +60 -8
- package/src/runtime/worker-heartbeat.ts +21 -21
- package/src/runtime/worker-startup.ts +57 -57
- package/src/runtime/workspace-tree.ts +298 -0
- package/src/runtime/yield-handler.ts +189 -0
- package/src/schema/config-schema.ts +6 -0
- package/src/schema/team-tool-schema.ts +11 -1
- package/src/skills/discover-skills.ts +67 -0
- package/src/state/active-run-registry.ts +4 -2
- package/src/state/artifact-store.ts +4 -1
- package/src/state/atomic-write.ts +50 -1
- package/src/state/blob-store.ts +117 -0
- package/src/state/contracts.ts +1 -0
- package/src/state/event-log-rotation.ts +158 -0
- package/src/state/event-log.ts +52 -2
- package/src/state/mailbox.ts +87 -7
- package/src/state/state-store.ts +24 -4
- package/src/state/task-claims.ts +44 -44
- package/src/state/types.ts +20 -0
- 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/team-serializer.ts +38 -38
- package/src/types/diff.d.ts +18 -18
- package/src/ui/agent-management-overlay.ts +144 -0
- package/src/ui/crew-footer.ts +101 -101
- package/src/ui/crew-select-list.ts +111 -111
- package/src/ui/crew-widget.ts +11 -2
- package/src/ui/dashboard-panes/cancellation-pane.ts +43 -0
- package/src/ui/dashboard-panes/capability-pane.ts +60 -0
- package/src/ui/dashboard-panes/mailbox-pane.ts +35 -11
- package/src/ui/dashboard-panes/metrics-pane.ts +34 -34
- package/src/ui/dynamic-border.ts +25 -25
- package/src/ui/layout-primitives.ts +106 -106
- package/src/ui/live-run-sidebar.ts +4 -0
- package/src/ui/loaders.ts +158 -158
- package/src/ui/powerbar-publisher.ts +77 -15
- package/src/ui/render-coalescer.ts +51 -0
- package/src/ui/render-diff.ts +119 -119
- package/src/ui/render-scheduler.ts +143 -143
- package/src/ui/run-dashboard.ts +4 -0
- package/src/ui/run-event-bus.ts +209 -0
- package/src/ui/run-snapshot-cache.ts +68 -16
- package/src/ui/snapshot-types.ts +8 -0
- package/src/ui/spinner.ts +17 -17
- package/src/ui/status-colors.ts +58 -58
- package/src/ui/syntax-highlight.ts +116 -116
- package/src/ui/transcript-entries.ts +258 -0
- package/src/utils/atomic-write.ts +33 -33
- package/src/utils/completion-dedupe.ts +63 -63
- package/src/utils/frontmatter.ts +68 -68
- package/src/utils/git.ts +262 -262
- package/src/utils/ids.ts +17 -12
- package/src/utils/incremental-reader.ts +104 -0
- package/src/utils/names.ts +27 -27
- package/src/utils/redaction.ts +44 -44
- package/src/utils/safe-paths.ts +47 -47
- package/src/utils/scan-cache.ts +137 -0
- package/src/utils/sleep.ts +32 -32
- package/src/utils/sse-parser.ts +134 -0
- package/src/utils/task-name-generator.ts +337 -0
- package/src/utils/visual.ts +33 -2
- package/src/workflows/validate-workflow.ts +40 -40
- package/src/worktree/branch-freshness.ts +45 -45
- package/src/worktree/cleanup.ts +2 -1
- 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/workflows/default.workflow.md +29 -29
- package/workflows/fast-fix.workflow.md +22 -22
- package/workflows/implementation.workflow.md +38 -38
- package/workflows/parallel-research.workflow.md +46 -46
- package/workflows/research.workflow.md +22 -22
- package/workflows/review.workflow.md +30 -30
|
@@ -1,34 +1,34 @@
|
|
|
1
|
-
import type { MetricRegistry } from "../../observability/metric-registry.ts";
|
|
2
|
-
import type { HistogramPoint, MetricLabels, MetricPoint } from "../../observability/metrics-primitives.ts";
|
|
3
|
-
import type { RunUiSnapshot } from "../snapshot-types.ts";
|
|
4
|
-
|
|
5
|
-
export interface MetricsPaneOptions {
|
|
6
|
-
registry?: MetricRegistry;
|
|
7
|
-
maxCounters?: number;
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
function labelsText(labels: MetricLabels): string {
|
|
11
|
-
const entries = Object.entries(labels);
|
|
12
|
-
return entries.length ? `{${entries.map(([key, value]) => `${key}=${value}`).join(",")}}` : "";
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
function isHistogramPoint(point: MetricPoint | HistogramPoint): point is HistogramPoint {
|
|
16
|
-
return "quantiles" in point;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
export function renderMetricsPane(_snapshot: RunUiSnapshot | undefined, opts: MetricsPaneOptions = {}): string[] {
|
|
20
|
-
if (!opts.registry) return ["Metrics pane: registry unavailable"];
|
|
21
|
-
const snapshots = opts.registry.snapshot();
|
|
22
|
-
if (!snapshots.length) return ["Metrics pane: no metrics recorded"];
|
|
23
|
-
const lines = ["Metrics pane: top metrics"];
|
|
24
|
-
for (const snapshot of snapshots.slice(0, opts.maxCounters ?? 10)) {
|
|
25
|
-
const first = snapshot.values[0];
|
|
26
|
-
if (!first) {
|
|
27
|
-
lines.push(`${snapshot.name}: empty`);
|
|
28
|
-
continue;
|
|
29
|
-
}
|
|
30
|
-
if (isHistogramPoint(first)) lines.push(`${snapshot.name}${labelsText(first.labels)} count=${first.count} p95=${Number.isFinite(first.quantiles.p95) ? Math.round(first.quantiles.p95) : "n/a"}`);
|
|
31
|
-
else lines.push(`${snapshot.name}${labelsText(first.labels)} ${first.value}`);
|
|
32
|
-
}
|
|
33
|
-
return lines;
|
|
34
|
-
}
|
|
1
|
+
import type { MetricRegistry } from "../../observability/metric-registry.ts";
|
|
2
|
+
import type { HistogramPoint, MetricLabels, MetricPoint } from "../../observability/metrics-primitives.ts";
|
|
3
|
+
import type { RunUiSnapshot } from "../snapshot-types.ts";
|
|
4
|
+
|
|
5
|
+
export interface MetricsPaneOptions {
|
|
6
|
+
registry?: MetricRegistry;
|
|
7
|
+
maxCounters?: number;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
function labelsText(labels: MetricLabels): string {
|
|
11
|
+
const entries = Object.entries(labels);
|
|
12
|
+
return entries.length ? `{${entries.map(([key, value]) => `${key}=${value}`).join(",")}}` : "";
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
function isHistogramPoint(point: MetricPoint | HistogramPoint): point is HistogramPoint {
|
|
16
|
+
return "quantiles" in point;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function renderMetricsPane(_snapshot: RunUiSnapshot | undefined, opts: MetricsPaneOptions = {}): string[] {
|
|
20
|
+
if (!opts.registry) return ["Metrics pane: registry unavailable"];
|
|
21
|
+
const snapshots = opts.registry.snapshot();
|
|
22
|
+
if (!snapshots.length) return ["Metrics pane: no metrics recorded"];
|
|
23
|
+
const lines = ["Metrics pane: top metrics"];
|
|
24
|
+
for (const snapshot of snapshots.slice(0, opts.maxCounters ?? 10)) {
|
|
25
|
+
const first = snapshot.values[0];
|
|
26
|
+
if (!first) {
|
|
27
|
+
lines.push(`${snapshot.name}: empty`);
|
|
28
|
+
continue;
|
|
29
|
+
}
|
|
30
|
+
if (isHistogramPoint(first)) lines.push(`${snapshot.name}${labelsText(first.labels)} count=${first.count} p95=${Number.isFinite(first.quantiles.p95) ? Math.round(first.quantiles.p95) : "n/a"}`);
|
|
31
|
+
else lines.push(`${snapshot.name}${labelsText(first.labels)} ${first.value}`);
|
|
32
|
+
}
|
|
33
|
+
return lines;
|
|
34
|
+
}
|
package/src/ui/dynamic-border.ts
CHANGED
|
@@ -1,25 +1,25 @@
|
|
|
1
|
-
import type { CrewTheme } from "./theme-adapter.ts";
|
|
2
|
-
|
|
3
|
-
export interface DynamicCrewBorderOptions {
|
|
4
|
-
color?: (value: string) => string;
|
|
5
|
-
char?: string;
|
|
6
|
-
}
|
|
7
|
-
|
|
8
|
-
export class DynamicCrewBorder {
|
|
9
|
-
private readonly theme: CrewTheme;
|
|
10
|
-
private readonly color?: (value: string) => string;
|
|
11
|
-
private readonly char: string;
|
|
12
|
-
|
|
13
|
-
constructor(theme: CrewTheme, options: DynamicCrewBorderOptions = {}) {
|
|
14
|
-
this.theme = theme;
|
|
15
|
-
this.color = options.color;
|
|
16
|
-
this.char = options.char && options.char.length > 0 ? options.char : "─";
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
render(width: number): string[] {
|
|
20
|
-
const line = this.char.repeat(Math.max(0, width));
|
|
21
|
-
return [this.color ? this.color(line) : this.theme.fg("border", line)];
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
invalidate(): void {}
|
|
25
|
-
}
|
|
1
|
+
import type { CrewTheme } from "./theme-adapter.ts";
|
|
2
|
+
|
|
3
|
+
export interface DynamicCrewBorderOptions {
|
|
4
|
+
color?: (value: string) => string;
|
|
5
|
+
char?: string;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
export class DynamicCrewBorder {
|
|
9
|
+
private readonly theme: CrewTheme;
|
|
10
|
+
private readonly color?: (value: string) => string;
|
|
11
|
+
private readonly char: string;
|
|
12
|
+
|
|
13
|
+
constructor(theme: CrewTheme, options: DynamicCrewBorderOptions = {}) {
|
|
14
|
+
this.theme = theme;
|
|
15
|
+
this.color = options.color;
|
|
16
|
+
this.char = options.char && options.char.length > 0 ? options.char : "─";
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
render(width: number): string[] {
|
|
20
|
+
const line = this.char.repeat(Math.max(0, width));
|
|
21
|
+
return [this.color ? this.color(line) : this.theme.fg("border", line)];
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
invalidate(): void {}
|
|
25
|
+
}
|
|
@@ -1,106 +1,106 @@
|
|
|
1
|
-
import { pad, wrapHard } from "../utils/visual.ts";
|
|
2
|
-
|
|
3
|
-
export interface RenderableComponent {
|
|
4
|
-
invalidate(): void;
|
|
5
|
-
render(width: number): string[];
|
|
6
|
-
}
|
|
7
|
-
|
|
8
|
-
export class Container implements RenderableComponent {
|
|
9
|
-
private children: RenderableComponent[] = [];
|
|
10
|
-
|
|
11
|
-
addChild(child: RenderableComponent): void {
|
|
12
|
-
this.children.push(child);
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
clear(): void {
|
|
16
|
-
this.children = [];
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
invalidate(): void {
|
|
20
|
-
for (const child of this.children) {
|
|
21
|
-
child.invalidate();
|
|
22
|
-
}
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
render(width: number): string[] {
|
|
26
|
-
const lines: string[] = [];
|
|
27
|
-
for (const child of this.children) {
|
|
28
|
-
lines.push(...child.render(width));
|
|
29
|
-
}
|
|
30
|
-
return lines;
|
|
31
|
-
}
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
export class Box extends Container {
|
|
35
|
-
private readonly paddingX: number;
|
|
36
|
-
private readonly paddingY: number;
|
|
37
|
-
|
|
38
|
-
constructor(paddingX = 0, paddingY = 0) {
|
|
39
|
-
super();
|
|
40
|
-
this.paddingX = paddingX;
|
|
41
|
-
this.paddingY = paddingY;
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
render(width: number): string[] {
|
|
45
|
-
const innerWidth = Math.max(1, width - this.paddingX * 2);
|
|
46
|
-
const rows = super.render(innerWidth);
|
|
47
|
-
const paddedRows: string[] = [];
|
|
48
|
-
const left = " ".repeat(this.paddingX);
|
|
49
|
-
const right = " ".repeat(this.paddingX);
|
|
50
|
-
for (const row of rows) {
|
|
51
|
-
paddedRows.push(pad(`${left}${row}${right}`, width));
|
|
52
|
-
}
|
|
53
|
-
const emptyRow = pad("", width);
|
|
54
|
-
if (this.paddingY <= 0) return paddedRows;
|
|
55
|
-
if (this.paddingY > 0) {
|
|
56
|
-
const topAndBottom = Array.from({ length: this.paddingY }, () => emptyRow);
|
|
57
|
-
return [...topAndBottom, ...paddedRows, ...topAndBottom];
|
|
58
|
-
}
|
|
59
|
-
return paddedRows;
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
export class Text implements RenderableComponent {
|
|
64
|
-
private text: string;
|
|
65
|
-
private cachedWidth = 0;
|
|
66
|
-
private cachedResult: string[] = [];
|
|
67
|
-
|
|
68
|
-
constructor(text = "") {
|
|
69
|
-
this.text = text;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
setText(text: string): void {
|
|
73
|
-
if (text === this.text) return;
|
|
74
|
-
this.text = text;
|
|
75
|
-
this.invalidate();
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
invalidate(): void {
|
|
79
|
-
this.cachedWidth = 0;
|
|
80
|
-
this.cachedResult = [];
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
render(width: number): string[] {
|
|
84
|
-
if (this.cachedWidth === width) return this.cachedResult;
|
|
85
|
-
const wrapped = wrapHard(this.text, Math.max(1, width));
|
|
86
|
-
const lines = wrapped.length ? wrapped : [""];
|
|
87
|
-
this.cachedWidth = width;
|
|
88
|
-
this.cachedResult = lines.map((line) => pad(line, width));
|
|
89
|
-
return this.cachedResult;
|
|
90
|
-
}
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
export class Spacer implements RenderableComponent {
|
|
94
|
-
private readonly rows: number;
|
|
95
|
-
|
|
96
|
-
constructor(rows = 0) {
|
|
97
|
-
this.rows = rows;
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
render(width: number): string[] {
|
|
101
|
-
if (this.rows <= 0) return [];
|
|
102
|
-
return Array.from({ length: Math.max(0, this.rows) }, () => pad("", width));
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
invalidate(): void {}
|
|
106
|
-
}
|
|
1
|
+
import { pad, wrapHard } from "../utils/visual.ts";
|
|
2
|
+
|
|
3
|
+
export interface RenderableComponent {
|
|
4
|
+
invalidate(): void;
|
|
5
|
+
render(width: number): string[];
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
export class Container implements RenderableComponent {
|
|
9
|
+
private children: RenderableComponent[] = [];
|
|
10
|
+
|
|
11
|
+
addChild(child: RenderableComponent): void {
|
|
12
|
+
this.children.push(child);
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
clear(): void {
|
|
16
|
+
this.children = [];
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
invalidate(): void {
|
|
20
|
+
for (const child of this.children) {
|
|
21
|
+
child.invalidate();
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
render(width: number): string[] {
|
|
26
|
+
const lines: string[] = [];
|
|
27
|
+
for (const child of this.children) {
|
|
28
|
+
lines.push(...child.render(width));
|
|
29
|
+
}
|
|
30
|
+
return lines;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export class Box extends Container {
|
|
35
|
+
private readonly paddingX: number;
|
|
36
|
+
private readonly paddingY: number;
|
|
37
|
+
|
|
38
|
+
constructor(paddingX = 0, paddingY = 0) {
|
|
39
|
+
super();
|
|
40
|
+
this.paddingX = paddingX;
|
|
41
|
+
this.paddingY = paddingY;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
render(width: number): string[] {
|
|
45
|
+
const innerWidth = Math.max(1, width - this.paddingX * 2);
|
|
46
|
+
const rows = super.render(innerWidth);
|
|
47
|
+
const paddedRows: string[] = [];
|
|
48
|
+
const left = " ".repeat(this.paddingX);
|
|
49
|
+
const right = " ".repeat(this.paddingX);
|
|
50
|
+
for (const row of rows) {
|
|
51
|
+
paddedRows.push(pad(`${left}${row}${right}`, width));
|
|
52
|
+
}
|
|
53
|
+
const emptyRow = pad("", width);
|
|
54
|
+
if (this.paddingY <= 0) return paddedRows;
|
|
55
|
+
if (this.paddingY > 0) {
|
|
56
|
+
const topAndBottom = Array.from({ length: this.paddingY }, () => emptyRow);
|
|
57
|
+
return [...topAndBottom, ...paddedRows, ...topAndBottom];
|
|
58
|
+
}
|
|
59
|
+
return paddedRows;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export class Text implements RenderableComponent {
|
|
64
|
+
private text: string;
|
|
65
|
+
private cachedWidth = 0;
|
|
66
|
+
private cachedResult: string[] = [];
|
|
67
|
+
|
|
68
|
+
constructor(text = "") {
|
|
69
|
+
this.text = text;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
setText(text: string): void {
|
|
73
|
+
if (text === this.text) return;
|
|
74
|
+
this.text = text;
|
|
75
|
+
this.invalidate();
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
invalidate(): void {
|
|
79
|
+
this.cachedWidth = 0;
|
|
80
|
+
this.cachedResult = [];
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
render(width: number): string[] {
|
|
84
|
+
if (this.cachedWidth === width) return this.cachedResult;
|
|
85
|
+
const wrapped = wrapHard(this.text, Math.max(1, width));
|
|
86
|
+
const lines = wrapped.length ? wrapped : [""];
|
|
87
|
+
this.cachedWidth = width;
|
|
88
|
+
this.cachedResult = lines.map((line) => pad(line, width));
|
|
89
|
+
return this.cachedResult;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
export class Spacer implements RenderableComponent {
|
|
94
|
+
private readonly rows: number;
|
|
95
|
+
|
|
96
|
+
constructor(rows = 0) {
|
|
97
|
+
this.rows = rows;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
render(width: number): string[] {
|
|
101
|
+
if (this.rows <= 0) return [];
|
|
102
|
+
return Array.from({ length: Math.max(0, this.rows) }, () => pad("", width));
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
invalidate(): void {}
|
|
106
|
+
}
|
|
@@ -14,6 +14,7 @@ import { asCrewTheme, subscribeThemeChange } from "./theme-adapter.ts";
|
|
|
14
14
|
import { Box, Text } from "./layout-primitives.ts";
|
|
15
15
|
import type { RunSnapshotCache, RunUiSnapshot } from "./snapshot-types.ts";
|
|
16
16
|
import { spinnerBucket, spinnerFrame } from "./spinner.ts";
|
|
17
|
+
import { runEventBus } from "./run-event-bus.ts";
|
|
17
18
|
|
|
18
19
|
const TASK_READ_TTL_MS = 200;
|
|
19
20
|
|
|
@@ -59,6 +60,7 @@ export class LiveRunSidebar {
|
|
|
59
60
|
private readonly theme: CrewTheme;
|
|
60
61
|
private readonly config: CrewUiConfig;
|
|
61
62
|
private readonly unsubscribeTheme: () => void;
|
|
63
|
+
private readonly unsubscribeEventBus: () => void;
|
|
62
64
|
private readonly snapshotCache?: RunSnapshotCache;
|
|
63
65
|
private cachedLines: string[] = [];
|
|
64
66
|
private cachedWidth = 0;
|
|
@@ -72,6 +74,7 @@ export class LiveRunSidebar {
|
|
|
72
74
|
this.config = input.config ?? {};
|
|
73
75
|
this.snapshotCache = input.snapshotCache;
|
|
74
76
|
this.unsubscribeTheme = subscribeThemeChange(input.theme, () => this.invalidate());
|
|
77
|
+
this.unsubscribeEventBus = runEventBus.onAny(() => this.invalidate());
|
|
75
78
|
}
|
|
76
79
|
|
|
77
80
|
private buildSignature(manifestStatus: string, tasks: TeamTaskState[], agents: ReturnType<typeof readCrewAgents>, waitingCount: number, snapshot?: RunUiSnapshot): string {
|
|
@@ -99,6 +102,7 @@ export class LiveRunSidebar {
|
|
|
99
102
|
|
|
100
103
|
dispose(): void {
|
|
101
104
|
this.unsubscribeTheme();
|
|
105
|
+
this.unsubscribeEventBus();
|
|
102
106
|
}
|
|
103
107
|
|
|
104
108
|
render(width: number): string[] {
|