@mariozechner/pi-coding-agent 0.49.2 → 0.50.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.
Files changed (237) hide show
  1. package/CHANGELOG.md +126 -1
  2. package/README.md +310 -1229
  3. package/dist/cli/args.d.ts +5 -0
  4. package/dist/cli/args.d.ts.map +1 -1
  5. package/dist/cli/args.js +57 -22
  6. package/dist/cli/args.js.map +1 -1
  7. package/dist/cli/config-selector.d.ts +14 -0
  8. package/dist/cli/config-selector.d.ts.map +1 -0
  9. package/dist/cli/config-selector.js +31 -0
  10. package/dist/cli/config-selector.js.map +1 -0
  11. package/dist/cli/session-picker.d.ts.map +1 -1
  12. package/dist/cli/session-picker.js +1 -1
  13. package/dist/cli/session-picker.js.map +1 -1
  14. package/dist/config.d.ts +2 -0
  15. package/dist/config.d.ts.map +1 -1
  16. package/dist/config.js +6 -0
  17. package/dist/config.js.map +1 -1
  18. package/dist/core/agent-session.d.ts +53 -34
  19. package/dist/core/agent-session.d.ts.map +1 -1
  20. package/dist/core/agent-session.js +262 -67
  21. package/dist/core/agent-session.js.map +1 -1
  22. package/dist/core/auth-storage.d.ts +8 -18
  23. package/dist/core/auth-storage.d.ts.map +1 -1
  24. package/dist/core/auth-storage.js +39 -55
  25. package/dist/core/auth-storage.js.map +1 -1
  26. package/dist/core/bash-executor.d.ts.map +1 -1
  27. package/dist/core/bash-executor.js +2 -1
  28. package/dist/core/bash-executor.js.map +1 -1
  29. package/dist/core/diagnostics.d.ts +15 -0
  30. package/dist/core/diagnostics.d.ts.map +1 -0
  31. package/dist/core/diagnostics.js +2 -0
  32. package/dist/core/diagnostics.js.map +1 -0
  33. package/dist/core/export-html/template.css +9 -0
  34. package/dist/core/export-html/template.js +6 -4
  35. package/dist/core/extensions/index.d.ts +1 -1
  36. package/dist/core/extensions/index.d.ts.map +1 -1
  37. package/dist/core/extensions/index.js.map +1 -1
  38. package/dist/core/extensions/loader.d.ts +1 -1
  39. package/dist/core/extensions/loader.d.ts.map +1 -1
  40. package/dist/core/extensions/loader.js +10 -1
  41. package/dist/core/extensions/loader.js.map +1 -1
  42. package/dist/core/extensions/runner.d.ts +9 -3
  43. package/dist/core/extensions/runner.d.ts.map +1 -1
  44. package/dist/core/extensions/runner.js +39 -12
  45. package/dist/core/extensions/runner.js.map +1 -1
  46. package/dist/core/extensions/types.d.ts +112 -1
  47. package/dist/core/extensions/types.d.ts.map +1 -1
  48. package/dist/core/extensions/types.js.map +1 -1
  49. package/dist/core/footer-data-provider.d.ts +9 -2
  50. package/dist/core/footer-data-provider.d.ts.map +1 -1
  51. package/dist/core/footer-data-provider.js +13 -0
  52. package/dist/core/footer-data-provider.js.map +1 -1
  53. package/dist/core/model-registry.d.ts +42 -2
  54. package/dist/core/model-registry.d.ts.map +1 -1
  55. package/dist/core/model-registry.js +154 -44
  56. package/dist/core/model-registry.js.map +1 -1
  57. package/dist/core/model-resolver.d.ts.map +1 -1
  58. package/dist/core/model-resolver.js +3 -2
  59. package/dist/core/model-resolver.js.map +1 -1
  60. package/dist/core/package-manager.d.ts +129 -0
  61. package/dist/core/package-manager.d.ts.map +1 -0
  62. package/dist/core/package-manager.js +1148 -0
  63. package/dist/core/package-manager.js.map +1 -0
  64. package/dist/core/prompt-templates.d.ts +6 -0
  65. package/dist/core/prompt-templates.d.ts.map +1 -1
  66. package/dist/core/prompt-templates.js +114 -54
  67. package/dist/core/prompt-templates.js.map +1 -1
  68. package/dist/core/resource-loader.d.ts +160 -0
  69. package/dist/core/resource-loader.d.ts.map +1 -0
  70. package/dist/core/resource-loader.js +604 -0
  71. package/dist/core/resource-loader.js.map +1 -0
  72. package/dist/core/sdk.d.ts +14 -105
  73. package/dist/core/sdk.d.ts.map +1 -1
  74. package/dist/core/sdk.js +52 -304
  75. package/dist/core/sdk.js.map +1 -1
  76. package/dist/core/session-manager.d.ts.map +1 -1
  77. package/dist/core/session-manager.js +45 -1
  78. package/dist/core/session-manager.js.map +1 -1
  79. package/dist/core/settings-manager.d.ts +39 -16
  80. package/dist/core/settings-manager.d.ts.map +1 -1
  81. package/dist/core/settings-manager.js +107 -25
  82. package/dist/core/settings-manager.js.map +1 -1
  83. package/dist/core/skills.d.ts +18 -10
  84. package/dist/core/skills.d.ts.map +1 -1
  85. package/dist/core/skills.js +126 -93
  86. package/dist/core/skills.js.map +1 -1
  87. package/dist/core/system-prompt.d.ts +3 -27
  88. package/dist/core/system-prompt.d.ts.map +1 -1
  89. package/dist/core/system-prompt.js +16 -103
  90. package/dist/core/system-prompt.js.map +1 -1
  91. package/dist/core/tools/bash.d.ts.map +1 -1
  92. package/dist/core/tools/bash.js +2 -1
  93. package/dist/core/tools/bash.js.map +1 -1
  94. package/dist/core/tools/read.d.ts.map +1 -1
  95. package/dist/core/tools/read.js +4 -4
  96. package/dist/core/tools/read.js.map +1 -1
  97. package/dist/index.d.ts +12 -7
  98. package/dist/index.d.ts.map +1 -1
  99. package/dist/index.js +8 -6
  100. package/dist/index.js.map +1 -1
  101. package/dist/main.d.ts.map +1 -1
  102. package/dist/main.js +209 -97
  103. package/dist/main.js.map +1 -1
  104. package/dist/modes/interactive/components/assistant-message.d.ts +3 -2
  105. package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  106. package/dist/modes/interactive/components/assistant-message.js +5 -3
  107. package/dist/modes/interactive/components/assistant-message.js.map +1 -1
  108. package/dist/modes/interactive/components/bordered-loader.d.ts +5 -1
  109. package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -1
  110. package/dist/modes/interactive/components/bordered-loader.js +29 -9
  111. package/dist/modes/interactive/components/bordered-loader.js.map +1 -1
  112. package/dist/modes/interactive/components/branch-summary-message.d.ts +3 -2
  113. package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -1
  114. package/dist/modes/interactive/components/branch-summary-message.js +4 -2
  115. package/dist/modes/interactive/components/branch-summary-message.js.map +1 -1
  116. package/dist/modes/interactive/components/compaction-summary-message.d.ts +3 -2
  117. package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -1
  118. package/dist/modes/interactive/components/compaction-summary-message.js +4 -2
  119. package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -1
  120. package/dist/modes/interactive/components/config-selector.d.ts +71 -0
  121. package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
  122. package/dist/modes/interactive/components/config-selector.js +468 -0
  123. package/dist/modes/interactive/components/config-selector.js.map +1 -0
  124. package/dist/modes/interactive/components/custom-message.d.ts +3 -2
  125. package/dist/modes/interactive/components/custom-message.d.ts.map +1 -1
  126. package/dist/modes/interactive/components/custom-message.js +4 -2
  127. package/dist/modes/interactive/components/custom-message.js.map +1 -1
  128. package/dist/modes/interactive/components/footer.d.ts.map +1 -1
  129. package/dist/modes/interactive/components/footer.js +9 -0
  130. package/dist/modes/interactive/components/footer.js.map +1 -1
  131. package/dist/modes/interactive/components/index.d.ts +1 -0
  132. package/dist/modes/interactive/components/index.d.ts.map +1 -1
  133. package/dist/modes/interactive/components/index.js +1 -0
  134. package/dist/modes/interactive/components/index.js.map +1 -1
  135. package/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -1
  136. package/dist/modes/interactive/components/oauth-selector.js +3 -4
  137. package/dist/modes/interactive/components/oauth-selector.js.map +1 -1
  138. package/dist/modes/interactive/components/session-selector.d.ts +18 -1
  139. package/dist/modes/interactive/components/session-selector.d.ts.map +1 -1
  140. package/dist/modes/interactive/components/session-selector.js +195 -87
  141. package/dist/modes/interactive/components/session-selector.js.map +1 -1
  142. package/dist/modes/interactive/components/skill-invocation-message.d.ts +17 -0
  143. package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
  144. package/dist/modes/interactive/components/skill-invocation-message.js +47 -0
  145. package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
  146. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  147. package/dist/modes/interactive/components/tool-execution.js +12 -5
  148. package/dist/modes/interactive/components/tool-execution.js.map +1 -1
  149. package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -1
  150. package/dist/modes/interactive/components/tree-selector.js +2 -2
  151. package/dist/modes/interactive/components/tree-selector.js.map +1 -1
  152. package/dist/modes/interactive/components/user-message.d.ts +2 -2
  153. package/dist/modes/interactive/components/user-message.d.ts.map +1 -1
  154. package/dist/modes/interactive/components/user-message.js +2 -2
  155. package/dist/modes/interactive/components/user-message.js.map +1 -1
  156. package/dist/modes/interactive/interactive-mode.d.ts +47 -2
  157. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  158. package/dist/modes/interactive/interactive-mode.js +566 -211
  159. package/dist/modes/interactive/interactive-mode.js.map +1 -1
  160. package/dist/modes/interactive/theme/dark.json +1 -1
  161. package/dist/modes/interactive/theme/light.json +1 -1
  162. package/dist/modes/interactive/theme/theme-schema.json +8 -1
  163. package/dist/modes/interactive/theme/theme.d.ts +8 -1
  164. package/dist/modes/interactive/theme/theme.d.ts.map +1 -1
  165. package/dist/modes/interactive/theme/theme.js +79 -28
  166. package/dist/modes/interactive/theme/theme.js.map +1 -1
  167. package/dist/modes/print-mode.d.ts.map +1 -1
  168. package/dist/modes/print-mode.js +25 -89
  169. package/dist/modes/print-mode.js.map +1 -1
  170. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  171. package/dist/modes/rpc/rpc-mode.js +32 -92
  172. package/dist/modes/rpc/rpc-mode.js.map +1 -1
  173. package/dist/utils/git.d.ts +2 -0
  174. package/dist/utils/git.d.ts.map +1 -0
  175. package/dist/utils/git.js +6 -0
  176. package/dist/utils/git.js.map +1 -0
  177. package/dist/utils/shell.d.ts +1 -0
  178. package/dist/utils/shell.d.ts.map +1 -1
  179. package/dist/utils/shell.js +16 -2
  180. package/dist/utils/shell.js.map +1 -1
  181. package/dist/utils/sleep.d.ts +5 -0
  182. package/dist/utils/sleep.d.ts.map +1 -0
  183. package/dist/utils/sleep.js +17 -0
  184. package/dist/utils/sleep.js.map +1 -0
  185. package/docs/compaction.md +23 -21
  186. package/docs/custom-provider.md +538 -0
  187. package/docs/development.md +69 -0
  188. package/docs/extensions.md +180 -118
  189. package/docs/images/doom-extension.png +0 -0
  190. package/docs/images/interactive-mode.png +0 -0
  191. package/docs/images/tree-view.png +0 -0
  192. package/docs/json.md +79 -0
  193. package/docs/keybindings.md +162 -0
  194. package/docs/models.md +193 -0
  195. package/docs/packages.md +163 -0
  196. package/docs/prompt-templates.md +67 -0
  197. package/docs/providers.md +147 -0
  198. package/docs/sdk.md +111 -178
  199. package/docs/session.md +167 -16
  200. package/docs/settings.md +216 -0
  201. package/docs/shell-aliases.md +13 -0
  202. package/docs/skills.md +111 -202
  203. package/docs/terminal-setup.md +65 -0
  204. package/docs/themes.md +295 -0
  205. package/docs/tui.md +36 -5
  206. package/docs/windows.md +17 -0
  207. package/examples/README.md +1 -0
  208. package/examples/extensions/README.md +24 -2
  209. package/examples/extensions/antigravity-image-gen.ts +413 -0
  210. package/examples/extensions/bookmark.ts +50 -0
  211. package/examples/extensions/custom-provider-anthropic/index.ts +604 -0
  212. package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  213. package/examples/extensions/custom-provider-anthropic/package.json +19 -0
  214. package/examples/extensions/custom-provider-gitlab-duo/index.ts +349 -0
  215. package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  216. package/examples/extensions/custom-provider-gitlab-duo/test.ts +82 -0
  217. package/examples/extensions/doom-overlay/doom/build.sh +1 -1
  218. package/examples/extensions/event-bus.ts +43 -0
  219. package/examples/extensions/inline-bash.ts +94 -0
  220. package/examples/extensions/message-renderer.ts +59 -0
  221. package/examples/extensions/session-name.ts +27 -0
  222. package/examples/extensions/space-invaders.ts +560 -0
  223. package/examples/extensions/with-deps/package-lock.json +2 -2
  224. package/examples/extensions/with-deps/package.json +1 -1
  225. package/examples/sdk/02-custom-model.ts +3 -3
  226. package/examples/sdk/03-custom-prompt.ts +20 -9
  227. package/examples/sdk/04-skills.ts +26 -27
  228. package/examples/sdk/06-extensions.ts +15 -6
  229. package/examples/sdk/07-context-files.ts +22 -18
  230. package/examples/sdk/08-prompt-templates.ts +19 -14
  231. package/examples/sdk/09-api-keys-and-oauth.ts +5 -12
  232. package/examples/sdk/10-settings.ts +3 -3
  233. package/examples/sdk/12-full-control.ts +16 -7
  234. package/examples/sdk/README.md +24 -30
  235. package/package.json +4 -4
  236. package/docs/theme.md +0 -617
  237. package/examples/extensions/chalk-logger.ts +0 -26
