@clanker-code/pi-subagents 0.10.5

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.
Files changed (130) hide show
  1. package/.plans/PLAN-next-changes.md +183 -0
  2. package/.plans/README.md +14 -0
  3. package/AGENTS.md +31 -0
  4. package/CHANGELOG.md +583 -0
  5. package/CLAUDE.md +1 -0
  6. package/LICENSE +21 -0
  7. package/README.md +630 -0
  8. package/RELEASE.md +39 -0
  9. package/dist/abort-resend.d.ts +35 -0
  10. package/dist/abort-resend.js +71 -0
  11. package/dist/agent-details.d.ts +17 -0
  12. package/dist/agent-details.js +22 -0
  13. package/dist/agent-manager.d.ts +132 -0
  14. package/dist/agent-manager.js +493 -0
  15. package/dist/agent-runner.d.ts +165 -0
  16. package/dist/agent-runner.js +732 -0
  17. package/dist/agent-tool-description.d.ts +9 -0
  18. package/dist/agent-tool-description.js +147 -0
  19. package/dist/agent-types.d.ts +60 -0
  20. package/dist/agent-types.js +157 -0
  21. package/dist/context.d.ts +12 -0
  22. package/dist/context.js +56 -0
  23. package/dist/cross-extension-rpc.d.ts +46 -0
  24. package/dist/cross-extension-rpc.js +76 -0
  25. package/dist/custom-agents.d.ts +14 -0
  26. package/dist/custom-agents.js +149 -0
  27. package/dist/default-agents.d.ts +7 -0
  28. package/dist/default-agents.js +119 -0
  29. package/dist/enabled-models.d.ts +49 -0
  30. package/dist/enabled-models.js +145 -0
  31. package/dist/env.d.ts +6 -0
  32. package/dist/env.js +28 -0
  33. package/dist/group-join.d.ts +32 -0
  34. package/dist/group-join.js +116 -0
  35. package/dist/index.d.ts +36 -0
  36. package/dist/index.js +1918 -0
  37. package/dist/invocation-config.d.ts +25 -0
  38. package/dist/invocation-config.js +19 -0
  39. package/dist/memory.d.ts +49 -0
  40. package/dist/memory.js +151 -0
  41. package/dist/model-resolver.d.ts +19 -0
  42. package/dist/model-resolver.js +62 -0
  43. package/dist/notifications.d.ts +6 -0
  44. package/dist/notifications.js +107 -0
  45. package/dist/output-file.d.ts +24 -0
  46. package/dist/output-file.js +86 -0
  47. package/dist/peek.d.ts +37 -0
  48. package/dist/peek.js +121 -0
  49. package/dist/prompts.d.ts +40 -0
  50. package/dist/prompts.js +95 -0
  51. package/dist/schedule-store.d.ts +38 -0
  52. package/dist/schedule-store.js +155 -0
  53. package/dist/schedule.d.ts +109 -0
  54. package/dist/schedule.js +338 -0
  55. package/dist/settings.d.ts +135 -0
  56. package/dist/settings.js +168 -0
  57. package/dist/skill-loader.d.ts +24 -0
  58. package/dist/skill-loader.js +93 -0
  59. package/dist/status-note.d.ts +13 -0
  60. package/dist/status-note.js +24 -0
  61. package/dist/types.d.ts +184 -0
  62. package/dist/types.js +7 -0
  63. package/dist/ui/agent-tool-rendering.d.ts +34 -0
  64. package/dist/ui/agent-tool-rendering.js +154 -0
  65. package/dist/ui/agent-widget-tree.d.ts +33 -0
  66. package/dist/ui/agent-widget-tree.js +130 -0
  67. package/dist/ui/agent-widget.d.ts +156 -0
  68. package/dist/ui/agent-widget.js +408 -0
  69. package/dist/ui/conversation-viewer.d.ts +47 -0
  70. package/dist/ui/conversation-viewer.js +290 -0
  71. package/dist/ui/menu-select.d.ts +20 -0
  72. package/dist/ui/menu-select.js +46 -0
  73. package/dist/ui/schedule-menu.d.ts +16 -0
  74. package/dist/ui/schedule-menu.js +99 -0
  75. package/dist/ui/viewer-keys.d.ts +20 -0
  76. package/dist/ui/viewer-keys.js +17 -0
  77. package/dist/usage.d.ts +50 -0
  78. package/dist/usage.js +49 -0
  79. package/dist/wait.d.ts +10 -0
  80. package/dist/wait.js +37 -0
  81. package/dist/worktree.d.ts +45 -0
  82. package/dist/worktree.js +160 -0
  83. package/docs/design/default-extension-tool-exposure.md +56 -0
  84. package/docs/superpowers/plans/2026-06-19-recursive-subagent-widget.md +600 -0
  85. package/docs/superpowers/specs/2026-06-19-recursive-subagent-widget-design.md +189 -0
  86. package/examples/agent-tool-description.md +45 -0
  87. package/package.json +56 -0
  88. package/reviews/proposal-structured-output-schema.md +135 -0
  89. package/reviews/recursive-subagent-widget-preview-rev2.png +0 -0
  90. package/reviews/recursive-subagent-widget-preview.html +137 -0
  91. package/reviews/recursive-subagent-widget-preview.png +0 -0
  92. package/reviews/subagent-features-comparison.md +350 -0
  93. package/src/abort-resend.ts +75 -0
  94. package/src/agent-details.ts +31 -0
  95. package/src/agent-manager.ts +596 -0
  96. package/src/agent-runner.ts +872 -0
  97. package/src/agent-tool-description.ts +163 -0
  98. package/src/agent-types.ts +189 -0
  99. package/src/context.ts +58 -0
  100. package/src/cross-extension-rpc.ts +122 -0
  101. package/src/custom-agents.ts +160 -0
  102. package/src/default-agents.ts +123 -0
  103. package/src/enabled-models.ts +180 -0
  104. package/src/env.ts +33 -0
  105. package/src/group-join.ts +141 -0
  106. package/src/index.ts +2115 -0
  107. package/src/invocation-config.ts +42 -0
  108. package/src/memory.ts +165 -0
  109. package/src/model-resolver.ts +81 -0
  110. package/src/notifications.ts +120 -0
  111. package/src/output-file.ts +96 -0
  112. package/src/peek.ts +155 -0
  113. package/src/prompts.ts +129 -0
  114. package/src/schedule-store.ts +153 -0
  115. package/src/schedule.ts +365 -0
  116. package/src/settings.ts +289 -0
  117. package/src/skill-loader.ts +102 -0
  118. package/src/status-note.ts +25 -0
  119. package/src/types.ts +195 -0
  120. package/src/ui/agent-tool-rendering.ts +175 -0
  121. package/src/ui/agent-widget-tree.ts +169 -0
  122. package/src/ui/agent-widget.ts +497 -0
  123. package/src/ui/conversation-viewer.ts +297 -0
  124. package/src/ui/menu-select.ts +68 -0
  125. package/src/ui/schedule-menu.ts +105 -0
  126. package/src/ui/viewer-keys.ts +39 -0
  127. package/src/usage.ts +60 -0
  128. package/src/wait.ts +44 -0
  129. package/src/worktree.ts +191 -0
  130. package/vitest.config.ts +25 -0
