@draht/coding-agent 2026.3.6 → 2026.3.14

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 (177) hide show
  1. package/CHANGELOG.md +88 -0
  2. package/README.md +6 -2
  3. package/bin/draht-tools.cjs +187 -32
  4. package/dist/cli/args.d.ts +6 -0
  5. package/dist/cli/args.d.ts.map +1 -1
  6. package/dist/cli/args.js +24 -0
  7. package/dist/cli/args.js.map +1 -1
  8. package/dist/cli/attach-mode.d.ts +13 -0
  9. package/dist/cli/attach-mode.d.ts.map +1 -0
  10. package/dist/cli/attach-mode.js +97 -0
  11. package/dist/cli/attach-mode.js.map +1 -0
  12. package/dist/cli/list-sessions.d.ts +8 -0
  13. package/dist/cli/list-sessions.d.ts.map +1 -0
  14. package/dist/cli/list-sessions.js +52 -0
  15. package/dist/cli/list-sessions.js.map +1 -0
  16. package/dist/config.d.ts.map +1 -1
  17. package/dist/config.js +2 -2
  18. package/dist/config.js.map +1 -1
  19. package/dist/core/agent-session.d.ts +1 -0
  20. package/dist/core/agent-session.d.ts.map +1 -1
  21. package/dist/core/agent-session.js +50 -17
  22. package/dist/core/agent-session.js.map +1 -1
  23. package/dist/core/auth-storage.d.ts +2 -1
  24. package/dist/core/auth-storage.d.ts.map +1 -1
  25. package/dist/core/auth-storage.js +25 -1
  26. package/dist/core/auth-storage.js.map +1 -1
  27. package/dist/core/compaction/utils.d.ts +3 -0
  28. package/dist/core/compaction/utils.d.ts.map +1 -1
  29. package/dist/core/compaction/utils.js +16 -1
  30. package/dist/core/compaction/utils.js.map +1 -1
  31. package/dist/core/export-html/index.d.ts +5 -2
  32. package/dist/core/export-html/index.d.ts.map +1 -1
  33. package/dist/core/export-html/index.js +4 -3
  34. package/dist/core/export-html/index.js.map +1 -1
  35. package/dist/core/export-html/template.js +11 -14
  36. package/dist/core/export-html/tool-renderer.d.ts +5 -2
  37. package/dist/core/export-html/tool-renderer.d.ts.map +1 -1
  38. package/dist/core/export-html/tool-renderer.js +12 -5
  39. package/dist/core/export-html/tool-renderer.js.map +1 -1
  40. package/dist/core/extensions/index.d.ts +1 -1
  41. package/dist/core/extensions/index.d.ts.map +1 -1
  42. package/dist/core/extensions/index.js.map +1 -1
  43. package/dist/core/extensions/loader.d.ts.map +1 -1
  44. package/dist/core/extensions/loader.js +6 -6
  45. package/dist/core/extensions/loader.js.map +1 -1
  46. package/dist/core/extensions/runner.d.ts +3 -2
  47. package/dist/core/extensions/runner.d.ts.map +1 -1
  48. package/dist/core/extensions/runner.js +32 -0
  49. package/dist/core/extensions/runner.js.map +1 -1
  50. package/dist/core/extensions/types.d.ts +21 -2
  51. package/dist/core/extensions/types.d.ts.map +1 -1
  52. package/dist/core/extensions/types.js.map +1 -1
  53. package/dist/core/model-resolver.d.ts.map +1 -1
  54. package/dist/core/model-resolver.js +2 -2
  55. package/dist/core/model-resolver.js.map +1 -1
  56. package/dist/core/package-manager.d.ts.map +1 -1
  57. package/dist/core/package-manager.js +2 -2
  58. package/dist/core/package-manager.js.map +1 -1
  59. package/dist/core/resource-loader.d.ts.map +1 -1
  60. package/dist/core/resource-loader.js +1 -1
  61. package/dist/core/resource-loader.js.map +1 -1
  62. package/dist/core/sdk.d.ts.map +1 -1
  63. package/dist/core/sdk.js +7 -0
  64. package/dist/core/sdk.js.map +1 -1
  65. package/dist/core/settings-manager.d.ts +4 -0
  66. package/dist/core/settings-manager.d.ts.map +1 -1
  67. package/dist/core/settings-manager.js +36 -2
  68. package/dist/core/settings-manager.js.map +1 -1
  69. package/dist/core/socket-server/discovery.d.ts +19 -0
  70. package/dist/core/socket-server/discovery.d.ts.map +1 -0
  71. package/dist/core/socket-server/discovery.js +91 -0
  72. package/dist/core/socket-server/discovery.js.map +1 -0
  73. package/dist/core/socket-server/index.d.ts +13 -0
  74. package/dist/core/socket-server/index.d.ts.map +1 -0
  75. package/dist/core/socket-server/index.js +11 -0
  76. package/dist/core/socket-server/index.js.map +1 -0
  77. package/dist/core/socket-server/session-integration.d.ts +17 -0
  78. package/dist/core/socket-server/session-integration.d.ts.map +1 -0
  79. package/dist/core/socket-server/session-integration.js +77 -0
  80. package/dist/core/socket-server/session-integration.js.map +1 -0
  81. package/dist/core/socket-server/socket-client.d.ts +65 -0
  82. package/dist/core/socket-server/socket-client.d.ts.map +1 -0
  83. package/dist/core/socket-server/socket-client.js +197 -0
  84. package/dist/core/socket-server/socket-client.js.map +1 -0
  85. package/dist/core/socket-server/socket-server.d.ts +60 -0
  86. package/dist/core/socket-server/socket-server.d.ts.map +1 -0
  87. package/dist/core/socket-server/socket-server.js +273 -0
  88. package/dist/core/socket-server/socket-server.js.map +1 -0
  89. package/dist/core/socket-server/types.d.ts +81 -0
  90. package/dist/core/socket-server/types.d.ts.map +1 -0
  91. package/dist/core/socket-server/types.js +8 -0
  92. package/dist/core/socket-server/types.js.map +1 -0
  93. package/dist/gsd/domain.d.ts +5 -1
  94. package/dist/gsd/domain.d.ts.map +1 -1
  95. package/dist/gsd/domain.js +71 -1
  96. package/dist/gsd/domain.js.map +1 -1
  97. package/dist/gsd/git.d.ts.map +1 -1
  98. package/dist/gsd/git.js +18 -0
  99. package/dist/gsd/git.js.map +1 -1
  100. package/dist/gsd/index.d.ts +1 -0
  101. package/dist/gsd/index.d.ts.map +1 -1
  102. package/dist/gsd/index.js.map +1 -1
  103. package/dist/index.d.ts +1 -1
  104. package/dist/index.d.ts.map +1 -1
  105. package/dist/index.js.map +1 -1
  106. package/dist/main.d.ts.map +1 -1
  107. package/dist/main.js +76 -11
  108. package/dist/main.js.map +1 -1
  109. package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -1
  110. package/dist/modes/interactive/components/extension-editor.js +1 -0
  111. package/dist/modes/interactive/components/extension-editor.js.map +1 -1
  112. package/dist/modes/interactive/components/footer.d.ts.map +1 -1
  113. package/dist/modes/interactive/components/footer.js +8 -23
  114. package/dist/modes/interactive/components/footer.js.map +1 -1
  115. package/dist/modes/interactive/components/settings-selector.d.ts +2 -0
  116. package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
  117. package/dist/modes/interactive/components/settings-selector.js +10 -0
  118. package/dist/modes/interactive/components/settings-selector.js.map +1 -1
  119. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  120. package/dist/modes/interactive/components/tool-execution.js +14 -4
  121. package/dist/modes/interactive/components/tool-execution.js.map +1 -1
  122. package/dist/modes/interactive/components/tree-selector.d.ts +21 -2
  123. package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -1
  124. package/dist/modes/interactive/components/tree-selector.js +115 -9
  125. package/dist/modes/interactive/components/tree-selector.js.map +1 -1
  126. package/dist/modes/interactive/interactive-mode.d.ts +1 -0
  127. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  128. package/dist/modes/interactive/interactive-mode.js +66 -5
  129. package/dist/modes/interactive/interactive-mode.js.map +1 -1
  130. package/dist/modes/rpc/jsonl.d.ts +17 -0
  131. package/dist/modes/rpc/jsonl.d.ts.map +1 -0
  132. package/dist/modes/rpc/jsonl.js +49 -0
  133. package/dist/modes/rpc/jsonl.js.map +1 -0
  134. package/dist/modes/rpc/rpc-client.d.ts +1 -1
  135. package/dist/modes/rpc/rpc-client.d.ts.map +1 -1
  136. package/dist/modes/rpc/rpc-client.js +7 -11
  137. package/dist/modes/rpc/rpc-client.js.map +1 -1
  138. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  139. package/dist/modes/rpc/rpc-mode.js +9 -11
  140. package/dist/modes/rpc/rpc-mode.js.map +1 -1
  141. package/dist/prompts/commands/execute-phase.md +2 -2
  142. package/dist/prompts/commands/fix.md +2 -2
  143. package/dist/prompts/commands/plan-phase.md +5 -1
  144. package/dist/prompts/commands/quick.md +5 -1
  145. package/dist/utils/changelog.d.ts +12 -0
  146. package/dist/utils/changelog.d.ts.map +1 -1
  147. package/dist/utils/changelog.js +25 -14
  148. package/dist/utils/changelog.js.map +1 -1
  149. package/dist/utils/notify.d.ts +12 -0
  150. package/dist/utils/notify.d.ts.map +1 -0
  151. package/dist/utils/notify.js +41 -0
  152. package/dist/utils/notify.js.map +1 -0
  153. package/docs/compaction.md +2 -0
  154. package/docs/custom-provider.md +11 -7
  155. package/docs/extensions.md +55 -3
  156. package/docs/keybindings.md +9 -1
  157. package/docs/models.md +5 -1
  158. package/docs/rpc.md +40 -3
  159. package/docs/session.md +2 -2
  160. package/docs/settings.md +1 -0
  161. package/docs/terminal-setup.md +28 -3
  162. package/docs/tmux.md +61 -0
  163. package/docs/tree.md +9 -0
  164. package/examples/extensions/antigravity-image-gen.ts +5 -4
  165. package/examples/extensions/custom-provider-gitlab-duo/test.ts +2 -2
  166. package/examples/extensions/notify.ts +9 -2
  167. package/examples/extensions/overlay-qa-tests.ts +468 -1
  168. package/examples/extensions/preset.ts +2 -3
  169. package/examples/extensions/provider-payload.ts +14 -0
  170. package/examples/extensions/sandbox/index.ts +2 -3
  171. package/examples/extensions/tool-override.ts +2 -3
  172. package/examples/extensions/with-deps/index.ts +1 -5
  173. package/package.json +7 -5
  174. package/prompts/commands/execute-phase.md +2 -2
  175. package/prompts/commands/fix.md +2 -2
  176. package/prompts/commands/plan-phase.md +5 -1
  177. package/prompts/commands/quick.md +5 -1
