indusagi-coding-agent 0.1.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 (240) hide show
  1. package/CHANGELOG.md +2249 -0
  2. package/README.md +546 -0
  3. package/dist/cli/args.js +282 -0
  4. package/dist/cli/config-selector.js +30 -0
  5. package/dist/cli/file-processor.js +78 -0
  6. package/dist/cli/list-models.js +91 -0
  7. package/dist/cli/session-picker.js +31 -0
  8. package/dist/cli.js +10 -0
  9. package/dist/config.js +158 -0
  10. package/dist/core/agent-session.js +2097 -0
  11. package/dist/core/auth-storage.js +278 -0
  12. package/dist/core/bash-executor.js +211 -0
  13. package/dist/core/compaction/branch-summarization.js +241 -0
  14. package/dist/core/compaction/compaction.js +606 -0
  15. package/dist/core/compaction/index.js +6 -0
  16. package/dist/core/compaction/utils.js +137 -0
  17. package/dist/core/diagnostics.js +1 -0
  18. package/dist/core/event-bus.js +24 -0
  19. package/dist/core/exec.js +70 -0
  20. package/dist/core/export-html/ansi-to-html.js +248 -0
  21. package/dist/core/export-html/index.js +221 -0
  22. package/dist/core/export-html/template.css +905 -0
  23. package/dist/core/export-html/template.html +54 -0
  24. package/dist/core/export-html/template.js +1549 -0
  25. package/dist/core/export-html/tool-renderer.js +56 -0
  26. package/dist/core/export-html/vendor/highlight.min.js +1213 -0
  27. package/dist/core/export-html/vendor/marked.min.js +6 -0
  28. package/dist/core/extensions/index.js +8 -0
  29. package/dist/core/extensions/loader.js +395 -0
  30. package/dist/core/extensions/runner.js +499 -0
  31. package/dist/core/extensions/types.js +31 -0
  32. package/dist/core/extensions/wrapper.js +101 -0
  33. package/dist/core/footer-data-provider.js +133 -0
  34. package/dist/core/index.js +8 -0
  35. package/dist/core/keybindings.js +140 -0
  36. package/dist/core/messages.js +122 -0
  37. package/dist/core/model-registry.js +454 -0
  38. package/dist/core/model-resolver.js +309 -0
  39. package/dist/core/package-manager.js +1142 -0
  40. package/dist/core/prompt-templates.js +250 -0
  41. package/dist/core/resource-loader.js +569 -0
  42. package/dist/core/sdk.js +225 -0
  43. package/dist/core/session-manager.js +1078 -0
  44. package/dist/core/settings-manager.js +430 -0
  45. package/dist/core/skills.js +339 -0
  46. package/dist/core/system-prompt.js +136 -0
  47. package/dist/core/timings.js +24 -0
  48. package/dist/core/tools/bash.js +226 -0
  49. package/dist/core/tools/edit-diff.js +242 -0
  50. package/dist/core/tools/edit.js +145 -0
  51. package/dist/core/tools/find.js +205 -0
  52. package/dist/core/tools/grep.js +238 -0
  53. package/dist/core/tools/index.js +60 -0
  54. package/dist/core/tools/ls.js +117 -0
  55. package/dist/core/tools/path-utils.js +52 -0
  56. package/dist/core/tools/read.js +165 -0
  57. package/dist/core/tools/truncate.js +204 -0
  58. package/dist/core/tools/write.js +77 -0
  59. package/dist/index.js +41 -0
  60. package/dist/main.js +565 -0
  61. package/dist/migrations.js +260 -0
  62. package/dist/modes/index.js +7 -0
  63. package/dist/modes/interactive/components/armin.js +328 -0
  64. package/dist/modes/interactive/components/assistant-message.js +86 -0
  65. package/dist/modes/interactive/components/bash-execution.js +155 -0
  66. package/dist/modes/interactive/components/bordered-loader.js +47 -0
  67. package/dist/modes/interactive/components/branch-summary-message.js +41 -0
  68. package/dist/modes/interactive/components/compaction-summary-message.js +42 -0
  69. package/dist/modes/interactive/components/config-selector.js +458 -0
  70. package/dist/modes/interactive/components/countdown-timer.js +27 -0
  71. package/dist/modes/interactive/components/custom-editor.js +61 -0
  72. package/dist/modes/interactive/components/custom-message.js +80 -0
  73. package/dist/modes/interactive/components/diff.js +132 -0
  74. package/dist/modes/interactive/components/dynamic-border.js +19 -0
  75. package/dist/modes/interactive/components/extension-editor.js +96 -0
  76. package/dist/modes/interactive/components/extension-input.js +54 -0
  77. package/dist/modes/interactive/components/extension-selector.js +70 -0
  78. package/dist/modes/interactive/components/footer.js +213 -0
  79. package/dist/modes/interactive/components/index.js +31 -0
  80. package/dist/modes/interactive/components/keybinding-hints.js +60 -0
  81. package/dist/modes/interactive/components/login-dialog.js +138 -0
  82. package/dist/modes/interactive/components/model-selector.js +253 -0
  83. package/dist/modes/interactive/components/oauth-selector.js +91 -0
  84. package/dist/modes/interactive/components/scoped-models-selector.js +262 -0
  85. package/dist/modes/interactive/components/session-selector-search.js +145 -0
  86. package/dist/modes/interactive/components/session-selector.js +698 -0
  87. package/dist/modes/interactive/components/settings-selector.js +250 -0
  88. package/dist/modes/interactive/components/show-images-selector.js +33 -0
  89. package/dist/modes/interactive/components/skill-invocation-message.js +44 -0
  90. package/dist/modes/interactive/components/theme-selector.js +43 -0
  91. package/dist/modes/interactive/components/thinking-selector.js +45 -0
  92. package/dist/modes/interactive/components/tool-execution.js +608 -0
  93. package/dist/modes/interactive/components/tree-selector.js +892 -0
  94. package/dist/modes/interactive/components/user-message-selector.js +109 -0
  95. package/dist/modes/interactive/components/user-message.js +15 -0
  96. package/dist/modes/interactive/components/visual-truncate.js +32 -0
  97. package/dist/modes/interactive/interactive-mode.js +3576 -0
  98. package/dist/modes/interactive/theme/dark.json +85 -0
  99. package/dist/modes/interactive/theme/light.json +84 -0
  100. package/dist/modes/interactive/theme/theme-schema.json +335 -0
  101. package/dist/modes/interactive/theme/theme.js +938 -0
  102. package/dist/modes/print-mode.js +96 -0
  103. package/dist/modes/rpc/rpc-client.js +390 -0
  104. package/dist/modes/rpc/rpc-mode.js +448 -0
  105. package/dist/modes/rpc/rpc-types.js +7 -0
  106. package/dist/utils/changelog.js +86 -0
  107. package/dist/utils/clipboard-image.js +116 -0
  108. package/dist/utils/clipboard.js +58 -0
  109. package/dist/utils/frontmatter.js +25 -0
  110. package/dist/utils/git.js +5 -0
  111. package/dist/utils/image-convert.js +34 -0
  112. package/dist/utils/image-resize.js +180 -0
  113. package/dist/utils/mime.js +25 -0
  114. package/dist/utils/photon.js +120 -0
  115. package/dist/utils/shell.js +164 -0
  116. package/dist/utils/sleep.js +16 -0
  117. package/dist/utils/tools-manager.js +186 -0
  118. package/docs/compaction.md +390 -0
  119. package/docs/custom-provider.md +538 -0
  120. package/docs/development.md +69 -0
  121. package/docs/extensions.md +1733 -0
  122. package/docs/images/doom-extension.png +0 -0
  123. package/docs/images/interactive-mode.png +0 -0
  124. package/docs/images/tree-view.png +0 -0
  125. package/docs/json.md +79 -0
  126. package/docs/keybindings.md +162 -0
  127. package/docs/models.md +193 -0
  128. package/docs/packages.md +163 -0
  129. package/docs/prompt-templates.md +67 -0
  130. package/docs/providers.md +147 -0
  131. package/docs/rpc.md +1048 -0
  132. package/docs/sdk.md +957 -0
  133. package/docs/session.md +412 -0
  134. package/docs/settings.md +216 -0
  135. package/docs/shell-aliases.md +13 -0
  136. package/docs/skills.md +226 -0
  137. package/docs/terminal-setup.md +65 -0
  138. package/docs/themes.md +295 -0
  139. package/docs/tree.md +219 -0
  140. package/docs/tui.md +887 -0
  141. package/docs/windows.md +17 -0
  142. package/examples/README.md +25 -0
  143. package/examples/extensions/README.md +192 -0
  144. package/examples/extensions/antigravity-image-gen.ts +414 -0
  145. package/examples/extensions/auto-commit-on-exit.ts +49 -0
  146. package/examples/extensions/bookmark.ts +50 -0
  147. package/examples/extensions/claude-rules.ts +86 -0
  148. package/examples/extensions/confirm-destructive.ts +59 -0
  149. package/examples/extensions/custom-compaction.ts +115 -0
  150. package/examples/extensions/custom-footer.ts +65 -0
  151. package/examples/extensions/custom-header.ts +73 -0
  152. package/examples/extensions/custom-provider-anthropic/index.ts +605 -0
  153. package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  154. package/examples/extensions/custom-provider-anthropic/package.json +19 -0
  155. package/examples/extensions/custom-provider-gitlab-duo/index.ts +350 -0
  156. package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  157. package/examples/extensions/custom-provider-gitlab-duo/test.ts +83 -0
  158. package/examples/extensions/dirty-repo-guard.ts +56 -0
  159. package/examples/extensions/doom-overlay/README.md +46 -0
  160. package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
  161. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  162. package/examples/extensions/doom-overlay/doom/build.sh +152 -0
  163. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  164. package/examples/extensions/doom-overlay/doom-component.ts +133 -0
  165. package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
  166. package/examples/extensions/doom-overlay/doom-keys.ts +105 -0
  167. package/examples/extensions/doom-overlay/index.ts +74 -0
  168. package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  169. package/examples/extensions/event-bus.ts +43 -0
  170. package/examples/extensions/file-trigger.ts +41 -0
  171. package/examples/extensions/git-checkpoint.ts +53 -0
  172. package/examples/extensions/handoff.ts +151 -0
  173. package/examples/extensions/hello.ts +25 -0
  174. package/examples/extensions/inline-bash.ts +94 -0
  175. package/examples/extensions/input-transform.ts +43 -0
  176. package/examples/extensions/interactive-shell.ts +196 -0
  177. package/examples/extensions/mac-system-theme.ts +47 -0
  178. package/examples/extensions/message-renderer.ts +60 -0
  179. package/examples/extensions/modal-editor.ts +86 -0
  180. package/examples/extensions/model-status.ts +31 -0
  181. package/examples/extensions/notify.ts +25 -0
  182. package/examples/extensions/overlay-qa-tests.ts +882 -0
  183. package/examples/extensions/overlay-test.ts +151 -0
  184. package/examples/extensions/permission-gate.ts +34 -0
  185. package/examples/extensions/pirate.ts +47 -0
  186. package/examples/extensions/plan-mode/README.md +65 -0
  187. package/examples/extensions/plan-mode/index.ts +341 -0
  188. package/examples/extensions/plan-mode/utils.ts +168 -0
  189. package/examples/extensions/preset.ts +399 -0
  190. package/examples/extensions/protected-paths.ts +30 -0
  191. package/examples/extensions/qna.ts +120 -0
  192. package/examples/extensions/question.ts +265 -0
  193. package/examples/extensions/questionnaire.ts +428 -0
  194. package/examples/extensions/rainbow-editor.ts +88 -0
  195. package/examples/extensions/sandbox/index.ts +318 -0
  196. package/examples/extensions/sandbox/package-lock.json +92 -0
  197. package/examples/extensions/sandbox/package.json +19 -0
  198. package/examples/extensions/send-user-message.ts +97 -0
  199. package/examples/extensions/session-name.ts +27 -0
  200. package/examples/extensions/shutdown-command.ts +63 -0
  201. package/examples/extensions/snake.ts +344 -0
  202. package/examples/extensions/space-invaders.ts +561 -0
  203. package/examples/extensions/ssh.ts +220 -0
  204. package/examples/extensions/status-line.ts +40 -0
  205. package/examples/extensions/subagent/README.md +172 -0
  206. package/examples/extensions/subagent/agents/planner.md +37 -0
  207. package/examples/extensions/subagent/agents/reviewer.md +35 -0
  208. package/examples/extensions/subagent/agents/scout.md +50 -0
  209. package/examples/extensions/subagent/agents/worker.md +24 -0
  210. package/examples/extensions/subagent/agents.ts +127 -0
  211. package/examples/extensions/subagent/index.ts +964 -0
  212. package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  213. package/examples/extensions/subagent/prompts/implement.md +10 -0
  214. package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  215. package/examples/extensions/summarize.ts +196 -0
  216. package/examples/extensions/timed-confirm.ts +70 -0
  217. package/examples/extensions/todo.ts +300 -0
  218. package/examples/extensions/tool-override.ts +144 -0
  219. package/examples/extensions/tools.ts +147 -0
  220. package/examples/extensions/trigger-compact.ts +40 -0
  221. package/examples/extensions/truncated-tool.ts +193 -0
  222. package/examples/extensions/widget-placement.ts +17 -0
  223. package/examples/extensions/with-deps/index.ts +36 -0
  224. package/examples/extensions/with-deps/package-lock.json +31 -0
  225. package/examples/extensions/with-deps/package.json +22 -0
  226. package/examples/sdk/01-minimal.ts +22 -0
  227. package/examples/sdk/02-custom-model.ts +50 -0
  228. package/examples/sdk/03-custom-prompt.ts +55 -0
  229. package/examples/sdk/04-skills.ts +46 -0
  230. package/examples/sdk/05-tools.ts +56 -0
  231. package/examples/sdk/06-extensions.ts +88 -0
  232. package/examples/sdk/07-context-files.ts +40 -0
  233. package/examples/sdk/08-prompt-templates.ts +47 -0
  234. package/examples/sdk/09-api-keys-and-oauth.ts +48 -0
  235. package/examples/sdk/10-settings.ts +38 -0
  236. package/examples/sdk/11-sessions.ts +48 -0
  237. package/examples/sdk/12-full-control.ts +82 -0
  238. package/examples/sdk/13-codex-oauth.ts +37 -0
  239. package/examples/sdk/README.md +144 -0
  240. package/package.json +85 -0
