@gotgenes/pi-subagents 1.0.0
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/.markdownlint-cli2.yaml +19 -0
- package/.prettierignore +5 -0
- package/.release-please-manifest.json +3 -0
- package/AGENTS.md +85 -0
- package/CHANGELOG.md +495 -0
- package/LICENSE +21 -0
- package/README.md +528 -0
- package/dist/agent-manager.d.ts +108 -0
- package/dist/agent-manager.js +390 -0
- package/dist/agent-runner.d.ts +93 -0
- package/dist/agent-runner.js +428 -0
- package/dist/agent-types.d.ts +48 -0
- package/dist/agent-types.js +136 -0
- package/dist/context.d.ts +12 -0
- package/dist/context.js +56 -0
- package/dist/cross-extension-rpc.d.ts +46 -0
- package/dist/cross-extension-rpc.js +54 -0
- package/dist/custom-agents.d.ts +14 -0
- package/dist/custom-agents.js +127 -0
- package/dist/default-agents.d.ts +7 -0
- package/dist/default-agents.js +119 -0
- package/dist/env.d.ts +6 -0
- package/dist/env.js +28 -0
- package/dist/group-join.d.ts +32 -0
- package/dist/group-join.js +116 -0
- package/dist/index.d.ts +13 -0
- package/dist/index.js +1731 -0
- package/dist/invocation-config.d.ts +22 -0
- package/dist/invocation-config.js +15 -0
- package/dist/memory.d.ts +49 -0
- package/dist/memory.js +151 -0
- package/dist/model-resolver.d.ts +19 -0
- package/dist/model-resolver.js +62 -0
- package/dist/output-file.d.ts +24 -0
- package/dist/output-file.js +86 -0
- package/dist/prompts.d.ts +29 -0
- package/dist/prompts.js +72 -0
- package/dist/schedule-store.d.ts +36 -0
- package/dist/schedule-store.js +144 -0
- package/dist/schedule.d.ts +109 -0
- package/dist/schedule.js +338 -0
- package/dist/settings.d.ts +66 -0
- package/dist/settings.js +130 -0
- package/dist/skill-loader.d.ts +24 -0
- package/dist/skill-loader.js +93 -0
- package/dist/types.d.ts +164 -0
- package/dist/types.js +5 -0
- package/dist/ui/agent-widget.d.ts +134 -0
- package/dist/ui/agent-widget.js +451 -0
- package/dist/ui/conversation-viewer.d.ts +35 -0
- package/dist/ui/conversation-viewer.js +252 -0
- package/dist/ui/schedule-menu.d.ts +16 -0
- package/dist/ui/schedule-menu.js +95 -0
- package/dist/usage.d.ts +50 -0
- package/dist/usage.js +49 -0
- package/dist/worktree.d.ts +36 -0
- package/dist/worktree.js +139 -0
- package/docs/decisions/0001-deferred-patches.md +75 -0
- package/package.json +68 -0
- package/prek.toml +24 -0
- package/release-please-config.json +22 -0
- package/src/agent-manager.ts +482 -0
- package/src/agent-runner.ts +625 -0
- package/src/agent-types.ts +164 -0
- package/src/context.ts +58 -0
- package/src/cross-extension-rpc.ts +95 -0
- package/src/custom-agents.ts +136 -0
- package/src/default-agents.ts +123 -0
- package/src/env.ts +33 -0
- package/src/group-join.ts +141 -0
- package/src/index.ts +1894 -0
- package/src/invocation-config.ts +40 -0
- package/src/memory.ts +165 -0
- package/src/model-resolver.ts +81 -0
- package/src/output-file.ts +96 -0
- package/src/prompts.ts +105 -0
- package/src/schedule-store.ts +143 -0
- package/src/schedule.ts +365 -0
- package/src/settings.ts +186 -0
- package/src/skill-loader.ts +102 -0
- package/src/types.ts +176 -0
- package/src/ui/agent-widget.ts +533 -0
- package/src/ui/conversation-viewer.ts +261 -0
- package/src/ui/schedule-menu.ts +104 -0
- package/src/usage.ts +60 -0
- package/src/worktree.ts +162 -0
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* group-join.ts — Manages grouped background agent completion notifications.
|
|
3
|
+
*
|
|
4
|
+
* Instead of each agent individually nudging the main agent on completion,
|
|
5
|
+
* agents in a group are held until all complete (or a timeout fires),
|
|
6
|
+
* then a single consolidated notification is sent.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type { AgentRecord } from "./types.js";
|
|
10
|
+
|
|
11
|
+
export type DeliveryCallback = (records: AgentRecord[], partial: boolean) => void;
|
|
12
|
+
|
|
13
|
+
interface AgentGroup {
|
|
14
|
+
groupId: string;
|
|
15
|
+
agentIds: Set<string>;
|
|
16
|
+
completedRecords: Map<string, AgentRecord>;
|
|
17
|
+
timeoutHandle?: ReturnType<typeof setTimeout>;
|
|
18
|
+
delivered: boolean;
|
|
19
|
+
/** Shorter timeout for stragglers after a partial delivery. */
|
|
20
|
+
isStraggler: boolean;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/** Default timeout: 30s after first completion in a group. */
|
|
24
|
+
const DEFAULT_TIMEOUT = 30_000;
|
|
25
|
+
/** Straggler re-batch timeout: 15s. */
|
|
26
|
+
const STRAGGLER_TIMEOUT = 15_000;
|
|
27
|
+
|
|
28
|
+
export class GroupJoinManager {
|
|
29
|
+
private groups = new Map<string, AgentGroup>();
|
|
30
|
+
private agentToGroup = new Map<string, string>();
|
|
31
|
+
|
|
32
|
+
constructor(
|
|
33
|
+
private deliverCb: DeliveryCallback,
|
|
34
|
+
private groupTimeout = DEFAULT_TIMEOUT,
|
|
35
|
+
) {}
|
|
36
|
+
|
|
37
|
+
/** Register a group of agent IDs that should be joined. */
|
|
38
|
+
registerGroup(groupId: string, agentIds: string[]): void {
|
|
39
|
+
const group: AgentGroup = {
|
|
40
|
+
groupId,
|
|
41
|
+
agentIds: new Set(agentIds),
|
|
42
|
+
completedRecords: new Map(),
|
|
43
|
+
delivered: false,
|
|
44
|
+
isStraggler: false,
|
|
45
|
+
};
|
|
46
|
+
this.groups.set(groupId, group);
|
|
47
|
+
for (const id of agentIds) {
|
|
48
|
+
this.agentToGroup.set(id, groupId);
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Called when an agent completes.
|
|
54
|
+
* Returns:
|
|
55
|
+
* - 'pass' — agent is not grouped, caller should send individual nudge
|
|
56
|
+
* - 'held' — result held, waiting for group completion
|
|
57
|
+
* - 'delivered' — this completion triggered the group notification
|
|
58
|
+
*/
|
|
59
|
+
onAgentComplete(record: AgentRecord): 'delivered' | 'held' | 'pass' {
|
|
60
|
+
const groupId = this.agentToGroup.get(record.id);
|
|
61
|
+
if (!groupId) return 'pass';
|
|
62
|
+
|
|
63
|
+
const group = this.groups.get(groupId);
|
|
64
|
+
if (!group || group.delivered) return 'pass';
|
|
65
|
+
|
|
66
|
+
group.completedRecords.set(record.id, record);
|
|
67
|
+
|
|
68
|
+
// All done — deliver immediately
|
|
69
|
+
if (group.completedRecords.size >= group.agentIds.size) {
|
|
70
|
+
this.deliver(group, false);
|
|
71
|
+
return 'delivered';
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// First completion in this batch — start timeout
|
|
75
|
+
if (!group.timeoutHandle) {
|
|
76
|
+
const timeout = group.isStraggler ? STRAGGLER_TIMEOUT : this.groupTimeout;
|
|
77
|
+
group.timeoutHandle = setTimeout(() => {
|
|
78
|
+
this.onTimeout(group);
|
|
79
|
+
}, timeout);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
return 'held';
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
private onTimeout(group: AgentGroup): void {
|
|
86
|
+
if (group.delivered) return;
|
|
87
|
+
group.timeoutHandle = undefined;
|
|
88
|
+
|
|
89
|
+
// Partial delivery — some agents still running
|
|
90
|
+
const remaining = new Set<string>();
|
|
91
|
+
for (const id of group.agentIds) {
|
|
92
|
+
if (!group.completedRecords.has(id)) remaining.add(id);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Clean up agentToGroup for delivered agents (they won't complete again)
|
|
96
|
+
for (const id of group.completedRecords.keys()) {
|
|
97
|
+
this.agentToGroup.delete(id);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// Deliver what we have
|
|
101
|
+
this.deliverCb([...group.completedRecords.values()], true);
|
|
102
|
+
|
|
103
|
+
// Set up straggler group for remaining agents
|
|
104
|
+
group.completedRecords.clear();
|
|
105
|
+
group.agentIds = remaining;
|
|
106
|
+
group.isStraggler = true;
|
|
107
|
+
// Timeout will be started when the next straggler completes
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
private deliver(group: AgentGroup, partial: boolean): void {
|
|
111
|
+
if (group.timeoutHandle) {
|
|
112
|
+
clearTimeout(group.timeoutHandle);
|
|
113
|
+
group.timeoutHandle = undefined;
|
|
114
|
+
}
|
|
115
|
+
group.delivered = true;
|
|
116
|
+
this.deliverCb([...group.completedRecords.values()], partial);
|
|
117
|
+
this.cleanupGroup(group.groupId);
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
private cleanupGroup(groupId: string): void {
|
|
121
|
+
const group = this.groups.get(groupId);
|
|
122
|
+
if (!group) return;
|
|
123
|
+
for (const id of group.agentIds) {
|
|
124
|
+
this.agentToGroup.delete(id);
|
|
125
|
+
}
|
|
126
|
+
this.groups.delete(groupId);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
/** Check if an agent is in a group. */
|
|
130
|
+
isGrouped(agentId: string): boolean {
|
|
131
|
+
return this.agentToGroup.has(agentId);
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
dispose(): void {
|
|
135
|
+
for (const group of this.groups.values()) {
|
|
136
|
+
if (group.timeoutHandle) clearTimeout(group.timeoutHandle);
|
|
137
|
+
}
|
|
138
|
+
this.groups.clear();
|
|
139
|
+
this.agentToGroup.clear();
|
|
140
|
+
}
|
|
141
|
+
}
|