@@ -0,0 +1,149 @@
1
+ /**
2
+ * custom-agents.ts — Load user-defined agents from project (.pi/agents/) and global ($PI_CODING_AGENT_DIR/agents/, default ~/.pi/agent/agents/) locations.
3
+ */
4
+ import { existsSync, readdirSync, readFileSync } from "node:fs";
5
+ import { basename, join } from "node:path";
6
+ import { getAgentDir, parseFrontmatter } from "@earendil-works/pi-coding-agent";
7
+ import { BUILTIN_TOOL_NAMES } from "./agent-types.js";
8
+ /**
9
+ * Scan for custom agent .md files from multiple locations.
10
+ * Discovery hierarchy (higher priority wins):
11
+ * 1. Project: <cwd>/.pi/agents/*.md
12
+ * 2. Global: $PI_CODING_AGENT_DIR/agents/*.md (default: ~/.pi/agent/agents/*.md)
13
+ *
14
+ * Project-level agents override global ones with the same name.
15
+ * Any name is allowed — names matching defaults (e.g. "Explore") override them.
16
+ */
17
+ export function loadCustomAgents(cwd) {
18
+ const globalDir = join(getAgentDir(), "agents");
19
+ const projectDir = join(cwd, ".pi", "agents");
20
+ const agents = new Map();
21
+ loadFromDir(globalDir, agents, "global"); // lower priority
22
+ loadFromDir(projectDir, agents, "project"); // higher priority (overwrites)
23
+ return agents;
24
+ }
25
+ /** Load agent configs from a directory into the map. */
26
+ function loadFromDir(dir, agents, source) {
27
+ if (!existsSync(dir))
28
+ return;
29
+ let files;
30
+ try {
31
+ files = readdirSync(dir).filter(f => f.endsWith(".md"));
32
+ }
33
+ catch {
34
+ return;
35
+ }
36
+ for (const file of files) {
37
+ const name = basename(file, ".md");
38
+ let content;
39
+ try {
40
+ content = readFileSync(join(dir, file), "utf-8");
41
+ }
42
+ catch {
43
+ continue;
44
+ }
45
+ const { frontmatter: fm, body } = parseFrontmatter(content);
46
+ const { builtinToolNames, extSelectors } = parseToolsField(fm.tools);
47
+ agents.set(name, {
48
+ name,
49
+ displayName: str(fm.display_name),
50
+ description: str(fm.description) ?? name,
51
+ builtinToolNames,
52
+ extSelectors,
53
+ disallowedTools: csvListOptional(fm.disallowed_tools),
54
+ extensions: inheritField(fm.extensions ?? fm.inherit_extensions),
55
+ excludeExtensions: csvListOptional(fm.exclude_extensions),
56
+ skills: inheritField(fm.skills ?? fm.inherit_skills),
57
+ model: str(fm.model),
58
+ thinking: str(fm.thinking),
59
+ maxTurns: nonNegativeInt(fm.max_turns),
60
+ systemPrompt: body.trim(),
61
+ promptMode: fm.prompt_mode === "append" ? "append" : "replace",
62
+ inheritContext: fm.inherit_context != null ? fm.inherit_context === true : undefined,
63
+ // Parsed for compatibility, but ignored at spawn time — this fork always runs in the background.
64
+ runInBackground: fm.run_in_background != null ? fm.run_in_background === true : undefined,
65
+ isolated: fm.isolated != null ? fm.isolated === true : undefined,
66
+ memory: parseMemory(fm.memory),
67
+ isolation: fm.isolation === "worktree" ? "worktree" : undefined,
68
+ enabled: fm.enabled !== false, // default true; explicitly false disables
69
+ source,
70
+ });
71
+ }
72
+ }
73
+ // ---- Field parsers ----
74
+ // All follow the same convention: omitted → default, "none"/empty → nothing, value → exact.
75
+ /** Extract a string or undefined. */
76
+ function str(val) {
77
+ return typeof val === "string" ? val : undefined;
78
+ }
79
+ /** Extract a non-negative integer or undefined. 0 means unlimited for max_turns. */
80
+ function nonNegativeInt(val) {
81
+ return typeof val === "number" && val >= 0 ? val : undefined;
82
+ }
83
+ /**
84
+ * Parse a raw CSV field value into items, or undefined if absent/empty/"none".
85
+ */
86
+ function parseCsvField(val) {
87
+ if (val === undefined || val === null)
88
+ return undefined;
89
+ const s = String(val).trim();
90
+ if (!s || s === "none")
91
+ return undefined;
92
+ const items = s.split(",").map(t => t.trim()).filter(Boolean);
93
+ return items.length > 0 ? items : undefined;
94
+ }
95
+ /**
96
+ * Parse a comma-separated list field with defaults.
97
+ * omitted → defaults; "none"/empty → []; csv → listed items.
98
+ */
99
+ function csvList(val, defaults) {
100
+ if (val === undefined || val === null)
101
+ return defaults;
102
+ return parseCsvField(val) ?? [];
103
+ }
104
+ /**
105
+ * Partition the `tools:` CSV into the built-in tool allowlist and raw `ext:` selectors.
106
+ * `*` (and the case-insensitive alias `all`, for `tools: all`) expands to all
107
+ * built-ins; plain entries are built-in names; `ext:` entries are extension-tool
108
+ * selectors parsed later by the runner. omitted → all built-ins, no selectors.
109
+ * `tools:` present with only `ext:` entries → zero built-ins (use `*`).
110
+ */
111
+ function parseToolsField(val) {
112
+ const entries = csvList(val, BUILTIN_TOOL_NAMES);
113
+ const isWildcard = (e) => e === "*" || e.toLowerCase() === "all";
114
+ const hasWildcard = entries.some(isWildcard);
115
+ const plain = entries.filter(e => !isWildcard(e) && !e.startsWith("ext:"));
116
+ const extEntries = entries.filter(e => e.startsWith("ext:"));
117
+ return {
118
+ builtinToolNames: hasWildcard ? [...new Set([...BUILTIN_TOOL_NAMES, ...plain])] : plain,
119
+ extSelectors: extEntries.length > 0 ? extEntries : undefined,
120
+ };
121
+ }
122
+ /**
123
+ * Parse an optional comma-separated list field.
124
+ * omitted → undefined; "none"/empty → undefined; csv → listed items.
125
+ */
126
+ function csvListOptional(val) {
127
+ return parseCsvField(val);
128
+ }
129
+ /**
130
+ * Parse a memory scope field.
131
+ * omitted → undefined; "user"/"project"/"local" → MemoryScope.
132
+ */
133
+ function parseMemory(val) {
134
+ if (val === "user" || val === "project" || val === "local")
135
+ return val;
136
+ return undefined;
137
+ }
138
+ /**
139
+ * Parse an inherit field (extensions, skills).
140
+ * omitted/true → true (inherit all); false/"none"/empty → false; csv → listed names.
141
+ */
142
+ function inheritField(val) {
143
+ if (val === undefined || val === null || val === true)
144
+ return true;
145
+ if (val === false || val === "none")
146
+ return false;
147
+ const items = csvList(val, []);
148
+ return items.length > 0 ? items : false;
149
+ }
@@ -0,0 +1,7 @@
1
+ /**
2
+ * default-agents.ts — Embedded default agent configurations.
3
+ *
4
+ * These are always available but can be overridden by user .md files with the same name.
5
+ */
6
+ import type { AgentConfig } from "./types.js";
7
+ export declare const DEFAULT_AGENTS: Map<string, AgentConfig>;
@@ -0,0 +1,119 @@
1
+ /**
2
+ * default-agents.ts — Embedded default agent configurations.
3
+ *
4
+ * These are always available but can be overridden by user .md files with the same name.
5
+ */
6
+ const READ_ONLY_TOOLS = ["read", "bash", "grep", "find", "ls"];
7
+ export const DEFAULT_AGENTS = new Map([
8
+ [
9
+ "general-purpose",
10
+ {
11
+ name: "general-purpose",
12
+ displayName: "Agent",
13
+ description: "General-purpose agent for researching complex questions, searching for code, and executing multi-step tasks. When you are searching for a keyword or file and are not confident that you will find the right match in the first few tries use this agent to perform the search for you.",
14
+ // builtinToolNames omitted — means "all available tools" (resolved at lookup time)
15
+ // inheritContext / runInBackground / isolated omitted — strategy fields, callers decide per-call.
16
+ // Setting them to false would lock callsite intent (see resolveAgentInvocationConfig in invocation-config.ts).
17
+ extensions: true,
18
+ skills: true,
19
+ systemPrompt: "",
20
+ promptMode: "append",
21
+ isDefault: true,
22
+ },
23
+ ],
24
+ [
25
+ "Explore",
26
+ {
27
+ name: "Explore",
28
+ displayName: "Explore",
29
+ description: "Fast read-only search agent for locating code. Use it to find files by pattern (eg. \"src/components/**/*.tsx\"), grep for symbols or keywords (eg. \"API endpoints\"), or answer \"where is X defined / which files reference Y.\" Do NOT use it for code review, design-doc auditing, cross-file consistency checks, or open-ended analysis — it reads excerpts rather than whole files and will miss content past its read window. When calling, specify search breadth: \"quick\" for a single targeted lookup, \"medium\" for moderate exploration, or \"very thorough\" to search across multiple locations and naming conventions.",
30
+ builtinToolNames: READ_ONLY_TOOLS,
31
+ extensions: true,
32
+ skills: true,
33
+ model: "anthropic/claude-haiku-4-5-20251001",
34
+ systemPrompt: `# CRITICAL: READ-ONLY MODE - NO FILE MODIFICATIONS
35
+ You are a file search specialist. You excel at thoroughly navigating and exploring codebases.
36
+ Your role is EXCLUSIVELY to search and analyze existing code. You do NOT have access to file editing tools.
37
+
38
+ You are STRICTLY PROHIBITED from:
39
+ - Creating new files
40
+ - Modifying existing files
41
+ - Deleting files
42
+ - Moving or copying files
43
+ - Creating temporary files anywhere, including /tmp
44
+ - Using redirect operators (>, >>, |) or heredocs to write to files
45
+ - Running ANY commands that change system state
46
+
47
+ Use Bash ONLY for read-only operations: ls, git status, git log, git diff, find, cat, head, tail.
48
+
49
+ # Tool Usage
50
+ - Use the find tool for file pattern matching (NOT the bash find command)
51
+ - Use the grep tool for content search (NOT bash grep/rg command)
52
+ - Use the read tool for reading files (NOT bash cat/head/tail)
53
+ - Use Bash ONLY for read-only operations
54
+ - Make independent tool calls in parallel for efficiency
55
+ - Adapt search approach based on thoroughness level specified
56
+
57
+ # Output
58
+ - Use absolute file paths in all references
59
+ - Report findings as regular messages
60
+ - Do not use emojis
61
+ - Be thorough and precise`,
62
+ promptMode: "replace",
63
+ isDefault: true,
64
+ },
65
+ ],
66
+ [
67
+ "Plan",
68
+ {
69
+ name: "Plan",
70
+ displayName: "Plan",
71
+ description: "Software architect agent for designing implementation plans. Use this when you need to plan the implementation strategy for a task. Returns step-by-step plans, identifies critical files, and considers architectural trade-offs.",
72
+ builtinToolNames: READ_ONLY_TOOLS,
73
+ extensions: true,
74
+ skills: true,
75
+ systemPrompt: `# CRITICAL: READ-ONLY MODE - NO FILE MODIFICATIONS
76
+ You are a software architect and planning specialist.
77
+ Your role is EXCLUSIVELY to explore the codebase and design implementation plans.
78
+ You do NOT have access to file editing tools — attempting to edit files will fail.
79
+
80
+ You are STRICTLY PROHIBITED from:
81
+ - Creating new files
82
+ - Modifying existing files
83
+ - Deleting files
84
+ - Moving or copying files
85
+ - Creating temporary files anywhere, including /tmp
86
+ - Using redirect operators (>, >>, |) or heredocs to write to files
87
+ - Running ANY commands that change system state
88
+
89
+ # Planning Process
90
+ 1. Understand requirements
91
+ 2. Explore thoroughly (read files, find patterns, understand architecture)
92
+ 3. Design solution based on your assigned perspective
93
+ 4. Detail the plan with step-by-step implementation strategy
94
+
95
+ # Requirements
96
+ - Consider trade-offs and architectural decisions
97
+ - Identify dependencies and sequencing
98
+ - Anticipate potential challenges
99
+ - Follow existing patterns where appropriate
100
+
101
+ # Tool Usage
102
+ - Use the find tool for file pattern matching (NOT the bash find command)
103
+ - Use the grep tool for content search (NOT bash grep/rg command)
104
+ - Use the read tool for reading files (NOT bash cat/head/tail)
105
+ - Use Bash ONLY for read-only operations
106
+
107
+ # Output Format
108
+ - Use absolute file paths
109
+ - Do not use emojis
110
+ - End your response with:
111
+
112
+ ### Critical Files for Implementation
113
+ List 3-5 files most critical for implementing this plan:
114
+ - /absolute/path/to/file.ts - [Brief reason]`,
115
+ promptMode: "replace",
116
+ isDefault: true,
117
+ },
118
+ ],
119
+ ]);
@@ -0,0 +1,49 @@
1
+ /**
2
+ * Reads `enabledModels` from pi's settings (global `<agentDir>/settings.json`
3
+ * + project-local `<cwd>/.pi/settings.json`, project wins) and resolves
4
+ * entries to concrete `provider/modelId` keys for scope validation.
5
+ *
6
+ * **Project overrides global**, mirroring pi's own `SettingsManager`
7
+ * deep-merge behavior and matching the precedence we use for our own
8
+ * `subagents.json` settings (see `src/settings.ts:loadSettings`). If
9
+ * project file has `enabledModels` set, it wholly replaces global's
10
+ * (array fields are replaced, not concatenated).
11
+ *
12
+ * **Limited subset of upstream's resolveModelScope.** We support exact
13
+ * `provider/modelId` matching only. Upstream (pi-coding-agent's
14
+ * `core/model-resolver.ts`) additionally supports glob patterns
15
+ * (`*sonnet*`, `anthropic/*`), bare model IDs without provider, and
16
+ * thinking-level suffixes (`provider/*:high`). Those forms are silently
17
+ * ignored here.
18
+ *
19
+ * In practice, pi's `/scoped-models` picker writes exact `provider/modelId`
20
+ * entries, so the limitation is invisible for users who configure scope
21
+ * through pi's UI. Hand-edited settings using globs or bare IDs will
22
+ * produce an empty allowed set (scope check becomes a no-op).
23
+ *
24
+ * Example:
25
+ * enabledModels = ["anthropic/claude-sonnet-4-6", "anthropic/claude-opus-4-6"]
26
+ * → resolves to { "anthropic/claude-sonnet-4-6", "anthropic/claude-opus-4-6" }
27
+ */
28
+ /** Minimal registry shape — only the methods resolveEnabledModels actually calls. */
29
+ export interface ModelRegistryRef {
30
+ getAll(): unknown[];
31
+ getAvailable?(): unknown[];
32
+ }
33
+ /**
34
+ * Read enabledModels from pi's settings — project-local overrides global.
35
+ * Mirrors pi's SettingsManager deep-merge for the `enabledModels` field
36
+ * (and matches our own loadSettings precedence in src/settings.ts).
37
+ * Returns undefined when neither file has the field.
38
+ */
39
+ export declare function readEnabledModels(cwd: string): string[] | undefined;
40
+ export declare function resolveEnabledModels(patterns: string[] | undefined, registry: ModelRegistryRef, cwd?: string): Set<string> | undefined;
41
+ /**
42
+ * True when `model` is in the allowed set. Centralizes the key format
43
+ * (`provider/id` lowercase) so callers don't have to reproduce it —
44
+ * both set-building (resolveExact) and lookup go through `modelKey`.
45
+ */
46
+ export declare function isModelInScope(model: {
47
+ provider: string;
48
+ id: string;
49
+ }, allowed: Set<string>): boolean;
@@ -0,0 +1,145 @@
1
+ /**
2
+ * Reads `enabledModels` from pi's settings (global `<agentDir>/settings.json`
3
+ * + project-local `<cwd>/.pi/settings.json`, project wins) and resolves
4
+ * entries to concrete `provider/modelId` keys for scope validation.
5
+ *
6
+ * **Project overrides global**, mirroring pi's own `SettingsManager`
7
+ * deep-merge behavior and matching the precedence we use for our own
8
+ * `subagents.json` settings (see `src/settings.ts:loadSettings`). If
9
+ * project file has `enabledModels` set, it wholly replaces global's
10
+ * (array fields are replaced, not concatenated).
11
+ *
12
+ * **Limited subset of upstream's resolveModelScope.** We support exact
13
+ * `provider/modelId` matching only. Upstream (pi-coding-agent's
14
+ * `core/model-resolver.ts`) additionally supports glob patterns
15
+ * (`*sonnet*`, `anthropic/*`), bare model IDs without provider, and
16
+ * thinking-level suffixes (`provider/*:high`). Those forms are silently
17
+ * ignored here.
18
+ *
19
+ * In practice, pi's `/scoped-models` picker writes exact `provider/modelId`
20
+ * entries, so the limitation is invisible for users who configure scope
21
+ * through pi's UI. Hand-edited settings using globs or bare IDs will
22
+ * produce an empty allowed set (scope check becomes a no-op).
23
+ *
24
+ * Example:
25
+ * enabledModels = ["anthropic/claude-sonnet-4-6", "anthropic/claude-opus-4-6"]
26
+ * → resolves to { "anthropic/claude-sonnet-4-6", "anthropic/claude-opus-4-6" }
27
+ */
28
+ import { existsSync, readFileSync, statSync } from "node:fs";
29
+ import { join } from "node:path";
30
+ import { getAgentDir } from "@earendil-works/pi-coding-agent";
31
+ /** Paths to pi's settings.json files: [project, global] (project takes precedence). */
32
+ function settingsPaths(cwd) {
33
+ return [
34
+ join(cwd, ".pi", "settings.json"),
35
+ join(getAgentDir(), "settings.json"),
36
+ ];
37
+ }
38
+ /** Read `enabledModels` from a single settings.json file. Undefined when missing or absent. */
39
+ function readField(path) {
40
+ if (!existsSync(path))
41
+ return undefined;
42
+ try {
43
+ const raw = JSON.parse(readFileSync(path, "utf-8"));
44
+ if (Array.isArray(raw?.enabledModels))
45
+ return raw.enabledModels;
46
+ }
47
+ catch {
48
+ /* corrupt file — silent */
49
+ }
50
+ return undefined;
51
+ }
52
+ /**
53
+ * Read enabledModels from pi's settings — project-local overrides global.
54
+ * Mirrors pi's SettingsManager deep-merge for the `enabledModels` field
55
+ * (and matches our own loadSettings precedence in src/settings.ts).
56
+ * Returns undefined when neither file has the field.
57
+ */
58
+ export function readEnabledModels(cwd) {
59
+ const [project, global] = settingsPaths(cwd);
60
+ return readField(project) ?? readField(global);
61
+ }
62
+ /**
63
+ * Resolve enabledModels patterns → Set<"provider/modelId"> (lowercase keys).
64
+ *
65
+ * Only exact `provider/modelId` patterns are matched (case-insensitive).
66
+ * Patterns without a slash, with glob characters, or with a `:thinking`
67
+ * suffix are silently dropped. See module-level docstring for rationale.
68
+ *
69
+ * Cache: keyed on JSON.stringify(patterns) + mtime/size of *both*
70
+ * project and global settings.json files. Re-resolves when either file
71
+ * changes or the patterns argument differs.
72
+ *
73
+ * Returns undefined when no patterns are provided or no patterns match
74
+ * (scope check becomes a no-op at the call site).
75
+ */
76
+ // Module-level cache — invalidated when either settings.json changes or patterns differ.
77
+ let cachedAllowed;
78
+ let cachedHash = "";
79
+ let cachedPatternsKey = "";
80
+ /** mtime+size hash of one file, or "missing" if absent. */
81
+ function hashOf(path) {
82
+ try {
83
+ const s = statSync(path);
84
+ return `${s.mtimeMs}-${s.size}`;
85
+ }
86
+ catch {
87
+ return "missing";
88
+ }
89
+ }
90
+ export function resolveEnabledModels(patterns, registry, cwd = process.cwd()) {
91
+ // Fast path: check cache (stat both project and global settings.json files)
92
+ const patternsKey = JSON.stringify(patterns);
93
+ const [project, global] = settingsPaths(cwd);
94
+ const fileHash = `${hashOf(project)};${hashOf(global)}`;
95
+ if (fileHash === cachedHash && patternsKey === cachedPatternsKey) {
96
+ return cachedAllowed;
97
+ }
98
+ // Cache miss — resolve
99
+ if (!patterns || patterns.length === 0) {
100
+ cachedHash = fileHash;
101
+ cachedPatternsKey = patternsKey;
102
+ cachedAllowed = undefined;
103
+ return undefined;
104
+ }
105
+ const available = (registry.getAvailable?.() ?? registry.getAll());
106
+ const allowed = new Set();
107
+ for (const pattern of patterns) {
108
+ const trimmed = pattern.trim();
109
+ if (!trimmed)
110
+ continue; // skip empty/whitespace
111
+ resolveExact(trimmed, available, allowed);
112
+ }
113
+ const result = allowed.size > 0 ? allowed : undefined;
114
+ cachedHash = fileHash;
115
+ cachedPatternsKey = patternsKey;
116
+ cachedAllowed = result;
117
+ return result;
118
+ }
119
+ /**
120
+ * True when `model` is in the allowed set. Centralizes the key format
121
+ * (`provider/id` lowercase) so callers don't have to reproduce it —
122
+ * both set-building (resolveExact) and lookup go through `modelKey`.
123
+ */
124
+ export function isModelInScope(model, allowed) {
125
+ return allowed.has(modelKey(model));
126
+ }
127
+ /** Canonical lowercase `provider/id` key for the allowed set. */
128
+ function modelKey(model) {
129
+ return `${model.provider}/${model.id}`.toLowerCase();
130
+ }
131
+ /**
132
+ * Resolve exact model pattern. Example: "google/gemma-4-31b-it".
133
+ */
134
+ function resolveExact(pattern, available, allowed) {
135
+ // "provider/modelId" — exact (colon is part of id, not split)
136
+ const slashIdx = pattern.indexOf("/");
137
+ if (slashIdx === -1)
138
+ return; // bare modelId not supported
139
+ const provider = pattern.slice(0, slashIdx).toLowerCase();
140
+ const modelId = pattern.slice(slashIdx + 1).toLowerCase();
141
+ const exact = available.find(m => m.provider.toLowerCase() === provider && m.id.toLowerCase() === modelId);
142
+ if (exact) {
143
+ allowed.add(modelKey(exact));
144
+ }
145
+ }
package/dist/env.d.ts ADDED
@@ -0,0 +1,6 @@
1
+ /**
2
+ * env.ts — Detect environment info (git, platform) for subagent system prompts.
3
+ */
4
+ import type { ExtensionAPI } from "@earendil-works/pi-coding-agent";
5
+ import type { EnvInfo } from "./types.js";
6
+ export declare function detectEnv(pi: ExtensionAPI, cwd: string): Promise<EnvInfo>;
package/dist/env.js ADDED
@@ -0,0 +1,28 @@
1
+ /**
2
+ * env.ts — Detect environment info (git, platform) for subagent system prompts.
3
+ */
4
+ export async function detectEnv(pi, cwd) {
5
+ let isGitRepo = false;
6
+ let branch = "";
7
+ try {
8
+ const result = await pi.exec("git", ["rev-parse", "--is-inside-work-tree"], { cwd, timeout: 5000 });
9
+ isGitRepo = result.code === 0 && result.stdout.trim() === "true";
10
+ }
11
+ catch {
12
+ // Not a git repo or git not installed
13
+ }
14
+ if (isGitRepo) {
15
+ try {
16
+ const result = await pi.exec("git", ["branch", "--show-current"], { cwd, timeout: 5000 });
17
+ branch = result.code === 0 ? result.stdout.trim() : "unknown";
18
+ }
19
+ catch {
20
+ branch = "unknown";
21
+ }
22
+ }
23
+ return {
24
+ isGitRepo,
25
+ branch,
26
+ platform: process.platform,
27
+ };
28
+ }
@@ -0,0 +1,32 @@
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
+ import type { AgentRecord } from "./types.js";
9
+ export type DeliveryCallback = (records: AgentRecord[], partial: boolean) => void;
10
+ export declare class GroupJoinManager {
11
+ private deliverCb;
12
+ private groupTimeout;
13
+ private groups;
14
+ private agentToGroup;
15
+ constructor(deliverCb: DeliveryCallback, groupTimeout?: number);
16
+ /** Register a group of agent IDs that should be joined. */
17
+ registerGroup(groupId: string, agentIds: string[]): void;
18
+ /**
19
+ * Called when an agent completes.
20
+ * Returns:
21
+ * - 'pass' — agent is not grouped, caller should send individual nudge
22
+ * - 'held' — result held, waiting for group completion
23
+ * - 'delivered' — this completion triggered the group notification
24
+ */
25
+ onAgentComplete(record: AgentRecord): 'delivered' | 'held' | 'pass';
26
+ private onTimeout;
27
+ private deliver;
28
+ private cleanupGroup;
29
+ /** Check if an agent is in a group. */
30
+ isGrouped(agentId: string): boolean;
31
+ dispose(): void;
32
+ }
@@ -0,0 +1,116 @@
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
+ /** Default timeout: 30s after first completion in a group. */
9
+ const DEFAULT_TIMEOUT = 30_000;
10
+ /** Straggler re-batch timeout: 15s. */
11
+ const STRAGGLER_TIMEOUT = 15_000;
12
+ export class GroupJoinManager {
13
+ deliverCb;
14
+ groupTimeout;
15
+ groups = new Map();
16
+ agentToGroup = new Map();
17
+ constructor(deliverCb, groupTimeout = DEFAULT_TIMEOUT) {
18
+ this.deliverCb = deliverCb;
19
+ this.groupTimeout = groupTimeout;
20
+ }
21
+ /** Register a group of agent IDs that should be joined. */
22
+ registerGroup(groupId, agentIds) {
23
+ const group = {
24
+ groupId,
25
+ agentIds: new Set(agentIds),
26
+ completedRecords: new Map(),
27
+ delivered: false,
28
+ isStraggler: false,
29
+ };
30
+ this.groups.set(groupId, group);
31
+ for (const id of agentIds) {
32
+ this.agentToGroup.set(id, groupId);
33
+ }
34
+ }
35
+ /**
36
+ * Called when an agent completes.
37
+ * Returns:
38
+ * - 'pass' — agent is not grouped, caller should send individual nudge
39
+ * - 'held' — result held, waiting for group completion
40
+ * - 'delivered' — this completion triggered the group notification
41
+ */
42
+ onAgentComplete(record) {
43
+ const groupId = this.agentToGroup.get(record.id);
44
+ if (!groupId)
45
+ return 'pass';
46
+ const group = this.groups.get(groupId);
47
+ if (!group || group.delivered)
48
+ return 'pass';
49
+ group.completedRecords.set(record.id, record);
50
+ // All done — deliver immediately
51
+ if (group.completedRecords.size >= group.agentIds.size) {
52
+ this.deliver(group, false);
53
+ return 'delivered';
54
+ }
55
+ // First completion in this batch — start timeout
56
+ if (!group.timeoutHandle) {
57
+ const timeout = group.isStraggler ? STRAGGLER_TIMEOUT : this.groupTimeout;
58
+ group.timeoutHandle = setTimeout(() => {
59
+ this.onTimeout(group);
60
+ }, timeout);
61
+ }
62
+ return 'held';
63
+ }
64
+ onTimeout(group) {
65
+ if (group.delivered)
66
+ return;
67
+ group.timeoutHandle = undefined;
68
+ // Partial delivery — some agents still running
69
+ const remaining = new Set();
70
+ for (const id of group.agentIds) {
71
+ if (!group.completedRecords.has(id))
72
+ remaining.add(id);
73
+ }
74
+ // Clean up agentToGroup for delivered agents (they won't complete again)
75
+ for (const id of group.completedRecords.keys()) {
76
+ this.agentToGroup.delete(id);
77
+ }
78
+ // Deliver what we have
79
+ this.deliverCb([...group.completedRecords.values()], true);
80
+ // Set up straggler group for remaining agents
81
+ group.completedRecords.clear();
82
+ group.agentIds = remaining;
83
+ group.isStraggler = true;
84
+ // Timeout will be started when the next straggler completes
85
+ }
86
+ deliver(group, partial) {
87
+ if (group.timeoutHandle) {
88
+ clearTimeout(group.timeoutHandle);
89
+ group.timeoutHandle = undefined;
90
+ }
91
+ group.delivered = true;
92
+ this.deliverCb([...group.completedRecords.values()], partial);
93
+ this.cleanupGroup(group.groupId);
94
+ }
95
+ cleanupGroup(groupId) {
96
+ const group = this.groups.get(groupId);
97
+ if (!group)
98
+ return;
99
+ for (const id of group.agentIds) {
100
+ this.agentToGroup.delete(id);
101
+ }
102
+ this.groups.delete(groupId);
103
+ }
104
+ /** Check if an agent is in a group. */
105
+ isGrouped(agentId) {
106
+ return this.agentToGroup.has(agentId);
107
+ }
108
+ dispose() {
109
+ for (const group of this.groups.values()) {
110
+ if (group.timeoutHandle)
111
+ clearTimeout(group.timeoutHandle);
112
+ }
113
+ this.groups.clear();
114
+ this.agentToGroup.clear();
115
+ }
116
+ }