pybao-cli 1.3.3
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/LICENSE +201 -0
- package/README.md +440 -0
- package/README.zh-CN.md +338 -0
- package/cli-acp.js +82 -0
- package/cli.js +105 -0
- package/dist/REPL-WPV32MTF.js +42 -0
- package/dist/REPL-WPV32MTF.js.map +7 -0
- package/dist/acp-75HO2LBV.js +1357 -0
- package/dist/acp-75HO2LBV.js.map +7 -0
- package/dist/agentsValidate-6Z57ARKC.js +373 -0
- package/dist/agentsValidate-6Z57ARKC.js.map +7 -0
- package/dist/ask-NXXXCGY4.js +125 -0
- package/dist/ask-NXXXCGY4.js.map +7 -0
- package/dist/autoUpdater-PJMGNPUG.js +17 -0
- package/dist/autoUpdater-PJMGNPUG.js.map +7 -0
- package/dist/chunk-27GYWUY2.js +72 -0
- package/dist/chunk-27GYWUY2.js.map +7 -0
- package/dist/chunk-3DFBSQIT.js +23 -0
- package/dist/chunk-3DFBSQIT.js.map +7 -0
- package/dist/chunk-3KNGJX7Q.js +794 -0
- package/dist/chunk-3KNGJX7Q.js.map +7 -0
- package/dist/chunk-3PDD7M4T.js +164 -0
- package/dist/chunk-3PDD7M4T.js.map +7 -0
- package/dist/chunk-3ZNSAB7B.js +515 -0
- package/dist/chunk-3ZNSAB7B.js.map +7 -0
- package/dist/chunk-4SNFQYCY.js +511 -0
- package/dist/chunk-4SNFQYCY.js.map +7 -0
- package/dist/chunk-4XPNRLJG.js +1609 -0
- package/dist/chunk-4XPNRLJG.js.map +7 -0
- package/dist/chunk-5P7HBXTD.js +12 -0
- package/dist/chunk-5P7HBXTD.js.map +7 -0
- package/dist/chunk-6RZIUY5K.js +191 -0
- package/dist/chunk-6RZIUY5K.js.map +7 -0
- package/dist/chunk-6WELHKDA.js +240 -0
- package/dist/chunk-6WELHKDA.js.map +7 -0
- package/dist/chunk-7AAE6EO2.js +145 -0
- package/dist/chunk-7AAE6EO2.js.map +7 -0
- package/dist/chunk-A3BVXXA3.js +47 -0
- package/dist/chunk-A3BVXXA3.js.map +7 -0
- package/dist/chunk-A6PUMROK.js +152 -0
- package/dist/chunk-A6PUMROK.js.map +7 -0
- package/dist/chunk-BH3Y62E3.js +11 -0
- package/dist/chunk-BH3Y62E3.js.map +7 -0
- package/dist/chunk-BJSWTHRM.js +16 -0
- package/dist/chunk-BJSWTHRM.js.map +7 -0
- package/dist/chunk-BQA2EOUU.js +124 -0
- package/dist/chunk-BQA2EOUU.js.map +7 -0
- package/dist/chunk-CZZKRPE2.js +19 -0
- package/dist/chunk-CZZKRPE2.js.map +7 -0
- package/dist/chunk-ERMQRV55.js +24 -0
- package/dist/chunk-ERMQRV55.js.map +7 -0
- package/dist/chunk-HB2P6645.js +34 -0
- package/dist/chunk-HB2P6645.js.map +7 -0
- package/dist/chunk-HIRIJ2LQ.js +1256 -0
- package/dist/chunk-HIRIJ2LQ.js.map +7 -0
- package/dist/chunk-ICTEVBLN.js +735 -0
- package/dist/chunk-ICTEVBLN.js.map +7 -0
- package/dist/chunk-JKGOGSFT.js +128 -0
- package/dist/chunk-JKGOGSFT.js.map +7 -0
- package/dist/chunk-JZDE77EH.js +836 -0
- package/dist/chunk-JZDE77EH.js.map +7 -0
- package/dist/chunk-M624LT6O.js +17 -0
- package/dist/chunk-M624LT6O.js.map +7 -0
- package/dist/chunk-OMELVAJD.js +96 -0
- package/dist/chunk-OMELVAJD.js.map +7 -0
- package/dist/chunk-OUXHGDLH.js +95 -0
- package/dist/chunk-OUXHGDLH.js.map +7 -0
- package/dist/chunk-PCXUZ6AT.js +249 -0
- package/dist/chunk-PCXUZ6AT.js.map +7 -0
- package/dist/chunk-Q24ZGKIE.js +1097 -0
- package/dist/chunk-Q24ZGKIE.js.map +7 -0
- package/dist/chunk-QBHEERCF.js +30254 -0
- package/dist/chunk-QBHEERCF.js.map +7 -0
- package/dist/chunk-QIHB5PYM.js +472 -0
- package/dist/chunk-QIHB5PYM.js.map +7 -0
- package/dist/chunk-RQVLBMP7.js +24 -0
- package/dist/chunk-RQVLBMP7.js.map +7 -0
- package/dist/chunk-SWYJOV5E.js +490 -0
- package/dist/chunk-SWYJOV5E.js.map +7 -0
- package/dist/chunk-T6GVXTNQ.js +21 -0
- package/dist/chunk-T6GVXTNQ.js.map +7 -0
- package/dist/chunk-T7GPUZVK.js +766 -0
- package/dist/chunk-T7GPUZVK.js.map +7 -0
- package/dist/chunk-TXFCNQDE.js +2934 -0
- package/dist/chunk-TXFCNQDE.js.map +7 -0
- package/dist/chunk-UNNVICVU.js +95 -0
- package/dist/chunk-UNNVICVU.js.map +7 -0
- package/dist/chunk-UUNVJZWA.js +515 -0
- package/dist/chunk-UUNVJZWA.js.map +7 -0
- package/dist/chunk-VRGR4ZTQ.js +49 -0
- package/dist/chunk-VRGR4ZTQ.js.map +7 -0
- package/dist/chunk-VTVTEE5N.js +2613 -0
- package/dist/chunk-VTVTEE5N.js.map +7 -0
- package/dist/chunk-WPTPPOYN.js +936 -0
- package/dist/chunk-WPTPPOYN.js.map +7 -0
- package/dist/chunk-XXFY63TM.js +196 -0
- package/dist/chunk-XXFY63TM.js.map +7 -0
- package/dist/chunk-Z3HMXDXP.js +654 -0
- package/dist/chunk-Z3HMXDXP.js.map +7 -0
- package/dist/chunk-ZJGXEWKF.js +138 -0
- package/dist/chunk-ZJGXEWKF.js.map +7 -0
- package/dist/cli-RFYBXM7F.js +3917 -0
- package/dist/cli-RFYBXM7F.js.map +7 -0
- package/dist/commands-YOXMODDO.js +46 -0
- package/dist/commands-YOXMODDO.js.map +7 -0
- package/dist/config-5OPX3H2K.js +81 -0
- package/dist/config-5OPX3H2K.js.map +7 -0
- package/dist/context-THRRBPFP.js +30 -0
- package/dist/context-THRRBPFP.js.map +7 -0
- package/dist/costTracker-ELNBZ2DN.js +19 -0
- package/dist/costTracker-ELNBZ2DN.js.map +7 -0
- package/dist/customCommands-4XOZH44N.js +25 -0
- package/dist/customCommands-4XOZH44N.js.map +7 -0
- package/dist/env-EL4KBHMB.js +22 -0
- package/dist/env-EL4KBHMB.js.map +7 -0
- package/dist/index.js +34 -0
- package/dist/index.js.map +7 -0
- package/dist/kodeAgentSessionId-PROTVRBR.js +13 -0
- package/dist/kodeAgentSessionId-PROTVRBR.js.map +7 -0
- package/dist/kodeAgentSessionLoad-UMPV7MC3.js +18 -0
- package/dist/kodeAgentSessionLoad-UMPV7MC3.js.map +7 -0
- package/dist/kodeAgentSessionResume-YJS4FVQM.js +16 -0
- package/dist/kodeAgentSessionResume-YJS4FVQM.js.map +7 -0
- package/dist/kodeAgentStreamJson-3T26CHCP.js +13 -0
- package/dist/kodeAgentStreamJson-3T26CHCP.js.map +7 -0
- package/dist/kodeAgentStreamJsonSession-BZS2VDCY.js +131 -0
- package/dist/kodeAgentStreamJsonSession-BZS2VDCY.js.map +7 -0
- package/dist/kodeAgentStructuredStdio-TNB6U6SP.js +10 -0
- package/dist/kodeAgentStructuredStdio-TNB6U6SP.js.map +7 -0
- package/dist/kodeHooks-VUAWIY2D.js +36 -0
- package/dist/kodeHooks-VUAWIY2D.js.map +7 -0
- package/dist/llm-A3BCM4Q2.js +3118 -0
- package/dist/llm-A3BCM4Q2.js.map +7 -0
- package/dist/llmLazy-ZJSRLZVD.js +15 -0
- package/dist/llmLazy-ZJSRLZVD.js.map +7 -0
- package/dist/loader-HZQBWO74.js +28 -0
- package/dist/loader-HZQBWO74.js.map +7 -0
- package/dist/mcp-XKOJ55B2.js +49 -0
- package/dist/mcp-XKOJ55B2.js.map +7 -0
- package/dist/mentionProcessor-ANYU5MLF.js +211 -0
- package/dist/mentionProcessor-ANYU5MLF.js.map +7 -0
- package/dist/messages-75DL5XBP.js +63 -0
- package/dist/messages-75DL5XBP.js.map +7 -0
- package/dist/model-OPJGJZRC.js +30 -0
- package/dist/model-OPJGJZRC.js.map +7 -0
- package/dist/openai-DT54BAFP.js +29 -0
- package/dist/openai-DT54BAFP.js.map +7 -0
- package/dist/outputStyles-TPFVI52O.js +28 -0
- package/dist/outputStyles-TPFVI52O.js.map +7 -0
- package/dist/package.json +4 -0
- package/dist/pluginRuntime-W74PYSZ4.js +218 -0
- package/dist/pluginRuntime-W74PYSZ4.js.map +7 -0
- package/dist/pluginValidation-FALYRVI2.js +17 -0
- package/dist/pluginValidation-FALYRVI2.js.map +7 -0
- package/dist/prompts-J4TPRMJ3.js +48 -0
- package/dist/prompts-J4TPRMJ3.js.map +7 -0
- package/dist/query-K3QKBVDN.js +50 -0
- package/dist/query-K3QKBVDN.js.map +7 -0
- package/dist/responsesStreaming-HMB74TRD.js +10 -0
- package/dist/responsesStreaming-HMB74TRD.js.map +7 -0
- package/dist/ripgrep-XJGSUBG7.js +17 -0
- package/dist/ripgrep-XJGSUBG7.js.map +7 -0
- package/dist/skillMarketplace-AUGKNCPW.js +37 -0
- package/dist/skillMarketplace-AUGKNCPW.js.map +7 -0
- package/dist/state-DQYRXKTG.js +16 -0
- package/dist/state-DQYRXKTG.js.map +7 -0
- package/dist/theme-MS5HDUBJ.js +14 -0
- package/dist/theme-MS5HDUBJ.js.map +7 -0
- package/dist/toolPermissionContext-GYD5LYFK.js +17 -0
- package/dist/toolPermissionContext-GYD5LYFK.js.map +7 -0
- package/dist/toolPermissionSettings-4MPZVYDR.js +18 -0
- package/dist/toolPermissionSettings-4MPZVYDR.js.map +7 -0
- package/dist/tools-QW6SIJLJ.js +47 -0
- package/dist/tools-QW6SIJLJ.js.map +7 -0
- package/dist/userInput-F2PGBRFU.js +311 -0
- package/dist/userInput-F2PGBRFU.js.map +7 -0
- package/dist/uuid-GYYCQ6QK.js +9 -0
- package/dist/uuid-GYYCQ6QK.js.map +7 -0
- package/dist/yoga.wasm +0 -0
- package/package.json +136 -0
- package/scripts/binary-utils.cjs +62 -0
- package/scripts/cli-acp-wrapper.cjs +82 -0
- package/scripts/cli-wrapper.cjs +105 -0
- package/scripts/postinstall.js +144 -0
- package/yoga.wasm +0 -0
|
@@ -0,0 +1,654 @@
|
|
|
1
|
+
import { createRequire as __pybCreateRequire } from "node:module";
|
|
2
|
+
const require = __pybCreateRequire(import.meta.url);
|
|
3
|
+
import {
|
|
4
|
+
getGlobalConfig,
|
|
5
|
+
saveGlobalConfig
|
|
6
|
+
} from "./chunk-JZDE77EH.js";
|
|
7
|
+
import {
|
|
8
|
+
debug
|
|
9
|
+
} from "./chunk-3KNGJX7Q.js";
|
|
10
|
+
import {
|
|
11
|
+
logError
|
|
12
|
+
} from "./chunk-TXFCNQDE.js";
|
|
13
|
+
|
|
14
|
+
// src/utils/model/index.ts
|
|
15
|
+
import { memoize } from "lodash-es";
|
|
16
|
+
var USE_BEDROCK = !!(process.env.KODE_USE_BEDROCK ?? process.env.CLAUDE_CODE_USE_BEDROCK);
|
|
17
|
+
var USE_VERTEX = !!(process.env.KODE_USE_VERTEX ?? process.env.CLAUDE_CODE_USE_VERTEX);
|
|
18
|
+
var DEFAULT_MODEL_CONFIG = {
|
|
19
|
+
bedrock: "us.anthropic.claude-3-7-sonnet-20250219-v1:0",
|
|
20
|
+
vertex: "claude-3-7-sonnet@20250219",
|
|
21
|
+
firstParty: "claude-sonnet-4-20250514"
|
|
22
|
+
};
|
|
23
|
+
async function getModelConfig() {
|
|
24
|
+
return DEFAULT_MODEL_CONFIG;
|
|
25
|
+
}
|
|
26
|
+
var getSlowAndCapableModel = memoize(async () => {
|
|
27
|
+
const config = await getGlobalConfig();
|
|
28
|
+
const modelManager = new ModelManager(config);
|
|
29
|
+
const model = modelManager.getMainAgentModel();
|
|
30
|
+
if (model) {
|
|
31
|
+
return model;
|
|
32
|
+
}
|
|
33
|
+
const modelConfig = await getModelConfig();
|
|
34
|
+
if (USE_BEDROCK) return modelConfig.bedrock;
|
|
35
|
+
if (USE_VERTEX) return modelConfig.vertex;
|
|
36
|
+
return modelConfig.firstParty;
|
|
37
|
+
});
|
|
38
|
+
async function isDefaultSlowAndCapableModel() {
|
|
39
|
+
return !process.env.ANTHROPIC_MODEL || process.env.ANTHROPIC_MODEL === await getSlowAndCapableModel();
|
|
40
|
+
}
|
|
41
|
+
function getVertexRegionForModel(model) {
|
|
42
|
+
if (model?.startsWith("claude-3-5-haiku")) {
|
|
43
|
+
return process.env.VERTEX_REGION_CLAUDE_3_5_HAIKU;
|
|
44
|
+
} else if (model?.startsWith("claude-3-5-sonnet")) {
|
|
45
|
+
return process.env.VERTEX_REGION_CLAUDE_3_5_SONNET;
|
|
46
|
+
} else if (model?.startsWith("claude-3-7-sonnet")) {
|
|
47
|
+
return process.env.VERTEX_REGION_CLAUDE_3_7_SONNET;
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
var ModelManager = class {
|
|
51
|
+
config;
|
|
52
|
+
modelProfiles;
|
|
53
|
+
constructor(config) {
|
|
54
|
+
this.config = config;
|
|
55
|
+
this.modelProfiles = config.modelProfiles || [];
|
|
56
|
+
}
|
|
57
|
+
getCurrentModel() {
|
|
58
|
+
const mainModelName = this.config.modelPointers?.main;
|
|
59
|
+
if (mainModelName) {
|
|
60
|
+
const profile = this.findModelProfile(mainModelName);
|
|
61
|
+
if (profile && profile.isActive) {
|
|
62
|
+
return profile.modelName;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
return this.getMainAgentModel();
|
|
66
|
+
}
|
|
67
|
+
getMainAgentModel() {
|
|
68
|
+
const mainModelName = this.config.modelPointers?.main;
|
|
69
|
+
if (mainModelName) {
|
|
70
|
+
const profile = this.findModelProfile(mainModelName);
|
|
71
|
+
if (profile && profile.isActive) {
|
|
72
|
+
return profile.modelName;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
const activeProfile = this.modelProfiles.find((p) => p.isActive);
|
|
76
|
+
if (activeProfile) {
|
|
77
|
+
return activeProfile.modelName;
|
|
78
|
+
}
|
|
79
|
+
return null;
|
|
80
|
+
}
|
|
81
|
+
getTaskToolModel() {
|
|
82
|
+
const taskModelName = this.config.modelPointers?.task;
|
|
83
|
+
if (taskModelName) {
|
|
84
|
+
const profile = this.findModelProfile(taskModelName);
|
|
85
|
+
if (profile && profile.isActive) {
|
|
86
|
+
return profile.modelName;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
return this.getMainAgentModel();
|
|
90
|
+
}
|
|
91
|
+
switchToNextModelWithContextCheck(currentContextTokens = 0) {
|
|
92
|
+
const allProfiles = this.getAllConfiguredModels();
|
|
93
|
+
if (allProfiles.length === 0) {
|
|
94
|
+
return {
|
|
95
|
+
success: false,
|
|
96
|
+
modelName: null,
|
|
97
|
+
previousModelName: null,
|
|
98
|
+
contextOverflow: false,
|
|
99
|
+
usagePercentage: 0,
|
|
100
|
+
currentContextTokens
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
allProfiles.sort((a, b) => a.createdAt - b.createdAt);
|
|
104
|
+
const currentMainModelName = this.config.modelPointers?.main;
|
|
105
|
+
const currentModel = currentMainModelName ? this.findModelProfile(currentMainModelName) : null;
|
|
106
|
+
const previousModelName = currentModel?.name || null;
|
|
107
|
+
const budgetForModel = (model) => {
|
|
108
|
+
const contextLength = Number(model.contextLength);
|
|
109
|
+
if (!Number.isFinite(contextLength) || contextLength <= 0) {
|
|
110
|
+
return { budgetTokens: null, usagePercentage: 0, compatible: true };
|
|
111
|
+
}
|
|
112
|
+
const budgetTokens = Math.floor(contextLength * 0.9);
|
|
113
|
+
const usagePercentage = budgetTokens > 0 ? currentContextTokens / budgetTokens * 100 : 0;
|
|
114
|
+
return {
|
|
115
|
+
budgetTokens,
|
|
116
|
+
usagePercentage,
|
|
117
|
+
compatible: budgetTokens > 0 ? currentContextTokens <= budgetTokens : true
|
|
118
|
+
};
|
|
119
|
+
};
|
|
120
|
+
const currentIndex = currentMainModelName ? allProfiles.findIndex((p) => p.modelName === currentMainModelName) : -1;
|
|
121
|
+
const startIndex = currentIndex >= 0 ? currentIndex : -1;
|
|
122
|
+
if (allProfiles.length === 1) {
|
|
123
|
+
return {
|
|
124
|
+
success: false,
|
|
125
|
+
modelName: null,
|
|
126
|
+
previousModelName,
|
|
127
|
+
contextOverflow: false,
|
|
128
|
+
usagePercentage: 0,
|
|
129
|
+
currentContextTokens
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
const maxOffsets = startIndex === -1 ? allProfiles.length : allProfiles.length - 1;
|
|
133
|
+
const skippedModels = [];
|
|
134
|
+
let selected = null;
|
|
135
|
+
let selectedUsagePercentage = 0;
|
|
136
|
+
for (let offset = 1; offset <= maxOffsets; offset++) {
|
|
137
|
+
const candidateIndex = (startIndex + offset + allProfiles.length) % allProfiles.length;
|
|
138
|
+
const candidate = allProfiles[candidateIndex];
|
|
139
|
+
if (!candidate) continue;
|
|
140
|
+
const { budgetTokens, usagePercentage, compatible } = budgetForModel(candidate);
|
|
141
|
+
if (compatible) {
|
|
142
|
+
selected = candidate;
|
|
143
|
+
selectedUsagePercentage = usagePercentage;
|
|
144
|
+
break;
|
|
145
|
+
}
|
|
146
|
+
skippedModels.push({
|
|
147
|
+
name: candidate.name,
|
|
148
|
+
provider: candidate.provider,
|
|
149
|
+
contextLength: candidate.contextLength,
|
|
150
|
+
budgetTokens,
|
|
151
|
+
usagePercentage
|
|
152
|
+
});
|
|
153
|
+
}
|
|
154
|
+
if (!selected) {
|
|
155
|
+
const firstSkipped = skippedModels[0];
|
|
156
|
+
return {
|
|
157
|
+
success: false,
|
|
158
|
+
modelName: null,
|
|
159
|
+
previousModelName,
|
|
160
|
+
contextOverflow: true,
|
|
161
|
+
usagePercentage: firstSkipped?.usagePercentage ?? 0,
|
|
162
|
+
currentContextTokens,
|
|
163
|
+
skippedModels
|
|
164
|
+
};
|
|
165
|
+
}
|
|
166
|
+
if (!selected.isActive) {
|
|
167
|
+
selected.isActive = true;
|
|
168
|
+
}
|
|
169
|
+
this.setPointer("main", selected.modelName);
|
|
170
|
+
this.updateLastUsed(selected.modelName);
|
|
171
|
+
return {
|
|
172
|
+
success: true,
|
|
173
|
+
modelName: selected.name,
|
|
174
|
+
previousModelName,
|
|
175
|
+
contextOverflow: false,
|
|
176
|
+
usagePercentage: selectedUsagePercentage,
|
|
177
|
+
currentContextTokens,
|
|
178
|
+
skippedModels
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
switchToNextModel(currentContextTokens = 0) {
|
|
182
|
+
const result = this.switchToNextModelWithContextCheck(currentContextTokens);
|
|
183
|
+
const formatTokens = (tokens) => {
|
|
184
|
+
if (!Number.isFinite(tokens)) return "unknown";
|
|
185
|
+
if (tokens >= 1e3) return `${Math.round(tokens / 1e3)}k`;
|
|
186
|
+
return String(Math.round(tokens));
|
|
187
|
+
};
|
|
188
|
+
const allModels = this.getAllConfiguredModels();
|
|
189
|
+
if (allModels.length === 0) {
|
|
190
|
+
return {
|
|
191
|
+
success: false,
|
|
192
|
+
modelName: null,
|
|
193
|
+
blocked: false,
|
|
194
|
+
message: "\u274C No models configured. Use /model to add models."
|
|
195
|
+
};
|
|
196
|
+
}
|
|
197
|
+
if (allModels.length === 1) {
|
|
198
|
+
return {
|
|
199
|
+
success: false,
|
|
200
|
+
modelName: null,
|
|
201
|
+
blocked: false,
|
|
202
|
+
message: `\u26A0\uFE0F Only one model configured (${allModels[0].modelName}). Use /model to add more models for switching.`
|
|
203
|
+
};
|
|
204
|
+
}
|
|
205
|
+
const currentModel = this.findModelProfile(this.config.modelPointers?.main);
|
|
206
|
+
const modelsSorted = [...allModels].sort(
|
|
207
|
+
(a, b) => a.createdAt - b.createdAt
|
|
208
|
+
);
|
|
209
|
+
const currentIndex = modelsSorted.findIndex(
|
|
210
|
+
(m) => m.modelName === currentModel?.modelName
|
|
211
|
+
);
|
|
212
|
+
const totalModels = modelsSorted.length;
|
|
213
|
+
if (result.success && result.modelName) {
|
|
214
|
+
const skippedCount = result.skippedModels?.length ?? 0;
|
|
215
|
+
const skippedSuffix = skippedCount > 0 ? ` \xB7 skipped ${skippedCount} incompatible` : "";
|
|
216
|
+
const contextSuffix = currentModel?.contextLength && result.currentContextTokens ? ` \xB7 context ~${formatTokens(result.currentContextTokens)}/${formatTokens(currentModel.contextLength)}` : "";
|
|
217
|
+
return {
|
|
218
|
+
success: true,
|
|
219
|
+
modelName: result.modelName,
|
|
220
|
+
blocked: false,
|
|
221
|
+
message: `\u2705 Switched to ${result.modelName} (${currentIndex + 1}/${totalModels})${currentModel?.provider ? ` [${currentModel.provider}]` : ""}${skippedSuffix}${contextSuffix}`
|
|
222
|
+
};
|
|
223
|
+
}
|
|
224
|
+
if (result.contextOverflow) {
|
|
225
|
+
const attempted = result.skippedModels?.[0];
|
|
226
|
+
const attemptedContext = attempted?.contextLength;
|
|
227
|
+
const attemptedBudget = attempted?.budgetTokens;
|
|
228
|
+
const currentLabel = currentModel?.name || currentModel?.modelName || "current model";
|
|
229
|
+
const attemptedText = attempted ? `Can't switch to ${attempted.name}: current ~${formatTokens(result.currentContextTokens)} tokens exceeds safe budget (~${formatTokens(attemptedBudget ?? 0)} tokens, 90% of ${formatTokens(attemptedContext ?? 0)}).` : `Can't switch models due to context size (~${formatTokens(result.currentContextTokens)} tokens).`;
|
|
230
|
+
return {
|
|
231
|
+
success: false,
|
|
232
|
+
modelName: null,
|
|
233
|
+
blocked: true,
|
|
234
|
+
message: `\u26A0\uFE0F ${attemptedText} Keeping ${currentLabel}.`
|
|
235
|
+
};
|
|
236
|
+
}
|
|
237
|
+
return {
|
|
238
|
+
success: false,
|
|
239
|
+
modelName: null,
|
|
240
|
+
blocked: false,
|
|
241
|
+
message: "\u274C Failed to switch models"
|
|
242
|
+
};
|
|
243
|
+
}
|
|
244
|
+
revertToPreviousModel(previousModelName) {
|
|
245
|
+
const previousModel = this.modelProfiles.find(
|
|
246
|
+
(p) => p.name === previousModelName && p.isActive
|
|
247
|
+
);
|
|
248
|
+
if (!previousModel) {
|
|
249
|
+
return false;
|
|
250
|
+
}
|
|
251
|
+
this.setPointer("main", previousModel.modelName);
|
|
252
|
+
this.updateLastUsed(previousModel.modelName);
|
|
253
|
+
return true;
|
|
254
|
+
}
|
|
255
|
+
analyzeContextCompatibility(model, contextTokens) {
|
|
256
|
+
const usableContext = Math.floor(model.contextLength * 0.8);
|
|
257
|
+
const usagePercentage = contextTokens / usableContext * 100;
|
|
258
|
+
if (usagePercentage <= 70) {
|
|
259
|
+
return {
|
|
260
|
+
compatible: true,
|
|
261
|
+
severity: "safe",
|
|
262
|
+
usagePercentage,
|
|
263
|
+
recommendation: "Full context preserved"
|
|
264
|
+
};
|
|
265
|
+
} else if (usagePercentage <= 90) {
|
|
266
|
+
return {
|
|
267
|
+
compatible: true,
|
|
268
|
+
severity: "warning",
|
|
269
|
+
usagePercentage,
|
|
270
|
+
recommendation: "Context usage high, consider compression"
|
|
271
|
+
};
|
|
272
|
+
} else {
|
|
273
|
+
return {
|
|
274
|
+
compatible: false,
|
|
275
|
+
severity: "critical",
|
|
276
|
+
usagePercentage,
|
|
277
|
+
recommendation: "Auto-compression or message truncation required"
|
|
278
|
+
};
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
switchToNextModelWithAnalysis(currentContextTokens = 0) {
|
|
282
|
+
const result = this.switchToNextModel(currentContextTokens);
|
|
283
|
+
if (!result.success || !result.modelName) {
|
|
284
|
+
return {
|
|
285
|
+
modelName: null,
|
|
286
|
+
contextAnalysis: null,
|
|
287
|
+
requiresCompression: false,
|
|
288
|
+
estimatedTokensAfterSwitch: 0
|
|
289
|
+
};
|
|
290
|
+
}
|
|
291
|
+
const newModel = this.getModel("main");
|
|
292
|
+
if (!newModel) {
|
|
293
|
+
return {
|
|
294
|
+
modelName: result.modelName,
|
|
295
|
+
contextAnalysis: null,
|
|
296
|
+
requiresCompression: false,
|
|
297
|
+
estimatedTokensAfterSwitch: currentContextTokens
|
|
298
|
+
};
|
|
299
|
+
}
|
|
300
|
+
const analysis = this.analyzeContextCompatibility(
|
|
301
|
+
newModel,
|
|
302
|
+
currentContextTokens
|
|
303
|
+
);
|
|
304
|
+
return {
|
|
305
|
+
modelName: result.modelName,
|
|
306
|
+
contextAnalysis: analysis,
|
|
307
|
+
requiresCompression: analysis.severity === "critical",
|
|
308
|
+
estimatedTokensAfterSwitch: currentContextTokens
|
|
309
|
+
};
|
|
310
|
+
}
|
|
311
|
+
canModelHandleContext(model, contextTokens) {
|
|
312
|
+
const analysis = this.analyzeContextCompatibility(model, contextTokens);
|
|
313
|
+
return analysis.compatible;
|
|
314
|
+
}
|
|
315
|
+
findModelWithSufficientContext(models, contextTokens) {
|
|
316
|
+
return models.find((model) => this.canModelHandleContext(model, contextTokens)) || null;
|
|
317
|
+
}
|
|
318
|
+
getModelForContext(contextType) {
|
|
319
|
+
switch (contextType) {
|
|
320
|
+
case "terminal":
|
|
321
|
+
return this.getCurrentModel();
|
|
322
|
+
case "main-agent":
|
|
323
|
+
return this.getMainAgentModel();
|
|
324
|
+
case "task-tool":
|
|
325
|
+
return this.getTaskToolModel();
|
|
326
|
+
default:
|
|
327
|
+
return this.getMainAgentModel();
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
getActiveModelProfiles() {
|
|
331
|
+
return this.modelProfiles.filter((p) => p.isActive);
|
|
332
|
+
}
|
|
333
|
+
hasConfiguredModels() {
|
|
334
|
+
return this.getActiveModelProfiles().length > 0;
|
|
335
|
+
}
|
|
336
|
+
getModel(pointer) {
|
|
337
|
+
const pointerId = this.config.modelPointers?.[pointer];
|
|
338
|
+
if (!pointerId) {
|
|
339
|
+
return this.getDefaultModel();
|
|
340
|
+
}
|
|
341
|
+
const profile = this.findModelProfile(pointerId);
|
|
342
|
+
return profile && profile.isActive ? profile : this.getDefaultModel();
|
|
343
|
+
}
|
|
344
|
+
getModelName(pointer) {
|
|
345
|
+
const profile = this.getModel(pointer);
|
|
346
|
+
return profile ? profile.modelName : null;
|
|
347
|
+
}
|
|
348
|
+
getCompactModel() {
|
|
349
|
+
return this.getModelName("compact") || this.getModelName("main");
|
|
350
|
+
}
|
|
351
|
+
getQuickModel() {
|
|
352
|
+
return this.getModelName("quick") || this.getModelName("task") || this.getModelName("main");
|
|
353
|
+
}
|
|
354
|
+
async addModel(config) {
|
|
355
|
+
const existingByModelName = this.modelProfiles.find(
|
|
356
|
+
(p) => p.modelName === config.modelName
|
|
357
|
+
);
|
|
358
|
+
if (existingByModelName) {
|
|
359
|
+
throw new Error(
|
|
360
|
+
`Model with modelName '${config.modelName}' already exists: ${existingByModelName.name}`
|
|
361
|
+
);
|
|
362
|
+
}
|
|
363
|
+
const existingByName = this.modelProfiles.find((p) => p.name === config.name);
|
|
364
|
+
if (existingByName) {
|
|
365
|
+
throw new Error(`Model with name '${config.name}' already exists`);
|
|
366
|
+
}
|
|
367
|
+
const newModel = {
|
|
368
|
+
...config,
|
|
369
|
+
createdAt: Date.now(),
|
|
370
|
+
isActive: true
|
|
371
|
+
};
|
|
372
|
+
this.modelProfiles.push(newModel);
|
|
373
|
+
if (this.modelProfiles.length === 1) {
|
|
374
|
+
this.config.modelPointers = {
|
|
375
|
+
main: config.modelName,
|
|
376
|
+
task: config.modelName,
|
|
377
|
+
compact: config.modelName,
|
|
378
|
+
quick: config.modelName
|
|
379
|
+
};
|
|
380
|
+
this.config.defaultModelName = config.modelName;
|
|
381
|
+
} else {
|
|
382
|
+
if (!this.config.modelPointers) {
|
|
383
|
+
this.config.modelPointers = {
|
|
384
|
+
main: config.modelName,
|
|
385
|
+
task: "",
|
|
386
|
+
compact: "",
|
|
387
|
+
quick: ""
|
|
388
|
+
};
|
|
389
|
+
} else {
|
|
390
|
+
this.config.modelPointers.main = config.modelName;
|
|
391
|
+
}
|
|
392
|
+
}
|
|
393
|
+
this.saveConfig();
|
|
394
|
+
return config.modelName;
|
|
395
|
+
}
|
|
396
|
+
setPointer(pointer, modelName) {
|
|
397
|
+
if (!this.findModelProfile(modelName)) {
|
|
398
|
+
throw new Error(`Model '${modelName}' not found`);
|
|
399
|
+
}
|
|
400
|
+
if (!this.config.modelPointers) {
|
|
401
|
+
this.config.modelPointers = {
|
|
402
|
+
main: "",
|
|
403
|
+
task: "",
|
|
404
|
+
compact: "",
|
|
405
|
+
quick: ""
|
|
406
|
+
};
|
|
407
|
+
}
|
|
408
|
+
this.config.modelPointers[pointer] = modelName;
|
|
409
|
+
this.saveConfig();
|
|
410
|
+
}
|
|
411
|
+
getAvailableModels() {
|
|
412
|
+
return this.modelProfiles.filter((p) => p.isActive);
|
|
413
|
+
}
|
|
414
|
+
getAllConfiguredModels() {
|
|
415
|
+
return this.modelProfiles;
|
|
416
|
+
}
|
|
417
|
+
getAllAvailableModelNames() {
|
|
418
|
+
return this.getAvailableModels().map((p) => p.modelName);
|
|
419
|
+
}
|
|
420
|
+
getAllConfiguredModelNames() {
|
|
421
|
+
return this.getAllConfiguredModels().map((p) => p.modelName);
|
|
422
|
+
}
|
|
423
|
+
getModelSwitchingDebugInfo() {
|
|
424
|
+
const availableModels = this.getAvailableModels();
|
|
425
|
+
const currentMainModelName = this.config.modelPointers?.main;
|
|
426
|
+
return {
|
|
427
|
+
totalModels: this.modelProfiles.length,
|
|
428
|
+
activeModels: availableModels.length,
|
|
429
|
+
inactiveModels: this.modelProfiles.length - availableModels.length,
|
|
430
|
+
currentMainModel: currentMainModelName || null,
|
|
431
|
+
availableModels: this.modelProfiles.map((p) => ({
|
|
432
|
+
name: p.name,
|
|
433
|
+
modelName: p.modelName,
|
|
434
|
+
provider: p.provider,
|
|
435
|
+
isActive: p.isActive,
|
|
436
|
+
lastUsed: p.lastUsed
|
|
437
|
+
})),
|
|
438
|
+
modelPointers: this.config.modelPointers || {}
|
|
439
|
+
};
|
|
440
|
+
}
|
|
441
|
+
removeModel(modelName) {
|
|
442
|
+
this.modelProfiles = this.modelProfiles.filter(
|
|
443
|
+
(p) => p.modelName !== modelName
|
|
444
|
+
);
|
|
445
|
+
if (this.config.modelPointers) {
|
|
446
|
+
Object.keys(this.config.modelPointers).forEach((pointer) => {
|
|
447
|
+
if (this.config.modelPointers[pointer] === modelName) {
|
|
448
|
+
this.config.modelPointers[pointer] = this.config.defaultModelName || "";
|
|
449
|
+
}
|
|
450
|
+
});
|
|
451
|
+
}
|
|
452
|
+
this.saveConfig();
|
|
453
|
+
}
|
|
454
|
+
getDefaultModel() {
|
|
455
|
+
if (this.config.defaultModelId) {
|
|
456
|
+
const profile = this.findModelProfile(this.config.defaultModelId);
|
|
457
|
+
if (profile && profile.isActive) {
|
|
458
|
+
return profile;
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
return this.modelProfiles.find((p) => p.isActive) || null;
|
|
462
|
+
}
|
|
463
|
+
saveConfig() {
|
|
464
|
+
const updatedConfig = {
|
|
465
|
+
...this.config,
|
|
466
|
+
modelProfiles: this.modelProfiles
|
|
467
|
+
};
|
|
468
|
+
saveGlobalConfig(updatedConfig);
|
|
469
|
+
}
|
|
470
|
+
async getFallbackModel() {
|
|
471
|
+
const modelConfig = await getModelConfig();
|
|
472
|
+
if (USE_BEDROCK) return modelConfig.bedrock;
|
|
473
|
+
if (USE_VERTEX) return modelConfig.vertex;
|
|
474
|
+
return modelConfig.firstParty;
|
|
475
|
+
}
|
|
476
|
+
resolveModel(modelParam) {
|
|
477
|
+
if (["main", "task", "compact", "quick"].includes(modelParam)) {
|
|
478
|
+
const pointerId = this.config.modelPointers?.[modelParam];
|
|
479
|
+
if (pointerId) {
|
|
480
|
+
let profile2 = this.findModelProfile(pointerId);
|
|
481
|
+
if (!profile2) {
|
|
482
|
+
profile2 = this.findModelProfileByModelName(pointerId);
|
|
483
|
+
}
|
|
484
|
+
if (profile2 && profile2.isActive) {
|
|
485
|
+
return profile2;
|
|
486
|
+
}
|
|
487
|
+
}
|
|
488
|
+
return this.getDefaultModel();
|
|
489
|
+
}
|
|
490
|
+
let profile = this.findModelProfile(modelParam);
|
|
491
|
+
if (profile && profile.isActive) {
|
|
492
|
+
return profile;
|
|
493
|
+
}
|
|
494
|
+
profile = this.findModelProfileByModelName(modelParam);
|
|
495
|
+
if (profile && profile.isActive) {
|
|
496
|
+
return profile;
|
|
497
|
+
}
|
|
498
|
+
profile = this.findModelProfileByName(modelParam);
|
|
499
|
+
if (profile && profile.isActive) {
|
|
500
|
+
return profile;
|
|
501
|
+
}
|
|
502
|
+
if (typeof modelParam === "string") {
|
|
503
|
+
const qualified = this.resolveProviderQualifiedModel(modelParam);
|
|
504
|
+
if (qualified && qualified.isActive) {
|
|
505
|
+
return qualified;
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
return this.getDefaultModel();
|
|
509
|
+
}
|
|
510
|
+
resolveModelWithInfo(modelParam) {
|
|
511
|
+
const isPointer = ["main", "task", "compact", "quick"].includes(modelParam);
|
|
512
|
+
if (isPointer) {
|
|
513
|
+
const pointerId = this.config.modelPointers?.[modelParam];
|
|
514
|
+
if (!pointerId) {
|
|
515
|
+
return {
|
|
516
|
+
success: false,
|
|
517
|
+
profile: null,
|
|
518
|
+
error: `Model pointer '${modelParam}' is not configured. Use /model to set up models.`
|
|
519
|
+
};
|
|
520
|
+
}
|
|
521
|
+
let profile = this.findModelProfile(pointerId);
|
|
522
|
+
if (!profile) {
|
|
523
|
+
profile = this.findModelProfileByModelName(pointerId);
|
|
524
|
+
}
|
|
525
|
+
if (!profile) {
|
|
526
|
+
return {
|
|
527
|
+
success: false,
|
|
528
|
+
profile: null,
|
|
529
|
+
error: `Model pointer '${modelParam}' points to invalid model '${pointerId}'. Use /model to reconfigure.`
|
|
530
|
+
};
|
|
531
|
+
}
|
|
532
|
+
if (!profile.isActive) {
|
|
533
|
+
return {
|
|
534
|
+
success: false,
|
|
535
|
+
profile: null,
|
|
536
|
+
error: `Model '${profile.name}' (pointed by '${modelParam}') is inactive. Use /model to activate it.`
|
|
537
|
+
};
|
|
538
|
+
}
|
|
539
|
+
return {
|
|
540
|
+
success: true,
|
|
541
|
+
profile
|
|
542
|
+
};
|
|
543
|
+
} else {
|
|
544
|
+
let profile = this.findModelProfile(modelParam);
|
|
545
|
+
if (!profile) {
|
|
546
|
+
profile = this.findModelProfileByModelName(modelParam);
|
|
547
|
+
}
|
|
548
|
+
if (!profile) {
|
|
549
|
+
profile = this.findModelProfileByName(modelParam);
|
|
550
|
+
}
|
|
551
|
+
if (!profile && typeof modelParam === "string") {
|
|
552
|
+
profile = this.resolveProviderQualifiedModel(modelParam);
|
|
553
|
+
}
|
|
554
|
+
if (!profile) {
|
|
555
|
+
return {
|
|
556
|
+
success: false,
|
|
557
|
+
profile: null,
|
|
558
|
+
error: `Model '${modelParam}' not found. Use /model to add models, or run 'kode models list' to see configured profiles.`
|
|
559
|
+
};
|
|
560
|
+
}
|
|
561
|
+
if (!profile.isActive) {
|
|
562
|
+
return {
|
|
563
|
+
success: false,
|
|
564
|
+
profile: null,
|
|
565
|
+
error: `Model '${profile.name}' is inactive. Use /model to activate it.`
|
|
566
|
+
};
|
|
567
|
+
}
|
|
568
|
+
return {
|
|
569
|
+
success: true,
|
|
570
|
+
profile
|
|
571
|
+
};
|
|
572
|
+
}
|
|
573
|
+
}
|
|
574
|
+
resolveProviderQualifiedModel(input) {
|
|
575
|
+
const trimmed = input.trim();
|
|
576
|
+
const colonIndex = trimmed.indexOf(":");
|
|
577
|
+
if (colonIndex <= 0 || colonIndex >= trimmed.length - 1) return null;
|
|
578
|
+
const provider = trimmed.slice(0, colonIndex).trim().toLowerCase();
|
|
579
|
+
const modelOrName = trimmed.slice(colonIndex + 1).trim();
|
|
580
|
+
if (!provider || !modelOrName) return null;
|
|
581
|
+
const providerProfiles = this.modelProfiles.filter(
|
|
582
|
+
(p) => String(p.provider).trim().toLowerCase() === provider
|
|
583
|
+
);
|
|
584
|
+
if (providerProfiles.length === 0) return null;
|
|
585
|
+
const byModelName = providerProfiles.find((p) => p.modelName === modelOrName);
|
|
586
|
+
if (byModelName) return byModelName;
|
|
587
|
+
const byName = providerProfiles.find((p) => p.name === modelOrName);
|
|
588
|
+
if (byName) return byName;
|
|
589
|
+
return null;
|
|
590
|
+
}
|
|
591
|
+
findModelProfile(modelName) {
|
|
592
|
+
return this.modelProfiles.find((p) => p.modelName === modelName) || null;
|
|
593
|
+
}
|
|
594
|
+
findModelProfileByModelName(modelName) {
|
|
595
|
+
return this.modelProfiles.find((p) => p.modelName === modelName) || null;
|
|
596
|
+
}
|
|
597
|
+
findModelProfileByName(name) {
|
|
598
|
+
return this.modelProfiles.find((p) => p.name === name) || null;
|
|
599
|
+
}
|
|
600
|
+
updateLastUsed(modelName) {
|
|
601
|
+
const profile = this.findModelProfile(modelName);
|
|
602
|
+
if (profile) {
|
|
603
|
+
profile.lastUsed = Date.now();
|
|
604
|
+
}
|
|
605
|
+
}
|
|
606
|
+
};
|
|
607
|
+
var globalModelManager = null;
|
|
608
|
+
var getModelManager = () => {
|
|
609
|
+
try {
|
|
610
|
+
if (!globalModelManager) {
|
|
611
|
+
const config = getGlobalConfig();
|
|
612
|
+
if (!config) {
|
|
613
|
+
debug.warn("MODEL_MANAGER_GLOBAL_CONFIG_MISSING", {});
|
|
614
|
+
globalModelManager = new ModelManager({
|
|
615
|
+
modelProfiles: [],
|
|
616
|
+
modelPointers: { main: "", task: "", compact: "", quick: "" }
|
|
617
|
+
});
|
|
618
|
+
} else {
|
|
619
|
+
globalModelManager = new ModelManager(config);
|
|
620
|
+
}
|
|
621
|
+
}
|
|
622
|
+
return globalModelManager;
|
|
623
|
+
} catch (error) {
|
|
624
|
+
logError(error);
|
|
625
|
+
debug.error("MODEL_MANAGER_CREATE_FAILED", {
|
|
626
|
+
error: error instanceof Error ? error.message : String(error)
|
|
627
|
+
});
|
|
628
|
+
return new ModelManager({
|
|
629
|
+
modelProfiles: [],
|
|
630
|
+
modelPointers: { main: "", task: "", compact: "", quick: "" }
|
|
631
|
+
});
|
|
632
|
+
}
|
|
633
|
+
};
|
|
634
|
+
var reloadModelManager = () => {
|
|
635
|
+
globalModelManager = null;
|
|
636
|
+
getModelManager();
|
|
637
|
+
};
|
|
638
|
+
var getQuickModel = () => {
|
|
639
|
+
const manager = getModelManager();
|
|
640
|
+
const quickModel = manager.getModel("quick");
|
|
641
|
+
return quickModel?.modelName || "quick";
|
|
642
|
+
};
|
|
643
|
+
|
|
644
|
+
export {
|
|
645
|
+
USE_BEDROCK,
|
|
646
|
+
USE_VERTEX,
|
|
647
|
+
getSlowAndCapableModel,
|
|
648
|
+
isDefaultSlowAndCapableModel,
|
|
649
|
+
getVertexRegionForModel,
|
|
650
|
+
ModelManager,
|
|
651
|
+
getModelManager,
|
|
652
|
+
reloadModelManager,
|
|
653
|
+
getQuickModel
|
|
654
|
+
};
|