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.
- package/CHANGELOG.md +2249 -0
- package/README.md +546 -0
- package/dist/cli/args.js +282 -0
- package/dist/cli/config-selector.js +30 -0
- package/dist/cli/file-processor.js +78 -0
- package/dist/cli/list-models.js +91 -0
- package/dist/cli/session-picker.js +31 -0
- package/dist/cli.js +10 -0
- package/dist/config.js +158 -0
- package/dist/core/agent-session.js +2097 -0
- package/dist/core/auth-storage.js +278 -0
- package/dist/core/bash-executor.js +211 -0
- package/dist/core/compaction/branch-summarization.js +241 -0
- package/dist/core/compaction/compaction.js +606 -0
- package/dist/core/compaction/index.js +6 -0
- package/dist/core/compaction/utils.js +137 -0
- package/dist/core/diagnostics.js +1 -0
- package/dist/core/event-bus.js +24 -0
- package/dist/core/exec.js +70 -0
- package/dist/core/export-html/ansi-to-html.js +248 -0
- package/dist/core/export-html/index.js +221 -0
- package/dist/core/export-html/template.css +905 -0
- package/dist/core/export-html/template.html +54 -0
- package/dist/core/export-html/template.js +1549 -0
- package/dist/core/export-html/tool-renderer.js +56 -0
- package/dist/core/export-html/vendor/highlight.min.js +1213 -0
- package/dist/core/export-html/vendor/marked.min.js +6 -0
- package/dist/core/extensions/index.js +8 -0
- package/dist/core/extensions/loader.js +395 -0
- package/dist/core/extensions/runner.js +499 -0
- package/dist/core/extensions/types.js +31 -0
- package/dist/core/extensions/wrapper.js +101 -0
- package/dist/core/footer-data-provider.js +133 -0
- package/dist/core/index.js +8 -0
- package/dist/core/keybindings.js +140 -0
- package/dist/core/messages.js +122 -0
- package/dist/core/model-registry.js +454 -0
- package/dist/core/model-resolver.js +309 -0
- package/dist/core/package-manager.js +1142 -0
- package/dist/core/prompt-templates.js +250 -0
- package/dist/core/resource-loader.js +569 -0
- package/dist/core/sdk.js +225 -0
- package/dist/core/session-manager.js +1078 -0
- package/dist/core/settings-manager.js +430 -0
- package/dist/core/skills.js +339 -0
- package/dist/core/system-prompt.js +136 -0
- package/dist/core/timings.js +24 -0
- package/dist/core/tools/bash.js +226 -0
- package/dist/core/tools/edit-diff.js +242 -0
- package/dist/core/tools/edit.js +145 -0
- package/dist/core/tools/find.js +205 -0
- package/dist/core/tools/grep.js +238 -0
- package/dist/core/tools/index.js +60 -0
- package/dist/core/tools/ls.js +117 -0
- package/dist/core/tools/path-utils.js +52 -0
- package/dist/core/tools/read.js +165 -0
- package/dist/core/tools/truncate.js +204 -0
- package/dist/core/tools/write.js +77 -0
- package/dist/index.js +41 -0
- package/dist/main.js +565 -0
- package/dist/migrations.js +260 -0
- package/dist/modes/index.js +7 -0
- package/dist/modes/interactive/components/armin.js +328 -0
- package/dist/modes/interactive/components/assistant-message.js +86 -0
- package/dist/modes/interactive/components/bash-execution.js +155 -0
- package/dist/modes/interactive/components/bordered-loader.js +47 -0
- package/dist/modes/interactive/components/branch-summary-message.js +41 -0
- package/dist/modes/interactive/components/compaction-summary-message.js +42 -0
- package/dist/modes/interactive/components/config-selector.js +458 -0
- package/dist/modes/interactive/components/countdown-timer.js +27 -0
- package/dist/modes/interactive/components/custom-editor.js +61 -0
- package/dist/modes/interactive/components/custom-message.js +80 -0
- package/dist/modes/interactive/components/diff.js +132 -0
- package/dist/modes/interactive/components/dynamic-border.js +19 -0
- package/dist/modes/interactive/components/extension-editor.js +96 -0
- package/dist/modes/interactive/components/extension-input.js +54 -0
- package/dist/modes/interactive/components/extension-selector.js +70 -0
- package/dist/modes/interactive/components/footer.js +213 -0
- package/dist/modes/interactive/components/index.js +31 -0
- package/dist/modes/interactive/components/keybinding-hints.js +60 -0
- package/dist/modes/interactive/components/login-dialog.js +138 -0
- package/dist/modes/interactive/components/model-selector.js +253 -0
- package/dist/modes/interactive/components/oauth-selector.js +91 -0
- package/dist/modes/interactive/components/scoped-models-selector.js +262 -0
- package/dist/modes/interactive/components/session-selector-search.js +145 -0
- package/dist/modes/interactive/components/session-selector.js +698 -0
- package/dist/modes/interactive/components/settings-selector.js +250 -0
- package/dist/modes/interactive/components/show-images-selector.js +33 -0
- package/dist/modes/interactive/components/skill-invocation-message.js +44 -0
- package/dist/modes/interactive/components/theme-selector.js +43 -0
- package/dist/modes/interactive/components/thinking-selector.js +45 -0
- package/dist/modes/interactive/components/tool-execution.js +608 -0
- package/dist/modes/interactive/components/tree-selector.js +892 -0
- package/dist/modes/interactive/components/user-message-selector.js +109 -0
- package/dist/modes/interactive/components/user-message.js +15 -0
- package/dist/modes/interactive/components/visual-truncate.js +32 -0
- package/dist/modes/interactive/interactive-mode.js +3576 -0
- package/dist/modes/interactive/theme/dark.json +85 -0
- package/dist/modes/interactive/theme/light.json +84 -0
- package/dist/modes/interactive/theme/theme-schema.json +335 -0
- package/dist/modes/interactive/theme/theme.js +938 -0
- package/dist/modes/print-mode.js +96 -0
- package/dist/modes/rpc/rpc-client.js +390 -0
- package/dist/modes/rpc/rpc-mode.js +448 -0
- package/dist/modes/rpc/rpc-types.js +7 -0
- package/dist/utils/changelog.js +86 -0
- package/dist/utils/clipboard-image.js +116 -0
- package/dist/utils/clipboard.js +58 -0
- package/dist/utils/frontmatter.js +25 -0
- package/dist/utils/git.js +5 -0
- package/dist/utils/image-convert.js +34 -0
- package/dist/utils/image-resize.js +180 -0
- package/dist/utils/mime.js +25 -0
- package/dist/utils/photon.js +120 -0
- package/dist/utils/shell.js +164 -0
- package/dist/utils/sleep.js +16 -0
- package/dist/utils/tools-manager.js +186 -0
- package/docs/compaction.md +390 -0
- package/docs/custom-provider.md +538 -0
- package/docs/development.md +69 -0
- package/docs/extensions.md +1733 -0
- package/docs/images/doom-extension.png +0 -0
- package/docs/images/interactive-mode.png +0 -0
- package/docs/images/tree-view.png +0 -0
- package/docs/json.md +79 -0
- package/docs/keybindings.md +162 -0
- package/docs/models.md +193 -0
- package/docs/packages.md +163 -0
- package/docs/prompt-templates.md +67 -0
- package/docs/providers.md +147 -0
- package/docs/rpc.md +1048 -0
- package/docs/sdk.md +957 -0
- package/docs/session.md +412 -0
- package/docs/settings.md +216 -0
- package/docs/shell-aliases.md +13 -0
- package/docs/skills.md +226 -0
- package/docs/terminal-setup.md +65 -0
- package/docs/themes.md +295 -0
- package/docs/tree.md +219 -0
- package/docs/tui.md +887 -0
- package/docs/windows.md +17 -0
- package/examples/README.md +25 -0
- package/examples/extensions/README.md +192 -0
- package/examples/extensions/antigravity-image-gen.ts +414 -0
- package/examples/extensions/auto-commit-on-exit.ts +49 -0
- package/examples/extensions/bookmark.ts +50 -0
- package/examples/extensions/claude-rules.ts +86 -0
- package/examples/extensions/confirm-destructive.ts +59 -0
- package/examples/extensions/custom-compaction.ts +115 -0
- package/examples/extensions/custom-footer.ts +65 -0
- package/examples/extensions/custom-header.ts +73 -0
- package/examples/extensions/custom-provider-anthropic/index.ts +605 -0
- package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
- package/examples/extensions/custom-provider-anthropic/package.json +19 -0
- package/examples/extensions/custom-provider-gitlab-duo/index.ts +350 -0
- package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
- package/examples/extensions/custom-provider-gitlab-duo/test.ts +83 -0
- package/examples/extensions/dirty-repo-guard.ts +56 -0
- package/examples/extensions/doom-overlay/README.md +46 -0
- package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
- package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
- package/examples/extensions/doom-overlay/doom/build.sh +152 -0
- package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
- package/examples/extensions/doom-overlay/doom-component.ts +133 -0
- package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
- package/examples/extensions/doom-overlay/doom-keys.ts +105 -0
- package/examples/extensions/doom-overlay/index.ts +74 -0
- package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
- package/examples/extensions/event-bus.ts +43 -0
- package/examples/extensions/file-trigger.ts +41 -0
- package/examples/extensions/git-checkpoint.ts +53 -0
- package/examples/extensions/handoff.ts +151 -0
- package/examples/extensions/hello.ts +25 -0
- package/examples/extensions/inline-bash.ts +94 -0
- package/examples/extensions/input-transform.ts +43 -0
- package/examples/extensions/interactive-shell.ts +196 -0
- package/examples/extensions/mac-system-theme.ts +47 -0
- package/examples/extensions/message-renderer.ts +60 -0
- package/examples/extensions/modal-editor.ts +86 -0
- package/examples/extensions/model-status.ts +31 -0
- package/examples/extensions/notify.ts +25 -0
- package/examples/extensions/overlay-qa-tests.ts +882 -0
- package/examples/extensions/overlay-test.ts +151 -0
- package/examples/extensions/permission-gate.ts +34 -0
- package/examples/extensions/pirate.ts +47 -0
- package/examples/extensions/plan-mode/README.md +65 -0
- package/examples/extensions/plan-mode/index.ts +341 -0
- package/examples/extensions/plan-mode/utils.ts +168 -0
- package/examples/extensions/preset.ts +399 -0
- package/examples/extensions/protected-paths.ts +30 -0
- package/examples/extensions/qna.ts +120 -0
- package/examples/extensions/question.ts +265 -0
- package/examples/extensions/questionnaire.ts +428 -0
- package/examples/extensions/rainbow-editor.ts +88 -0
- package/examples/extensions/sandbox/index.ts +318 -0
- package/examples/extensions/sandbox/package-lock.json +92 -0
- package/examples/extensions/sandbox/package.json +19 -0
- package/examples/extensions/send-user-message.ts +97 -0
- package/examples/extensions/session-name.ts +27 -0
- package/examples/extensions/shutdown-command.ts +63 -0
- package/examples/extensions/snake.ts +344 -0
- package/examples/extensions/space-invaders.ts +561 -0
- package/examples/extensions/ssh.ts +220 -0
- package/examples/extensions/status-line.ts +40 -0
- package/examples/extensions/subagent/README.md +172 -0
- package/examples/extensions/subagent/agents/planner.md +37 -0
- package/examples/extensions/subagent/agents/reviewer.md +35 -0
- package/examples/extensions/subagent/agents/scout.md +50 -0
- package/examples/extensions/subagent/agents/worker.md +24 -0
- package/examples/extensions/subagent/agents.ts +127 -0
- package/examples/extensions/subagent/index.ts +964 -0
- package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
- package/examples/extensions/subagent/prompts/implement.md +10 -0
- package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
- package/examples/extensions/summarize.ts +196 -0
- package/examples/extensions/timed-confirm.ts +70 -0
- package/examples/extensions/todo.ts +300 -0
- package/examples/extensions/tool-override.ts +144 -0
- package/examples/extensions/tools.ts +147 -0
- package/examples/extensions/trigger-compact.ts +40 -0
- package/examples/extensions/truncated-tool.ts +193 -0
- package/examples/extensions/widget-placement.ts +17 -0
- package/examples/extensions/with-deps/index.ts +36 -0
- package/examples/extensions/with-deps/package-lock.json +31 -0
- package/examples/extensions/with-deps/package.json +22 -0
- package/examples/sdk/01-minimal.ts +22 -0
- package/examples/sdk/02-custom-model.ts +50 -0
- package/examples/sdk/03-custom-prompt.ts +55 -0
- package/examples/sdk/04-skills.ts +46 -0
- package/examples/sdk/05-tools.ts +56 -0
- package/examples/sdk/06-extensions.ts +88 -0
- package/examples/sdk/07-context-files.ts +40 -0
- package/examples/sdk/08-prompt-templates.ts +47 -0
- package/examples/sdk/09-api-keys-and-oauth.ts +48 -0
- package/examples/sdk/10-settings.ts +38 -0
- package/examples/sdk/11-sessions.ts +48 -0
- package/examples/sdk/12-full-control.ts +82 -0
- package/examples/sdk/13-codex-oauth.ts +37 -0
- package/examples/sdk/README.md +144 -0
- package/package.json +85 -0
|
@@ -0,0 +1,309 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Model resolution, scoping, and initial selection
|
|
3
|
+
*/
|
|
4
|
+
import { modelsAreEqual } from "indusagi/ai";
|
|
5
|
+
import chalk from "chalk";
|
|
6
|
+
import { minimatch } from "minimatch";
|
|
7
|
+
import { isValidThinkingLevel } from "../cli/args.js";
|
|
8
|
+
/** Default model IDs for each known provider */
|
|
9
|
+
export const defaultModelPerProvider = {
|
|
10
|
+
"amazon-bedrock": "us.anthropic.claude-opus-4-20250514-v1:0",
|
|
11
|
+
anthropic: "claude-opus-4-5",
|
|
12
|
+
openai: "gpt-5.1-codex",
|
|
13
|
+
"azure-openai-responses": "gpt-5.2",
|
|
14
|
+
"openai-codex": "gpt-5.2-codex",
|
|
15
|
+
google: "gemini-2.5-pro",
|
|
16
|
+
"google-gemini-cli": "gemini-2.5-pro",
|
|
17
|
+
"google-antigravity": "gemini-3-pro-high",
|
|
18
|
+
"google-vertex": "gemini-3-pro-preview",
|
|
19
|
+
"github-copilot": "gpt-4o",
|
|
20
|
+
openrouter: "openai/gpt-5.1-codex",
|
|
21
|
+
"vercel-ai-gateway": "anthropic/claude-opus-4.5",
|
|
22
|
+
xai: "grok-4-fast-non-reasoning",
|
|
23
|
+
groq: "openai/gpt-oss-120b",
|
|
24
|
+
cerebras: "zai-glm-4.6",
|
|
25
|
+
zai: "glm-4.6",
|
|
26
|
+
mistral: "devstral-medium-latest",
|
|
27
|
+
minimax: "MiniMax-M2.1",
|
|
28
|
+
"minimax-cn": "MiniMax-M2.1",
|
|
29
|
+
opencode: "claude-opus-4-5",
|
|
30
|
+
};
|
|
31
|
+
/**
|
|
32
|
+
* Helper to check if a model ID looks like an alias (no date suffix)
|
|
33
|
+
* Dates are typically in format: -20241022 or -20250929
|
|
34
|
+
*/
|
|
35
|
+
function isAlias(id) {
|
|
36
|
+
// Check if ID ends with -latest
|
|
37
|
+
if (id.endsWith("-latest"))
|
|
38
|
+
return true;
|
|
39
|
+
// Check if ID ends with a date pattern (-YYYYMMDD)
|
|
40
|
+
const datePattern = /-\d{8}$/;
|
|
41
|
+
return !datePattern.test(id);
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Try to match a pattern to a model from the available models list.
|
|
45
|
+
* Returns the matched model or undefined if no match found.
|
|
46
|
+
*/
|
|
47
|
+
function tryMatchModel(modelPattern, availableModels) {
|
|
48
|
+
// Check for provider/modelId format (provider is everything before the first /)
|
|
49
|
+
const slashIndex = modelPattern.indexOf("/");
|
|
50
|
+
if (slashIndex !== -1) {
|
|
51
|
+
const provider = modelPattern.substring(0, slashIndex);
|
|
52
|
+
const modelId = modelPattern.substring(slashIndex + 1);
|
|
53
|
+
const providerMatch = availableModels.find((m) => m.provider.toLowerCase() === provider.toLowerCase() && m.id.toLowerCase() === modelId.toLowerCase());
|
|
54
|
+
if (providerMatch) {
|
|
55
|
+
return providerMatch;
|
|
56
|
+
}
|
|
57
|
+
// No exact provider/model match - fall through to other matching
|
|
58
|
+
}
|
|
59
|
+
// Check for exact ID match (case-insensitive)
|
|
60
|
+
const exactMatch = availableModels.find((m) => m.id.toLowerCase() === modelPattern.toLowerCase());
|
|
61
|
+
if (exactMatch) {
|
|
62
|
+
return exactMatch;
|
|
63
|
+
}
|
|
64
|
+
// No exact match - fall back to partial matching
|
|
65
|
+
const matches = availableModels.filter((m) => m.id.toLowerCase().includes(modelPattern.toLowerCase()) ||
|
|
66
|
+
m.name?.toLowerCase().includes(modelPattern.toLowerCase()));
|
|
67
|
+
if (matches.length === 0) {
|
|
68
|
+
return undefined;
|
|
69
|
+
}
|
|
70
|
+
// Separate into aliases and dated versions
|
|
71
|
+
const aliases = matches.filter((m) => isAlias(m.id));
|
|
72
|
+
const datedVersions = matches.filter((m) => !isAlias(m.id));
|
|
73
|
+
if (aliases.length > 0) {
|
|
74
|
+
// Prefer alias - if multiple aliases, pick the one that sorts highest
|
|
75
|
+
aliases.sort((a, b) => b.id.localeCompare(a.id));
|
|
76
|
+
return aliases[0];
|
|
77
|
+
}
|
|
78
|
+
else {
|
|
79
|
+
// No alias found, pick latest dated version
|
|
80
|
+
datedVersions.sort((a, b) => b.id.localeCompare(a.id));
|
|
81
|
+
return datedVersions[0];
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Parse a pattern to extract model and thinking level.
|
|
86
|
+
* Handles models with colons in their IDs (e.g., OpenRouter's :exacto suffix).
|
|
87
|
+
*
|
|
88
|
+
* Algorithm:
|
|
89
|
+
* 1. Try to match full pattern as a model
|
|
90
|
+
* 2. If found, return it with "off" thinking level
|
|
91
|
+
* 3. If not found and has colons, split on last colon:
|
|
92
|
+
* - If suffix is valid thinking level, use it and recurse on prefix
|
|
93
|
+
* - If suffix is invalid, warn and recurse on prefix with "off"
|
|
94
|
+
*
|
|
95
|
+
* @internal Exported for testing
|
|
96
|
+
*/
|
|
97
|
+
export function parseModelPattern(pattern, availableModels) {
|
|
98
|
+
// Try exact match first
|
|
99
|
+
const exactMatch = tryMatchModel(pattern, availableModels);
|
|
100
|
+
if (exactMatch) {
|
|
101
|
+
return { model: exactMatch, thinkingLevel: undefined, warning: undefined };
|
|
102
|
+
}
|
|
103
|
+
// No match - try splitting on last colon if present
|
|
104
|
+
const lastColonIndex = pattern.lastIndexOf(":");
|
|
105
|
+
if (lastColonIndex === -1) {
|
|
106
|
+
// No colons, pattern simply doesn't match any model
|
|
107
|
+
return { model: undefined, thinkingLevel: undefined, warning: undefined };
|
|
108
|
+
}
|
|
109
|
+
const prefix = pattern.substring(0, lastColonIndex);
|
|
110
|
+
const suffix = pattern.substring(lastColonIndex + 1);
|
|
111
|
+
if (isValidThinkingLevel(suffix)) {
|
|
112
|
+
// Valid thinking level - recurse on prefix and use this level
|
|
113
|
+
const result = parseModelPattern(prefix, availableModels);
|
|
114
|
+
if (result.model) {
|
|
115
|
+
// Only use this thinking level if no warning from inner recursion
|
|
116
|
+
return {
|
|
117
|
+
model: result.model,
|
|
118
|
+
thinkingLevel: result.warning ? undefined : suffix,
|
|
119
|
+
warning: result.warning,
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
return result;
|
|
123
|
+
}
|
|
124
|
+
else {
|
|
125
|
+
// Invalid suffix - recurse on prefix and warn
|
|
126
|
+
const result = parseModelPattern(prefix, availableModels);
|
|
127
|
+
if (result.model) {
|
|
128
|
+
return {
|
|
129
|
+
model: result.model,
|
|
130
|
+
thinkingLevel: undefined,
|
|
131
|
+
warning: `Invalid thinking level "${suffix}" in pattern "${pattern}". Using default instead.`,
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
return result;
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Resolve model patterns to actual Model objects with optional thinking levels
|
|
139
|
+
* Format: "pattern:level" where :level is optional
|
|
140
|
+
* For each pattern, finds all matching models and picks the best version:
|
|
141
|
+
* 1. Prefer alias (e.g., claude-sonnet-4-5) over dated versions (claude-sonnet-4-5-20250929)
|
|
142
|
+
* 2. If no alias, pick the latest dated version
|
|
143
|
+
*
|
|
144
|
+
* Supports models with colons in their IDs (e.g., OpenRouter's model:exacto).
|
|
145
|
+
* The algorithm tries to match the full pattern first, then progressively
|
|
146
|
+
* strips colon-suffixes to find a match.
|
|
147
|
+
*/
|
|
148
|
+
export async function resolveModelScope(patterns, modelRegistry) {
|
|
149
|
+
const availableModels = await modelRegistry.getAvailable();
|
|
150
|
+
const scopedModels = [];
|
|
151
|
+
for (const pattern of patterns) {
|
|
152
|
+
// Check if pattern contains glob characters
|
|
153
|
+
if (pattern.includes("*") || pattern.includes("?") || pattern.includes("[")) {
|
|
154
|
+
// Extract optional thinking level suffix (e.g., "provider/*:high")
|
|
155
|
+
const colonIdx = pattern.lastIndexOf(":");
|
|
156
|
+
let globPattern = pattern;
|
|
157
|
+
let thinkingLevel;
|
|
158
|
+
if (colonIdx !== -1) {
|
|
159
|
+
const suffix = pattern.substring(colonIdx + 1);
|
|
160
|
+
if (isValidThinkingLevel(suffix)) {
|
|
161
|
+
thinkingLevel = suffix;
|
|
162
|
+
globPattern = pattern.substring(0, colonIdx);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
// Match against "provider/modelId" format OR just model ID
|
|
166
|
+
// This allows "*sonnet*" to match without requiring "anthropic/*sonnet*"
|
|
167
|
+
const matchingModels = availableModels.filter((m) => {
|
|
168
|
+
const fullId = `${m.provider}/${m.id}`;
|
|
169
|
+
return minimatch(fullId, globPattern, { nocase: true }) || minimatch(m.id, globPattern, { nocase: true });
|
|
170
|
+
});
|
|
171
|
+
if (matchingModels.length === 0) {
|
|
172
|
+
console.warn(chalk.yellow(`Warning: No models match pattern "${pattern}"`));
|
|
173
|
+
continue;
|
|
174
|
+
}
|
|
175
|
+
for (const model of matchingModels) {
|
|
176
|
+
if (!scopedModels.find((sm) => modelsAreEqual(sm.model, model))) {
|
|
177
|
+
scopedModels.push({ model, thinkingLevel });
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
continue;
|
|
181
|
+
}
|
|
182
|
+
const { model, thinkingLevel, warning } = parseModelPattern(pattern, availableModels);
|
|
183
|
+
if (warning) {
|
|
184
|
+
console.warn(chalk.yellow(`Warning: ${warning}`));
|
|
185
|
+
}
|
|
186
|
+
if (!model) {
|
|
187
|
+
console.warn(chalk.yellow(`Warning: No models match pattern "${pattern}"`));
|
|
188
|
+
continue;
|
|
189
|
+
}
|
|
190
|
+
// Avoid duplicates
|
|
191
|
+
if (!scopedModels.find((sm) => modelsAreEqual(sm.model, model))) {
|
|
192
|
+
scopedModels.push({ model, thinkingLevel });
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
return scopedModels;
|
|
196
|
+
}
|
|
197
|
+
/**
|
|
198
|
+
* Find the initial model to use based on priority:
|
|
199
|
+
* 1. CLI args (provider + model)
|
|
200
|
+
* 2. First model from scoped models (if not continuing/resuming)
|
|
201
|
+
* 3. Restored from session (if continuing/resuming)
|
|
202
|
+
* 4. Saved default from settings
|
|
203
|
+
* 5. First available model with valid API key
|
|
204
|
+
*/
|
|
205
|
+
export async function findInitialModel(options) {
|
|
206
|
+
const { cliProvider, cliModel, scopedModels, isContinuing, defaultProvider, defaultModelId, defaultThinkingLevel, modelRegistry, } = options;
|
|
207
|
+
let model;
|
|
208
|
+
let thinkingLevel = "off";
|
|
209
|
+
// 1. CLI args take priority
|
|
210
|
+
if (cliProvider && cliModel) {
|
|
211
|
+
const found = modelRegistry.find(cliProvider, cliModel);
|
|
212
|
+
if (!found) {
|
|
213
|
+
console.error(chalk.red(`Model ${cliProvider}/${cliModel} not found`));
|
|
214
|
+
process.exit(1);
|
|
215
|
+
}
|
|
216
|
+
return { model: found, thinkingLevel: "off", fallbackMessage: undefined };
|
|
217
|
+
}
|
|
218
|
+
// 2. Use first model from scoped models (skip if continuing/resuming)
|
|
219
|
+
if (scopedModels.length > 0 && !isContinuing) {
|
|
220
|
+
return {
|
|
221
|
+
model: scopedModels[0].model,
|
|
222
|
+
thinkingLevel: scopedModels[0].thinkingLevel ?? defaultThinkingLevel ?? "off",
|
|
223
|
+
fallbackMessage: undefined,
|
|
224
|
+
};
|
|
225
|
+
}
|
|
226
|
+
// 3. Try saved default from settings
|
|
227
|
+
if (defaultProvider && defaultModelId) {
|
|
228
|
+
const found = modelRegistry.find(defaultProvider, defaultModelId);
|
|
229
|
+
if (found) {
|
|
230
|
+
model = found;
|
|
231
|
+
if (defaultThinkingLevel) {
|
|
232
|
+
thinkingLevel = defaultThinkingLevel;
|
|
233
|
+
}
|
|
234
|
+
return { model, thinkingLevel, fallbackMessage: undefined };
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
// 4. Try first available model with valid API key
|
|
238
|
+
const availableModels = await modelRegistry.getAvailable();
|
|
239
|
+
if (availableModels.length > 0) {
|
|
240
|
+
// Try to find a default model from known providers
|
|
241
|
+
for (const provider of Object.keys(defaultModelPerProvider)) {
|
|
242
|
+
const defaultId = defaultModelPerProvider[provider];
|
|
243
|
+
const match = availableModels.find((m) => m.provider === provider && m.id === defaultId);
|
|
244
|
+
if (match) {
|
|
245
|
+
return { model: match, thinkingLevel: "off", fallbackMessage: undefined };
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
// If no default found, use first available
|
|
249
|
+
return { model: availableModels[0], thinkingLevel: "off", fallbackMessage: undefined };
|
|
250
|
+
}
|
|
251
|
+
// 5. No model found
|
|
252
|
+
return { model: undefined, thinkingLevel: "off", fallbackMessage: undefined };
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* Restore model from session, with fallback to available models
|
|
256
|
+
*/
|
|
257
|
+
export async function restoreModelFromSession(savedProvider, savedModelId, currentModel, shouldPrintMessages, modelRegistry) {
|
|
258
|
+
const restoredModel = modelRegistry.find(savedProvider, savedModelId);
|
|
259
|
+
// Check if restored model exists and has a valid API key
|
|
260
|
+
const hasApiKey = restoredModel ? !!(await modelRegistry.getApiKey(restoredModel)) : false;
|
|
261
|
+
if (restoredModel && hasApiKey) {
|
|
262
|
+
if (shouldPrintMessages) {
|
|
263
|
+
console.log(chalk.dim(`Restored model: ${savedProvider}/${savedModelId}`));
|
|
264
|
+
}
|
|
265
|
+
return { model: restoredModel, fallbackMessage: undefined };
|
|
266
|
+
}
|
|
267
|
+
// Model not found or no API key - fall back
|
|
268
|
+
const reason = !restoredModel ? "model no longer exists" : "no API key available";
|
|
269
|
+
if (shouldPrintMessages) {
|
|
270
|
+
console.error(chalk.yellow(`Warning: Could not restore model ${savedProvider}/${savedModelId} (${reason}).`));
|
|
271
|
+
}
|
|
272
|
+
// If we already have a model, use it as fallback
|
|
273
|
+
if (currentModel) {
|
|
274
|
+
if (shouldPrintMessages) {
|
|
275
|
+
console.log(chalk.dim(`Falling back to: ${currentModel.provider}/${currentModel.id}`));
|
|
276
|
+
}
|
|
277
|
+
return {
|
|
278
|
+
model: currentModel,
|
|
279
|
+
fallbackMessage: `Could not restore model ${savedProvider}/${savedModelId} (${reason}). Using ${currentModel.provider}/${currentModel.id}.`,
|
|
280
|
+
};
|
|
281
|
+
}
|
|
282
|
+
// Try to find any available model
|
|
283
|
+
const availableModels = await modelRegistry.getAvailable();
|
|
284
|
+
if (availableModels.length > 0) {
|
|
285
|
+
// Try to find a default model from known providers
|
|
286
|
+
let fallbackModel;
|
|
287
|
+
for (const provider of Object.keys(defaultModelPerProvider)) {
|
|
288
|
+
const defaultId = defaultModelPerProvider[provider];
|
|
289
|
+
const match = availableModels.find((m) => m.provider === provider && m.id === defaultId);
|
|
290
|
+
if (match) {
|
|
291
|
+
fallbackModel = match;
|
|
292
|
+
break;
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
// If no default found, use first available
|
|
296
|
+
if (!fallbackModel) {
|
|
297
|
+
fallbackModel = availableModels[0];
|
|
298
|
+
}
|
|
299
|
+
if (shouldPrintMessages) {
|
|
300
|
+
console.log(chalk.dim(`Falling back to: ${fallbackModel.provider}/${fallbackModel.id}`));
|
|
301
|
+
}
|
|
302
|
+
return {
|
|
303
|
+
model: fallbackModel,
|
|
304
|
+
fallbackMessage: `Could not restore model ${savedProvider}/${savedModelId} (${reason}). Using ${fallbackModel.provider}/${fallbackModel.id}.`,
|
|
305
|
+
};
|
|
306
|
+
}
|
|
307
|
+
// No models available
|
|
308
|
+
return { model: undefined, fallbackMessage: undefined };
|
|
309
|
+
}
|