@@ -1 +1 @@
1
- {"version":3,"file":"main.d.ts","sourceRoot":"","sources":["../src/main.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AA6rBH,wBAAsB,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,iBAmPxC","sourcesContent":["/**\n * Main entry point for the coding agent CLI.\n *\n * This file handles CLI argument parsing and translates them into\n * createAgentSession() options. The SDK does the heavy lifting.\n */\n\nimport { type ImageContent, modelsAreEqual, supportsXhigh } from \"@draht/ai\";\nimport chalk from \"chalk\";\nimport { createInterface } from \"readline\";\nimport { type Args, parseArgs, printHelp } from \"./cli/args.js\";\nimport { selectConfig } from \"./cli/config-selector.js\";\nimport { processFileArguments } from \"./cli/file-processor.js\";\nimport { listModels } from \"./cli/list-models.js\";\nimport { selectSession } from \"./cli/session-picker.js\";\nimport { APP_NAME, getAgentDir, getModelsPath, VERSION } from \"./config.js\";\nimport { AuthStorage } from \"./core/auth-storage.js\";\nimport { exportFromFile } from \"./core/export-html/index.js\";\nimport type { LoadExtensionsResult } from \"./core/extensions/index.js\";\nimport { KeybindingsManager } from \"./core/keybindings.js\";\nimport { ModelRegistry } from \"./core/model-registry.js\";\nimport { resolveCliModel, resolveModelScope, type ScopedModel } from \"./core/model-resolver.js\";\nimport { DefaultPackageManager } from \"./core/package-manager.js\";\nimport { DefaultResourceLoader } from \"./core/resource-loader.js\";\nimport { type CreateAgentSessionOptions, createAgentSession } from \"./core/sdk.js\";\nimport { SessionManager } from \"./core/session-manager.js\";\nimport { SettingsManager } from \"./core/settings-manager.js\";\nimport { printTimings, time } from \"./core/timings.js\";\nimport { allTools } from \"./core/tools/index.js\";\nimport { runMigrations, showDeprecationWarnings } from \"./migrations.js\";\nimport { InteractiveMode, runPrintMode, runRpcMode } from \"./modes/index.js\";\nimport { initTheme, stopThemeWatcher } from \"./modes/interactive/theme/theme.js\";\n\n/**\n * Read all content from piped stdin.\n * Returns undefined if stdin is a TTY (interactive terminal).\n */\nasync function readPipedStdin(): Promise<string | undefined> {\n\t// If stdin is a TTY, we're running interactively - don't read stdin\n\tif (process.stdin.isTTY) {\n\t\treturn undefined;\n\t}\n\n\treturn new Promise((resolve) => {\n\t\tlet data = \"\";\n\t\tprocess.stdin.setEncoding(\"utf8\");\n\t\tprocess.stdin.on(\"data\", (chunk) => {\n\t\t\tdata += chunk;\n\t\t});\n\t\tprocess.stdin.on(\"end\", () => {\n\t\t\tresolve(data.trim() || undefined);\n\t\t});\n\t\tprocess.stdin.resume();\n\t});\n}\n\nfunction reportSettingsErrors(settingsManager: SettingsManager, context: string): void {\n\tconst errors = settingsManager.drainErrors();\n\tfor (const { scope, error } of errors) {\n\t\tconsole.error(chalk.yellow(`Warning (${context}, ${scope} settings): ${error.message}`));\n\t\tif (error.stack) {\n\t\t\tconsole.error(chalk.dim(error.stack));\n\t\t}\n\t}\n}\n\nfunction isTruthyEnvFlag(value: string | undefined): boolean {\n\tif (!value) return false;\n\treturn value === \"1\" || value.toLowerCase() === \"true\" || value.toLowerCase() === \"yes\";\n}\n\ntype PackageCommand = \"install\" | \"remove\" | \"update\" | \"list\";\n\ninterface PackageCommandOptions {\n\tcommand: PackageCommand;\n\tsource?: string;\n\tlocal: boolean;\n\thelp: boolean;\n\tinvalidOption?: string;\n}\n\nfunction getPackageCommandUsage(command: PackageCommand): string {\n\tswitch (command) {\n\t\tcase \"install\":\n\t\t\treturn `${APP_NAME} install <source> [-l]`;\n\t\tcase \"remove\":\n\t\t\treturn `${APP_NAME} remove <source> [-l]`;\n\t\tcase \"update\":\n\t\t\treturn `${APP_NAME} update [source]`;\n\t\tcase \"list\":\n\t\t\treturn `${APP_NAME} list`;\n\t}\n}\n\nfunction printPackageCommandHelp(command: PackageCommand): void {\n\tswitch (command) {\n\t\tcase \"install\":\n\t\t\tconsole.log(`${chalk.bold(\"Usage:\")}\n ${getPackageCommandUsage(\"install\")}\n\nInstall a package and add it to settings.\n\nOptions:\n -l, --local Install project-locally (.draht/settings.json)\n\nExamples:\n ${APP_NAME} install npm:@foo/bar\n ${APP_NAME} install git:github.com/user/repo\n ${APP_NAME} install git:git@github.com:user/repo\n ${APP_NAME} install https://github.com/user/repo\n ${APP_NAME} install ssh://git@github.com/user/repo\n ${APP_NAME} install ./local/path\n`);\n\t\t\treturn;\n\n\t\tcase \"remove\":\n\t\t\tconsole.log(`${chalk.bold(\"Usage:\")}\n ${getPackageCommandUsage(\"remove\")}\n\nRemove a package and its source from settings.\n\nOptions:\n -l, --local Remove from project settings (.draht/settings.json)\n\nExample:\n ${APP_NAME} remove npm:@foo/bar\n`);\n\t\t\treturn;\n\n\t\tcase \"update\":\n\t\t\tconsole.log(`${chalk.bold(\"Usage:\")}\n ${getPackageCommandUsage(\"update\")}\n\nUpdate installed packages.\nIf <source> is provided, only that package is updated.\n`);\n\t\t\treturn;\n\n\t\tcase \"list\":\n\t\t\tconsole.log(`${chalk.bold(\"Usage:\")}\n ${getPackageCommandUsage(\"list\")}\n\nList installed packages from user and project settings.\n`);\n\t\t\treturn;\n\t}\n}\n\nfunction parsePackageCommand(args: string[]): PackageCommandOptions | undefined {\n\tconst [command, ...rest] = args;\n\tif (command !== \"install\" && command !== \"remove\" && command !== \"update\" && command !== \"list\") {\n\t\treturn undefined;\n\t}\n\n\tlet local = false;\n\tlet help = false;\n\tlet invalidOption: string | undefined;\n\tlet source: string | undefined;\n\n\tfor (const arg of rest) {\n\t\tif (arg === \"-h\" || arg === \"--help\") {\n\t\t\thelp = true;\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (arg === \"-l\" || arg === \"--local\") {\n\t\t\tif (command === \"install\" || command === \"remove\") {\n\t\t\t\tlocal = true;\n\t\t\t} else {\n\t\t\t\tinvalidOption = invalidOption ?? arg;\n\t\t\t}\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (arg.startsWith(\"-\")) {\n\t\t\tinvalidOption = invalidOption ?? arg;\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (!source) {\n\t\t\tsource = arg;\n\t\t}\n\t}\n\n\treturn { command, source, local, help, invalidOption };\n}\n\nasync function handlePackageCommand(args: string[]): Promise<boolean> {\n\tconst options = parsePackageCommand(args);\n\tif (!options) {\n\t\treturn false;\n\t}\n\n\tif (options.help) {\n\t\tprintPackageCommandHelp(options.command);\n\t\treturn true;\n\t}\n\n\tif (options.invalidOption) {\n\t\tconsole.error(chalk.red(`Unknown option ${options.invalidOption} for \"${options.command}\".`));\n\t\tconsole.error(chalk.dim(`Use \"${APP_NAME} --help\" or \"${getPackageCommandUsage(options.command)}\".`));\n\t\tprocess.exitCode = 1;\n\t\treturn true;\n\t}\n\n\tconst source = options.source;\n\tif ((options.command === \"install\" || options.command === \"remove\") && !source) {\n\t\tconsole.error(chalk.red(`Missing ${options.command} source.`));\n\t\tconsole.error(chalk.dim(`Usage: ${getPackageCommandUsage(options.command)}`));\n\t\tprocess.exitCode = 1;\n\t\treturn true;\n\t}\n\n\tconst cwd = process.cwd();\n\tconst agentDir = getAgentDir();\n\tconst settingsManager = SettingsManager.create(cwd, agentDir);\n\treportSettingsErrors(settingsManager, \"package command\");\n\tconst packageManager = new DefaultPackageManager({ cwd, agentDir, settingsManager });\n\n\tpackageManager.setProgressCallback((event) => {\n\t\tif (event.type === \"start\") {\n\t\t\tprocess.stdout.write(chalk.dim(`${event.message}\\n`));\n\t\t}\n\t});\n\n\ttry {\n\t\tswitch (options.command) {\n\t\t\tcase \"install\":\n\t\t\t\tawait packageManager.install(source!, { local: options.local });\n\t\t\t\tpackageManager.addSourceToSettings(source!, { local: options.local });\n\t\t\t\tconsole.log(chalk.green(`Installed ${source}`));\n\t\t\t\treturn true;\n\n\t\t\tcase \"remove\": {\n\t\t\t\tawait packageManager.remove(source!, { local: options.local });\n\t\t\t\tconst removed = packageManager.removeSourceFromSettings(source!, { local: options.local });\n\t\t\t\tif (!removed) {\n\t\t\t\t\tconsole.error(chalk.red(`No matching package found for ${source}`));\n\t\t\t\t\tprocess.exitCode = 1;\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\tconsole.log(chalk.green(`Removed ${source}`));\n\t\t\t\treturn true;\n\t\t\t}\n\n\t\t\tcase \"list\": {\n\t\t\t\tconst globalSettings = settingsManager.getGlobalSettings();\n\t\t\t\tconst projectSettings = settingsManager.getProjectSettings();\n\t\t\t\tconst globalPackages = globalSettings.packages ?? [];\n\t\t\t\tconst projectPackages = projectSettings.packages ?? [];\n\n\t\t\t\tif (globalPackages.length === 0 && projectPackages.length === 0) {\n\t\t\t\t\tconsole.log(chalk.dim(\"No packages installed.\"));\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\tconst formatPackage = (pkg: (typeof globalPackages)[number], scope: \"user\" | \"project\") => {\n\t\t\t\t\tconst source = typeof pkg === \"string\" ? pkg : pkg.source;\n\t\t\t\t\tconst filtered = typeof pkg === \"object\";\n\t\t\t\t\tconst display = filtered ? `${source} (filtered)` : source;\n\t\t\t\t\tconsole.log(` ${display}`);\n\t\t\t\t\tconst path = packageManager.getInstalledPath(source, scope);\n\t\t\t\t\tif (path) {\n\t\t\t\t\t\tconsole.log(chalk.dim(` ${path}`));\n\t\t\t\t\t}\n\t\t\t\t};\n\n\t\t\t\tif (globalPackages.length > 0) {\n\t\t\t\t\tconsole.log(chalk.bold(\"User packages:\"));\n\t\t\t\t\tfor (const pkg of globalPackages) {\n\t\t\t\t\t\tformatPackage(pkg, \"user\");\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif (projectPackages.length > 0) {\n\t\t\t\t\tif (globalPackages.length > 0) console.log();\n\t\t\t\t\tconsole.log(chalk.bold(\"Project packages:\"));\n\t\t\t\t\tfor (const pkg of projectPackages) {\n\t\t\t\t\t\tformatPackage(pkg, \"project\");\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\treturn true;\n\t\t\t}\n\n\t\t\tcase \"update\":\n\t\t\t\tawait packageManager.update(source);\n\t\t\t\tif (source) {\n\t\t\t\t\tconsole.log(chalk.green(`Updated ${source}`));\n\t\t\t\t} else {\n\t\t\t\t\tconsole.log(chalk.green(\"Updated packages\"));\n\t\t\t\t}\n\t\t\t\treturn true;\n\t\t}\n\t} catch (error: unknown) {\n\t\tconst message = error instanceof Error ? error.message : \"Unknown package command error\";\n\t\tconsole.error(chalk.red(`Error: ${message}`));\n\t\tprocess.exitCode = 1;\n\t\treturn true;\n\t}\n}\n\nasync function prepareInitialMessage(\n\tparsed: Args,\n\tautoResizeImages: boolean,\n): Promise<{\n\tinitialMessage?: string;\n\tinitialImages?: ImageContent[];\n}> {\n\tif (parsed.fileArgs.length === 0) {\n\t\treturn {};\n\t}\n\n\tconst { text, images } = await processFileArguments(parsed.fileArgs, { autoResizeImages });\n\n\tlet initialMessage: string;\n\tif (parsed.messages.length > 0) {\n\t\tinitialMessage = text + parsed.messages[0];\n\t\tparsed.messages.shift();\n\t} else {\n\t\tinitialMessage = text;\n\t}\n\n\treturn {\n\t\tinitialMessage,\n\t\tinitialImages: images.length > 0 ? images : undefined,\n\t};\n}\n\n/** Result from resolving a session argument */\ntype ResolvedSession =\n\t| { type: \"path\"; path: string } // Direct file path\n\t| { type: \"local\"; path: string } // Found in current project\n\t| { type: \"global\"; path: string; cwd: string } // Found in different project\n\t| { type: \"not_found\"; arg: string }; // Not found anywhere\n\n/**\n * Resolve a session argument to a file path.\n * If it looks like a path, use as-is. Otherwise try to match as session ID prefix.\n */\nasync function resolveSessionPath(sessionArg: string, cwd: string, sessionDir?: string): Promise<ResolvedSession> {\n\t// If it looks like a file path, use as-is\n\tif (sessionArg.includes(\"/\") || sessionArg.includes(\"\\\\\") || sessionArg.endsWith(\".jsonl\")) {\n\t\treturn { type: \"path\", path: sessionArg };\n\t}\n\n\t// Try to match as session ID in current project first\n\tconst localSessions = await SessionManager.list(cwd, sessionDir);\n\tconst localMatches = localSessions.filter((s) => s.id.startsWith(sessionArg));\n\n\tif (localMatches.length >= 1) {\n\t\treturn { type: \"local\", path: localMatches[0].path };\n\t}\n\n\t// Try global search across all projects\n\tconst allSessions = await SessionManager.listAll();\n\tconst globalMatches = allSessions.filter((s) => s.id.startsWith(sessionArg));\n\n\tif (globalMatches.length >= 1) {\n\t\tconst match = globalMatches[0];\n\t\treturn { type: \"global\", path: match.path, cwd: match.cwd };\n\t}\n\n\t// Not found anywhere\n\treturn { type: \"not_found\", arg: sessionArg };\n}\n\n/** Prompt user for yes/no confirmation */\nasync function promptConfirm(message: string): Promise<boolean> {\n\treturn new Promise((resolve) => {\n\t\tconst rl = createInterface({\n\t\t\tinput: process.stdin,\n\t\t\toutput: process.stdout,\n\t\t});\n\t\trl.question(`${message} [y/N] `, (answer) => {\n\t\t\trl.close();\n\t\t\tresolve(answer.toLowerCase() === \"y\" || answer.toLowerCase() === \"yes\");\n\t\t});\n\t});\n}\n\nasync function createSessionManager(parsed: Args, cwd: string): Promise<SessionManager | undefined> {\n\tif (parsed.noSession) {\n\t\treturn SessionManager.inMemory();\n\t}\n\tif (parsed.session) {\n\t\tconst resolved = await resolveSessionPath(parsed.session, cwd, parsed.sessionDir);\n\n\t\tswitch (resolved.type) {\n\t\t\tcase \"path\":\n\t\t\tcase \"local\":\n\t\t\t\treturn SessionManager.open(resolved.path, parsed.sessionDir);\n\n\t\t\tcase \"global\": {\n\t\t\t\t// Session found in different project - ask user if they want to fork\n\t\t\t\tconsole.log(chalk.yellow(`Session found in different project: ${resolved.cwd}`));\n\t\t\t\tconst shouldFork = await promptConfirm(\"Fork this session into current directory?\");\n\t\t\t\tif (!shouldFork) {\n\t\t\t\t\tconsole.log(chalk.dim(\"Aborted.\"));\n\t\t\t\t\tprocess.exit(0);\n\t\t\t\t}\n\t\t\t\treturn SessionManager.forkFrom(resolved.path, cwd, parsed.sessionDir);\n\t\t\t}\n\n\t\t\tcase \"not_found\":\n\t\t\t\tconsole.error(chalk.red(`No session found matching '${resolved.arg}'`));\n\t\t\t\tprocess.exit(1);\n\t\t}\n\t}\n\tif (parsed.continue) {\n\t\treturn SessionManager.continueRecent(cwd, parsed.sessionDir);\n\t}\n\t// --resume is handled separately (needs picker UI)\n\t// If --session-dir provided without --continue/--resume, create new session there\n\tif (parsed.sessionDir) {\n\t\treturn SessionManager.create(cwd, parsed.sessionDir);\n\t}\n\t// Default case (new session) returns undefined, SDK will create one\n\treturn undefined;\n}\n\nfunction buildSessionOptions(\n\tparsed: Args,\n\tscopedModels: ScopedModel[],\n\tsessionManager: SessionManager | undefined,\n\tmodelRegistry: ModelRegistry,\n\tsettingsManager: SettingsManager,\n): { options: CreateAgentSessionOptions; cliThinkingFromModel: boolean } {\n\tconst options: CreateAgentSessionOptions = {};\n\tlet cliThinkingFromModel = false;\n\n\tif (sessionManager) {\n\t\toptions.sessionManager = sessionManager;\n\t}\n\n\t// Model from CLI\n\t// - supports --provider <name> --model <pattern>\n\t// - supports --model <provider>/<pattern>\n\tif (parsed.model) {\n\t\tconst resolved = resolveCliModel({\n\t\t\tcliProvider: parsed.provider,\n\t\t\tcliModel: parsed.model,\n\t\t\tmodelRegistry,\n\t\t});\n\t\tif (resolved.warning) {\n\t\t\tconsole.warn(chalk.yellow(`Warning: ${resolved.warning}`));\n\t\t}\n\t\tif (resolved.error) {\n\t\t\tconsole.error(chalk.red(resolved.error));\n\t\t\tprocess.exit(1);\n\t\t}\n\t\tif (resolved.model) {\n\t\t\toptions.model = resolved.model;\n\t\t\t// Allow \"--model <pattern>:<thinking>\" as a shorthand.\n\t\t\t// Explicit --thinking still takes precedence (applied later).\n\t\t\tif (!parsed.thinking && resolved.thinkingLevel) {\n\t\t\t\toptions.thinkingLevel = resolved.thinkingLevel;\n\t\t\t\tcliThinkingFromModel = true;\n\t\t\t}\n\t\t}\n\t}\n\n\tif (!options.model && scopedModels.length > 0 && !parsed.continue && !parsed.resume) {\n\t\t// Check if saved default is in scoped models - use it if so, otherwise first scoped model\n\t\tconst savedProvider = settingsManager.getDefaultProvider();\n\t\tconst savedModelId = settingsManager.getDefaultModel();\n\t\tconst savedModel = savedProvider && savedModelId ? modelRegistry.find(savedProvider, savedModelId) : undefined;\n\t\tconst savedInScope = savedModel ? scopedModels.find((sm) => modelsAreEqual(sm.model, savedModel)) : undefined;\n\n\t\tif (savedInScope) {\n\t\t\toptions.model = savedInScope.model;\n\t\t\t// Use thinking level from scoped model config if explicitly set\n\t\t\tif (!parsed.thinking && savedInScope.thinkingLevel) {\n\t\t\t\toptions.thinkingLevel = savedInScope.thinkingLevel;\n\t\t\t}\n\t\t} else {\n\t\t\toptions.model = scopedModels[0].model;\n\t\t\t// Use thinking level from first scoped model if explicitly set\n\t\t\tif (!parsed.thinking && scopedModels[0].thinkingLevel) {\n\t\t\t\toptions.thinkingLevel = scopedModels[0].thinkingLevel;\n\t\t\t}\n\t\t}\n\t}\n\n\t// Thinking level from CLI (takes precedence over scoped model thinking levels set above)\n\tif (parsed.thinking) {\n\t\toptions.thinkingLevel = parsed.thinking;\n\t}\n\n\t// Scoped models for Ctrl+P cycling\n\t// Keep thinking level undefined when not explicitly set in the model pattern.\n\t// Undefined means \"inherit current session thinking level\" during cycling.\n\tif (scopedModels.length > 0) {\n\t\toptions.scopedModels = scopedModels.map((sm) => ({\n\t\t\tmodel: sm.model,\n\t\t\tthinkingLevel: sm.thinkingLevel,\n\t\t}));\n\t}\n\n\t// API key from CLI - set in authStorage\n\t// (handled by caller before createAgentSession)\n\n\t// Tools\n\tif (parsed.noTools) {\n\t\t// --no-tools: start with no built-in tools\n\t\t// --tools can still add specific ones back\n\t\tif (parsed.tools && parsed.tools.length > 0) {\n\t\t\toptions.tools = parsed.tools.map((name) => allTools[name]);\n\t\t} else {\n\t\t\toptions.tools = [];\n\t\t}\n\t} else if (parsed.tools) {\n\t\toptions.tools = parsed.tools.map((name) => allTools[name]);\n\t}\n\n\treturn { options, cliThinkingFromModel };\n}\n\nasync function handleConfigCommand(args: string[]): Promise<boolean> {\n\tif (args[0] !== \"config\") {\n\t\treturn false;\n\t}\n\n\tconst cwd = process.cwd();\n\tconst agentDir = getAgentDir();\n\tconst settingsManager = SettingsManager.create(cwd, agentDir);\n\treportSettingsErrors(settingsManager, \"config command\");\n\tconst packageManager = new DefaultPackageManager({ cwd, agentDir, settingsManager });\n\n\tconst resolvedPaths = await packageManager.resolve();\n\n\tawait selectConfig({\n\t\tresolvedPaths,\n\t\tsettingsManager,\n\t\tcwd,\n\t\tagentDir,\n\t});\n\n\tprocess.exit(0);\n}\n\nasync function handleLoginCommand(args: string[]): Promise<boolean> {\n\tif (args[0] !== \"login\" && args[0] !== \"--login\") {\n\t\treturn false;\n\t}\n\n\tconst { getOAuthProviders } = await import(\"@draht/ai/oauth\");\n\tconst authMod = await import(\"./core/auth-storage.js\");\n\tconst AuthStorageClass = authMod.AuthStorage;\n\t// open browser - use dynamic import, fallback to exec\n\tconst openBrowser = async (url: string) => {\n\t\ttry {\n\t\t\tconst { exec } = await import(\"child_process\");\n\t\t\texec(`open \"${url}\" 2>/dev/null || xdg-open \"${url}\" 2>/dev/null`);\n\t\t} catch {\n\t\t\t/* ignore */\n\t\t}\n\t};\n\n\tconst providers = getOAuthProviders();\n\tconst requestedProvider = args[1]?.toLowerCase();\n\n\t// Map friendly names to provider IDs\n\tconst PROVIDER_ALIASES: Record<string, string> = {\n\t\tanthropic: \"anthropic\",\n\t\tclaude: \"anthropic\",\n\t\tgoogle: \"google-gemini-cli\",\n\t\tgemini: \"google-gemini-cli\",\n\t\topenai: \"openai-codex\",\n\t\tcodex: \"openai-codex\",\n\t\tcopilot: \"github-copilot\",\n\t\tgithub: \"github-copilot\",\n\t\tantigravity: \"google-antigravity\",\n\t};\n\n\tif (requestedProvider === \"--help\" || requestedProvider === \"-h\" || requestedProvider === \"help\") {\n\t\tconsole.log(chalk.bold(\"\\n🔌 draht login\\n\"));\n\t\tconsole.log(\"Authenticate with AI providers.\\n\");\n\t\tconsole.log(\"Usage:\");\n\t\tconsole.log(\" draht login Interactive provider selection\");\n\t\tconsole.log(\" draht login anthropic Login to Anthropic (Claude Pro/Max)\");\n\t\tconsole.log(\" draht login gemini Login to Google Gemini CLI\");\n\t\tconsole.log(\" draht login openai Login to OpenAI (ChatGPT/Codex)\");\n\t\tconsole.log(\" draht login copilot Login to GitHub Copilot\");\n\t\tconsole.log(\" draht login all Login to all providers\\n\");\n\t\tconsole.log(\"Aliases: claude, google, codex, github\\n\");\n\t\tprocess.exit(0);\n\t}\n\n\tif (requestedProvider && requestedProvider !== \"all\") {\n\t\tconst providerId = PROVIDER_ALIASES[requestedProvider] ?? requestedProvider;\n\t\tconst provider = providers.find((p) => p.id === providerId);\n\t\tif (!provider) {\n\t\t\tconsole.log(chalk.red(`Unknown provider: ${requestedProvider}`));\n\t\t\tconsole.log(`\\nAvailable providers:`);\n\t\t\tfor (const p of providers) {\n\t\t\t\tconst aliases = Object.entries(PROVIDER_ALIASES)\n\t\t\t\t\t.filter(([_, v]) => v === p.id)\n\t\t\t\t\t.map(([k]) => k);\n\t\t\t\tconsole.log(` ${chalk.bold(p.name)} (${p.id}) — aliases: ${aliases.join(\", \")}`);\n\t\t\t}\n\t\t\tprocess.exit(1);\n\t\t}\n\t\tawait loginProvider(provider, openBrowser);\n\t\tprocess.exit(0);\n\t}\n\n\t// Interactive: show menu\n\tconsole.log(chalk.bold(\"\\n🔌 Draht Login\\n\"));\n\tconsole.log(\"Select a provider to authenticate:\\n\");\n\n\tconst providerList = providers.filter((p) =>\n\t\t[\"anthropic\", \"gemini-cli\", \"openai-codex\", \"github-copilot\"].includes(p.id),\n\t);\n\n\tfor (let i = 0; i < providerList.length; i++) {\n\t\tconst p = providerList[i];\n\t\tconst as = AuthStorageClass.create();\n\t\tconst hasCredentials = as.has(p.id);\n\t\tconst status = hasCredentials ? chalk.green(\"✓ logged in\") : chalk.dim(\"not connected\");\n\t\tconsole.log(` ${chalk.bold(i + 1)}. ${p.name} ${status}`);\n\t}\n\tconsole.log(` ${chalk.bold(\"a\")}. Login to all`);\n\tconsole.log(` ${chalk.bold(\"q\")}. Cancel\\n`);\n\n\tconst rl = createInterface({ input: process.stdin, output: process.stdout });\n\tconst answer = await new Promise<string>((resolve) => {\n\t\trl.question(\"Choice: \", (ans) => {\n\t\t\trl.close();\n\t\t\tresolve(ans.trim());\n\t\t});\n\t});\n\n\tif (answer === \"q\" || answer === \"\") {\n\t\tprocess.exit(0);\n\t}\n\n\tif (answer === \"a\") {\n\t\tfor (const p of providerList) {\n\t\t\tawait loginProvider(p, openBrowser);\n\t\t}\n\t} else {\n\t\tconst idx = parseInt(answer, 10) - 1;\n\t\tif (idx >= 0 && idx < providerList.length) {\n\t\t\tawait loginProvider(providerList[idx], openBrowser);\n\t\t} else {\n\t\t\tconsole.log(chalk.red(\"Invalid choice\"));\n\t\t}\n\t}\n\n\tprocess.exit(0);\n}\n\nasync function loginProvider(\n\tprovider: { id: string; name: string; login: (cb: any) => Promise<any>; usesCallbackServer?: boolean },\n\topenBrowser: (url: string) => Promise<any>,\n): Promise<void> {\n\tconst { AuthStorage } = await import(\"./core/auth-storage.js\");\n\tconst authStorage = AuthStorage.create();\n\n\tconsole.log(`\\n${chalk.bold(`Logging in to ${provider.name}...`)}`);\n\n\ttry {\n\t\tconst credentials = await provider.login({\n\t\t\tonAuth: (info: { url: string; instructions?: string }) => {\n\t\t\t\tconsole.log(`\\n${chalk.blue(\"→\")} Opening browser for authentication...`);\n\t\t\t\tif (info.instructions) console.log(chalk.dim(info.instructions));\n\t\t\t\topenBrowser(info.url).catch(() => {\n\t\t\t\t\tconsole.log(`\\n${chalk.yellow(\"Could not open browser. Visit manually:\")}`);\n\t\t\t\t\tconsole.log(chalk.underline(info.url));\n\t\t\t\t});\n\t\t\t},\n\t\t\tonPrompt: async (prompt: { message: string; placeholder?: string }) => {\n\t\t\t\tconst { createInterface } = await import(\"readline\");\n\t\t\t\tconst rl = createInterface({ input: process.stdin, output: process.stdout });\n\t\t\t\treturn new Promise<string>((resolve) => {\n\t\t\t\t\trl.question(`${prompt.message} `, (ans: string) => {\n\t\t\t\t\t\trl.close();\n\t\t\t\t\t\tresolve(ans.trim());\n\t\t\t\t\t});\n\t\t\t\t});\n\t\t\t},\n\t\t\tonProgress: (message: string) => {\n\t\t\t\tconsole.log(chalk.dim(` ${message}`));\n\t\t\t},\n\t\t\tonManualCodeInput: provider.usesCallbackServer\n\t\t\t\t? async () => {\n\t\t\t\t\t\tconst { createInterface } = await import(\"readline\");\n\t\t\t\t\t\tconst rl = createInterface({ input: process.stdin, output: process.stdout });\n\t\t\t\t\t\treturn new Promise<string>((resolve) => {\n\t\t\t\t\t\t\trl.question(\"Enter the authorization code: \", (ans: string) => {\n\t\t\t\t\t\t\t\trl.close();\n\t\t\t\t\t\t\t\tresolve(ans.trim());\n\t\t\t\t\t\t\t});\n\t\t\t\t\t\t});\n\t\t\t\t\t}\n\t\t\t\t: undefined,\n\t\t});\n\n\t\tauthStorage.set(provider.id, { type: \"oauth\", ...credentials });\n\t\tconsole.log(chalk.green(`✓ ${provider.name} — logged in successfully`));\n\t} catch (error) {\n\t\tconsole.log(chalk.red(`✗ ${provider.name} — login failed: ${error instanceof Error ? error.message : error}`));\n\t}\n}\n\nexport async function main(args: string[]) {\n\tconst offlineMode = args.includes(\"--offline\") || isTruthyEnvFlag(process.env.DRAHT_OFFLINE);\n\tif (offlineMode) {\n\t\tprocess.env.DRAHT_OFFLINE = \"1\";\n\t\tprocess.env.DRAHT_SKIP_VERSION_CHECK = \"1\";\n\t}\n\n\tif (await handlePackageCommand(args)) {\n\t\treturn;\n\t}\n\n\tif (await handleConfigCommand(args)) {\n\t\treturn;\n\t}\n\n\tif (await handleLoginCommand(args)) {\n\t\treturn;\n\t}\n\n\t// Run migrations (pass cwd for project-local migrations)\n\tconst { migratedAuthProviders: migratedProviders, deprecationWarnings } = runMigrations(process.cwd());\n\n\t// First pass: parse args to get --extension paths\n\tconst firstPass = parseArgs(args);\n\n\t// Early load extensions to discover their CLI flags\n\tconst cwd = process.cwd();\n\tconst agentDir = getAgentDir();\n\tconst settingsManager = SettingsManager.create(cwd, agentDir);\n\treportSettingsErrors(settingsManager, \"startup\");\n\tconst authStorage = AuthStorage.create();\n\tconst modelRegistry = new ModelRegistry(authStorage, getModelsPath());\n\n\tconst resourceLoader = new DefaultResourceLoader({\n\t\tcwd,\n\t\tagentDir,\n\t\tsettingsManager,\n\t\tadditionalExtensionPaths: firstPass.extensions,\n\t\tadditionalSkillPaths: firstPass.skills,\n\t\tadditionalPromptTemplatePaths: firstPass.promptTemplates,\n\t\tadditionalThemePaths: firstPass.themes,\n\t\tnoExtensions: firstPass.noExtensions,\n\t\tnoSkills: firstPass.noSkills,\n\t\tnoPromptTemplates: firstPass.noPromptTemplates,\n\t\tnoThemes: firstPass.noThemes,\n\t\tsystemPrompt: firstPass.systemPrompt,\n\t\tappendSystemPrompt: firstPass.appendSystemPrompt,\n\t});\n\tawait resourceLoader.reload();\n\ttime(\"resourceLoader.reload\");\n\n\tconst extensionsResult: LoadExtensionsResult = resourceLoader.getExtensions();\n\tfor (const { path, error } of extensionsResult.errors) {\n\t\tconsole.error(chalk.red(`Failed to load extension \"${path}\": ${error}`));\n\t}\n\n\t// Apply pending provider registrations from extensions immediately\n\t// so they're available for model resolution before AgentSession is created\n\tfor (const { name, config } of extensionsResult.runtime.pendingProviderRegistrations) {\n\t\tmodelRegistry.registerProvider(name, config);\n\t}\n\textensionsResult.runtime.pendingProviderRegistrations = [];\n\n\tconst extensionFlags = new Map<string, { type: \"boolean\" | \"string\" }>();\n\tfor (const ext of extensionsResult.extensions) {\n\t\tfor (const [name, flag] of ext.flags) {\n\t\t\textensionFlags.set(name, { type: flag.type });\n\t\t}\n\t}\n\n\t// Second pass: parse args with extension flags\n\tconst parsed = parseArgs(args, extensionFlags);\n\n\t// Pass flag values to extensions via runtime\n\tfor (const [name, value] of parsed.unknownFlags) {\n\t\textensionsResult.runtime.flagValues.set(name, value);\n\t}\n\n\tif (parsed.version) {\n\t\tconsole.log(VERSION);\n\t\tprocess.exit(0);\n\t}\n\n\tif (parsed.help) {\n\t\tprintHelp();\n\t\tprocess.exit(0);\n\t}\n\n\tif (parsed.listModels !== undefined) {\n\t\tconst searchPattern = typeof parsed.listModels === \"string\" ? parsed.listModels : undefined;\n\t\tawait listModels(modelRegistry, searchPattern);\n\t\tprocess.exit(0);\n\t}\n\n\t// Read piped stdin content (if any) - skip for RPC mode which uses stdin for JSON-RPC\n\tif (parsed.mode !== \"rpc\") {\n\t\tconst stdinContent = await readPipedStdin();\n\t\tif (stdinContent !== undefined) {\n\t\t\t// Force print mode since interactive mode requires a TTY for keyboard input\n\t\t\tparsed.print = true;\n\t\t\t// Prepend stdin content to messages\n\t\t\tparsed.messages.unshift(stdinContent);\n\t\t}\n\t}\n\n\tif (parsed.export) {\n\t\tlet result: string;\n\t\ttry {\n\t\t\tconst outputPath = parsed.messages.length > 0 ? parsed.messages[0] : undefined;\n\t\t\tresult = await exportFromFile(parsed.export, outputPath);\n\t\t} catch (error: unknown) {\n\t\t\tconst message = error instanceof Error ? error.message : \"Failed to export session\";\n\t\t\tconsole.error(chalk.red(`Error: ${message}`));\n\t\t\tprocess.exit(1);\n\t\t}\n\t\tconsole.log(`Exported to: ${result}`);\n\t\tprocess.exit(0);\n\t}\n\n\tif (parsed.mode === \"rpc\" && parsed.fileArgs.length > 0) {\n\t\tconsole.error(chalk.red(\"Error: @file arguments are not supported in RPC mode\"));\n\t\tprocess.exit(1);\n\t}\n\n\tconst { initialMessage, initialImages } = await prepareInitialMessage(parsed, settingsManager.getImageAutoResize());\n\tconst isInteractive = !parsed.print && parsed.mode === undefined;\n\tconst mode = parsed.mode || \"text\";\n\tinitTheme(settingsManager.getTheme(), isInteractive);\n\n\t// Show deprecation warnings in interactive mode\n\tif (isInteractive && deprecationWarnings.length > 0) {\n\t\tawait showDeprecationWarnings(deprecationWarnings);\n\t}\n\n\tlet scopedModels: ScopedModel[] = [];\n\tconst modelPatterns = parsed.models ?? settingsManager.getEnabledModels();\n\tif (modelPatterns && modelPatterns.length > 0) {\n\t\tscopedModels = await resolveModelScope(modelPatterns, modelRegistry);\n\t}\n\n\t// Create session manager based on CLI flags\n\tlet sessionManager = await createSessionManager(parsed, cwd);\n\n\t// Handle --resume: show session picker\n\tif (parsed.resume) {\n\t\t// Initialize keybindings so session picker respects user config\n\t\tKeybindingsManager.create();\n\n\t\tconst selectedPath = await selectSession(\n\t\t\t(onProgress) => SessionManager.list(cwd, parsed.sessionDir, onProgress),\n\t\t\tSessionManager.listAll,\n\t\t);\n\t\tif (!selectedPath) {\n\t\t\tconsole.log(chalk.dim(\"No session selected\"));\n\t\t\tstopThemeWatcher();\n\t\t\tprocess.exit(0);\n\t\t}\n\t\tsessionManager = SessionManager.open(selectedPath);\n\t}\n\n\tconst { options: sessionOptions, cliThinkingFromModel } = buildSessionOptions(\n\t\tparsed,\n\t\tscopedModels,\n\t\tsessionManager,\n\t\tmodelRegistry,\n\t\tsettingsManager,\n\t);\n\tsessionOptions.authStorage = authStorage;\n\tsessionOptions.modelRegistry = modelRegistry;\n\tsessionOptions.resourceLoader = resourceLoader;\n\n\t// Handle CLI --api-key as runtime override (not persisted)\n\tif (parsed.apiKey) {\n\t\tif (!sessionOptions.model) {\n\t\t\tconsole.error(\n\t\t\t\tchalk.red(\"--api-key requires a model to be specified via --model, --provider/--model, or --models\"),\n\t\t\t);\n\t\t\tprocess.exit(1);\n\t\t}\n\t\tauthStorage.setRuntimeApiKey(sessionOptions.model.provider, parsed.apiKey);\n\t}\n\n\tconst { session, modelFallbackMessage } = await createAgentSession(sessionOptions);\n\n\tif (!isInteractive && !session.model) {\n\t\tconsole.error(chalk.red(\"No models available.\"));\n\t\tconsole.error(chalk.yellow(\"\\nSet an API key environment variable:\"));\n\t\tconsole.error(\" ANTHROPIC_API_KEY, OPENAI_API_KEY, GEMINI_API_KEY, etc.\");\n\t\tconsole.error(chalk.yellow(`\\nOr create ${getModelsPath()}`));\n\t\tprocess.exit(1);\n\t}\n\n\t// Clamp thinking level to model capabilities for CLI-provided thinking levels.\n\t// This covers both --thinking <level> and --model <pattern>:<thinking>.\n\tconst cliThinkingOverride = parsed.thinking !== undefined || cliThinkingFromModel;\n\tif (session.model && cliThinkingOverride) {\n\t\tlet effectiveThinking = session.thinkingLevel;\n\t\tif (!session.model.reasoning) {\n\t\t\teffectiveThinking = \"off\";\n\t\t} else if (effectiveThinking === \"xhigh\" && !supportsXhigh(session.model)) {\n\t\t\teffectiveThinking = \"high\";\n\t\t}\n\t\tif (effectiveThinking !== session.thinkingLevel) {\n\t\t\tsession.setThinkingLevel(effectiveThinking);\n\t\t}\n\t}\n\n\tif (mode === \"rpc\") {\n\t\tawait runRpcMode(session);\n\t} else if (isInteractive) {\n\t\tif (scopedModels.length > 0 && (parsed.verbose || !settingsManager.getQuietStartup())) {\n\t\t\tconst modelList = scopedModels\n\t\t\t\t.map((sm) => {\n\t\t\t\t\tconst thinkingStr = sm.thinkingLevel ? `:${sm.thinkingLevel}` : \"\";\n\t\t\t\t\treturn `${sm.model.id}${thinkingStr}`;\n\t\t\t\t})\n\t\t\t\t.join(\", \");\n\t\t\tconsole.log(chalk.dim(`Model scope: ${modelList} ${chalk.gray(\"(Ctrl+P to cycle)\")}`));\n\t\t}\n\n\t\tprintTimings();\n\t\tconst mode = new InteractiveMode(session, {\n\t\t\tmigratedProviders,\n\t\t\tmodelFallbackMessage,\n\t\t\tinitialMessage,\n\t\t\tinitialImages,\n\t\t\tinitialMessages: parsed.messages,\n\t\t\tverbose: parsed.verbose,\n\t\t});\n\t\tawait mode.run();\n\t} else {\n\t\tawait runPrintMode(session, {\n\t\t\tmode,\n\t\t\tmessages: parsed.messages,\n\t\t\tinitialMessage,\n\t\t\tinitialImages,\n\t\t});\n\t\tstopThemeWatcher();\n\t\tif (process.stdout.writableLength > 0) {\n\t\t\tawait new Promise<void>((resolve) => process.stdout.once(\"drain\", resolve));\n\t\t}\n\t\tprocess.exit(0);\n\t}\n}\n"]}
1
+ {"version":3,"file":"main.d.ts","sourceRoot":"","sources":["../src/main.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAkuBH,wBAAsB,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,iBA6RxC","sourcesContent":["/**\n * Main entry point for the coding agent CLI.\n *\n * This file handles CLI argument parsing and translates them into\n * createAgentSession() options. The SDK does the heavy lifting.\n */\n\nimport { type ImageContent, modelsAreEqual, supportsXhigh } from \"@draht/ai\";\nimport chalk from \"chalk\";\nimport { createInterface } from \"readline\";\nimport { type Args, parseArgs, printHelp } from \"./cli/args.js\";\nimport { selectConfig } from \"./cli/config-selector.js\";\nimport { processFileArguments } from \"./cli/file-processor.js\";\nimport { listModels } from \"./cli/list-models.js\";\nimport { selectSession } from \"./cli/session-picker.js\";\nimport { APP_NAME, getAgentDir, getModelsPath, VERSION } from \"./config.js\";\nimport { AuthStorage } from \"./core/auth-storage.js\";\nimport { exportFromFile } from \"./core/export-html/index.js\";\nimport type { LoadExtensionsResult } from \"./core/extensions/index.js\";\nimport { KeybindingsManager } from \"./core/keybindings.js\";\nimport { ModelRegistry } from \"./core/model-registry.js\";\nimport { resolveCliModel, resolveModelScope, type ScopedModel } from \"./core/model-resolver.js\";\nimport { DefaultPackageManager } from \"./core/package-manager.js\";\nimport { DefaultResourceLoader } from \"./core/resource-loader.js\";\nimport { type CreateAgentSessionOptions, createAgentSession } from \"./core/sdk.js\";\nimport { SessionManager } from \"./core/session-manager.js\";\nimport { SettingsManager } from \"./core/settings-manager.js\";\nimport { printTimings, time } from \"./core/timings.js\";\nimport { allTools } from \"./core/tools/index.js\";\nimport { runMigrations, showDeprecationWarnings } from \"./migrations.js\";\nimport { InteractiveMode, runPrintMode, runRpcMode } from \"./modes/index.js\";\nimport { initTheme, stopThemeWatcher } from \"./modes/interactive/theme/theme.js\";\n\n/**\n * Read all content from piped stdin.\n * Returns undefined if stdin is a TTY (interactive terminal).\n */\nasync function readPipedStdin(): Promise<string | undefined> {\n\t// If stdin is a TTY, we're running interactively - don't read stdin\n\tif (process.stdin.isTTY) {\n\t\treturn undefined;\n\t}\n\n\treturn new Promise((resolve) => {\n\t\tlet data = \"\";\n\t\tprocess.stdin.setEncoding(\"utf8\");\n\t\tprocess.stdin.on(\"data\", (chunk) => {\n\t\t\tdata += chunk;\n\t\t});\n\t\tprocess.stdin.on(\"end\", () => {\n\t\t\tresolve(data.trim() || undefined);\n\t\t});\n\t\tprocess.stdin.resume();\n\t});\n}\n\nfunction reportSettingsErrors(settingsManager: SettingsManager, context: string): void {\n\tconst errors = settingsManager.drainErrors();\n\tfor (const { scope, error } of errors) {\n\t\tconsole.error(chalk.yellow(`Warning (${context}, ${scope} settings): ${error.message}`));\n\t\tif (error.stack) {\n\t\t\tconsole.error(chalk.dim(error.stack));\n\t\t}\n\t}\n}\n\nfunction isTruthyEnvFlag(value: string | undefined): boolean {\n\tif (!value) return false;\n\treturn value === \"1\" || value.toLowerCase() === \"true\" || value.toLowerCase() === \"yes\";\n}\n\ntype PackageCommand = \"install\" | \"remove\" | \"update\" | \"list\";\n\ninterface PackageCommandOptions {\n\tcommand: PackageCommand;\n\tsource?: string;\n\tlocal: boolean;\n\thelp: boolean;\n\tinvalidOption?: string;\n}\n\nfunction getPackageCommandUsage(command: PackageCommand): string {\n\tswitch (command) {\n\t\tcase \"install\":\n\t\t\treturn `${APP_NAME} install <source> [-l]`;\n\t\tcase \"remove\":\n\t\t\treturn `${APP_NAME} remove <source> [-l]`;\n\t\tcase \"update\":\n\t\t\treturn `${APP_NAME} update [source]`;\n\t\tcase \"list\":\n\t\t\treturn `${APP_NAME} list`;\n\t}\n}\n\nfunction printPackageCommandHelp(command: PackageCommand): void {\n\tswitch (command) {\n\t\tcase \"install\":\n\t\t\tconsole.log(`${chalk.bold(\"Usage:\")}\n ${getPackageCommandUsage(\"install\")}\n\nInstall a package and add it to settings.\n\nOptions:\n -l, --local Install project-locally (.draht/settings.json)\n\nExamples:\n ${APP_NAME} install npm:@foo/bar\n ${APP_NAME} install git:github.com/user/repo\n ${APP_NAME} install git:git@github.com:user/repo\n ${APP_NAME} install https://github.com/user/repo\n ${APP_NAME} install ssh://git@github.com/user/repo\n ${APP_NAME} install ./local/path\n`);\n\t\t\treturn;\n\n\t\tcase \"remove\":\n\t\t\tconsole.log(`${chalk.bold(\"Usage:\")}\n ${getPackageCommandUsage(\"remove\")}\n\nRemove a package and its source from settings.\n\nOptions:\n -l, --local Remove from project settings (.draht/settings.json)\n\nExample:\n ${APP_NAME} remove npm:@foo/bar\n`);\n\t\t\treturn;\n\n\t\tcase \"update\":\n\t\t\tconsole.log(`${chalk.bold(\"Usage:\")}\n ${getPackageCommandUsage(\"update\")}\n\nUpdate installed packages.\nIf <source> is provided, only that package is updated.\n`);\n\t\t\treturn;\n\n\t\tcase \"list\":\n\t\t\tconsole.log(`${chalk.bold(\"Usage:\")}\n ${getPackageCommandUsage(\"list\")}\n\nList installed packages from user and project settings.\n`);\n\t\t\treturn;\n\t}\n}\n\nfunction parsePackageCommand(args: string[]): PackageCommandOptions | undefined {\n\tconst [command, ...rest] = args;\n\tif (command !== \"install\" && command !== \"remove\" && command !== \"update\" && command !== \"list\") {\n\t\treturn undefined;\n\t}\n\n\tlet local = false;\n\tlet help = false;\n\tlet invalidOption: string | undefined;\n\tlet source: string | undefined;\n\n\tfor (const arg of rest) {\n\t\tif (arg === \"-h\" || arg === \"--help\") {\n\t\t\thelp = true;\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (arg === \"-l\" || arg === \"--local\") {\n\t\t\tif (command === \"install\" || command === \"remove\") {\n\t\t\t\tlocal = true;\n\t\t\t} else {\n\t\t\t\tinvalidOption = invalidOption ?? arg;\n\t\t\t}\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (arg.startsWith(\"-\")) {\n\t\t\tinvalidOption = invalidOption ?? arg;\n\t\t\tcontinue;\n\t\t}\n\n\t\tif (!source) {\n\t\t\tsource = arg;\n\t\t}\n\t}\n\n\treturn { command, source, local, help, invalidOption };\n}\n\nasync function handlePackageCommand(args: string[]): Promise<boolean> {\n\tconst options = parsePackageCommand(args);\n\tif (!options) {\n\t\treturn false;\n\t}\n\n\tif (options.help) {\n\t\tprintPackageCommandHelp(options.command);\n\t\treturn true;\n\t}\n\n\tif (options.invalidOption) {\n\t\tconsole.error(chalk.red(`Unknown option ${options.invalidOption} for \"${options.command}\".`));\n\t\tconsole.error(chalk.dim(`Use \"${APP_NAME} --help\" or \"${getPackageCommandUsage(options.command)}\".`));\n\t\tprocess.exitCode = 1;\n\t\treturn true;\n\t}\n\n\tconst source = options.source;\n\tif ((options.command === \"install\" || options.command === \"remove\") && !source) {\n\t\tconsole.error(chalk.red(`Missing ${options.command} source.`));\n\t\tconsole.error(chalk.dim(`Usage: ${getPackageCommandUsage(options.command)}`));\n\t\tprocess.exitCode = 1;\n\t\treturn true;\n\t}\n\n\tconst cwd = process.cwd();\n\tconst agentDir = getAgentDir();\n\tconst settingsManager = SettingsManager.create(cwd, agentDir);\n\treportSettingsErrors(settingsManager, \"package command\");\n\tconst packageManager = new DefaultPackageManager({ cwd, agentDir, settingsManager });\n\n\tpackageManager.setProgressCallback((event) => {\n\t\tif (event.type === \"start\") {\n\t\t\tprocess.stdout.write(chalk.dim(`${event.message}\\n`));\n\t\t}\n\t});\n\n\ttry {\n\t\tswitch (options.command) {\n\t\t\tcase \"install\":\n\t\t\t\tawait packageManager.install(source!, { local: options.local });\n\t\t\t\tpackageManager.addSourceToSettings(source!, { local: options.local });\n\t\t\t\tconsole.log(chalk.green(`Installed ${source}`));\n\t\t\t\treturn true;\n\n\t\t\tcase \"remove\": {\n\t\t\t\tawait packageManager.remove(source!, { local: options.local });\n\t\t\t\tconst removed = packageManager.removeSourceFromSettings(source!, { local: options.local });\n\t\t\t\tif (!removed) {\n\t\t\t\t\tconsole.error(chalk.red(`No matching package found for ${source}`));\n\t\t\t\t\tprocess.exitCode = 1;\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\tconsole.log(chalk.green(`Removed ${source}`));\n\t\t\t\treturn true;\n\t\t\t}\n\n\t\t\tcase \"list\": {\n\t\t\t\tconst globalSettings = settingsManager.getGlobalSettings();\n\t\t\t\tconst projectSettings = settingsManager.getProjectSettings();\n\t\t\t\tconst globalPackages = globalSettings.packages ?? [];\n\t\t\t\tconst projectPackages = projectSettings.packages ?? [];\n\n\t\t\t\tif (globalPackages.length === 0 && projectPackages.length === 0) {\n\t\t\t\t\tconsole.log(chalk.dim(\"No packages installed.\"));\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\tconst formatPackage = (pkg: (typeof globalPackages)[number], scope: \"user\" | \"project\") => {\n\t\t\t\t\tconst source = typeof pkg === \"string\" ? pkg : pkg.source;\n\t\t\t\t\tconst filtered = typeof pkg === \"object\";\n\t\t\t\t\tconst display = filtered ? `${source} (filtered)` : source;\n\t\t\t\t\tconsole.log(` ${display}`);\n\t\t\t\t\tconst path = packageManager.getInstalledPath(source, scope);\n\t\t\t\t\tif (path) {\n\t\t\t\t\t\tconsole.log(chalk.dim(` ${path}`));\n\t\t\t\t\t}\n\t\t\t\t};\n\n\t\t\t\tif (globalPackages.length > 0) {\n\t\t\t\t\tconsole.log(chalk.bold(\"User packages:\"));\n\t\t\t\t\tfor (const pkg of globalPackages) {\n\t\t\t\t\t\tformatPackage(pkg, \"user\");\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif (projectPackages.length > 0) {\n\t\t\t\t\tif (globalPackages.length > 0) console.log();\n\t\t\t\t\tconsole.log(chalk.bold(\"Project packages:\"));\n\t\t\t\t\tfor (const pkg of projectPackages) {\n\t\t\t\t\t\tformatPackage(pkg, \"project\");\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\treturn true;\n\t\t\t}\n\n\t\t\tcase \"update\":\n\t\t\t\tawait packageManager.update(source);\n\t\t\t\tif (source) {\n\t\t\t\t\tconsole.log(chalk.green(`Updated ${source}`));\n\t\t\t\t} else {\n\t\t\t\t\tconsole.log(chalk.green(\"Updated packages\"));\n\t\t\t\t}\n\t\t\t\treturn true;\n\t\t}\n\t} catch (error: unknown) {\n\t\tconst message = error instanceof Error ? error.message : \"Unknown package command error\";\n\t\tconsole.error(chalk.red(`Error: ${message}`));\n\t\tprocess.exitCode = 1;\n\t\treturn true;\n\t}\n}\n\nasync function prepareInitialMessage(\n\tparsed: Args,\n\tautoResizeImages: boolean,\n): Promise<{\n\tinitialMessage?: string;\n\tinitialImages?: ImageContent[];\n}> {\n\tif (parsed.fileArgs.length === 0) {\n\t\treturn {};\n\t}\n\n\tconst { text, images } = await processFileArguments(parsed.fileArgs, { autoResizeImages });\n\n\tlet initialMessage: string;\n\tif (parsed.messages.length > 0) {\n\t\tinitialMessage = text + parsed.messages[0];\n\t\tparsed.messages.shift();\n\t} else {\n\t\tinitialMessage = text;\n\t}\n\n\treturn {\n\t\tinitialMessage,\n\t\tinitialImages: images.length > 0 ? images : undefined,\n\t};\n}\n\n/** Result from resolving a session argument */\ntype ResolvedSession =\n\t| { type: \"path\"; path: string } // Direct file path\n\t| { type: \"local\"; path: string } // Found in current project\n\t| { type: \"global\"; path: string; cwd: string } // Found in different project\n\t| { type: \"not_found\"; arg: string }; // Not found anywhere\n\n/**\n * Resolve a session argument to a file path.\n * If it looks like a path, use as-is. Otherwise try to match as session ID prefix.\n */\nasync function resolveSessionPath(sessionArg: string, cwd: string, sessionDir?: string): Promise<ResolvedSession> {\n\t// If it looks like a file path, use as-is\n\tif (sessionArg.includes(\"/\") || sessionArg.includes(\"\\\\\") || sessionArg.endsWith(\".jsonl\")) {\n\t\treturn { type: \"path\", path: sessionArg };\n\t}\n\n\t// Try to match as session ID in current project first\n\tconst localSessions = await SessionManager.list(cwd, sessionDir);\n\tconst localMatches = localSessions.filter((s) => s.id.startsWith(sessionArg));\n\n\tif (localMatches.length >= 1) {\n\t\treturn { type: \"local\", path: localMatches[0].path };\n\t}\n\n\t// Try global search across all projects\n\tconst allSessions = await SessionManager.listAll();\n\tconst globalMatches = allSessions.filter((s) => s.id.startsWith(sessionArg));\n\n\tif (globalMatches.length >= 1) {\n\t\tconst match = globalMatches[0];\n\t\treturn { type: \"global\", path: match.path, cwd: match.cwd };\n\t}\n\n\t// Not found anywhere\n\treturn { type: \"not_found\", arg: sessionArg };\n}\n\n/** Prompt user for yes/no confirmation */\nasync function promptConfirm(message: string): Promise<boolean> {\n\treturn new Promise((resolve) => {\n\t\tconst rl = createInterface({\n\t\t\tinput: process.stdin,\n\t\t\toutput: process.stdout,\n\t\t});\n\t\trl.question(`${message} [y/N] `, (answer) => {\n\t\t\trl.close();\n\t\t\tresolve(answer.toLowerCase() === \"y\" || answer.toLowerCase() === \"yes\");\n\t\t});\n\t});\n}\n\n/** Helper to call CLI-only session_directory handlers before the initial session manager is created */\nasync function callSessionDirectoryHook(extensions: LoadExtensionsResult, cwd: string): Promise<string | undefined> {\n\tlet customSessionDir: string | undefined;\n\n\tfor (const ext of extensions.extensions) {\n\t\tconst handlers = ext.handlers.get(\"session_directory\");\n\t\tif (!handlers || handlers.length === 0) continue;\n\n\t\tfor (const handler of handlers) {\n\t\t\ttry {\n\t\t\t\tconst event = { type: \"session_directory\" as const, cwd };\n\t\t\t\tconst result = (await handler(event)) as { sessionDir?: string } | undefined;\n\n\t\t\t\tif (result?.sessionDir) {\n\t\t\t\t\tcustomSessionDir = result.sessionDir;\n\t\t\t\t}\n\t\t\t} catch (err) {\n\t\t\t\tconst message = err instanceof Error ? err.message : String(err);\n\t\t\t\tconsole.error(chalk.red(`Extension \"${ext.path}\" session_directory handler failed: ${message}`));\n\t\t\t}\n\t\t}\n\t}\n\n\treturn customSessionDir;\n}\n\nasync function createSessionManager(\n\tparsed: Args,\n\tcwd: string,\n\textensions: LoadExtensionsResult,\n): Promise<SessionManager | undefined> {\n\tif (parsed.noSession) {\n\t\treturn SessionManager.inMemory();\n\t}\n\n\t// CLI flag takes precedence, otherwise ask extensions for custom session directory\n\tlet effectiveSessionDir = parsed.sessionDir;\n\tif (!effectiveSessionDir) {\n\t\teffectiveSessionDir = await callSessionDirectoryHook(extensions, cwd);\n\t}\n\n\tif (parsed.session) {\n\t\tconst resolved = await resolveSessionPath(parsed.session, cwd, effectiveSessionDir);\n\n\t\tswitch (resolved.type) {\n\t\t\tcase \"path\":\n\t\t\tcase \"local\":\n\t\t\t\treturn SessionManager.open(resolved.path, effectiveSessionDir);\n\n\t\t\tcase \"global\": {\n\t\t\t\t// Session found in different project - ask user if they want to fork\n\t\t\t\tconsole.log(chalk.yellow(`Session found in different project: ${resolved.cwd}`));\n\t\t\t\tconst shouldFork = await promptConfirm(\"Fork this session into current directory?\");\n\t\t\t\tif (!shouldFork) {\n\t\t\t\t\tconsole.log(chalk.dim(\"Aborted.\"));\n\t\t\t\t\tprocess.exit(0);\n\t\t\t\t}\n\t\t\t\treturn SessionManager.forkFrom(resolved.path, cwd, effectiveSessionDir);\n\t\t\t}\n\n\t\t\tcase \"not_found\":\n\t\t\t\tconsole.error(chalk.red(`No session found matching '${resolved.arg}'`));\n\t\t\t\tprocess.exit(1);\n\t\t}\n\t}\n\tif (parsed.continue) {\n\t\treturn SessionManager.continueRecent(cwd, effectiveSessionDir);\n\t}\n\t// --resume is handled separately (needs picker UI)\n\t// If effective session dir is set, create new session there\n\tif (effectiveSessionDir) {\n\t\treturn SessionManager.create(cwd, effectiveSessionDir);\n\t}\n\t// Default case (new session) returns undefined, SDK will create one\n\treturn undefined;\n}\n\nfunction buildSessionOptions(\n\tparsed: Args,\n\tscopedModels: ScopedModel[],\n\tsessionManager: SessionManager | undefined,\n\tmodelRegistry: ModelRegistry,\n\tsettingsManager: SettingsManager,\n): { options: CreateAgentSessionOptions; cliThinkingFromModel: boolean } {\n\tconst options: CreateAgentSessionOptions = {};\n\tlet cliThinkingFromModel = false;\n\n\tif (sessionManager) {\n\t\toptions.sessionManager = sessionManager;\n\t}\n\n\t// Model from CLI\n\t// - supports --provider <name> --model <pattern>\n\t// - supports --model <provider>/<pattern>\n\tif (parsed.model) {\n\t\tconst resolved = resolveCliModel({\n\t\t\tcliProvider: parsed.provider,\n\t\t\tcliModel: parsed.model,\n\t\t\tmodelRegistry,\n\t\t});\n\t\tif (resolved.warning) {\n\t\t\tconsole.warn(chalk.yellow(`Warning: ${resolved.warning}`));\n\t\t}\n\t\tif (resolved.error) {\n\t\t\tconsole.error(chalk.red(resolved.error));\n\t\t\tprocess.exit(1);\n\t\t}\n\t\tif (resolved.model) {\n\t\t\toptions.model = resolved.model;\n\t\t\t// Allow \"--model <pattern>:<thinking>\" as a shorthand.\n\t\t\t// Explicit --thinking still takes precedence (applied later).\n\t\t\tif (!parsed.thinking && resolved.thinkingLevel) {\n\t\t\t\toptions.thinkingLevel = resolved.thinkingLevel;\n\t\t\t\tcliThinkingFromModel = true;\n\t\t\t}\n\t\t}\n\t}\n\n\tif (!options.model && scopedModels.length > 0 && !parsed.continue && !parsed.resume) {\n\t\t// Check if saved default is in scoped models - use it if so, otherwise first scoped model\n\t\tconst savedProvider = settingsManager.getDefaultProvider();\n\t\tconst savedModelId = settingsManager.getDefaultModel();\n\t\tconst savedModel = savedProvider && savedModelId ? modelRegistry.find(savedProvider, savedModelId) : undefined;\n\t\tconst savedInScope = savedModel ? scopedModels.find((sm) => modelsAreEqual(sm.model, savedModel)) : undefined;\n\n\t\tif (savedInScope) {\n\t\t\toptions.model = savedInScope.model;\n\t\t\t// Use thinking level from scoped model config if explicitly set\n\t\t\tif (!parsed.thinking && savedInScope.thinkingLevel) {\n\t\t\t\toptions.thinkingLevel = savedInScope.thinkingLevel;\n\t\t\t}\n\t\t} else {\n\t\t\toptions.model = scopedModels[0].model;\n\t\t\t// Use thinking level from first scoped model if explicitly set\n\t\t\tif (!parsed.thinking && scopedModels[0].thinkingLevel) {\n\t\t\t\toptions.thinkingLevel = scopedModels[0].thinkingLevel;\n\t\t\t}\n\t\t}\n\t}\n\n\t// Thinking level from CLI (takes precedence over scoped model thinking levels set above)\n\tif (parsed.thinking) {\n\t\toptions.thinkingLevel = parsed.thinking;\n\t}\n\n\t// Scoped models for Ctrl+P cycling\n\t// Keep thinking level undefined when not explicitly set in the model pattern.\n\t// Undefined means \"inherit current session thinking level\" during cycling.\n\tif (scopedModels.length > 0) {\n\t\toptions.scopedModels = scopedModels.map((sm) => ({\n\t\t\tmodel: sm.model,\n\t\t\tthinkingLevel: sm.thinkingLevel,\n\t\t}));\n\t}\n\n\t// API key from CLI - set in authStorage\n\t// (handled by caller before createAgentSession)\n\n\t// Tools\n\tif (parsed.noTools) {\n\t\t// --no-tools: start with no built-in tools\n\t\t// --tools can still add specific ones back\n\t\tif (parsed.tools && parsed.tools.length > 0) {\n\t\t\toptions.tools = parsed.tools.map((name) => allTools[name]);\n\t\t} else {\n\t\t\toptions.tools = [];\n\t\t}\n\t} else if (parsed.tools) {\n\t\toptions.tools = parsed.tools.map((name) => allTools[name]);\n\t}\n\n\treturn { options, cliThinkingFromModel };\n}\n\nasync function handleConfigCommand(args: string[]): Promise<boolean> {\n\tif (args[0] !== \"config\") {\n\t\treturn false;\n\t}\n\n\tconst cwd = process.cwd();\n\tconst agentDir = getAgentDir();\n\tconst settingsManager = SettingsManager.create(cwd, agentDir);\n\treportSettingsErrors(settingsManager, \"config command\");\n\tconst packageManager = new DefaultPackageManager({ cwd, agentDir, settingsManager });\n\n\tconst resolvedPaths = await packageManager.resolve();\n\n\tawait selectConfig({\n\t\tresolvedPaths,\n\t\tsettingsManager,\n\t\tcwd,\n\t\tagentDir,\n\t});\n\n\tprocess.exit(0);\n}\n\nasync function handleLoginCommand(args: string[]): Promise<boolean> {\n\tif (args[0] !== \"login\" && args[0] !== \"--login\") {\n\t\treturn false;\n\t}\n\n\tconst { getOAuthProviders } = await import(\"@draht/ai/oauth\");\n\tconst authMod = await import(\"./core/auth-storage.js\");\n\tconst AuthStorageClass = authMod.AuthStorage;\n\t// open browser - use dynamic import, fallback to exec\n\tconst openBrowser = async (url: string) => {\n\t\ttry {\n\t\t\tconst { exec } = await import(\"child_process\");\n\t\t\texec(`open \"${url}\" 2>/dev/null || xdg-open \"${url}\" 2>/dev/null`);\n\t\t} catch {\n\t\t\t/* ignore */\n\t\t}\n\t};\n\n\tconst providers = getOAuthProviders();\n\tconst requestedProvider = args[1]?.toLowerCase();\n\n\t// Map friendly names to provider IDs\n\tconst PROVIDER_ALIASES: Record<string, string> = {\n\t\tanthropic: \"anthropic\",\n\t\tclaude: \"anthropic\",\n\t\tgoogle: \"google-gemini-cli\",\n\t\tgemini: \"google-gemini-cli\",\n\t\topenai: \"openai-codex\",\n\t\tcodex: \"openai-codex\",\n\t\tcopilot: \"github-copilot\",\n\t\tgithub: \"github-copilot\",\n\t\tantigravity: \"google-antigravity\",\n\t};\n\n\tif (requestedProvider === \"--help\" || requestedProvider === \"-h\" || requestedProvider === \"help\") {\n\t\tconsole.log(chalk.bold(\"\\n🔌 draht login\\n\"));\n\t\tconsole.log(\"Authenticate with AI providers.\\n\");\n\t\tconsole.log(\"Usage:\");\n\t\tconsole.log(\" draht login Interactive provider selection\");\n\t\tconsole.log(\" draht login anthropic Login to Anthropic (Claude Pro/Max)\");\n\t\tconsole.log(\" draht login gemini Login to Google Gemini CLI\");\n\t\tconsole.log(\" draht login openai Login to OpenAI (ChatGPT/Codex)\");\n\t\tconsole.log(\" draht login copilot Login to GitHub Copilot\");\n\t\tconsole.log(\" draht login all Login to all providers\\n\");\n\t\tconsole.log(\"Aliases: claude, google, codex, github\\n\");\n\t\tprocess.exit(0);\n\t}\n\n\tif (requestedProvider && requestedProvider !== \"all\") {\n\t\tconst providerId = PROVIDER_ALIASES[requestedProvider] ?? requestedProvider;\n\t\tconst provider = providers.find((p) => p.id === providerId);\n\t\tif (!provider) {\n\t\t\tconsole.log(chalk.red(`Unknown provider: ${requestedProvider}`));\n\t\t\tconsole.log(`\\nAvailable providers:`);\n\t\t\tfor (const p of providers) {\n\t\t\t\tconst aliases = Object.entries(PROVIDER_ALIASES)\n\t\t\t\t\t.filter(([_, v]) => v === p.id)\n\t\t\t\t\t.map(([k]) => k);\n\t\t\t\tconsole.log(` ${chalk.bold(p.name)} (${p.id}) — aliases: ${aliases.join(\", \")}`);\n\t\t\t}\n\t\t\tprocess.exit(1);\n\t\t}\n\t\tawait loginProvider(provider, openBrowser);\n\t\tprocess.exit(0);\n\t}\n\n\t// Interactive: show menu\n\tconsole.log(chalk.bold(\"\\n🔌 Draht Login\\n\"));\n\tconsole.log(\"Select a provider to authenticate:\\n\");\n\n\tconst providerList = providers.filter((p) =>\n\t\t[\"anthropic\", \"gemini-cli\", \"openai-codex\", \"github-copilot\"].includes(p.id),\n\t);\n\n\tfor (let i = 0; i < providerList.length; i++) {\n\t\tconst p = providerList[i];\n\t\tconst as = AuthStorageClass.create();\n\t\tconst hasCredentials = as.has(p.id);\n\t\tconst status = hasCredentials ? chalk.green(\"✓ logged in\") : chalk.dim(\"not connected\");\n\t\tconsole.log(` ${chalk.bold(i + 1)}. ${p.name} ${status}`);\n\t}\n\tconsole.log(` ${chalk.bold(\"a\")}. Login to all`);\n\tconsole.log(` ${chalk.bold(\"q\")}. Cancel\\n`);\n\n\tconst rl = createInterface({ input: process.stdin, output: process.stdout });\n\tconst answer = await new Promise<string>((resolve) => {\n\t\trl.question(\"Choice: \", (ans) => {\n\t\t\trl.close();\n\t\t\tresolve(ans.trim());\n\t\t});\n\t});\n\n\tif (answer === \"q\" || answer === \"\") {\n\t\tprocess.exit(0);\n\t}\n\n\tif (answer === \"a\") {\n\t\tfor (const p of providerList) {\n\t\t\tawait loginProvider(p, openBrowser);\n\t\t}\n\t} else {\n\t\tconst idx = parseInt(answer, 10) - 1;\n\t\tif (idx >= 0 && idx < providerList.length) {\n\t\t\tawait loginProvider(providerList[idx], openBrowser);\n\t\t} else {\n\t\t\tconsole.log(chalk.red(\"Invalid choice\"));\n\t\t}\n\t}\n\n\tprocess.exit(0);\n}\n\nasync function loginProvider(\n\tprovider: { id: string; name: string; login: (cb: any) => Promise<any>; usesCallbackServer?: boolean },\n\topenBrowser: (url: string) => Promise<any>,\n): Promise<void> {\n\tconst { AuthStorage } = await import(\"./core/auth-storage.js\");\n\tconst authStorage = AuthStorage.create();\n\n\tconsole.log(`\\n${chalk.bold(`Logging in to ${provider.name}...`)}`);\n\n\ttry {\n\t\tconst credentials = await provider.login({\n\t\t\tonAuth: (info: { url: string; instructions?: string }) => {\n\t\t\t\tconsole.log(`\\n${chalk.blue(\"→\")} Opening browser for authentication...`);\n\t\t\t\tif (info.instructions) console.log(chalk.dim(info.instructions));\n\t\t\t\topenBrowser(info.url).catch(() => {\n\t\t\t\t\tconsole.log(`\\n${chalk.yellow(\"Could not open browser. Visit manually:\")}`);\n\t\t\t\t\tconsole.log(chalk.underline(info.url));\n\t\t\t\t});\n\t\t\t},\n\t\t\tonPrompt: async (prompt: { message: string; placeholder?: string }) => {\n\t\t\t\tconst { createInterface } = await import(\"readline\");\n\t\t\t\tconst rl = createInterface({ input: process.stdin, output: process.stdout });\n\t\t\t\treturn new Promise<string>((resolve) => {\n\t\t\t\t\trl.question(`${prompt.message} `, (ans: string) => {\n\t\t\t\t\t\trl.close();\n\t\t\t\t\t\tresolve(ans.trim());\n\t\t\t\t\t});\n\t\t\t\t});\n\t\t\t},\n\t\t\tonProgress: (message: string) => {\n\t\t\t\tconsole.log(chalk.dim(` ${message}`));\n\t\t\t},\n\t\t\tonManualCodeInput: provider.usesCallbackServer\n\t\t\t\t? async () => {\n\t\t\t\t\t\tconst { createInterface } = await import(\"readline\");\n\t\t\t\t\t\tconst rl = createInterface({ input: process.stdin, output: process.stdout });\n\t\t\t\t\t\treturn new Promise<string>((resolve) => {\n\t\t\t\t\t\t\trl.question(\"Enter the authorization code: \", (ans: string) => {\n\t\t\t\t\t\t\t\trl.close();\n\t\t\t\t\t\t\t\tresolve(ans.trim());\n\t\t\t\t\t\t\t});\n\t\t\t\t\t\t});\n\t\t\t\t\t}\n\t\t\t\t: undefined,\n\t\t});\n\n\t\tauthStorage.set(provider.id, { type: \"oauth\", ...credentials });\n\t\tconsole.log(chalk.green(`✓ ${provider.name} — logged in successfully`));\n\t} catch (error) {\n\t\tconsole.log(chalk.red(`✗ ${provider.name} — login failed: ${error instanceof Error ? error.message : error}`));\n\t}\n}\n\nexport async function main(args: string[]) {\n\tconst offlineMode = args.includes(\"--offline\") || isTruthyEnvFlag(process.env.DRAHT_OFFLINE);\n\tif (offlineMode) {\n\t\tprocess.env.DRAHT_OFFLINE = \"1\";\n\t\tprocess.env.DRAHT_SKIP_VERSION_CHECK = \"1\";\n\t}\n\n\tif (await handlePackageCommand(args)) {\n\t\treturn;\n\t}\n\n\tif (await handleConfigCommand(args)) {\n\t\treturn;\n\t}\n\n\tif (await handleLoginCommand(args)) {\n\t\treturn;\n\t}\n\n\t// Run migrations (pass cwd for project-local migrations)\n\tconst { migratedAuthProviders: migratedProviders, deprecationWarnings } = runMigrations(process.cwd());\n\n\t// First pass: parse args to get --extension paths\n\tconst firstPass = parseArgs(args);\n\n\t// Early load extensions to discover their CLI flags\n\tconst cwd = process.cwd();\n\tconst agentDir = getAgentDir();\n\tconst settingsManager = SettingsManager.create(cwd, agentDir);\n\treportSettingsErrors(settingsManager, \"startup\");\n\tconst authStorage = AuthStorage.create();\n\tconst modelRegistry = new ModelRegistry(authStorage, getModelsPath());\n\n\tconst resourceLoader = new DefaultResourceLoader({\n\t\tcwd,\n\t\tagentDir,\n\t\tsettingsManager,\n\t\tadditionalExtensionPaths: firstPass.extensions,\n\t\tadditionalSkillPaths: firstPass.skills,\n\t\tadditionalPromptTemplatePaths: firstPass.promptTemplates,\n\t\tadditionalThemePaths: firstPass.themes,\n\t\tnoExtensions: firstPass.noExtensions,\n\t\tnoSkills: firstPass.noSkills,\n\t\tnoPromptTemplates: firstPass.noPromptTemplates,\n\t\tnoThemes: firstPass.noThemes,\n\t\tsystemPrompt: firstPass.systemPrompt,\n\t\tappendSystemPrompt: firstPass.appendSystemPrompt,\n\t});\n\tawait resourceLoader.reload();\n\ttime(\"resourceLoader.reload\");\n\n\tconst extensionsResult: LoadExtensionsResult = resourceLoader.getExtensions();\n\tfor (const { path, error } of extensionsResult.errors) {\n\t\tconsole.error(chalk.red(`Failed to load extension \"${path}\": ${error}`));\n\t}\n\n\t// Apply pending provider registrations from extensions immediately\n\t// so they're available for model resolution before AgentSession is created\n\tfor (const { name, config } of extensionsResult.runtime.pendingProviderRegistrations) {\n\t\tmodelRegistry.registerProvider(name, config);\n\t}\n\textensionsResult.runtime.pendingProviderRegistrations = [];\n\n\tconst extensionFlags = new Map<string, { type: \"boolean\" | \"string\" }>();\n\tfor (const ext of extensionsResult.extensions) {\n\t\tfor (const [name, flag] of ext.flags) {\n\t\t\textensionFlags.set(name, { type: flag.type });\n\t\t}\n\t}\n\n\t// Second pass: parse args with extension flags\n\tconst parsed = parseArgs(args, extensionFlags);\n\n\t// Pass flag values to extensions via runtime\n\tfor (const [name, value] of parsed.unknownFlags) {\n\t\textensionsResult.runtime.flagValues.set(name, value);\n\t}\n\n\tif (parsed.version) {\n\t\tconsole.log(VERSION);\n\t\tprocess.exit(0);\n\t}\n\n\tif (parsed.help) {\n\t\tprintHelp();\n\t\tprocess.exit(0);\n\t}\n\n\tif (parsed.listModels !== undefined) {\n\t\tconst searchPattern = typeof parsed.listModels === \"string\" ? parsed.listModels : undefined;\n\t\tawait listModels(modelRegistry, searchPattern);\n\t\tprocess.exit(0);\n\t}\n\n\t// Experimental: List attachable sessions\n\tif (parsed.listSessions) {\n\t\tconst { listSessions } = await import(\"./cli/list-sessions.js\");\n\t\tawait listSessions();\n\t\tprocess.exit(0);\n\t}\n\n\t// Experimental: Attach to existing session\n\tif (parsed.attach) {\n\t\tconst { runAttachMode } = await import(\"./cli/attach-mode.js\");\n\t\tawait runAttachMode(parsed.attach);\n\t\tprocess.exit(0);\n\t}\n\n\t// Read piped stdin content (if any) - skip for RPC mode which uses stdin for JSON-RPC\n\tif (parsed.mode !== \"rpc\") {\n\t\tconst stdinContent = await readPipedStdin();\n\t\tif (stdinContent !== undefined) {\n\t\t\t// Force print mode since interactive mode requires a TTY for keyboard input\n\t\t\tparsed.print = true;\n\t\t\t// Prepend stdin content to messages\n\t\t\tparsed.messages.unshift(stdinContent);\n\t\t}\n\t}\n\n\tif (parsed.export) {\n\t\tlet result: string;\n\t\ttry {\n\t\t\tconst outputPath = parsed.messages.length > 0 ? parsed.messages[0] : undefined;\n\t\t\tresult = await exportFromFile(parsed.export, outputPath);\n\t\t} catch (error: unknown) {\n\t\t\tconst message = error instanceof Error ? error.message : \"Failed to export session\";\n\t\t\tconsole.error(chalk.red(`Error: ${message}`));\n\t\t\tprocess.exit(1);\n\t\t}\n\t\tconsole.log(`Exported to: ${result}`);\n\t\tprocess.exit(0);\n\t}\n\n\tif (parsed.mode === \"rpc\" && parsed.fileArgs.length > 0) {\n\t\tconsole.error(chalk.red(\"Error: @file arguments are not supported in RPC mode\"));\n\t\tprocess.exit(1);\n\t}\n\n\tconst { initialMessage, initialImages } = await prepareInitialMessage(parsed, settingsManager.getImageAutoResize());\n\tconst isInteractive = !parsed.print && parsed.mode === undefined;\n\tconst mode = parsed.mode || \"text\";\n\tinitTheme(settingsManager.getTheme(), isInteractive);\n\n\t// Show deprecation warnings in interactive mode\n\tif (isInteractive && deprecationWarnings.length > 0) {\n\t\tawait showDeprecationWarnings(deprecationWarnings);\n\t}\n\n\tlet scopedModels: ScopedModel[] = [];\n\tconst modelPatterns = parsed.models ?? settingsManager.getEnabledModels();\n\tif (modelPatterns && modelPatterns.length > 0) {\n\t\tscopedModels = await resolveModelScope(modelPatterns, modelRegistry);\n\t}\n\n\t// Create session manager based on CLI flags\n\tlet sessionManager = await createSessionManager(parsed, cwd, extensionsResult);\n\n\t// Handle --resume: show session picker\n\tif (parsed.resume) {\n\t\t// Initialize keybindings so session picker respects user config\n\t\tKeybindingsManager.create();\n\n\t\t// Compute effective session dir for resume (same logic as createSessionManager)\n\t\tconst effectiveSessionDir = parsed.sessionDir || (await callSessionDirectoryHook(extensionsResult, cwd));\n\n\t\tconst selectedPath = await selectSession(\n\t\t\t(onProgress) => SessionManager.list(cwd, effectiveSessionDir, onProgress),\n\t\t\tSessionManager.listAll,\n\t\t);\n\t\tif (!selectedPath) {\n\t\t\tconsole.log(chalk.dim(\"No session selected\"));\n\t\t\tstopThemeWatcher();\n\t\t\tprocess.exit(0);\n\t\t}\n\t\tsessionManager = SessionManager.open(selectedPath, effectiveSessionDir);\n\t}\n\n\tconst { options: sessionOptions, cliThinkingFromModel } = buildSessionOptions(\n\t\tparsed,\n\t\tscopedModels,\n\t\tsessionManager,\n\t\tmodelRegistry,\n\t\tsettingsManager,\n\t);\n\tsessionOptions.authStorage = authStorage;\n\tsessionOptions.modelRegistry = modelRegistry;\n\tsessionOptions.resourceLoader = resourceLoader;\n\n\t// Handle CLI --api-key as runtime override (not persisted)\n\tif (parsed.apiKey) {\n\t\tif (!sessionOptions.model) {\n\t\t\tconsole.error(\n\t\t\t\tchalk.red(\"--api-key requires a model to be specified via --model, --provider/--model, or --models\"),\n\t\t\t);\n\t\t\tprocess.exit(1);\n\t\t}\n\t\tauthStorage.setRuntimeApiKey(sessionOptions.model.provider, parsed.apiKey);\n\t}\n\n\tconst { session, modelFallbackMessage } = await createAgentSession(sessionOptions);\n\n\t// Experimental: Make session attachable if --attachable flag is set\n\tlet cleanupSocketServer: (() => Promise<void>) | null = null;\n\tif (parsed.attachable) {\n\t\tconst { makeSessionAttachable } = await import(\"./core/socket-server/index.js\");\n\t\tcleanupSocketServer = await makeSessionAttachable({\n\t\t\tsession,\n\t\t\tenabled: true,\n\t\t});\n\t}\n\n\t// Ensure socket server is cleaned up on exit\n\tconst cleanup = async () => {\n\t\tif (cleanupSocketServer) {\n\t\t\tawait cleanupSocketServer();\n\t\t}\n\t};\n\tprocess.on(\"SIGINT\", cleanup);\n\tprocess.on(\"SIGTERM\", cleanup);\n\tprocess.on(\"exit\", () => {\n\t\tif (cleanupSocketServer) {\n\t\t\t// Synchronous cleanup attempt\n\t\t\tcleanupSocketServer().catch(() => {});\n\t\t}\n\t});\n\n\tif (!isInteractive && !session.model) {\n\t\tconsole.error(chalk.red(\"No models available.\"));\n\t\tconsole.error(chalk.yellow(\"\\nSet an API key environment variable:\"));\n\t\tconsole.error(\" ANTHROPIC_API_KEY, OPENAI_API_KEY, GEMINI_API_KEY, etc.\");\n\t\tconsole.error(chalk.yellow(`\\nOr create ${getModelsPath()}`));\n\t\tprocess.exit(1);\n\t}\n\n\t// Clamp thinking level to model capabilities for CLI-provided thinking levels.\n\t// This covers both --thinking <level> and --model <pattern>:<thinking>.\n\tconst cliThinkingOverride = parsed.thinking !== undefined || cliThinkingFromModel;\n\tif (session.model && cliThinkingOverride) {\n\t\tlet effectiveThinking = session.thinkingLevel;\n\t\tif (!session.model.reasoning) {\n\t\t\teffectiveThinking = \"off\";\n\t\t} else if (effectiveThinking === \"xhigh\" && !supportsXhigh(session.model)) {\n\t\t\teffectiveThinking = \"high\";\n\t\t}\n\t\tif (effectiveThinking !== session.thinkingLevel) {\n\t\t\tsession.setThinkingLevel(effectiveThinking);\n\t\t}\n\t}\n\n\tif (mode === \"rpc\") {\n\t\tawait runRpcMode(session);\n\t} else if (isInteractive) {\n\t\tif (scopedModels.length > 0 && (parsed.verbose || !settingsManager.getQuietStartup())) {\n\t\t\tconst modelList = scopedModels\n\t\t\t\t.map((sm) => {\n\t\t\t\t\tconst thinkingStr = sm.thinkingLevel ? `:${sm.thinkingLevel}` : \"\";\n\t\t\t\t\treturn `${sm.model.id}${thinkingStr}`;\n\t\t\t\t})\n\t\t\t\t.join(\", \");\n\t\t\tconsole.log(chalk.dim(`Model scope: ${modelList} ${chalk.gray(\"(Ctrl+P to cycle)\")}`));\n\t\t}\n\n\t\tprintTimings();\n\t\tconst mode = new InteractiveMode(session, {\n\t\t\tmigratedProviders,\n\t\t\tmodelFallbackMessage,\n\t\t\tinitialMessage,\n\t\t\tinitialImages,\n\t\t\tinitialMessages: parsed.messages,\n\t\t\tverbose: parsed.verbose,\n\t\t});\n\t\tawait mode.run();\n\t} else {\n\t\tawait runPrintMode(session, {\n\t\t\tmode,\n\t\t\tmessages: parsed.messages,\n\t\t\tinitialMessage,\n\t\t\tinitialImages,\n\t\t});\n\t\tstopThemeWatcher();\n\t\tif (process.stdout.writableLength > 0) {\n\t\t\tawait new Promise<void>((resolve) => process.stdout.once(\"drain\", resolve));\n\t\t}\n\t\tprocess.exit(0);\n\t}\n}\n"]}
package/dist/main.js CHANGED
@@ -317,16 +317,44 @@ async function promptConfirm(message) {
317
317
  });
318
318
  });