@@ -0,0 +1,147 @@
1
+ # Providers
2
+
3
+ Pi supports subscription-based providers via OAuth and API key providers via environment variables or auth file. For each provider, pi knows all available models. The list is updated with every pi release.
4
+
5
+ ## Table of Contents
6
+
7
+ - [Subscriptions](#subscriptions)
8
+ - [API Keys](#api-keys)
9
+ - [Auth File](#auth-file)
10
+ - [Cloud Providers](#cloud-providers)
11
+ - [Custom Providers](#custom-providers)
12
+ - [Resolution Order](#resolution-order)
13
+
14
+ ## Subscriptions
15
+
16
+ Use `/login` in interactive mode, then select a provider:
17
+
18
+ - Claude Pro/Max
19
+ - ChatGPT Plus/Pro (Codex)
20
+ - GitHub Copilot
21
+ - Google Gemini CLI
22
+ - Google Antigravity
23
+
24
+ Use `/logout` to clear credentials. Tokens are stored in `~/.pi/agent/auth.json` and auto-refresh when expired.
25
+
26
+ ### GitHub Copilot
27
+
28
+ - Press Enter for github.com, or enter your GitHub Enterprise Server domain
29
+ - If you get "model not supported", enable it in VS Code: Copilot Chat → model selector → select model → "Enable"
30
+
31
+ ### Google Providers
32
+
33
+ - **Gemini CLI**: Standard Gemini models via Cloud Code Assist
34
+ - **Antigravity**: Sandbox with Gemini 3, Claude, and GPT-OSS models
35
+ - Both free with any Google account, subject to rate limits
36
+ - For paid Cloud Code Assist: set `GOOGLE_CLOUD_PROJECT` env var
37
+
38
+ ### OpenAI Codex
39
+
40
+ - Requires ChatGPT Plus or Pro subscription
41
+ - Personal use only; for production, use the OpenAI Platform API
42
+
43
+ ## API Keys
44
+
45
+ Set via environment variable:
46
+
47
+ ```bash
48
+ export ANTHROPIC_API_KEY=sk-ant-...
49
+ pi
50
+ ```
51
+
52
+ | Provider | Environment Variable |
53
+ |----------|---------------------|
54
+ | Anthropic | `ANTHROPIC_API_KEY` |
55
+ | OpenAI | `OPENAI_API_KEY` |
56
+ | Google Gemini | `GEMINI_API_KEY` |
57
+ | Mistral | `MISTRAL_API_KEY` |
58
+ | Groq | `GROQ_API_KEY` |
59
+ | Cerebras | `CEREBRAS_API_KEY` |
60
+ | xAI | `XAI_API_KEY` |
61
+ | OpenRouter | `OPENROUTER_API_KEY` |
62
+ | Vercel AI Gateway | `AI_GATEWAY_API_KEY` |
63
+ | ZAI | `ZAI_API_KEY` |
64
+ | OpenCode Zen | `OPENCODE_API_KEY` |
65
+ | MiniMax | `MINIMAX_API_KEY` |
66
+ | MiniMax (China) | `MINIMAX_CN_API_KEY` |
67
+
68
+ ## Auth File
69
+
70
+ Store credentials in `~/.pi/agent/auth.json`:
71
+
72
+ ```json
73
+ {
74
+ "anthropic": { "type": "api_key", "key": "sk-ant-..." },
75
+ "openai": { "type": "api_key", "key": "sk-..." },
76
+ "google": { "type": "api_key", "key": "..." }
77
+ }
78
+ ```
79
+
80
+ The file is created with `0600` permissions (user read/write only). Auth file credentials take priority over environment variables.
81
+
82
+ OAuth credentials are also stored here after `/login` and managed automatically.
83
+
84
+ ## Cloud Providers
85
+
86
+ ### Azure OpenAI
87
+
88
+ ```bash
89
+ export AZURE_OPENAI_API_KEY=...
90
+ export AZURE_OPENAI_BASE_URL=https://your-resource.openai.azure.com
91
+ # or use resource name instead of base URL
92
+ export AZURE_OPENAI_RESOURCE_NAME=your-resource
93
+
94
+ # Optional
95
+ export AZURE_OPENAI_API_VERSION=2024-02-01
96
+ export AZURE_OPENAI_DEPLOYMENT_NAME_MAP=gpt-4=my-gpt4,gpt-4o=my-gpt4o
97
+ ```
98
+
99
+ ### Amazon Bedrock
100
+
101
+ ```bash
102
+ # Option 1: AWS Profile
103
+ export AWS_PROFILE=your-profile
104
+
105
+ # Option 2: IAM Keys
106
+ export AWS_ACCESS_KEY_ID=AKIA...
107
+ export AWS_SECRET_ACCESS_KEY=...
108
+
109
+ # Option 3: Bearer Token
110
+ export AWS_BEARER_TOKEN_BEDROCK=...
111
+
112
+ # Optional region (defaults to us-east-1)
113
+ export AWS_REGION=us-west-2
114
+ ```
115
+
116
+ Also supports ECS task roles (`AWS_CONTAINER_CREDENTIALS_*`) and IRSA (`AWS_WEB_IDENTITY_TOKEN_FILE`).
117
+
118
+ ```bash
119
+ pi --provider amazon-bedrock --model us.anthropic.claude-sonnet-4-20250514-v1:0
120
+ ```
121
+
122
+ ### Google Vertex AI
123
+
124
+ Uses Application Default Credentials:
125
+
126
+ ```bash
127
+ gcloud auth application-default login
128
+ export GOOGLE_CLOUD_PROJECT=your-project
129
+ export GOOGLE_CLOUD_LOCATION=us-central1
130
+ ```
131
+
132
+ Or set `GOOGLE_APPLICATION_CREDENTIALS` to a service account key file.
133
+
134
+ ## Custom Providers
135
+
136
+ **Via models.json:** Add Ollama, LM Studio, vLLM, or any provider that speaks a supported API (OpenAI Completions, OpenAI Responses, Anthropic Messages, Google Generative AI). See [models.md](models.md).
137
+
138
+ **Via extensions:** For providers that need custom API implementations or OAuth flows, create an extension. See [custom-provider.md](custom-provider.md) and [examples/extensions/custom-provider-gitlab-duo](../examples/extensions/custom-provider-gitlab-duo/).
139
+
140
+ ## Resolution Order
141
+
142
+ When resolving credentials for a provider:
143
+
144
+ 1. CLI `--api-key` flag
145
+ 2. `auth.json` entry (API key or OAuth token)
146
+ 3. Environment variable
147
+ 4. Custom provider keys from `models.json`
package/docs/sdk.md CHANGED
@@ -16,11 +16,11 @@ See [examples/sdk/](../examples/sdk/) for working examples from minimal to full
16
16
  ## Quick Start
17
17
 
18
18
  ```typescript
19
- import { createAgentSession, discoverAuthStorage, discoverModels, SessionManager } from "@mariozechner/pi-coding-agent";
19
+ import { AuthStorage, createAgentSession, ModelRegistry, SessionManager } from "@mariozechner/pi-coding-agent";
20
20
 
21
21
  // Set up credential storage and model registry
22
- const authStorage = discoverAuthStorage();
23
- const modelRegistry = discoverModels(authStorage);
22
+ const authStorage = new AuthStorage();
23
+ const modelRegistry = new ModelRegistry(authStorage);
24
24
 
25
25
  const { session } = await createAgentSession({
26
26
  sessionManager: SessionManager.inMemory(),
@@ -51,20 +51,17 @@ The SDK is included in the main package. No separate installation needed.
51
51
 
52
52
  The main factory function. Creates an `AgentSession` with configurable options.
53
53
 
54
- **Philosophy:** "Omit to discover, provide to override."
55
- - Omit an option → pi discovers/loads from standard locations
56
- - Provide an option → your value is used, discovery skipped for that option
54
+ `createAgentSession()` uses a `ResourceLoader` to supply extensions, skills, prompt templates, themes, and context files. If you do not provide one, it uses `DefaultResourceLoader` with standard discovery.
57
55
 
58
56
  ```typescript
59
57
  import { createAgentSession } from "@mariozechner/pi-coding-agent";
60
58
 
61
- // Minimal: all defaults (discovers everything from cwd and ~/.pi/agent)
59
+ // Minimal: defaults with DefaultResourceLoader
62
60
  const { session } = await createAgentSession();
63
61
 
64
62
  // Custom: override specific options
65
63
  const { session } = await createAgentSession({
66
64
  model: myModel,
67
- systemPrompt: "You are helpful.",
68
65
  tools: [readTool, bashTool],
69
66
  sessionManager: SessionManager.inMemory(),
70
67
  });
@@ -251,7 +248,7 @@ session.subscribe((event) => {
251
248
 
252
249
  ```typescript
253
250
  const { session } = await createAgentSession({
254
- // Working directory for project-local discovery
251
+ // Working directory for DefaultResourceLoader discovery
255
252
  cwd: process.cwd(), // default
256
253
 
257
254
  // Global config directory
@@ -259,14 +256,14 @@ const { session } = await createAgentSession({
259
256
  });
260
257
  ```
261
258
 
262
- `cwd` is used for:
259
+ `cwd` is used by `DefaultResourceLoader` for:
263
260
  - Project extensions (`.pi/extensions/`)
264
261
  - Project skills (`.pi/skills/`)
265
262
  - Project prompts (`.pi/prompts/`)
266
263
  - Context files (`AGENTS.md` walking up from cwd)
267
264
  - Session directory naming
268
265
 
269
- `agentDir` is used for:
266
+ `agentDir` is used by `DefaultResourceLoader` for:
270
267
  - Global extensions (`extensions/`)
271
268
  - Global skills (`skills/`)
272
269
  - Global prompts (`prompts/`)
@@ -276,14 +273,16 @@ const { session } = await createAgentSession({
276
273
  - Credentials (`auth.json`)
277
274
  - Sessions (`sessions/`)
278
275
 
276
+ When you pass a custom `ResourceLoader`, `cwd` and `agentDir` no longer control resource discovery. They still influence session naming and tool path resolution.
277
+
279
278
  ### Model
280
279
 
281
280
  ```typescript
282
281
  import { getModel } from "@mariozechner/pi-ai";
283
- import { discoverAuthStorage, discoverModels } from "@mariozechner/pi-coding-agent";
282
+ import { AuthStorage, ModelRegistry } from "@mariozechner/pi-coding-agent";
284
283
 
285
- const authStorage = discoverAuthStorage();
286
- const modelRegistry = discoverModels(authStorage);
284
+ const authStorage = new AuthStorage();
285
+ const modelRegistry = new ModelRegistry(authStorage);
287
286
 
288
287
  // Find specific built-in model (doesn't check if API key exists)
289
288
  const opus = getModel("anthropic", "claude-opus-4-5");
@@ -327,11 +326,11 @@ API key resolution priority (handled by AuthStorage):
327
326
  4. Fallback resolver (for custom provider keys from `models.json`)
328
327
 
329
328
  ```typescript
330
- import { AuthStorage, ModelRegistry, discoverAuthStorage, discoverModels } from "@mariozechner/pi-coding-agent";
329
+ import { AuthStorage, ModelRegistry } from "@mariozechner/pi-coding-agent";
331
330
 
332
331
  // Default: uses ~/.pi/agent/auth.json and ~/.pi/agent/models.json
333
- const authStorage = discoverAuthStorage();
334
- const modelRegistry = discoverModels(authStorage);
332
+ const authStorage = new AuthStorage();
333
+ const modelRegistry = new ModelRegistry(authStorage);
335
334
 
336
335
  const { session } = await createAgentSession({
337
336
  sessionManager: SessionManager.inMemory(),
@@ -360,16 +359,17 @@ const simpleRegistry = new ModelRegistry(authStorage);
360
359
 
361
360
  ### System Prompt
362
361
 
362
+ Use a `ResourceLoader` to override the system prompt:
363
+
363
364
  ```typescript
364
- const { session } = await createAgentSession({
365
- // Replace entirely
366
- systemPrompt: "You are a helpful assistant.",
367
-
368
- // Or modify default (receives default, returns modified)
369
- systemPrompt: (defaultPrompt) => {
370
- return `${defaultPrompt}\n\n## Additional Rules\n- Be concise`;
371
- },
365
+ import { createAgentSession, DefaultResourceLoader } from "@mariozechner/pi-coding-agent";
366
+
367
+ const loader = new DefaultResourceLoader({
368
+ systemPromptOverride: () => "You are a helpful assistant.",
372
369
  });
370
+ await loader.reload();
371
+
372
+ const { session } = await createAgentSession({ resourceLoader: loader });
373
373
  ```
374
374
 
375
375
  > See [examples/sdk/03-custom-prompt.ts](../examples/sdk/03-custom-prompt.ts)
@@ -462,61 +462,45 @@ const { session } = await createAgentSession({
462
462
  });
463
463
  ```
464
464
 
465
- Custom tools passed via `customTools` are combined with extension-registered tools. Extensions discovered from `~/.pi/agent/extensions/` and `.pi/extensions/` can also register tools via `pi.registerTool()`.
465
+ Custom tools passed via `customTools` are combined with extension-registered tools. Extensions loaded by the ResourceLoader can also register tools via `pi.registerTool()`.
466
466
 
467
467
  > See [examples/sdk/05-tools.ts](../examples/sdk/05-tools.ts)
468
468
 
469
469
  ### Extensions
470
470
 
471
- By default, extensions are discovered from multiple locations:
472
- - `~/.pi/agent/extensions/` (global)
473
- - `.pi/extensions/` (project-local)
474
- - Paths listed in `settings.json` `"extensions"` array
471
+ Extensions are loaded by the `ResourceLoader`. `DefaultResourceLoader` discovers extensions from `~/.pi/agent/extensions/`, `.pi/extensions/`, and settings.json extension sources.
475
472
 
476
473
  ```typescript
477
- import { createAgentSession, type ExtensionFactory } from "@mariozechner/pi-coding-agent";
478
-
479
- // Inline extension factory
480
- const myExtension: ExtensionFactory = (pi) => {
481
- pi.on("tool_call", async (event, ctx) => {
482
- console.log(`Tool: ${event.toolName}`);
483
- });
484
-
485
- pi.registerCommand("hello", {
486
- description: "Say hello",
487
- handler: async (args, ctx) => {
488
- ctx.ui.notify("Hello!", "info");
489
- },
490
- });
491
- };
492
-
493
- // Pass inline extensions (skips file discovery)
494
- const { session } = await createAgentSession({
495
- extensions: [myExtension],
496
- });
474
+ import { createAgentSession, DefaultResourceLoader } from "@mariozechner/pi-coding-agent";
497
475
 
498
- // Add paths to load (merged with discovery)
499
- const { session } = await createAgentSession({
476
+ const loader = new DefaultResourceLoader({
500
477
  additionalExtensionPaths: ["/path/to/my-extension.ts"],
478
+ extensionFactories: [
479
+ (pi) => {
480
+ pi.on("agent_start", () => {
481
+ console.log("[Inline Extension] Agent starting");
482
+ });
483
+ },
484
+ ],
501
485
  });
486
+ await loader.reload();
502
487
 
503
- // Disable extension discovery entirely
504
- const { session } = await createAgentSession({
505
- extensions: [],
506
- });
488
+ const { session } = await createAgentSession({ resourceLoader: loader });
507
489
  ```
508
490
 
509
491
  Extensions can register tools, subscribe to events, add commands, and more. See [extensions.md](extensions.md) for the full API.
510
492
 
511
- **Event Bus:** Extensions can communicate via `pi.events`. Pass a shared `eventBus` to `createAgentSession()` if you need to emit/listen from outside:
493
+ **Event Bus:** Extensions can communicate via `pi.events`. Pass a shared `eventBus` to `DefaultResourceLoader` if you need to emit or listen from outside:
512
494
 
513
495
  ```typescript
514
- import { createAgentSession, createEventBus } from "@mariozechner/pi-coding-agent";
496
+ import { createEventBus, DefaultResourceLoader } from "@mariozechner/pi-coding-agent";
515
497
 
516
498
  const eventBus = createEventBus();
517
- const { session } = await createAgentSession({ eventBus });
499
+ const loader = new DefaultResourceLoader({
500
+ eventBus,
501
+ });
502
+ await loader.reload();
518
503
 
519
- // Listen for events from extensions
520
504
  eventBus.on("my-extension:status", (data) => console.log(data));
521
505
  ```
522
506
 
@@ -525,14 +509,13 @@ eventBus.on("my-extension:status", (data) => console.log(data));
525
509
  ### Skills
526
510
 
527
511
  ```typescript
528
- import { createAgentSession, discoverSkills, type Skill } from "@mariozechner/pi-coding-agent";
529
-
530
- // Discover and filter
531
- const { skills: allSkills, warnings } = discoverSkills();
532
- const filtered = allSkills.filter(s => s.name.includes("search"));
512
+ import {
513
+ createAgentSession,
514
+ DefaultResourceLoader,
515
+ type Skill,
516
+ } from "@mariozechner/pi-coding-agent";
533
517
 
534
- // Custom skill
535
- const mySkill: Skill = {
518
+ const customSkill: Skill = {
536
519
  name: "my-skill",
537
520
  description: "Custom instructions",
538
521
  filePath: "/path/to/SKILL.md",
@@ -540,20 +523,15 @@ const mySkill: Skill = {
540
523
  source: "custom",
541
524
  };
542
525
 
543
- const { session } = await createAgentSession({
544
- skills: [...filtered, mySkill],
545
- });
546
-
547
- // Disable skills
548
- const { session } = await createAgentSession({
549
- skills: [],
526
+ const loader = new DefaultResourceLoader({
527
+ skillsOverride: (current) => ({
528
+ skills: [...current.skills, customSkill],
529
+ diagnostics: current.diagnostics,
530
+ }),
550
531
  });
532
+ await loader.reload();
551
533
 
552
- // Discovery with settings filter
553
- const { skills } = discoverSkills(process.cwd(), undefined, {
554
- ignoredSkills: ["browser-*"], // glob patterns to exclude
555
- includeSkills: ["search-*"], // glob patterns to include (empty = all)
556
- });
534
+ const { session } = await createAgentSession({ resourceLoader: loader });
557
535
  ```
558
536
 
559
537
  > See [examples/sdk/04-skills.ts](../examples/sdk/04-skills.ts)
@@ -561,26 +539,19 @@ const { skills } = discoverSkills(process.cwd(), undefined, {
561
539
  ### Context Files
562
540
 
563
541
  ```typescript
564
- import { createAgentSession, discoverContextFiles } from "@mariozechner/pi-coding-agent";
565
-
566
- // Discover AGENTS.md files
567
- const discovered = discoverContextFiles();
568
-
569
- // Add custom context
570
- const { session } = await createAgentSession({
571
- contextFiles: [
572
- ...discovered,
573
- {
574
- path: "/virtual/AGENTS.md",
575
- content: "# Guidelines\n\n- Be concise\n- Use TypeScript",
576
- },
577
- ],
542
+ import { createAgentSession, DefaultResourceLoader } from "@mariozechner/pi-coding-agent";
543
+
544
+ const loader = new DefaultResourceLoader({
545
+ agentsFilesOverride: (current) => ({
546
+ agentsFiles: [
547
+ ...current.agentsFiles,
548
+ { path: "/virtual/AGENTS.md", content: "# Guidelines\n\n- Be concise" },
549
+ ],
550
+ }),
578
551
  });
552
+ await loader.reload();
579
553
 
580
- // Disable context files
581
- const { session } = await createAgentSession({
582
- contextFiles: [],
583
- });
554
+ const { session } = await createAgentSession({ resourceLoader: loader });
584
555
  ```
585
556
 
586
557
  > See [examples/sdk/07-context-files.ts](../examples/sdk/07-context-files.ts)
@@ -588,9 +559,11 @@ const { session } = await createAgentSession({
588
559
  ### Slash Commands
589
560
 
590
561
  ```typescript
591
- import { createAgentSession, discoverPromptTemplates, type PromptTemplate } from "@mariozechner/pi-coding-agent";
592
-
593
- const discovered = discoverPromptTemplates();
562
+ import {
563
+ createAgentSession,
564
+ DefaultResourceLoader,
565
+ type PromptTemplate,
566
+ } from "@mariozechner/pi-coding-agent";
594
567
 
595
568
  const customCommand: PromptTemplate = {
596
569
  name: "deploy",
@@ -599,9 +572,15 @@ const customCommand: PromptTemplate = {
599
572
  content: "# Deploy\n\n1. Build\n2. Test\n3. Deploy",
600
573
  };
601
574
 
602
- const { session } = await createAgentSession({
603
- promptTemplates: [...discovered, customCommand],
575
+ const loader = new DefaultResourceLoader({
576
+ promptsOverride: (current) => ({
577
+ prompts: [...current.prompts, customCommand],
578
+ diagnostics: current.diagnostics,
579
+ }),
604
580
  });
581
+ await loader.reload();
582
+
583
+ const { session } = await createAgentSession({ resourceLoader: loader });
605
584
  ```
606
585
 
607
586
  > See [examples/sdk/08-prompt-templates.ts](../examples/sdk/08-prompt-templates.ts)
@@ -719,62 +698,31 @@ Settings load from two locations and merge:
719
698
  1. Global: `~/.pi/agent/settings.json`
720
699
  2. Project: `<cwd>/.pi/settings.json`
721
700
 
722
- Project overrides global. Nested objects merge keys. Setters only modify global (project is read-only for version control).
701
+ Project overrides global. Nested objects merge keys. Setters modify global settings by default.
723
702
 
724
703
  > See [examples/sdk/10-settings.ts](../examples/sdk/10-settings.ts)
725
704
 
726
- ## Discovery Functions
705
+ ## ResourceLoader
727
706
 
728
- All discovery functions accept optional `cwd` and `agentDir` parameters.
707
+ Use `DefaultResourceLoader` to discover extensions, skills, prompts, themes, and context files.
729
708
 
730
709
  ```typescript
731
- import { getModel } from "@mariozechner/pi-ai";
732
710
  import {
733
- AuthStorage,
734
- ModelRegistry,
735
- discoverAuthStorage,
736
- discoverModels,
737
- discoverSkills,
738
- discoverExtensions,
739
- discoverContextFiles,
740
- discoverPromptTemplates,
741
- loadSettings,
742
- buildSystemPrompt,
743
- createEventBus,
711
+ DefaultResourceLoader,
712
+ getAgentDir,
744
713
  } from "@mariozechner/pi-coding-agent";
745
714
 
746
- // Auth and Models
747
- const authStorage = discoverAuthStorage(); // ~/.pi/agent/auth.json
748
- const modelRegistry = discoverModels(authStorage); // + ~/.pi/agent/models.json
749
- const allModels = modelRegistry.getAll(); // All models (built-in + custom)
750
- const available = await modelRegistry.getAvailable(); // Only models with API keys
751
- const model = modelRegistry.find("provider", "id"); // Find specific model
752
- const builtIn = getModel("anthropic", "claude-opus-4-5"); // Built-in only
753
-
754
- // Skills
755
- const { skills, warnings } = discoverSkills(cwd, agentDir, skillsSettings);
756
-
757
- // Extensions (async - loads TypeScript)
758
- // Pass eventBus to share pi.events across extensions
759
- const eventBus = createEventBus();
760
- const { extensions, errors } = await discoverExtensions(eventBus, cwd, agentDir);
761
-
762
- // Context files
763
- const contextFiles = discoverContextFiles(cwd, agentDir);
764
-
765
- // Prompt templates
766
- const templates = discoverPromptTemplates(cwd, agentDir);
767
-
768
- // Settings (global + project merged)
769
- const settings = loadSettings(cwd, agentDir);
770
-
771
- // Build system prompt manually
772
- const prompt = buildSystemPrompt({
773
- skills,
774
- contextFiles,
775
- appendPrompt: "Additional instructions",
715
+ const loader = new DefaultResourceLoader({
776
716
  cwd,
717
+ agentDir: getAgentDir(),
777
718
  });
719
+ await loader.reload();
720
+
721
+ const extensions = loader.getExtensions();
722
+ const skills = loader.getSkills();
723
+ const prompts = loader.getPrompts();
724
+ const themes = loader.getThemes();
725
+ const contextFiles = loader.getAgentsFiles().agentsFiles;
778
726
  ```
779
727
 
780
728
  ## Return Value
@@ -808,12 +756,12 @@ import { Type } from "@sinclair/typebox";
808
756
  import {
809
757
  AuthStorage,
810
758
  createAgentSession,
759
+ DefaultResourceLoader,
811
760
  ModelRegistry,
812
761
  SessionManager,
813
762
  SettingsManager,
814
763
  readTool,
815
764
  bashTool,
816
- type ExtensionFactory,
817
765
  type ToolDefinition,
818
766
  } from "@mariozechner/pi-coding-agent";
819
767
 
@@ -828,14 +776,6 @@ if (process.env.MY_KEY) {
828
776
  // Model registry (no custom models.json)
829
777
  const modelRegistry = new ModelRegistry(authStorage);
830
778
 
831
- // Inline extension
832
- const auditExtension: ExtensionFactory = (pi) => {
833
- pi.on("tool_call", async (event) => {
834
- console.log(`[Audit] ${event.toolName}`);
835
- return undefined;
836
- });
837
- };
838
-
839
779
  // Inline tool
840
780
  const statusTool: ToolDefinition = {
841
781
  name: "status",
@@ -857,24 +797,27 @@ const settingsManager = SettingsManager.inMemory({
857
797
  retry: { enabled: true, maxRetries: 2 },
858
798
  });
859
799
 
800
+ const loader = new DefaultResourceLoader({
801
+ cwd: process.cwd(),
802
+ agentDir: "/custom/agent",
803
+ settingsManager,
804
+ systemPromptOverride: () => "You are a minimal assistant. Be concise.",
805
+ });
806
+ await loader.reload();
807
+
860
808
  const { session } = await createAgentSession({
861
809
  cwd: process.cwd(),
862
810
  agentDir: "/custom/agent",
863
-
811
+
864
812
  model,
865
813
  thinkingLevel: "off",
866
814
  authStorage,
867
815
  modelRegistry,
868
-
869
- systemPrompt: "You are a minimal assistant. Be concise.",
870
-
816
+
871
817
  tools: [readTool, bashTool],
872
818
  customTools: [statusTool],
873
- extensions: [auditExtension],
874
- skills: [],
875
- contextFiles: [],
876
- promptTemplates: [],
877
-
819
+ resourceLoader: loader,
820
+
878
821
  sessionManager: SessionManager.inMemory(),
879
822
  settingsManager,
880
823
  });
@@ -976,21 +919,13 @@ createAgentSession
976
919
  // Auth and Models
977
920
  AuthStorage
978
921
  ModelRegistry
979
- discoverAuthStorage
980
- discoverModels
981
-
982
- // Discovery
983
- discoverSkills
984
- discoverExtensions
985
- discoverContextFiles
986
- discoverPromptTemplates
987
922
 
988
- // Event Bus (for shared extension communication)
923
+ // Resource loading
924
+ DefaultResourceLoader
925
+ type ResourceLoader
989
926
  createEventBus
990
927
 
991
928
  // Helpers
992
- loadSettings
993
- buildSystemPrompt
994
929
 
995
930
  // Session management
996
931
  SessionManager
@@ -1016,8 +951,6 @@ type ExtensionAPI
1016
951
  type ToolDefinition
1017
952
  type Skill
1018
953
  type PromptTemplate
1019
- type Settings
1020
- type SkillsSettings
1021
954
  type Tool
1022
955
  ```
1023
956