@@ -0,0 +1,50 @@
1
+ /**
2
+ * Custom Model Selection
3
+ *
4
+ * Shows how to select a specific model and thinking level.
5
+ */
6
+
7
+ import { getModel } from "indusagi/ai";
8
+ import { AuthStorage, createAgentSession, ModelRegistry } from "indusagi-coding-agent";
9
+
10
+ // Set up auth storage and model registry
11
+ const authStorage = new AuthStorage();
12
+ const modelRegistry = new ModelRegistry(authStorage);
13
+
14
+ // Option 1: Find a specific built-in model by provider/id
15
+ const opus = getModel("anthropic", "claude-opus-4-5");
16
+ if (opus) {
17
+ console.log(`Found model: ${opus.provider}/${opus.id}`);
18
+ }
19
+
20
+ // Option 2: Find model via registry (includes custom models from models.json)
21
+ const customModel = modelRegistry.find("my-provider", "my-model");
22
+ if (customModel) {
23
+ console.log(`Found custom model: ${customModel.provider}/${customModel.id}`);
24
+ }
25
+
26
+ // Option 3: Pick from available models (have valid API keys)
27
+ const available = await modelRegistry.getAvailable();
28
+ console.log(
29
+ "Available models:",
30
+ available.map((m) => `${m.provider}/${m.id}`),
31
+ );
32
+
33
+ if (available.length > 0) {
34
+ const { session } = await createAgentSession({
35
+ model: available[0],
36
+ thinkingLevel: "medium", // off, low, medium, high
37
+ authStorage,
38
+ modelRegistry,
39
+ });
40
+
41
+ session.subscribe((event) => {
42
+ if (event.type === "message_update" && event.assistantMessageEvent.type === "text_delta") {
43
+ process.stdout.write(event.assistantMessageEvent.delta);
44
+ }
45
+ });
46
+
47
+ await session.prompt("Say hello in one sentence.");
48
+ console.log();
49
+ }
50
+
@@ -0,0 +1,55 @@
1
+ /**
2
+ * Custom System Prompt
3
+ *
4
+ * Shows how to replace or modify the default system prompt.
5
+ */
6
+
7
+ import { createAgentSession, DefaultResourceLoader, SessionManager } from "indusagi-coding-agent";
8
+
9
+ // Option 1: Replace prompt entirely
10
+ const loader1 = new DefaultResourceLoader({
11
+ systemPromptOverride: () => `You are a helpful assistant that speaks like a pirate.
12
+ Always end responses with "Arrr!"`,
13
+ // Needed to avoid DefaultResourceLoader appending APPEND_SYSTEM.md from ~/.indusagi/agent or <cwd>/.indusagi.
14
+ appendSystemPromptOverride: () => [],
15
+ });
16
+ await loader1.reload();
17
+
18
+ const { session: session1 } = await createAgentSession({
19
+ resourceLoader: loader1,
20
+ sessionManager: SessionManager.inMemory(),
21
+ });
22
+
23
+ session1.subscribe((event) => {
24
+ if (event.type === "message_update" && event.assistantMessageEvent.type === "text_delta") {
25
+ process.stdout.write(event.assistantMessageEvent.delta);
26
+ }
27
+ });
28
+
29
+ console.log("=== Replace prompt ===");
30
+ await session1.prompt("What is 2 + 2?");
31
+ console.log("\n");
32
+
33
+ // Option 2: Append instructions to the default prompt
34
+ const loader2 = new DefaultResourceLoader({
35
+ appendSystemPromptOverride: (base) => [
36
+ ...base,
37
+ "## Additional Instructions\n- Always be concise\n- Use bullet points when listing things",
38
+ ],
39
+ });
40
+ await loader2.reload();
41
+
42
+ const { session: session2 } = await createAgentSession({
43
+ resourceLoader: loader2,
44
+ sessionManager: SessionManager.inMemory(),
45
+ });
46
+
47
+ session2.subscribe((event) => {
48
+ if (event.type === "message_update" && event.assistantMessageEvent.type === "text_delta") {
49
+ process.stdout.write(event.assistantMessageEvent.delta);
50
+ }
51
+ });
52
+
53
+ console.log("=== Modify prompt ===");
54
+ await session2.prompt("List 3 benefits of TypeScript.");
55
+ console.log();
@@ -0,0 +1,46 @@
1
+ /**
2
+ * Skills Configuration
3
+ *
4
+ * Skills provide specialized instructions loaded into the system prompt.
5
+ * Discover, filter, merge, or replace them.
6
+ */
7
+
8
+ import { createAgentSession, DefaultResourceLoader, SessionManager, type Skill } from "indusagi-coding-agent";
9
+
10
+ // Or define custom skills inline
11
+ const customSkill: Skill = {
12
+ name: "my-skill",
13
+ description: "Custom project instructions",
14
+ filePath: "/virtual/SKILL.md",
15
+ baseDir: "/virtual",
16
+ source: "path",
17
+ disableModelInvocation: false,
18
+ };
19
+
20
+ const loader = new DefaultResourceLoader({
21
+ skillsOverride: (current) => {
22
+ const filteredSkills = current.skills.filter((s) => s.name.includes("browser") || s.name.includes("search"));
23
+ return {
24
+ skills: [...filteredSkills, customSkill],
25
+ diagnostics: current.diagnostics,
26
+ };
27
+ },
28
+ });
29
+ await loader.reload();
30
+
31
+ // Discover all skills from cwd/.indusagi/skills, ~/.indusagi/agent/skills, etc.
32
+ const { skills: allSkills, diagnostics } = loader.getSkills();
33
+ console.log(
34
+ "Discovered skills:",
35
+ allSkills.map((s) => s.name),
36
+ );
37
+ if (diagnostics.length > 0) {
38
+ console.log("Warnings:", diagnostics);
39
+ }
40
+
41
+ await createAgentSession({
42
+ resourceLoader: loader,
43
+ sessionManager: SessionManager.inMemory(),
44
+ });
45
+
46
+ console.log("Session created with filtered skills");
@@ -0,0 +1,56 @@
1
+ /**
2
+ * Tools Configuration
3
+ *
4
+ * Use built-in tool sets or individual tools.
5
+ *
6
+ * IMPORTANT: When using a custom `cwd`, you must use the tool factory functions
7
+ * (createCodingTools, createReadOnlyTools, createReadTool, etc.) to ensure
8
+ * tools resolve paths relative to your cwd, not process.cwd().
9
+ *
10
+ * For custom tools, see 06-extensions.ts - custom tools are now registered
11
+ * via the extensions system using indusagi.registerTool().
12
+ */
13
+
14
+ import {
15
+ bashTool,
16
+ createAgentSession,
17
+ createBashTool,
18
+ createCodingTools,
19
+ createGrepTool,
20
+ createReadTool,
21
+ grepTool,
22
+ readOnlyTools,
23
+ readTool,
24
+ SessionManager,
25
+ } from "indusagi-coding-agent";
26
+
27
+ // Read-only mode (no edit/write) - uses process.cwd()
28
+ await createAgentSession({
29
+ tools: readOnlyTools,
30
+ sessionManager: SessionManager.inMemory(),
31
+ });
32
+ console.log("Read-only session created");
33
+
34
+ // Custom tool selection - uses process.cwd()
35
+ await createAgentSession({
36
+ tools: [readTool, bashTool, grepTool],
37
+ sessionManager: SessionManager.inMemory(),
38
+ });
39
+ console.log("Custom tools session created");
40
+
41
+ // With custom cwd - MUST use factory functions!
42
+ const customCwd = "/path/to/project";
43
+ await createAgentSession({
44
+ cwd: customCwd,
45
+ tools: createCodingTools(customCwd), // Tools resolve paths relative to customCwd
46
+ sessionManager: SessionManager.inMemory(),
47
+ });
48
+ console.log("Custom cwd session created");
49
+
50
+ // Or pick specific tools for custom cwd
51
+ await createAgentSession({
52
+ cwd: customCwd,
53
+ tools: [createReadTool(customCwd), createBashTool(customCwd), createGrepTool(customCwd)],
54
+ sessionManager: SessionManager.inMemory(),
55
+ });
56
+ console.log("Specific tools with custom cwd session created");
@@ -0,0 +1,88 @@
1
+ /**
2
+ * Extensions Configuration
3
+ *
4
+ * Extensions intercept agent events and can register custom tools.
5
+ * They provide a unified system for extensions, custom tools, commands, and more.
6
+ *
7
+ * By default, extension files are discovered from:
8
+ * - ~/.indusagi/agent/extensions/
9
+ * - <cwd>/.indusagi/extensions/
10
+ * - Paths specified in settings.json "extensions" array
11
+ *
12
+ * An extension is a TypeScript file that exports a default function:
13
+ * export default function (indusagi: ExtensionAPI) { ... }
14
+ */
15
+
16
+ import { createAgentSession, DefaultResourceLoader, SessionManager } from "indusagi-coding-agent";
17
+
18
+ // Extensions are discovered automatically from standard locations.
19
+ // You can also add paths via settings.json or DefaultResourceLoader options.
20
+
21
+ const resourceLoader = new DefaultResourceLoader({
22
+ additionalExtensionPaths: ["./my-logging-extension.ts", "./my-safety-extension.ts"],
23
+ extensionFactories: [
24
+ (indusagi) => {
25
+ indusagi.on("agent_start", () => {
26
+ console.log("[Inline Extension] Agent starting");
27
+ });
28
+ },
29
+ ],
30
+ });
31
+ await resourceLoader.reload();
32
+
33
+ const { session } = await createAgentSession({
34
+ resourceLoader,
35
+ sessionManager: SessionManager.inMemory(),
36
+ });
37
+
38
+ session.subscribe((event) => {
39
+ if (event.type === "message_update" && event.assistantMessageEvent.type === "text_delta") {
40
+ process.stdout.write(event.assistantMessageEvent.delta);
41
+ }
42
+ });
43
+
44
+ await session.prompt("List files in the current directory.");
45
+ console.log();
46
+
47
+ // Example extension file (./my-logging-extension.ts):
48
+ /*
49
+ import type { ExtensionAPI } from "indusagi-coding-agent";
50
+
51
+ export default function (indusagi: ExtensionAPI) {
52
+ indusagi.on("agent_start", async () => {
53
+ console.log("[Extension] Agent starting");
54
+ });
55
+
56
+ indusagi.on("tool_call", async (event) => {
57
+ console.log(\`[Extension] Tool: \${event.toolName}\`);
58
+ // Return { block: true, reason: "..." } to block execution
59
+ return undefined;
60
+ });
61
+
62
+ indusagi.on("agent_end", async (event) => {
63
+ console.log(\`[Extension] Done, \${event.messages.length} messages\`);
64
+ });
65
+
66
+ // Register a custom tool
67
+ indusagi.registerTool({
68
+ name: "my_tool",
69
+ label: "My Tool",
70
+ description: "Does something useful",
71
+ parameters: Type.Object({
72
+ input: Type.String(),
73
+ }),
74
+ execute: async (_toolCallId, params, _onUpdate, _ctx, _signal) => ({
75
+ content: [{ type: "text", text: \`Processed: \${params.input}\` }],
76
+ details: {},
77
+ }),
78
+ });
79
+
80
+ // Register a command
81
+ indusagi.registerCommand("mycommand", {
82
+ description: "Do something",
83
+ handler: async (args, ctx) => {
84
+ ctx.ui.notify(\`Command executed with: \${args}\`);
85
+ },
86
+ });
87
+ }
88
+ */
@@ -0,0 +1,40 @@
1
+ /**
2
+ * Context Files (AGENTS.md)
3
+ *
4
+ * Context files provide project-specific instructions loaded into the system prompt.
5
+ */
6
+
7
+ import { createAgentSession, DefaultResourceLoader, SessionManager } from "indusagi-coding-agent";
8
+
9
+ // Disable context files entirely by returning an empty list in agentsFilesOverride.
10
+ const loader = new DefaultResourceLoader({
11
+ agentsFilesOverride: (current) => ({
12
+ agentsFiles: [
13
+ ...current.agentsFiles,
14
+ {
15
+ path: "/virtual/AGENTS.md",
16
+ content: `# Project Guidelines
17
+
18
+ ## Code Style
19
+ - Use TypeScript strict mode
20
+ - No any types
21
+ - Prefer const over let`,
22
+ },
23
+ ],
24
+ }),
25
+ });
26
+ await loader.reload();
27
+
28
+ // Discover AGENTS.md files walking up from cwd
29
+ const discovered = loader.getAgentsFiles().agentsFiles;
30
+ console.log("Discovered context files:");
31
+ for (const file of discovered) {
32
+ console.log(` - ${file.path} (${file.content.length} chars)`);
33
+ }
34
+
35
+ await createAgentSession({
36
+ resourceLoader: loader,
37
+ sessionManager: SessionManager.inMemory(),
38
+ });
39
+
40
+ console.log(`Session created with ${discovered.length + 1} context files`);
@@ -0,0 +1,47 @@
1
+ /**
2
+ * Prompt Templates
3
+ *
4
+ * File-based templates that inject content when invoked with /templatename.
5
+ */
6
+
7
+ import {
8
+ createAgentSession,
9
+ DefaultResourceLoader,
10
+ type PromptTemplate,
11
+ SessionManager,
12
+ } from "indusagi-coding-agent";
13
+
14
+ // Define custom templates
15
+ const deployTemplate: PromptTemplate = {
16
+ name: "deploy",
17
+ description: "Deploy the application",
18
+ source: "path",
19
+ filePath: "/virtual/prompts/deploy.md",
20
+ content: `# Deploy Instructions
21
+
22
+ 1. Build: npm run build
23
+ 2. Test: npm test
24
+ 3. Deploy: npm run deploy`,
25
+ };
26
+
27
+ const loader = new DefaultResourceLoader({
28
+ promptsOverride: (current) => ({
29
+ prompts: [...current.prompts, deployTemplate],
30
+ diagnostics: current.diagnostics,
31
+ }),
32
+ });
33
+ await loader.reload();
34
+
35
+ // Discover templates from cwd/.indusagi/prompts/ and ~/.indusagi/agent/prompts/
36
+ const discovered = loader.getPrompts().prompts;
37
+ console.log("Discovered prompt templates:");
38
+ for (const template of discovered) {
39
+ console.log(` /${template.name}: ${template.description}`);
40
+ }
41
+
42
+ await createAgentSession({
43
+ resourceLoader: loader,
44
+ sessionManager: SessionManager.inMemory(),
45
+ });
46
+
47
+ console.log(`Session created with ${discovered.length + 1} prompt templates`);
@@ -0,0 +1,48 @@
1
+ /**
2
+ * API Keys and OAuth
3
+ *
4
+ * Configure API key resolution via AuthStorage and ModelRegistry.
5
+ */
6
+
7
+ import { AuthStorage, createAgentSession, ModelRegistry, SessionManager } from "indusagi-coding-agent";
8
+
9
+ // Default: AuthStorage uses ~/.indusagi/agent/auth.json
10
+ // ModelRegistry loads built-in + custom models from ~/.indusagi/agent/models.json
11
+ const authStorage = new AuthStorage();
12
+ const modelRegistry = new ModelRegistry(authStorage);
13
+
14
+ await createAgentSession({
15
+ sessionManager: SessionManager.inMemory(),
16
+ authStorage,
17
+ modelRegistry,
18
+ });
19
+ console.log("Session with default auth storage and model registry");
20
+
21
+ // Custom auth storage location
22
+ const customAuthStorage = new AuthStorage("/tmp/my-app/auth.json");
23
+ const customModelRegistry = new ModelRegistry(customAuthStorage, "/tmp/my-app/models.json");
24
+
25
+ await createAgentSession({
26
+ sessionManager: SessionManager.inMemory(),
27
+ authStorage: customAuthStorage,
28
+ modelRegistry: customModelRegistry,
29
+ });
30
+ console.log("Session with custom auth storage location");
31
+
32
+ // Runtime API key override (not persisted to disk)
33
+ authStorage.setRuntimeApiKey("anthropic", "sk-my-temp-key");
34
+ await createAgentSession({
35
+ sessionManager: SessionManager.inMemory(),
36
+ authStorage,
37
+ modelRegistry,
38
+ });
39
+ console.log("Session with runtime API key override");
40
+
41
+ // No models.json - only built-in models
42
+ const simpleRegistry = new ModelRegistry(authStorage); // null = no models.json
43
+ await createAgentSession({
44
+ sessionManager: SessionManager.inMemory(),
45
+ authStorage,
46
+ modelRegistry: simpleRegistry,
47
+ });
48
+ console.log("Session with only built-in models");
@@ -0,0 +1,38 @@
1
+ /**
2
+ * Settings Configuration
3
+ *
4
+ * Override settings using SettingsManager.
5
+ */
6
+
7
+ import { createAgentSession, SessionManager, SettingsManager } from "indusagi-coding-agent";
8
+
9
+ // Load current settings (merged global + project)
10
+ const settingsManagerFromDisk = SettingsManager.create();
11
+ console.log("Current settings:", JSON.stringify(settingsManagerFromDisk.getGlobalSettings(), null, 2));
12
+
13
+ // Override specific settings
14
+ const settingsManager = SettingsManager.create();
15
+ settingsManager.applyOverrides({
16
+ compaction: { enabled: false },
17
+ retry: { enabled: true, maxRetries: 5, baseDelayMs: 1000 },
18
+ });
19
+
20
+ await createAgentSession({
21
+ settingsManager,
22
+ sessionManager: SessionManager.inMemory(),
23
+ });
24
+
25
+ console.log("Session created with custom settings");
26
+
27
+ // For testing without file I/O:
28
+ const inMemorySettings = SettingsManager.inMemory({
29
+ compaction: { enabled: false },
30
+ retry: { enabled: false },
31
+ });
32
+
33
+ await createAgentSession({
34
+ settingsManager: inMemorySettings,
35
+ sessionManager: SessionManager.inMemory(),
36
+ });
37
+
38
+ console.log("Test session created with in-memory settings");
@@ -0,0 +1,48 @@
1
+ /**
2
+ * Session Management
3
+ *
4
+ * Control session persistence: in-memory, new file, continue, or open specific.
5
+ */
6
+
7
+ import { createAgentSession, SessionManager } from "indusagi-coding-agent";
8
+
9
+ // In-memory (no persistence)
10
+ const { session: inMemory } = await createAgentSession({
11
+ sessionManager: SessionManager.inMemory(),
12
+ });
13
+ console.log("In-memory session:", inMemory.sessionFile ?? "(none)");
14
+
15
+ // New persistent session
16
+ const { session: newSession } = await createAgentSession({
17
+ sessionManager: SessionManager.create(process.cwd()),
18
+ });
19
+ console.log("New session file:", newSession.sessionFile);
20
+
21
+ // Continue most recent session (or create new if none)
22
+ const { session: continued, modelFallbackMessage } = await createAgentSession({
23
+ sessionManager: SessionManager.continueRecent(process.cwd()),
24
+ });
25
+ if (modelFallbackMessage) console.log("Note:", modelFallbackMessage);
26
+ console.log("Continued session:", continued.sessionFile);
27
+
28
+ // List and open specific session
29
+ const sessions = await SessionManager.list(process.cwd());
30
+ console.log(`\nFound ${sessions.length} sessions:`);
31
+ for (const info of sessions.slice(0, 3)) {
32
+ console.log(` ${info.id.slice(0, 8)}... - "${info.firstMessage.slice(0, 30)}..."`);
33
+ }
34
+
35
+ if (sessions.length > 0) {
36
+ const { session: opened } = await createAgentSession({
37
+ sessionManager: SessionManager.open(sessions[0].path),
38
+ });
39
+ console.log(`\nOpened: ${opened.sessionId}`);
40
+ }
41
+
42
+ // Custom session directory (no cwd encoding)
43
+ // const customDir = "/path/to/my-sessions";
44
+ // const { session } = await createAgentSession({
45
+ // sessionManager: SessionManager.create(process.cwd(), customDir),
46
+ // });
47
+ // SessionManager.list(process.cwd(), customDir);
48
+ // SessionManager.continueRecent(process.cwd(), customDir);
@@ -0,0 +1,82 @@
1
+ /**
2
+ * Full Control
3
+ *
4
+ * Replace everything - no discovery, explicit configuration.
5
+ *
6
+ * IMPORTANT: When providing `tools` with a custom `cwd`, use the tool factory
7
+ * functions (createReadTool, createBashTool, etc.) to ensure tools resolve
8
+ * paths relative to your cwd.
9
+ */
10
+
11
+ import { getModel } from "indusagi/ai";
12
+ import {
13
+ AuthStorage,
14
+ createAgentSession,
15
+ createBashTool,
16
+ createExtensionRuntime,
17
+ createReadTool,
18
+ ModelRegistry,
19
+ type ResourceLoader,
20
+ SessionManager,
21
+ SettingsManager,
22
+ } from "indusagi-coding-agent";
23
+
24
+ // Custom auth storage location
25
+ const authStorage = new AuthStorage("/tmp/my-agent/auth.json");
26
+
27
+ // Runtime API key override (not persisted)
28
+ if (process.env.MY_ANTHROPIC_KEY) {
29
+ authStorage.setRuntimeApiKey("anthropic", process.env.MY_ANTHROPIC_KEY);
30
+ }
31
+
32
+ // Model registry with no custom models.json
33
+ const modelRegistry = new ModelRegistry(authStorage);
34
+
35
+ const model = getModel("anthropic", "claude-sonnet-4-20250514");
36
+ if (!model) throw new Error("Model not found");
37
+
38
+ // In-memory settings with overrides
39
+ const settingsManager = SettingsManager.inMemory({
40
+ compaction: { enabled: false },
41
+ retry: { enabled: true, maxRetries: 2 },
42
+ });
43
+
44
+ // When using a custom cwd with explicit tools, use the factory functions
45
+ const cwd = process.cwd();
46
+
47
+ const resourceLoader: ResourceLoader = {
48
+ getExtensions: () => ({ extensions: [], errors: [], runtime: createExtensionRuntime() }),
49
+ getSkills: () => ({ skills: [], diagnostics: [] }),
50
+ getPrompts: () => ({ prompts: [], diagnostics: [] }),
51
+ getThemes: () => ({ themes: [], diagnostics: [] }),
52
+ getAgentsFiles: () => ({ agentsFiles: [] }),
53
+ getSystemPrompt: () => `You are a minimal assistant.
54
+ Available: read, bash. Be concise.`,
55
+ getAppendSystemPrompt: () => [],
56
+ getPathMetadata: () => new Map(),
57
+ reload: async () => {},
58
+ };
59
+
60
+ const { session } = await createAgentSession({
61
+ cwd,
62
+ agentDir: "/tmp/my-agent",
63
+ model,
64
+ thinkingLevel: "off",
65
+ authStorage,
66
+ modelRegistry,
67
+ resourceLoader,
68
+ // Use factory functions with the same cwd to ensure path resolution works correctly
69
+ tools: [createReadTool(cwd), createBashTool(cwd)],
70
+ sessionManager: SessionManager.inMemory(),
71
+ settingsManager,
72
+ });
73
+
74
+ session.subscribe((event) => {
75
+ if (event.type === "message_update" && event.assistantMessageEvent.type === "text_delta") {
76
+ process.stdout.write(event.assistantMessageEvent.delta);
77
+ }
78
+ });
79
+
80
+ await session.prompt("List files in the current directory.");
81
+ console.log();
82
+
@@ -0,0 +1,37 @@
1
+ /**
2
+ * Codex OAuth Example
3
+ *
4
+ * Requires login via the CLI:
5
+ * indusagi /login (select ChatGPT Plus/Pro - Codex)
6
+ *
7
+ * This stores OAuth tokens in ~/.indusagi/agent/auth.json which AuthStorage reads.
8
+ */
9
+
10
+ import { getModel } from "indusagi/ai";
11
+ import { AuthStorage, createAgentSession, ModelRegistry, SessionManager } from "indusagi-coding-agent";
12
+
13
+ const authStorage = new AuthStorage();
14
+ const modelRegistry = new ModelRegistry(authStorage);
15
+
16
+ const model = getModel("openai-codex", "gpt-5.2-codex") ?? getModel("openai-codex", "gpt-5.1-codex-mini");
17
+ if (!model) {
18
+ throw new Error("Codex model not found. Check your indusagi/ai models list.");
19
+ }
20
+
21
+ const { session } = await createAgentSession({
22
+ model,
23
+ thinkingLevel: "medium",
24
+ sessionManager: SessionManager.inMemory(),
25
+ authStorage,
26
+ modelRegistry,
27
+ });
28
+
29
+ session.subscribe((event) => {
30
+ if (event.type === "message_update" && event.assistantMessageEvent.type === "text_delta") {
31
+ process.stdout.write(event.assistantMessageEvent.delta);
32
+ }
33
+ });
34
+
35
+ await session.prompt("Write a short plan for a CLI todo app.");
36
+ console.log();
37
+