pi-crew 0.1.37 → 0.1.39
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 +1 -1
- package/CHANGELOG.md +27 -0
- package/README.md +5 -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/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-extension-examples.md +297 -297
- package/docs/research-extension-system.md +324 -324
- package/docs/research-optimization-plan.md +548 -548
- package/docs/research-pi-coding-agent.md +357 -357
- package/docs/research-source-pi-crew-reference.md +174 -174
- package/docs/resource-formats.md +10 -8
- package/docs/runtime-flow.md +148 -148
- package/docs/source-runtime-refactor-map.md +83 -83
- package/docs/usage.md +6 -0
- package/index.ts +6 -6
- package/package.json +3 -3
- package/schema.json +2 -2
- package/src/agents/agent-serializer.ts +34 -34
- package/src/config/config.ts +8 -4
- package/src/extension/cross-extension-rpc.ts +82 -82
- package/src/extension/import-index.ts +18 -2
- package/src/extension/register.ts +11 -1
- package/src/extension/registration/compaction-guard.ts +125 -125
- package/src/extension/registration/subagent-helpers.ts +30 -6
- package/src/extension/registration/subagent-tools.ts +8 -3
- package/src/extension/result-watcher.ts +98 -98
- package/src/extension/run-import.ts +12 -2
- package/src/extension/run-index.ts +12 -2
- package/src/extension/run-maintenance.ts +24 -24
- package/src/extension/team-tool/api.ts +54 -14
- package/src/extension/team-tool/cancel.ts +31 -31
- package/src/extension/team-tool/doctor.ts +179 -179
- package/src/extension/team-tool/inspect.ts +41 -41
- package/src/extension/team-tool/lifecycle-actions.ts +79 -79
- package/src/extension/team-tool/plan.ts +19 -19
- package/src/extension/team-tool/status.ts +73 -73
- package/src/observability/correlation.ts +35 -35
- package/src/observability/event-to-metric.ts +54 -54
- package/src/observability/exporters/adapter.ts +24 -24
- package/src/observability/exporters/otlp-exporter.ts +65 -65
- package/src/observability/exporters/prometheus-exporter.ts +47 -47
- package/src/observability/metric-registry.ts +72 -72
- package/src/observability/metric-retention.ts +46 -46
- package/src/observability/metric-sink.ts +51 -51
- package/src/observability/metrics-primitives.ts +166 -166
- package/src/prompt/prompt-runtime.ts +68 -68
- package/src/runtime/agent-control.ts +64 -64
- package/src/runtime/agent-memory.ts +72 -72
- package/src/runtime/agent-observability.ts +114 -113
- package/src/runtime/async-marker.ts +26 -26
- package/src/runtime/background-runner.ts +53 -53
- package/src/runtime/crash-recovery.ts +56 -56
- package/src/runtime/crew-agent-records.ts +54 -9
- package/src/runtime/crew-agent-runtime.ts +58 -58
- package/src/runtime/deadletter.ts +36 -36
- package/src/runtime/direct-run.ts +35 -35
- package/src/runtime/foreground-control.ts +82 -82
- package/src/runtime/green-contract.ts +46 -46
- package/src/runtime/group-join.ts +88 -88
- package/src/runtime/heartbeat-gradient.ts +28 -28
- package/src/runtime/heartbeat-watcher.ts +80 -80
- package/src/runtime/live-agent-control.ts +87 -78
- package/src/runtime/live-agent-manager.ts +85 -85
- package/src/runtime/live-control-realtime.ts +36 -36
- package/src/runtime/live-session-runtime.ts +299 -299
- package/src/runtime/manifest-cache.ts +248 -212
- package/src/runtime/model-fallback.ts +261 -261
- package/src/runtime/parallel-research.ts +44 -44
- package/src/runtime/parallel-utils.ts +99 -99
- package/src/runtime/pi-json-output.ts +111 -111
- package/src/runtime/policy-engine.ts +78 -78
- package/src/runtime/post-exit-stdio-guard.ts +86 -86
- package/src/runtime/process-status.ts +56 -56
- package/src/runtime/progress-event-coalescer.ts +43 -43
- package/src/runtime/recovery-recipes.ts +74 -74
- package/src/runtime/retry-executor.ts +59 -59
- package/src/runtime/role-permission.ts +39 -39
- package/src/runtime/session-usage.ts +79 -79
- package/src/runtime/sidechain-output.ts +28 -28
- package/src/runtime/subagent-manager.ts +80 -12
- package/src/runtime/task-display.ts +38 -38
- package/src/runtime/task-output-context.ts +127 -106
- package/src/runtime/task-runner/live-executor.ts +98 -98
- package/src/runtime/task-runner/progress.ts +111 -111
- package/src/runtime/task-runner/result-utils.ts +14 -14
- package/src/runtime/task-runner/state-helpers.ts +22 -22
- package/src/runtime/team-runner.ts +1 -1
- package/src/runtime/worker-heartbeat.ts +21 -21
- package/src/runtime/worker-startup.ts +57 -57
- package/src/schema/config-schema.ts +21 -21
- package/src/schema/team-tool-schema.ts +100 -100
- package/src/state/artifact-store.ts +122 -108
- package/src/state/contracts.ts +105 -105
- package/src/state/jsonl-writer.ts +77 -77
- package/src/state/mailbox.ts +67 -22
- package/src/state/state-store.ts +36 -5
- package/src/state/task-claims.ts +42 -42
- 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 +27 -5
- package/src/teams/team-serializer.ts +38 -36
- package/src/types/diff.d.ts +18 -18
- package/src/ui/crew-footer.ts +101 -101
- package/src/ui/crew-select-list.ts +111 -111
- 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/loaders.ts +158 -158
- package/src/ui/mascot.ts +441 -441
- package/src/ui/render-diff.ts +119 -119
- package/src/ui/run-dashboard.ts +5 -2
- package/src/ui/run-snapshot-cache.ts +19 -8
- package/src/ui/spinner.ts +17 -17
- package/src/ui/status-colors.ts +54 -54
- package/src/ui/syntax-highlight.ts +116 -116
- package/src/ui/transcript-viewer.ts +15 -1
- package/src/utils/completion-dedupe.ts +63 -63
- package/src/utils/file-coalescer.ts +84 -84
- package/src/utils/frontmatter.ts +36 -36
- package/src/utils/fs-watch.ts +31 -31
- package/src/utils/git.ts +262 -262
- package/src/utils/ids.ts +12 -12
- package/src/utils/names.ts +26 -26
- package/src/utils/paths.ts +3 -2
- package/src/utils/safe-paths.ts +34 -0
- package/src/utils/sleep.ts +32 -32
- package/src/utils/timings.ts +31 -31
- package/src/utils/visual.ts +159 -159
- package/src/workflows/discover-workflows.ts +30 -3
- package/src/workflows/validate-workflow.ts +40 -40
- package/src/worktree/branch-freshness.ts +45 -45
- 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,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
|
+
}
|
package/src/ui/loaders.ts
CHANGED
|
@@ -1,158 +1,158 @@
|
|
|
1
|
-
import { pad, truncate } from "../utils/visual.ts";
|
|
2
|
-
import type { CrewTheme } from "./theme-adapter.ts";
|
|
3
|
-
import { asCrewTheme } from "./theme-adapter.ts";
|
|
4
|
-
import { DynamicCrewBorder } from "./dynamic-border.ts";
|
|
5
|
-
|
|
6
|
-
export interface BorderedLoaderOptions {
|
|
7
|
-
message: string;
|
|
8
|
-
cancellable?: boolean;
|
|
9
|
-
frames?: string[];
|
|
10
|
-
intervalMs?: number;
|
|
11
|
-
minWidth?: number;
|
|
12
|
-
onAbort?: () => void;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
const DEFAULT_FRAMES = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"];
|
|
16
|
-
|
|
17
|
-
export class CrewBorderedLoader {
|
|
18
|
-
private readonly abortController = new AbortController();
|
|
19
|
-
private readonly frameOptions: string[];
|
|
20
|
-
private readonly intervalMs: number;
|
|
21
|
-
private readonly minWidth: number;
|
|
22
|
-
private readonly onAbort?: () => void;
|
|
23
|
-
private theme: CrewTheme;
|
|
24
|
-
private message: string;
|
|
25
|
-
private lineCache = "";
|
|
26
|
-
private width = 0;
|
|
27
|
-
private startedAt = Date.now();
|
|
28
|
-
|
|
29
|
-
constructor(_ui: unknown, themeLike: unknown, options: BorderedLoaderOptions) {
|
|
30
|
-
const theme = asCrewTheme(themeLike);
|
|
31
|
-
this.theme = theme;
|
|
32
|
-
this.message = options.message;
|
|
33
|
-
this.minWidth = Math.max(12, options.minWidth ?? 24);
|
|
34
|
-
this.onAbort = options.onAbort;
|
|
35
|
-
this.frameOptions = options.frames ?? DEFAULT_FRAMES;
|
|
36
|
-
this.intervalMs = Math.max(40, options.intervalMs ?? 120);
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
private spinnerFrame(): string {
|
|
40
|
-
if (this.frameOptions.length === 0) return "•";
|
|
41
|
-
const elapsed = Date.now() - this.startedAt;
|
|
42
|
-
const index = Math.floor(elapsed / this.intervalMs) % this.frameOptions.length;
|
|
43
|
-
return this.frameOptions[Math.max(0, index)];
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
setMessage(message: string): void {
|
|
47
|
-
this.message = message;
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
get signal(): AbortSignal {
|
|
51
|
-
return this.abortController.signal;
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
handleInput(data: string): void {
|
|
55
|
-
if (!this.onAbort || this.abortController.signal.aborted) return;
|
|
56
|
-
if (data === "c" || data === "q" || data === "\u001b" || data === "\u0003") {
|
|
57
|
-
this.abortController.abort();
|
|
58
|
-
this.onAbort();
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
render(width: number): string[] {
|
|
63
|
-
if (width === this.width && this.lineCache) {
|
|
64
|
-
return this.lineCache.split("\n");
|
|
65
|
-
}
|
|
66
|
-
const innerWidth = Math.max(this.minWidth - 4, 1);
|
|
67
|
-
const contentWidth = Math.max(1, Math.min(width - 4, innerWidth));
|
|
68
|
-
const frame = this.spinnerFrame();
|
|
69
|
-
const loaderLine = ` ${frame} ${truncate(this.message, Math.max(1, contentWidth - 4))} `;
|
|
70
|
-
const body = ` ${truncate(loaderLine, contentWidth - 2)} `;
|
|
71
|
-
const inner = ` ${pad(body, contentWidth - 1)} `;
|
|
72
|
-
const padWidth = Math.max(0, width - (contentWidth + 4));
|
|
73
|
-
const leftRightPad = " ".repeat(Math.floor(padWidth / 2));
|
|
74
|
-
const widthAwareInner = contentWidth + padWidth;
|
|
75
|
-
const border = new DynamicCrewBorder(this.theme).render(widthAwareInner + 2)[0];
|
|
76
|
-
const top = `${leftRightPad}${this.theme.fg("border", "┌")}${border}${this.theme.fg("border", "┐")}`;
|
|
77
|
-
const line = `${leftRightPad}${this.theme.fg("border", "│")} ${truncate(inner, widthAwareInner)} ${this.theme.fg("border", "│")}`;
|
|
78
|
-
const hint = `${leftRightPad}${this.theme.fg("border", "│")}${" ".repeat(widthAwareInner + 2)}${this.theme.fg("border", "│")}`;
|
|
79
|
-
const bottom = `${leftRightPad}${this.theme.fg("border", "└")}${border}${this.theme.fg("border", "┘")}`;
|
|
80
|
-
const lineWithHint = optionsHint(this.theme, this.message, widthAwareInner);
|
|
81
|
-
this.width = width;
|
|
82
|
-
const lines = [
|
|
83
|
-
top,
|
|
84
|
-
line,
|
|
85
|
-
`${leftRightPad}│ ${pad(lineWithHint, widthAwareInner)} │`,
|
|
86
|
-
hint,
|
|
87
|
-
bottom,
|
|
88
|
-
];
|
|
89
|
-
this.lineCache = lines.join("\n");
|
|
90
|
-
return lines;
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
invalidate(): void {
|
|
94
|
-
this.lineCache = "";
|
|
95
|
-
this.width = 0;
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
dispose(): void {
|
|
99
|
-
this.abortController.abort();
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
export interface CountdownTimerOptions {
|
|
104
|
-
timeoutMs: number;
|
|
105
|
-
onTick: (seconds: number) => void;
|
|
106
|
-
onExpire: () => void;
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
export class CountdownTimer {
|
|
110
|
-
private readonly onExpire: () => void;
|
|
111
|
-
private readonly onTick: (seconds: number) => void;
|
|
112
|
-
private readonly startedAt: number;
|
|
113
|
-
private readonly timeoutMs: number;
|
|
114
|
-
private timer: ReturnType<typeof setTimeout> | undefined;
|
|
115
|
-
private expired = false;
|
|
116
|
-
|
|
117
|
-
constructor(options: CountdownTimerOptions) {
|
|
118
|
-
this.timeoutMs = Math.max(0, options.timeoutMs);
|
|
119
|
-
this.onTick = options.onTick;
|
|
120
|
-
this.onExpire = options.onExpire;
|
|
121
|
-
this.startedAt = Date.now();
|
|
122
|
-
this.onTick(this.secondsLeft());
|
|
123
|
-
if (this.timeoutMs === 0) {
|
|
124
|
-
this.emitExpire();
|
|
125
|
-
return;
|
|
126
|
-
}
|
|
127
|
-
this.timer = setInterval(() => {
|
|
128
|
-
const seconds = this.secondsLeft();
|
|
129
|
-
this.onTick(seconds);
|
|
130
|
-
if (seconds <= 0) {
|
|
131
|
-
this.emitExpire();
|
|
132
|
-
}
|
|
133
|
-
}, 1000);
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
private emitExpire(): void {
|
|
137
|
-
if (this.expired) return;
|
|
138
|
-
this.expired = true;
|
|
139
|
-
this.dispose();
|
|
140
|
-
this.onExpire();
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
private secondsLeft(): number {
|
|
144
|
-
const remainingMs = this.startedAt + this.timeoutMs - Date.now();
|
|
145
|
-
return Math.max(0, Math.ceil(remainingMs / 1000));
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
dispose(): void {
|
|
149
|
-
if (this.timer === undefined) return;
|
|
150
|
-
clearInterval(this.timer);
|
|
151
|
-
this.timer = undefined;
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
function optionsHint(theme: CrewTheme, message: string, width: number): string {
|
|
156
|
-
if (!message) return "";
|
|
157
|
-
return truncate(theme.fg("muted", message), width);
|
|
158
|
-
}
|
|
1
|
+
import { pad, truncate } from "../utils/visual.ts";
|
|
2
|
+
import type { CrewTheme } from "./theme-adapter.ts";
|
|
3
|
+
import { asCrewTheme } from "./theme-adapter.ts";
|
|
4
|
+
import { DynamicCrewBorder } from "./dynamic-border.ts";
|
|
5
|
+
|
|
6
|
+
export interface BorderedLoaderOptions {
|
|
7
|
+
message: string;
|
|
8
|
+
cancellable?: boolean;
|
|
9
|
+
frames?: string[];
|
|
10
|
+
intervalMs?: number;
|
|
11
|
+
minWidth?: number;
|
|
12
|
+
onAbort?: () => void;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
const DEFAULT_FRAMES = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"];
|
|
16
|
+
|
|
17
|
+
export class CrewBorderedLoader {
|
|
18
|
+
private readonly abortController = new AbortController();
|
|
19
|
+
private readonly frameOptions: string[];
|
|
20
|
+
private readonly intervalMs: number;
|
|
21
|
+
private readonly minWidth: number;
|
|
22
|
+
private readonly onAbort?: () => void;
|
|
23
|
+
private theme: CrewTheme;
|
|
24
|
+
private message: string;
|
|
25
|
+
private lineCache = "";
|
|
26
|
+
private width = 0;
|
|
27
|
+
private startedAt = Date.now();
|
|
28
|
+
|
|
29
|
+
constructor(_ui: unknown, themeLike: unknown, options: BorderedLoaderOptions) {
|
|
30
|
+
const theme = asCrewTheme(themeLike);
|
|
31
|
+
this.theme = theme;
|
|
32
|
+
this.message = options.message;
|
|
33
|
+
this.minWidth = Math.max(12, options.minWidth ?? 24);
|
|
34
|
+
this.onAbort = options.onAbort;
|
|
35
|
+
this.frameOptions = options.frames ?? DEFAULT_FRAMES;
|
|
36
|
+
this.intervalMs = Math.max(40, options.intervalMs ?? 120);
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
private spinnerFrame(): string {
|
|
40
|
+
if (this.frameOptions.length === 0) return "•";
|
|
41
|
+
const elapsed = Date.now() - this.startedAt;
|
|
42
|
+
const index = Math.floor(elapsed / this.intervalMs) % this.frameOptions.length;
|
|
43
|
+
return this.frameOptions[Math.max(0, index)];
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
setMessage(message: string): void {
|
|
47
|
+
this.message = message;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
get signal(): AbortSignal {
|
|
51
|
+
return this.abortController.signal;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
handleInput(data: string): void {
|
|
55
|
+
if (!this.onAbort || this.abortController.signal.aborted) return;
|
|
56
|
+
if (data === "c" || data === "q" || data === "\u001b" || data === "\u0003") {
|
|
57
|
+
this.abortController.abort();
|
|
58
|
+
this.onAbort();
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
render(width: number): string[] {
|
|
63
|
+
if (width === this.width && this.lineCache) {
|
|
64
|
+
return this.lineCache.split("\n");
|
|
65
|
+
}
|
|
66
|
+
const innerWidth = Math.max(this.minWidth - 4, 1);
|
|
67
|
+
const contentWidth = Math.max(1, Math.min(width - 4, innerWidth));
|
|
68
|
+
const frame = this.spinnerFrame();
|
|
69
|
+
const loaderLine = ` ${frame} ${truncate(this.message, Math.max(1, contentWidth - 4))} `;
|
|
70
|
+
const body = ` ${truncate(loaderLine, contentWidth - 2)} `;
|
|
71
|
+
const inner = ` ${pad(body, contentWidth - 1)} `;
|
|
72
|
+
const padWidth = Math.max(0, width - (contentWidth + 4));
|
|
73
|
+
const leftRightPad = " ".repeat(Math.floor(padWidth / 2));
|
|
74
|
+
const widthAwareInner = contentWidth + padWidth;
|
|
75
|
+
const border = new DynamicCrewBorder(this.theme).render(widthAwareInner + 2)[0];
|
|
76
|
+
const top = `${leftRightPad}${this.theme.fg("border", "┌")}${border}${this.theme.fg("border", "┐")}`;
|
|
77
|
+
const line = `${leftRightPad}${this.theme.fg("border", "│")} ${truncate(inner, widthAwareInner)} ${this.theme.fg("border", "│")}`;
|
|
78
|
+
const hint = `${leftRightPad}${this.theme.fg("border", "│")}${" ".repeat(widthAwareInner + 2)}${this.theme.fg("border", "│")}`;
|
|
79
|
+
const bottom = `${leftRightPad}${this.theme.fg("border", "└")}${border}${this.theme.fg("border", "┘")}`;
|
|
80
|
+
const lineWithHint = optionsHint(this.theme, this.message, widthAwareInner);
|
|
81
|
+
this.width = width;
|
|
82
|
+
const lines = [
|
|
83
|
+
top,
|
|
84
|
+
line,
|
|
85
|
+
`${leftRightPad}│ ${pad(lineWithHint, widthAwareInner)} │`,
|
|
86
|
+
hint,
|
|
87
|
+
bottom,
|
|
88
|
+
];
|
|
89
|
+
this.lineCache = lines.join("\n");
|
|
90
|
+
return lines;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
invalidate(): void {
|
|
94
|
+
this.lineCache = "";
|
|
95
|
+
this.width = 0;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
dispose(): void {
|
|
99
|
+
this.abortController.abort();
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
export interface CountdownTimerOptions {
|
|
104
|
+
timeoutMs: number;
|
|
105
|
+
onTick: (seconds: number) => void;
|
|
106
|
+
onExpire: () => void;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
export class CountdownTimer {
|
|
110
|
+
private readonly onExpire: () => void;
|
|
111
|
+
private readonly onTick: (seconds: number) => void;
|
|
112
|
+
private readonly startedAt: number;
|
|
113
|
+
private readonly timeoutMs: number;
|
|
114
|
+
private timer: ReturnType<typeof setTimeout> | undefined;
|
|
115
|
+
private expired = false;
|
|
116
|
+
|
|
117
|
+
constructor(options: CountdownTimerOptions) {
|
|
118
|
+
this.timeoutMs = Math.max(0, options.timeoutMs);
|
|
119
|
+
this.onTick = options.onTick;
|
|
120
|
+
this.onExpire = options.onExpire;
|
|
121
|
+
this.startedAt = Date.now();
|
|
122
|
+
this.onTick(this.secondsLeft());
|
|
123
|
+
if (this.timeoutMs === 0) {
|
|
124
|
+
this.emitExpire();
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
this.timer = setInterval(() => {
|
|
128
|
+
const seconds = this.secondsLeft();
|
|
129
|
+
this.onTick(seconds);
|
|
130
|
+
if (seconds <= 0) {
|
|
131
|
+
this.emitExpire();
|
|
132
|
+
}
|
|
133
|
+
}, 1000);
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
private emitExpire(): void {
|
|
137
|
+
if (this.expired) return;
|
|
138
|
+
this.expired = true;
|
|
139
|
+
this.dispose();
|
|
140
|
+
this.onExpire();
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
private secondsLeft(): number {
|
|
144
|
+
const remainingMs = this.startedAt + this.timeoutMs - Date.now();
|
|
145
|
+
return Math.max(0, Math.ceil(remainingMs / 1000));
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
dispose(): void {
|
|
149
|
+
if (this.timer === undefined) return;
|
|
150
|
+
clearInterval(this.timer);
|
|
151
|
+
this.timer = undefined;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
function optionsHint(theme: CrewTheme, message: string, width: number): string {
|
|
156
|
+
if (!message) return "";
|
|
157
|
+
return truncate(theme.fg("muted", message), width);
|
|
158
|
+
}
|