319
319
  }
320
- async function createSessionManager(parsed, cwd) {
320
+ /** Helper to call CLI-only session_directory handlers before the initial session manager is created */
321
+ async function callSessionDirectoryHook(extensions, cwd) {
322
+ let customSessionDir;
323
+ for (const ext of extensions.extensions) {
324
+ const handlers = ext.handlers.get("session_directory");
325
+ if (!handlers || handlers.length === 0)
326
+ continue;
327
+ for (const handler of handlers) {
328
+ try {
329
+ const event = { type: "session_directory", cwd };
330
+ const result = (await handler(event));
331
+ if (result?.sessionDir) {
332
+ customSessionDir = result.sessionDir;
333
+ }
334
+ }
335
+ catch (err) {
336
+ const message = err instanceof Error ? err.message : String(err);
337
+ console.error(chalk.red(`Extension "${ext.path}" session_directory handler failed: ${message}`));
338
+ }
339
+ }
340
+ }
341
+ return customSessionDir;
342
+ }
343
+ async function createSessionManager(parsed, cwd, extensions) {
321
344
  if (parsed.noSession) {
322
345
  return SessionManager.inMemory();
323
346
  }
347
+ // CLI flag takes precedence, otherwise ask extensions for custom session directory
348
+ let effectiveSessionDir = parsed.sessionDir;
349
+ if (!effectiveSessionDir) {
350
+ effectiveSessionDir = await callSessionDirectoryHook(extensions, cwd);
351
+ }
324
352
  if (parsed.session) {
325
- const resolved = await resolveSessionPath(parsed.session, cwd, parsed.sessionDir);
353
+ const resolved = await resolveSessionPath(parsed.session, cwd, effectiveSessionDir);
326
354
  switch (resolved.type) {
327
355
  case "path":
328
356
  case "local":
329
- return SessionManager.open(resolved.path, parsed.sessionDir);
357
+ return SessionManager.open(resolved.path, effectiveSessionDir);
330
358
  case "global": {
331
359
  // Session found in different project - ask user if they want to fork
332
360
  console.log(chalk.yellow(`Session found in different project: ${resolved.cwd}`));
@@ -335,7 +363,7 @@ async function createSessionManager(parsed, cwd) {
335
363
  console.log(chalk.dim("Aborted."));
336
364
  process.exit(0);
337
365
  }
338
- return SessionManager.forkFrom(resolved.path, cwd, parsed.sessionDir);
366
+ return SessionManager.forkFrom(resolved.path, cwd, effectiveSessionDir);
339
367
  }
340
368
  case "not_found":
341
369
  console.error(chalk.red(`No session found matching '${resolved.arg}'`));
@@ -343,12 +371,12 @@ async function createSessionManager(parsed, cwd) {
343
371
  }
344
372
  }
345
373
  if (parsed.continue) {
346
- return SessionManager.continueRecent(cwd, parsed.sessionDir);
374
+ return SessionManager.continueRecent(cwd, effectiveSessionDir);
347
375
  }
348
376
  // --resume is handled separately (needs picker UI)
349
- // If --session-dir provided without --continue/--resume, create new session there
350
- if (parsed.sessionDir) {
351
- return SessionManager.create(cwd, parsed.sessionDir);
377
+ // If effective session dir is set, create new session there
378
+ if (effectiveSessionDir) {
379
+ return SessionManager.create(cwd, effectiveSessionDir);
352
380
  }
353
381
  // Default case (new session) returns undefined, SDK will create one
354
382
  return undefined;
@@ -681,6 +709,18 @@ export async function main(args) {
681
709
  await listModels(modelRegistry, searchPattern);
682
710
  process.exit(0);
683
711
  }
712
+ // Experimental: List attachable sessions
713
+ if (parsed.listSessions) {
714
+ const { listSessions } = await import("./cli/list-sessions.js");
715
+ await listSessions();
716
+ process.exit(0);
717
+ }
718
+ // Experimental: Attach to existing session
719
+ if (parsed.attach) {
720
+ const { runAttachMode } = await import("./cli/attach-mode.js");
721
+ await runAttachMode(parsed.attach);
722
+ process.exit(0);
723
+ }
684
724
  // Read piped stdin content (if any) - skip for RPC mode which uses stdin for JSON-RPC
685
725
  if (parsed.mode !== "rpc") {
686
726
  const stdinContent = await readPipedStdin();
@@ -723,18 +763,20 @@ export async function main(args) {
723
763
  scopedModels = await resolveModelScope(modelPatterns, modelRegistry);
724
764
  }
725
765
  // Create session manager based on CLI flags
726
- let sessionManager = await createSessionManager(parsed, cwd);
766
+ let sessionManager = await createSessionManager(parsed, cwd, extensionsResult);
727
767
  // Handle --resume: show session picker
728
768
  if (parsed.resume) {
729
769
  // Initialize keybindings so session picker respects user config
730
770
  KeybindingsManager.create();
731
- const selectedPath = await selectSession((onProgress) => SessionManager.list(cwd, parsed.sessionDir, onProgress), SessionManager.listAll);
771
+ // Compute effective session dir for resume (same logic as createSessionManager)
772
+ const effectiveSessionDir = parsed.sessionDir || (await callSessionDirectoryHook(extensionsResult, cwd));
773
+ const selectedPath = await selectSession((onProgress) => SessionManager.list(cwd, effectiveSessionDir, onProgress), SessionManager.listAll);
732
774
  if (!selectedPath) {
733
775
  console.log(chalk.dim("No session selected"));
734
776
  stopThemeWatcher();
735
777
  process.exit(0);
736
778
  }
737
- sessionManager = SessionManager.open(selectedPath);
779
+ sessionManager = SessionManager.open(selectedPath, effectiveSessionDir);
738
780
  }
739
781
  const { options: sessionOptions, cliThinkingFromModel } = buildSessionOptions(parsed, scopedModels, sessionManager, modelRegistry, settingsManager);
740
782
  sessionOptions.authStorage = authStorage;
@@ -749,6 +791,29 @@ export async function main(args) {
749
791
  authStorage.setRuntimeApiKey(sessionOptions.model.provider, parsed.apiKey);
750
792
  }
751
793
  const { session, modelFallbackMessage } = await createAgentSession(sessionOptions);
794
+ // Experimental: Make session attachable if --attachable flag is set
795
+ let cleanupSocketServer = null;
796
+ if (parsed.attachable) {
797
+ const { makeSessionAttachable } = await import("./core/socket-server/index.js");
798
+ cleanupSocketServer = await makeSessionAttachable({
799
+ session,
800
+ enabled: true,
801
+ });
802
+ }
803
+ // Ensure socket server is cleaned up on exit
804
+ const cleanup = async () => {
805
+ if (cleanupSocketServer) {
806
+ await cleanupSocketServer();
807
+ }
808
+ };
809
+ process.on("SIGINT", cleanup);
810
+ process.on("SIGTERM", cleanup);
811
+ process.on("exit", () => {
812
+ if (cleanupSocketServer) {
813
+ // Synchronous cleanup attempt
814
+ cleanupSocketServer().catch(() => { });
815
+ }
816
+ });
752
817
  if (!isInteractive && !session.model) {
753
818
  console.error(chalk.red("No models available."));
754
819
  console.error(chalk.yellow("\nSet an API key environment variable:"));