@clinebot/core 0.0.22 → 0.0.24

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 (260) hide show
  1. package/dist/ClineCore.d.ts +110 -0
  2. package/dist/ClineCore.d.ts.map +1 -0
  3. package/dist/account/cline-account-service.d.ts +2 -1
  4. package/dist/account/cline-account-service.d.ts.map +1 -1
  5. package/dist/account/index.d.ts +1 -1
  6. package/dist/account/index.d.ts.map +1 -1
  7. package/dist/account/rpc.d.ts +3 -1
  8. package/dist/account/rpc.d.ts.map +1 -1
  9. package/dist/account/types.d.ts +3 -0
  10. package/dist/account/types.d.ts.map +1 -1
  11. package/dist/agents/plugin-loader.d.ts.map +1 -1
  12. package/dist/agents/plugin-sandbox-bootstrap.js +17 -17
  13. package/dist/auth/client.d.ts +1 -1
  14. package/dist/auth/client.d.ts.map +1 -1
  15. package/dist/auth/cline.d.ts +1 -1
  16. package/dist/auth/cline.d.ts.map +1 -1
  17. package/dist/auth/codex.d.ts +1 -1
  18. package/dist/auth/codex.d.ts.map +1 -1
  19. package/dist/auth/oca.d.ts +1 -1
  20. package/dist/auth/oca.d.ts.map +1 -1
  21. package/dist/auth/utils.d.ts +2 -2
  22. package/dist/auth/utils.d.ts.map +1 -1
  23. package/dist/index.d.ts +50 -5
  24. package/dist/index.d.ts.map +1 -1
  25. package/dist/index.js +949 -0
  26. package/dist/providers/local-provider-service.d.ts +4 -4
  27. package/dist/providers/local-provider-service.d.ts.map +1 -1
  28. package/dist/runtime/runtime-builder.d.ts +1 -0
  29. package/dist/runtime/runtime-builder.d.ts.map +1 -1
  30. package/dist/runtime/session-runtime.d.ts +2 -1
  31. package/dist/runtime/session-runtime.d.ts.map +1 -1
  32. package/dist/runtime/team-runtime-registry.d.ts +13 -0
  33. package/dist/runtime/team-runtime-registry.d.ts.map +1 -0
  34. package/dist/session/default-session-manager.d.ts +2 -2
  35. package/dist/session/default-session-manager.d.ts.map +1 -1
  36. package/dist/session/rpc-runtime-ensure.d.ts +53 -0
  37. package/dist/session/rpc-runtime-ensure.d.ts.map +1 -0
  38. package/dist/session/session-config-builder.d.ts +2 -3
  39. package/dist/session/session-config-builder.d.ts.map +1 -1
  40. package/dist/session/session-host.d.ts +8 -18
  41. package/dist/session/session-host.d.ts.map +1 -1
  42. package/dist/session/session-manager.d.ts +1 -1
  43. package/dist/session/session-manager.d.ts.map +1 -1
  44. package/dist/session/session-manifest.d.ts +1 -2
  45. package/dist/session/session-manifest.d.ts.map +1 -1
  46. package/dist/session/unified-session-persistence-service.d.ts +2 -2
  47. package/dist/session/unified-session-persistence-service.d.ts.map +1 -1
  48. package/dist/session/utils/helpers.d.ts +1 -1
  49. package/dist/session/utils/helpers.d.ts.map +1 -1
  50. package/dist/session/utils/types.d.ts +1 -1
  51. package/dist/session/utils/types.d.ts.map +1 -1
  52. package/dist/storage/provider-settings-legacy-migration.d.ts.map +1 -1
  53. package/dist/telemetry/OpenTelemetryProvider.d.ts.map +1 -1
  54. package/dist/telemetry/distinct-id.d.ts +2 -0
  55. package/dist/telemetry/distinct-id.d.ts.map +1 -0
  56. package/dist/telemetry/{opentelemetry.d.ts → index.d.ts} +1 -1
  57. package/dist/telemetry/index.d.ts.map +1 -0
  58. package/dist/telemetry/index.js +28 -0
  59. package/dist/tools/constants.d.ts +1 -1
  60. package/dist/tools/constants.d.ts.map +1 -1
  61. package/dist/tools/definitions.d.ts +3 -3
  62. package/dist/tools/definitions.d.ts.map +1 -1
  63. package/dist/tools/executors/apply-patch.d.ts +1 -1
  64. package/dist/tools/executors/apply-patch.d.ts.map +1 -1
  65. package/dist/tools/executors/bash.d.ts +1 -1
  66. package/dist/tools/executors/bash.d.ts.map +1 -1
  67. package/dist/tools/executors/editor.d.ts +1 -1
  68. package/dist/tools/executors/editor.d.ts.map +1 -1
  69. package/dist/tools/executors/file-read.d.ts +1 -1
  70. package/dist/tools/executors/file-read.d.ts.map +1 -1
  71. package/dist/tools/executors/index.d.ts +14 -14
  72. package/dist/tools/executors/index.d.ts.map +1 -1
  73. package/dist/tools/executors/search.d.ts +1 -1
  74. package/dist/tools/executors/search.d.ts.map +1 -1
  75. package/dist/tools/executors/web-fetch.d.ts +1 -1
  76. package/dist/tools/executors/web-fetch.d.ts.map +1 -1
  77. package/dist/tools/helpers.d.ts +1 -1
  78. package/dist/tools/helpers.d.ts.map +1 -1
  79. package/dist/tools/index.d.ts +10 -10
  80. package/dist/tools/index.d.ts.map +1 -1
  81. package/dist/tools/model-tool-routing.d.ts +1 -1
  82. package/dist/tools/model-tool-routing.d.ts.map +1 -1
  83. package/dist/tools/presets.d.ts +1 -1
  84. package/dist/tools/presets.d.ts.map +1 -1
  85. package/dist/types/common.d.ts +17 -8
  86. package/dist/types/common.d.ts.map +1 -1
  87. package/dist/types/config.d.ts +4 -3
  88. package/dist/types/config.d.ts.map +1 -1
  89. package/dist/types/provider-settings.d.ts +1 -1
  90. package/dist/types/provider-settings.d.ts.map +1 -1
  91. package/dist/types.d.ts +5 -2
  92. package/dist/types.d.ts.map +1 -1
  93. package/dist/version.d.ts +2 -0
  94. package/dist/version.d.ts.map +1 -0
  95. package/package.json +44 -38
  96. package/src/ClineCore.ts +137 -0
  97. package/src/account/cline-account-service.test.ts +101 -0
  98. package/src/account/cline-account-service.ts +300 -0
  99. package/src/account/featurebase-token.test.ts +175 -0
  100. package/src/account/index.ts +23 -0
  101. package/src/account/rpc.test.ts +63 -0
  102. package/src/account/rpc.ts +185 -0
  103. package/src/account/types.ts +102 -0
  104. package/src/agents/agent-config-loader.test.ts +236 -0
  105. package/src/agents/agent-config-loader.ts +108 -0
  106. package/src/agents/agent-config-parser.ts +198 -0
  107. package/src/agents/hooks-config-loader.test.ts +20 -0
  108. package/src/agents/hooks-config-loader.ts +118 -0
  109. package/src/agents/index.ts +85 -0
  110. package/src/agents/plugin-config-loader.test.ts +140 -0
  111. package/src/agents/plugin-config-loader.ts +97 -0
  112. package/src/agents/plugin-loader.test.ts +210 -0
  113. package/src/agents/plugin-loader.ts +175 -0
  114. package/src/agents/plugin-sandbox-bootstrap.ts +448 -0
  115. package/src/agents/plugin-sandbox.test.ts +296 -0
  116. package/src/agents/plugin-sandbox.ts +341 -0
  117. package/src/agents/unified-config-file-watcher.test.ts +196 -0
  118. package/src/agents/unified-config-file-watcher.ts +483 -0
  119. package/src/agents/user-instruction-config-loader.test.ts +158 -0
  120. package/src/agents/user-instruction-config-loader.ts +438 -0
  121. package/src/auth/client.test.ts +40 -0
  122. package/src/auth/client.ts +25 -0
  123. package/src/auth/cline.test.ts +130 -0
  124. package/src/auth/cline.ts +420 -0
  125. package/src/auth/codex.test.ts +170 -0
  126. package/src/auth/codex.ts +491 -0
  127. package/src/auth/oca.test.ts +215 -0
  128. package/src/auth/oca.ts +573 -0
  129. package/src/auth/server.ts +216 -0
  130. package/src/auth/types.ts +81 -0
  131. package/src/auth/utils.test.ts +128 -0
  132. package/src/auth/utils.ts +247 -0
  133. package/src/chat/chat-schema.ts +82 -0
  134. package/src/index.ts +479 -0
  135. package/src/input/file-indexer.d.ts +11 -0
  136. package/src/input/file-indexer.test.ts +127 -0
  137. package/src/input/file-indexer.ts +327 -0
  138. package/src/input/index.ts +7 -0
  139. package/src/input/mention-enricher.test.ts +85 -0
  140. package/src/input/mention-enricher.ts +122 -0
  141. package/src/mcp/config-loader.test.ts +238 -0
  142. package/src/mcp/config-loader.ts +219 -0
  143. package/src/mcp/index.ts +26 -0
  144. package/src/mcp/manager.test.ts +106 -0
  145. package/src/mcp/manager.ts +262 -0
  146. package/src/mcp/types.ts +88 -0
  147. package/src/providers/local-provider-registry.ts +232 -0
  148. package/src/providers/local-provider-service.test.ts +783 -0
  149. package/src/providers/local-provider-service.ts +471 -0
  150. package/src/runtime/commands.test.ts +98 -0
  151. package/src/runtime/commands.ts +83 -0
  152. package/src/runtime/hook-file-hooks.test.ts +237 -0
  153. package/src/runtime/hook-file-hooks.ts +859 -0
  154. package/src/runtime/index.ts +37 -0
  155. package/src/runtime/rules.ts +34 -0
  156. package/src/runtime/runtime-builder.team-persistence.test.ts +245 -0
  157. package/src/runtime/runtime-builder.test.ts +371 -0
  158. package/src/runtime/runtime-builder.ts +631 -0
  159. package/src/runtime/runtime-parity.test.ts +143 -0
  160. package/src/runtime/sandbox/subprocess-sandbox.ts +231 -0
  161. package/src/runtime/session-runtime.ts +49 -0
  162. package/src/runtime/skills.ts +44 -0
  163. package/src/runtime/team-runtime-registry.ts +46 -0
  164. package/src/runtime/tool-approval.ts +104 -0
  165. package/src/runtime/workflows.test.ts +119 -0
  166. package/src/runtime/workflows.ts +45 -0
  167. package/src/session/default-session-manager.e2e.test.ts +384 -0
  168. package/src/session/default-session-manager.test.ts +1931 -0
  169. package/src/session/default-session-manager.ts +1422 -0
  170. package/src/session/file-session-service.ts +280 -0
  171. package/src/session/index.ts +45 -0
  172. package/src/session/rpc-runtime-ensure.ts +521 -0
  173. package/src/session/rpc-session-service.ts +107 -0
  174. package/src/session/rpc-spawn-lease.test.ts +49 -0
  175. package/src/session/rpc-spawn-lease.ts +122 -0
  176. package/src/session/runtime-oauth-token-manager.test.ts +137 -0
  177. package/src/session/runtime-oauth-token-manager.ts +272 -0
  178. package/src/session/session-agent-events.ts +248 -0
  179. package/src/session/session-artifacts.ts +106 -0
  180. package/src/session/session-config-builder.ts +113 -0
  181. package/src/session/session-graph.ts +92 -0
  182. package/src/session/session-host.test.ts +89 -0
  183. package/src/session/session-host.ts +205 -0
  184. package/src/session/session-manager.ts +69 -0
  185. package/src/session/session-manifest.ts +29 -0
  186. package/src/session/session-service.team-persistence.test.ts +48 -0
  187. package/src/session/session-service.ts +673 -0
  188. package/src/session/session-team-coordination.ts +229 -0
  189. package/src/session/session-telemetry.ts +100 -0
  190. package/src/session/sqlite-rpc-session-backend.ts +303 -0
  191. package/src/session/unified-session-persistence-service.test.ts +85 -0
  192. package/src/session/unified-session-persistence-service.ts +994 -0
  193. package/src/session/utils/helpers.ts +139 -0
  194. package/src/session/utils/types.ts +57 -0
  195. package/src/session/utils/usage.ts +32 -0
  196. package/src/session/workspace-manager.ts +98 -0
  197. package/src/session/workspace-manifest.ts +100 -0
  198. package/src/storage/artifact-store.ts +1 -0
  199. package/src/storage/file-team-store.ts +257 -0
  200. package/src/storage/index.ts +11 -0
  201. package/src/storage/provider-settings-legacy-migration.test.ts +424 -0
  202. package/src/storage/provider-settings-legacy-migration.ts +826 -0
  203. package/src/storage/provider-settings-manager.test.ts +191 -0
  204. package/src/storage/provider-settings-manager.ts +152 -0
  205. package/src/storage/session-store.ts +1 -0
  206. package/src/storage/sqlite-session-store.ts +275 -0
  207. package/src/storage/sqlite-team-store.ts +454 -0
  208. package/src/storage/team-store.ts +40 -0
  209. package/src/team/index.ts +4 -0
  210. package/src/team/projections.ts +285 -0
  211. package/src/telemetry/ITelemetryAdapter.ts +94 -0
  212. package/src/telemetry/LoggerTelemetryAdapter.test.ts +42 -0
  213. package/src/telemetry/LoggerTelemetryAdapter.ts +114 -0
  214. package/src/telemetry/OpenTelemetryAdapter.test.ts +157 -0
  215. package/src/telemetry/OpenTelemetryAdapter.ts +348 -0
  216. package/src/telemetry/OpenTelemetryProvider.test.ts +113 -0
  217. package/src/telemetry/OpenTelemetryProvider.ts +325 -0
  218. package/src/telemetry/TelemetryService.test.ts +134 -0
  219. package/src/telemetry/TelemetryService.ts +141 -0
  220. package/src/telemetry/core-events.ts +400 -0
  221. package/src/telemetry/distinct-id.test.ts +57 -0
  222. package/src/telemetry/distinct-id.ts +58 -0
  223. package/src/telemetry/index.ts +20 -0
  224. package/src/tools/constants.ts +35 -0
  225. package/src/tools/definitions.test.ts +704 -0
  226. package/src/tools/definitions.ts +709 -0
  227. package/src/tools/executors/apply-patch-parser.ts +520 -0
  228. package/src/tools/executors/apply-patch.ts +359 -0
  229. package/src/tools/executors/bash.test.ts +87 -0
  230. package/src/tools/executors/bash.ts +207 -0
  231. package/src/tools/executors/editor.test.ts +35 -0
  232. package/src/tools/executors/editor.ts +219 -0
  233. package/src/tools/executors/file-read.test.ts +49 -0
  234. package/src/tools/executors/file-read.ts +110 -0
  235. package/src/tools/executors/index.ts +87 -0
  236. package/src/tools/executors/search.ts +278 -0
  237. package/src/tools/executors/web-fetch.ts +259 -0
  238. package/src/tools/helpers.ts +130 -0
  239. package/src/tools/index.ts +169 -0
  240. package/src/tools/model-tool-routing.test.ts +86 -0
  241. package/src/tools/model-tool-routing.ts +132 -0
  242. package/src/tools/presets.test.ts +62 -0
  243. package/src/tools/presets.ts +168 -0
  244. package/src/tools/schemas.ts +327 -0
  245. package/src/tools/types.ts +329 -0
  246. package/src/types/common.ts +26 -0
  247. package/src/types/config.ts +86 -0
  248. package/src/types/events.ts +74 -0
  249. package/src/types/index.ts +24 -0
  250. package/src/types/provider-settings.ts +43 -0
  251. package/src/types/sessions.ts +16 -0
  252. package/src/types/storage.ts +64 -0
  253. package/src/types/workspace.ts +7 -0
  254. package/src/types.ts +132 -0
  255. package/src/version.ts +3 -0
  256. package/dist/index.node.d.ts +0 -47
  257. package/dist/index.node.d.ts.map +0 -1
  258. package/dist/index.node.js +0 -948
  259. package/dist/telemetry/opentelemetry.d.ts.map +0 -1
  260. package/dist/telemetry/opentelemetry.js +0 -27
@@ -0,0 +1,826 @@
1
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
2
+ import { dirname, join } from "node:path";
3
+ import * as LlmsModels from "@clinebot/llms/models";
4
+ import * as LlmsProviders from "@clinebot/llms/providers";
5
+ import { resolveClineDataDir } from "@clinebot/shared/storage";
6
+ import type { ProviderSettings } from "../types/provider-settings";
7
+ import { emptyStoredProviderSettings } from "../types/provider-settings";
8
+ import type { ProviderSettingsManager } from "./provider-settings-manager";
9
+
10
+ type LegacyMode = "plan" | "act";
11
+
12
+ interface LegacyGlobalState {
13
+ mode?: LegacyMode;
14
+ planModeApiProvider?: string;
15
+ actModeApiProvider?: string;
16
+ planModeApiModelId?: string;
17
+ actModeApiModelId?: string;
18
+ planModeReasoningEffort?: string;
19
+ actModeReasoningEffort?: string;
20
+ planModeThinkingBudgetTokens?: number;
21
+ actModeThinkingBudgetTokens?: number;
22
+ geminiPlanModeThinkingLevel?: string;
23
+ geminiActModeThinkingLevel?: string;
24
+ anthropicBaseUrl?: string;
25
+ openAiBaseUrl?: string;
26
+ ollamaBaseUrl?: string;
27
+ lmStudioBaseUrl?: string;
28
+ liteLlmBaseUrl?: string;
29
+ geminiBaseUrl?: string;
30
+ requestyBaseUrl?: string;
31
+ asksageApiUrl?: string;
32
+ difyBaseUrl?: string;
33
+ ocaBaseUrl?: string;
34
+ aihubmixBaseUrl?: string;
35
+ openAiHeaders?: Record<string, string>;
36
+ requestTimeoutMs?: number;
37
+ awsRegion?: string;
38
+ awsAuthentication?: "iam" | "api-key" | "profile";
39
+ awsUseProfile?: boolean;
40
+ awsProfile?: string;
41
+ awsUseCrossRegionInference?: boolean;
42
+ awsUseGlobalInference?: boolean;
43
+ awsBedrockUsePromptCache?: boolean;
44
+ awsBedrockEndpoint?: string;
45
+ planModeAwsBedrockCustomModelBaseId?: string;
46
+ actModeAwsBedrockCustomModelBaseId?: string;
47
+ vertexProjectId?: string;
48
+ vertexRegion?: string;
49
+ azureApiVersion?: string;
50
+ azureIdentity?: boolean;
51
+ sapAiCoreTokenUrl?: string;
52
+ sapAiCoreBaseUrl?: string;
53
+ sapAiResourceGroup?: string;
54
+ sapAiCoreUseOrchestrationMode?: boolean;
55
+ ocaMode?: "internal" | "external";
56
+ qwenApiLine?: "china" | "international";
57
+ moonshotApiLine?: "china" | "international";
58
+ zaiApiLine?: "china" | "international";
59
+ minimaxApiLine?: "china" | "international";
60
+ planModeOpenRouterModelId?: string;
61
+ actModeOpenRouterModelId?: string;
62
+ planModeClineModelId?: string;
63
+ actModeClineModelId?: string;
64
+ planModeOpenAiModelId?: string;
65
+ actModeOpenAiModelId?: string;
66
+ planModeOllamaModelId?: string;
67
+ actModeOllamaModelId?: string;
68
+ planModeLmStudioModelId?: string;
69
+ actModeLmStudioModelId?: string;
70
+ planModeLiteLlmModelId?: string;
71
+ actModeLiteLlmModelId?: string;
72
+ planModeRequestyModelId?: string;
73
+ actModeRequestyModelId?: string;
74
+ planModeTogetherModelId?: string;
75
+ actModeTogetherModelId?: string;
76
+ planModeFireworksModelId?: string;
77
+ actModeFireworksModelId?: string;
78
+ planModeSapAiCoreModelId?: string;
79
+ actModeSapAiCoreModelId?: string;
80
+ planModeSapAiCoreDeploymentId?: string;
81
+ actModeSapAiCoreDeploymentId?: string;
82
+ planModeGroqModelId?: string;
83
+ actModeGroqModelId?: string;
84
+ planModeBasetenModelId?: string;
85
+ actModeBasetenModelId?: string;
86
+ planModeHuggingFaceModelId?: string;
87
+ actModeHuggingFaceModelId?: string;
88
+ planModeHuaweiCloudMaasModelId?: string;
89
+ actModeHuaweiCloudMaasModelId?: string;
90
+ planModeOcaModelId?: string;
91
+ actModeOcaModelId?: string;
92
+ planModeAihubmixModelId?: string;
93
+ actModeAihubmixModelId?: string;
94
+ planModeHicapModelId?: string;
95
+ actModeHicapModelId?: string;
96
+ planModeNousResearchModelId?: string;
97
+ actModeNousResearchModelId?: string;
98
+ planModeVercelAiGatewayModelId?: string;
99
+ actModeVercelAiGatewayModelId?: string;
100
+ }
101
+
102
+ interface LegacySecrets {
103
+ apiKey?: string;
104
+ clineApiKey?: string;
105
+ "cline:clineAccountId"?: string;
106
+ clineAccountId?: string;
107
+ openRouterApiKey?: string;
108
+ awsAccessKey?: string;
109
+ awsSecretKey?: string;
110
+ awsSessionToken?: string;
111
+ awsBedrockApiKey?: string;
112
+ openAiApiKey?: string;
113
+ geminiApiKey?: string;
114
+ openAiNativeApiKey?: string;
115
+ ollamaApiKey?: string;
116
+ deepSeekApiKey?: string;
117
+ requestyApiKey?: string;
118
+ togetherApiKey?: string;
119
+ fireworksApiKey?: string;
120
+ qwenApiKey?: string;
121
+ doubaoApiKey?: string;
122
+ mistralApiKey?: string;
123
+ liteLlmApiKey?: string;
124
+ asksageApiKey?: string;
125
+ xaiApiKey?: string;
126
+ moonshotApiKey?: string;
127
+ zaiApiKey?: string;
128
+ huggingFaceApiKey?: string;
129
+ nebiusApiKey?: string;
130
+ sambanovaApiKey?: string;
131
+ cerebrasApiKey?: string;
132
+ sapAiCoreClientId?: string;
133
+ sapAiCoreClientSecret?: string;
134
+ groqApiKey?: string;
135
+ huaweiCloudMaasApiKey?: string;
136
+ basetenApiKey?: string;
137
+ vercelAiGatewayApiKey?: string;
138
+ difyApiKey?: string;
139
+ minimaxApiKey?: string;
140
+ hicapApiKey?: string;
141
+ aihubmixApiKey?: string;
142
+ nousResearchApiKey?: string;
143
+ ocaApiKey?: string;
144
+ ocaRefreshToken?: string;
145
+ "openai-codex-oauth-credentials"?: string;
146
+ }
147
+
148
+ interface LegacyProviderStorage {
149
+ globalState: LegacyGlobalState;
150
+ secrets: LegacySecrets;
151
+ }
152
+
153
+ type StoredModelsFile = {
154
+ version: 1;
155
+ providers: Record<
156
+ string,
157
+ {
158
+ provider: {
159
+ name: string;
160
+ baseUrl: string;
161
+ defaultModelId?: string;
162
+ };
163
+ models: Record<
164
+ string,
165
+ {
166
+ id: string;
167
+ name: string;
168
+ }
169
+ >;
170
+ }
171
+ >;
172
+ };
173
+
174
+ const LEGACY_OPENAI_COMPATIBLE_PROVIDER_ID = "openai";
175
+
176
+ export interface MigrateLegacyProviderSettingsOptions {
177
+ providerSettingsManager: ProviderSettingsManager;
178
+ dataDir?: string;
179
+ globalStatePath?: string;
180
+ secretsPath?: string;
181
+ }
182
+
183
+ export interface MigrateLegacyProviderSettingsResult {
184
+ migrated: boolean;
185
+ providerCount: number;
186
+ lastUsedProvider?: string;
187
+ }
188
+
189
+ export type LegacyClineUserInfo = {
190
+ idToken: string;
191
+ expiresAt: number;
192
+ refreshToken: string;
193
+ userInfo: {
194
+ id: string;
195
+ email: string;
196
+ displayName: string;
197
+ termsAcceptedAt: string;
198
+ clineBenchConsent: boolean;
199
+ createdAt: string;
200
+ updatedAt: string;
201
+ };
202
+ provider: string;
203
+ startedAt: number;
204
+ };
205
+
206
+ /**
207
+ * Resolves legacy Cline account auth data from the raw `cline:clineAccountId`
208
+ * secret string into the auth fields used by `ProviderSettings`.
209
+ *
210
+ * Returns `undefined` when the input is missing, empty, whitespace-only, or
211
+ * unparseable JSON.
212
+ */
213
+ export function resolveLegacyClineAuth(
214
+ rawAccountData: string | undefined,
215
+ ): ProviderSettings["auth"] | undefined {
216
+ const trimmed = rawAccountData?.trim();
217
+ if (!trimmed) {
218
+ return undefined;
219
+ }
220
+ try {
221
+ const data = JSON.parse(trimmed) as LegacyClineUserInfo;
222
+ if (!data) {
223
+ return undefined;
224
+ }
225
+ return {
226
+ accessToken: data.idToken,
227
+ refreshToken: data.refreshToken,
228
+ expiresAt: data.expiresAt,
229
+ accountId: data.userInfo?.id,
230
+ };
231
+ } catch {
232
+ return undefined;
233
+ }
234
+ }
235
+
236
+ function trimNonEmpty(value: string | undefined): string | undefined {
237
+ const trimmed = value?.trim();
238
+ return trimmed ? trimmed : undefined;
239
+ }
240
+
241
+ function readJsonObject<T extends object>(filePath: string): T | undefined {
242
+ if (!existsSync(filePath)) {
243
+ return undefined;
244
+ }
245
+ try {
246
+ const raw = readFileSync(filePath, "utf8");
247
+ const parsed = JSON.parse(raw) as unknown;
248
+ if (parsed && typeof parsed === "object" && !Array.isArray(parsed)) {
249
+ return parsed as T;
250
+ }
251
+ } catch {
252
+ // Invalid legacy file should not block startup.
253
+ }
254
+ return undefined;
255
+ }
256
+
257
+ function readModelsFile(filePath: string): StoredModelsFile {
258
+ const parsed = readJsonObject<StoredModelsFile>(filePath);
259
+ if (
260
+ parsed?.version === 1 &&
261
+ parsed.providers &&
262
+ typeof parsed.providers === "object"
263
+ ) {
264
+ return parsed;
265
+ }
266
+ return {
267
+ version: 1,
268
+ providers: {},
269
+ };
270
+ }
271
+
272
+ function writeModelsFile(filePath: string, state: StoredModelsFile): void {
273
+ const dir = dirname(filePath);
274
+ if (!existsSync(dir)) {
275
+ mkdirSync(dir, { recursive: true });
276
+ }
277
+ writeFileSync(filePath, `${JSON.stringify(state, null, 2)}\n`, "utf8");
278
+ }
279
+
280
+ function resolveLegacyStorage(
281
+ options: MigrateLegacyProviderSettingsOptions,
282
+ ): LegacyProviderStorage | undefined {
283
+ const dataDir = options.dataDir ?? resolveClineDataDir();
284
+ const globalStatePath =
285
+ options.globalStatePath ?? join(dataDir, "globalState.json");
286
+ const secretsPath = options.secretsPath ?? join(dataDir, "secrets.json");
287
+ const globalState = readJsonObject<LegacyGlobalState>(globalStatePath);
288
+ const secrets = readJsonObject<LegacySecrets>(secretsPath);
289
+ if (!globalState && !secrets) {
290
+ return undefined;
291
+ }
292
+ return {
293
+ globalState: globalState ?? {},
294
+ secrets: secrets ?? {},
295
+ };
296
+ }
297
+
298
+ function isOfficialOpenAiBaseUrl(baseUrl: string): boolean {
299
+ try {
300
+ const url = new URL(baseUrl);
301
+ const hostname = url.hostname.toLowerCase();
302
+ return (
303
+ hostname === "api.openai.com" ||
304
+ hostname.endsWith(".openai.azure.com") ||
305
+ hostname.endsWith(".services.ai.azure.com")
306
+ );
307
+ } catch {
308
+ return false;
309
+ }
310
+ }
311
+
312
+ function shouldMigrateLegacyOpenAiAsCustomProvider(
313
+ legacyGlobalState: LegacyGlobalState,
314
+ ): boolean {
315
+ const baseUrl = trimNonEmpty(legacyGlobalState.openAiBaseUrl);
316
+ if (!baseUrl) {
317
+ return false;
318
+ }
319
+ if (legacyGlobalState.azureApiVersion || legacyGlobalState.azureIdentity) {
320
+ return false;
321
+ }
322
+ return !isOfficialOpenAiBaseUrl(baseUrl);
323
+ }
324
+
325
+ function resolveMigratedProviderId(
326
+ providerId: string,
327
+ legacyGlobalState: LegacyGlobalState,
328
+ ): string {
329
+ if (
330
+ providerId === "openai" &&
331
+ shouldMigrateLegacyOpenAiAsCustomProvider(legacyGlobalState)
332
+ ) {
333
+ return LEGACY_OPENAI_COMPATIBLE_PROVIDER_ID;
334
+ }
335
+ return providerId;
336
+ }
337
+
338
+ function resolveModelForProvider(
339
+ legacy: LegacyGlobalState,
340
+ providerId: string,
341
+ mode: LegacyMode,
342
+ activeProviderForMode: string | undefined,
343
+ ): string | undefined {
344
+ const modePrefix = mode === "plan" ? "planMode" : "actMode";
345
+ const fallbackModel =
346
+ providerId === activeProviderForMode
347
+ ? trimNonEmpty(
348
+ mode === "plan"
349
+ ? legacy.planModeApiModelId
350
+ : legacy.actModeApiModelId,
351
+ )
352
+ : undefined;
353
+ const providerModelKeyById: Record<string, keyof LegacyGlobalState> = {
354
+ openrouter: `${modePrefix}OpenRouterModelId` as keyof LegacyGlobalState,
355
+ cline: `${modePrefix}ClineModelId` as keyof LegacyGlobalState,
356
+ openai: `${modePrefix}OpenAiModelId` as keyof LegacyGlobalState,
357
+ ollama: `${modePrefix}OllamaModelId` as keyof LegacyGlobalState,
358
+ lmstudio: `${modePrefix}LmStudioModelId` as keyof LegacyGlobalState,
359
+ litellm: `${modePrefix}LiteLlmModelId` as keyof LegacyGlobalState,
360
+ requesty: `${modePrefix}RequestyModelId` as keyof LegacyGlobalState,
361
+ together: `${modePrefix}TogetherModelId` as keyof LegacyGlobalState,
362
+ fireworks: `${modePrefix}FireworksModelId` as keyof LegacyGlobalState,
363
+ sapaicore: `${modePrefix}SapAiCoreModelId` as keyof LegacyGlobalState,
364
+ groq: `${modePrefix}GroqModelId` as keyof LegacyGlobalState,
365
+ baseten: `${modePrefix}BasetenModelId` as keyof LegacyGlobalState,
366
+ huggingface: `${modePrefix}HuggingFaceModelId` as keyof LegacyGlobalState,
367
+ "huawei-cloud-maas":
368
+ `${modePrefix}HuaweiCloudMaasModelId` as keyof LegacyGlobalState,
369
+ oca: `${modePrefix}OcaModelId` as keyof LegacyGlobalState,
370
+ aihubmix: `${modePrefix}AihubmixModelId` as keyof LegacyGlobalState,
371
+ hicap: `${modePrefix}HicapModelId` as keyof LegacyGlobalState,
372
+ nousResearch: `${modePrefix}NousResearchModelId` as keyof LegacyGlobalState,
373
+ "vercel-ai-gateway":
374
+ `${modePrefix}VercelAiGatewayModelId` as keyof LegacyGlobalState,
375
+ };
376
+ const providerModelKey = providerModelKeyById[providerId];
377
+ const providerModel = providerModelKey
378
+ ? trimNonEmpty(
379
+ typeof legacy[providerModelKey] === "string"
380
+ ? (legacy[providerModelKey] as string)
381
+ : undefined,
382
+ )
383
+ : undefined;
384
+ return providerModel ?? fallbackModel;
385
+ }
386
+
387
+ function resolveReasoning(
388
+ legacy: LegacyGlobalState,
389
+ providerId: string,
390
+ mode: LegacyMode,
391
+ ): ProviderSettings["reasoning"] | undefined {
392
+ const effortCandidate =
393
+ mode === "plan"
394
+ ? legacy.planModeReasoningEffort
395
+ : legacy.actModeReasoningEffort;
396
+ const geminiLevel =
397
+ mode === "plan"
398
+ ? legacy.geminiPlanModeThinkingLevel
399
+ : legacy.geminiActModeThinkingLevel;
400
+ const budgetTokens =
401
+ mode === "plan"
402
+ ? legacy.planModeThinkingBudgetTokens
403
+ : legacy.actModeThinkingBudgetTokens;
404
+ const rawEffort =
405
+ (providerId === "gemini" ? geminiLevel : undefined) ?? effortCandidate;
406
+ const effort =
407
+ rawEffort === "none" ||
408
+ rawEffort === "low" ||
409
+ rawEffort === "medium" ||
410
+ rawEffort === "high"
411
+ ? rawEffort
412
+ : undefined;
413
+ const normalizedBudget =
414
+ typeof budgetTokens === "number" &&
415
+ Number.isInteger(budgetTokens) &&
416
+ budgetTokens > 0
417
+ ? budgetTokens
418
+ : undefined;
419
+ if (!effort && normalizedBudget === undefined) {
420
+ return undefined;
421
+ }
422
+ return {
423
+ ...(effort ? { effort } : {}),
424
+ ...(normalizedBudget !== undefined
425
+ ? { budgetTokens: normalizedBudget }
426
+ : {}),
427
+ };
428
+ }
429
+
430
+ function resolveLegacyCodexAuth(
431
+ legacySecrets: LegacySecrets,
432
+ ): Pick<ProviderSettings, "apiKey" | "auth"> | undefined {
433
+ const raw = legacySecrets["openai-codex-oauth-credentials"];
434
+ if (!raw) {
435
+ return undefined;
436
+ }
437
+ try {
438
+ const parsed = JSON.parse(raw) as {
439
+ access_token?: string;
440
+ refresh_token?: string;
441
+ accountId?: string;
442
+ };
443
+ const access = trimNonEmpty(parsed.access_token);
444
+ const refresh = trimNonEmpty(parsed.refresh_token);
445
+ const accountId = trimNonEmpty(parsed.accountId);
446
+ if (!access && !refresh && !accountId) {
447
+ return undefined;
448
+ }
449
+ return {
450
+ ...(access ? { apiKey: access } : {}),
451
+ auth: {
452
+ ...(access ? { accessToken: access } : {}),
453
+ ...(refresh ? { refreshToken: refresh } : {}),
454
+ ...(accountId ? { accountId } : {}),
455
+ },
456
+ };
457
+ } catch {
458
+ return undefined;
459
+ }
460
+ }
461
+
462
+ function getDefaultModelForProvider(providerId: string): string | undefined {
463
+ const builtInModels = LlmsModels.getGeneratedModelsForProvider(providerId);
464
+ const firstModelId = Object.keys(builtInModels)[0];
465
+ return firstModelId ?? undefined;
466
+ }
467
+
468
+ function buildLegacyProviderSettings(
469
+ providerId: string,
470
+ legacyGlobalState: LegacyGlobalState,
471
+ legacySecrets: LegacySecrets,
472
+ mode: LegacyMode,
473
+ ): ProviderSettings | undefined {
474
+ const targetProviderId = resolveMigratedProviderId(
475
+ providerId,
476
+ legacyGlobalState,
477
+ );
478
+ const activeProviderForMode = trimNonEmpty(
479
+ mode === "plan"
480
+ ? legacyGlobalState.planModeApiProvider
481
+ : legacyGlobalState.actModeApiProvider,
482
+ );
483
+ const model =
484
+ resolveModelForProvider(
485
+ legacyGlobalState,
486
+ providerId,
487
+ mode,
488
+ activeProviderForMode,
489
+ ) ?? getDefaultModelForProvider(targetProviderId);
490
+ const reasoning = resolveReasoning(legacyGlobalState, targetProviderId, mode);
491
+ const timeout =
492
+ typeof legacyGlobalState.requestTimeoutMs === "number" &&
493
+ Number.isInteger(legacyGlobalState.requestTimeoutMs) &&
494
+ legacyGlobalState.requestTimeoutMs > 0
495
+ ? legacyGlobalState.requestTimeoutMs
496
+ : undefined;
497
+
498
+ const secretByProvider: Record<string, string | undefined> = {
499
+ anthropic: legacySecrets.apiKey,
500
+ cline: legacySecrets.clineApiKey,
501
+ openai: legacySecrets.openAiApiKey,
502
+ "openai-native": legacySecrets.openAiNativeApiKey,
503
+ openrouter: legacySecrets.openRouterApiKey,
504
+ bedrock: legacySecrets.awsBedrockApiKey,
505
+ gemini: legacySecrets.geminiApiKey,
506
+ ollama: legacySecrets.ollamaApiKey,
507
+ deepseek: legacySecrets.deepSeekApiKey,
508
+ requesty: legacySecrets.requestyApiKey,
509
+ together: legacySecrets.togetherApiKey,
510
+ fireworks: legacySecrets.fireworksApiKey,
511
+ qwen: legacySecrets.qwenApiKey,
512
+ doubao: legacySecrets.doubaoApiKey,
513
+ mistral: legacySecrets.mistralApiKey,
514
+ litellm: legacySecrets.liteLlmApiKey,
515
+ asksage: legacySecrets.asksageApiKey,
516
+ xai: legacySecrets.xaiApiKey,
517
+ moonshot: legacySecrets.moonshotApiKey,
518
+ zai: legacySecrets.zaiApiKey,
519
+ huggingface: legacySecrets.huggingFaceApiKey,
520
+ nebius: legacySecrets.nebiusApiKey,
521
+ sambanova: legacySecrets.sambanovaApiKey,
522
+ cerebras: legacySecrets.cerebrasApiKey,
523
+ groq: legacySecrets.groqApiKey,
524
+ "huawei-cloud-maas": legacySecrets.huaweiCloudMaasApiKey,
525
+ baseten: legacySecrets.basetenApiKey,
526
+ "vercel-ai-gateway": legacySecrets.vercelAiGatewayApiKey,
527
+ dify: legacySecrets.difyApiKey,
528
+ minimax: legacySecrets.minimaxApiKey,
529
+ hicap: legacySecrets.hicapApiKey,
530
+ aihubmix: legacySecrets.aihubmixApiKey,
531
+ nousResearch: legacySecrets.nousResearchApiKey,
532
+ oca: legacySecrets.ocaApiKey,
533
+ sapaicore: legacySecrets.sapAiCoreClientId,
534
+ };
535
+
536
+ const providerSpecific: Partial<ProviderSettings> = {};
537
+ if (providerId === "openai-codex") {
538
+ Object.assign(providerSpecific, resolveLegacyCodexAuth(legacySecrets));
539
+ }
540
+ if (providerId === "cline") {
541
+ try {
542
+ const legacyAuthString = trimNonEmpty(
543
+ legacySecrets["cline:clineAccountId"],
544
+ );
545
+
546
+ if (legacyAuthString) {
547
+ providerSpecific.auth = {
548
+ ...(providerSpecific.auth ?? {}),
549
+ ...resolveLegacyClineAuth(legacyAuthString),
550
+ };
551
+ }
552
+ } catch {
553
+ // Failed to parse stored cline auth data
554
+ }
555
+ }
556
+ if (providerId === "openai" && legacyGlobalState.openAiHeaders) {
557
+ providerSpecific.headers = legacyGlobalState.openAiHeaders;
558
+ }
559
+ if (providerId === "bedrock") {
560
+ providerSpecific.aws = {
561
+ accessKey: trimNonEmpty(legacySecrets.awsAccessKey),
562
+ secretKey: trimNonEmpty(legacySecrets.awsSecretKey),
563
+ sessionToken: trimNonEmpty(legacySecrets.awsSessionToken),
564
+ region: trimNonEmpty(legacyGlobalState.awsRegion),
565
+ authentication: legacyGlobalState.awsAuthentication,
566
+ profile: legacyGlobalState.awsUseProfile
567
+ ? trimNonEmpty(legacyGlobalState.awsProfile)
568
+ : undefined,
569
+ usePromptCache: legacyGlobalState.awsBedrockUsePromptCache,
570
+ useCrossRegionInference: legacyGlobalState.awsUseCrossRegionInference,
571
+ useGlobalInference: legacyGlobalState.awsUseGlobalInference,
572
+ endpoint: trimNonEmpty(legacyGlobalState.awsBedrockEndpoint),
573
+ customModelBaseId: trimNonEmpty(
574
+ mode === "plan"
575
+ ? legacyGlobalState.planModeAwsBedrockCustomModelBaseId
576
+ : legacyGlobalState.actModeAwsBedrockCustomModelBaseId,
577
+ ),
578
+ };
579
+ }
580
+ if (providerId === "vertex") {
581
+ providerSpecific.gcp = {
582
+ projectId: trimNonEmpty(legacyGlobalState.vertexProjectId),
583
+ region: trimNonEmpty(legacyGlobalState.vertexRegion),
584
+ };
585
+ }
586
+ if (
587
+ providerId === "openai" &&
588
+ (legacyGlobalState.azureApiVersion ||
589
+ legacyGlobalState.azureIdentity !== undefined)
590
+ ) {
591
+ providerSpecific.azure = {
592
+ apiVersion: trimNonEmpty(legacyGlobalState.azureApiVersion),
593
+ useIdentity: legacyGlobalState.azureIdentity,
594
+ };
595
+ }
596
+ if (providerId === "sapaicore") {
597
+ providerSpecific.sap = {
598
+ clientId: trimNonEmpty(legacySecrets.sapAiCoreClientId),
599
+ clientSecret: trimNonEmpty(legacySecrets.sapAiCoreClientSecret),
600
+ tokenUrl: trimNonEmpty(legacyGlobalState.sapAiCoreTokenUrl),
601
+ resourceGroup: trimNonEmpty(legacyGlobalState.sapAiResourceGroup),
602
+ deploymentId: trimNonEmpty(
603
+ mode === "plan"
604
+ ? legacyGlobalState.planModeSapAiCoreDeploymentId
605
+ : legacyGlobalState.actModeSapAiCoreDeploymentId,
606
+ ),
607
+ useOrchestrationMode: legacyGlobalState.sapAiCoreUseOrchestrationMode,
608
+ };
609
+ }
610
+ if (providerId === "oca") {
611
+ providerSpecific.oca = {
612
+ mode: legacyGlobalState.ocaMode,
613
+ };
614
+ const refreshToken = trimNonEmpty(legacySecrets.ocaRefreshToken);
615
+ if (refreshToken) {
616
+ providerSpecific.auth = {
617
+ ...(providerSpecific.auth ?? {}),
618
+ refreshToken,
619
+ };
620
+ }
621
+ }
622
+ if (providerId === "qwen") {
623
+ providerSpecific.apiLine = legacyGlobalState.qwenApiLine;
624
+ }
625
+ if (providerId === "moonshot") {
626
+ providerSpecific.apiLine = legacyGlobalState.moonshotApiLine;
627
+ }
628
+ if (providerId === "zai") {
629
+ providerSpecific.apiLine = legacyGlobalState.zaiApiLine;
630
+ }
631
+ if (providerId === "minimax") {
632
+ providerSpecific.apiLine = legacyGlobalState.minimaxApiLine;
633
+ }
634
+
635
+ const baseUrlByProvider: Record<string, string | undefined> = {
636
+ anthropic: legacyGlobalState.anthropicBaseUrl,
637
+ openai: legacyGlobalState.openAiBaseUrl,
638
+ ollama: legacyGlobalState.ollamaBaseUrl,
639
+ lmstudio: legacyGlobalState.lmStudioBaseUrl,
640
+ litellm: legacyGlobalState.liteLlmBaseUrl,
641
+ gemini: legacyGlobalState.geminiBaseUrl,
642
+ requesty: legacyGlobalState.requestyBaseUrl,
643
+ asksage: legacyGlobalState.asksageApiUrl,
644
+ dify: legacyGlobalState.difyBaseUrl,
645
+ oca: legacyGlobalState.ocaBaseUrl,
646
+ aihubmix: legacyGlobalState.aihubmixBaseUrl,
647
+ sapaicore: legacyGlobalState.sapAiCoreBaseUrl,
648
+ };
649
+
650
+ const apiKey = trimNonEmpty(secretByProvider[providerId]);
651
+ const baseUrl = trimNonEmpty(baseUrlByProvider[providerId]);
652
+
653
+ const settings: ProviderSettings = {
654
+ provider: targetProviderId as ProviderSettings["provider"],
655
+ ...(apiKey ? { apiKey } : {}),
656
+ ...(model ? { model } : {}),
657
+ ...(baseUrl ? { baseUrl } : {}),
658
+ ...(reasoning ? { reasoning } : {}),
659
+ ...(timeout ? { timeout } : {}),
660
+ ...providerSpecific,
661
+ };
662
+ const parsed = LlmsProviders.ProviderSettingsSchema.safeParse(settings);
663
+ if (!parsed.success) {
664
+ return undefined;
665
+ }
666
+ const hasNonProviderFields =
667
+ Object.keys(settings).filter((key) => key !== "provider").length > 0;
668
+ return hasNonProviderFields ? parsed.data : undefined;
669
+ }
670
+
671
+ function resolveLegacyCustomProviderRegistration(
672
+ providerId: string,
673
+ settings: ProviderSettings,
674
+ ): StoredModelsFile["providers"][string] | undefined {
675
+ if (providerId !== LEGACY_OPENAI_COMPATIBLE_PROVIDER_ID) {
676
+ return undefined;
677
+ }
678
+ if (!settings.baseUrl || !settings.model) {
679
+ return undefined;
680
+ }
681
+ return {
682
+ provider: {
683
+ name: "OpenAI Compatible",
684
+ baseUrl: settings.baseUrl,
685
+ defaultModelId: settings.model,
686
+ },
687
+ models: {
688
+ [settings.model]: {
689
+ id: settings.model,
690
+ name: settings.model,
691
+ },
692
+ },
693
+ };
694
+ }
695
+
696
+ function collectCandidateProviderIds(
697
+ legacyGlobalState: LegacyGlobalState,
698
+ legacySecrets: LegacySecrets,
699
+ ): Set<string> {
700
+ const candidates = new Set<string>();
701
+ for (const maybeProvider of [
702
+ legacyGlobalState.actModeApiProvider,
703
+ legacyGlobalState.planModeApiProvider,
704
+ ]) {
705
+ const provider = trimNonEmpty(maybeProvider);
706
+ if (provider) {
707
+ candidates.add(provider);
708
+ }
709
+ }
710
+ if (trimNonEmpty(legacySecrets.apiKey)) candidates.add("anthropic");
711
+ if (trimNonEmpty(legacySecrets.openRouterApiKey))
712
+ candidates.add("openrouter");
713
+ if (trimNonEmpty(legacySecrets.openAiApiKey)) candidates.add("openai");
714
+ if (trimNonEmpty(legacySecrets.openAiNativeApiKey))
715
+ candidates.add("openai-native");
716
+ if (trimNonEmpty(legacySecrets["openai-codex-oauth-credentials"]))
717
+ candidates.add("openai-codex");
718
+ if (trimNonEmpty(legacySecrets.geminiApiKey)) candidates.add("gemini");
719
+ if (trimNonEmpty(legacySecrets.ollamaApiKey)) candidates.add("ollama");
720
+ if (
721
+ trimNonEmpty(legacySecrets.awsAccessKey) ||
722
+ trimNonEmpty(legacySecrets.awsBedrockApiKey)
723
+ )
724
+ candidates.add("bedrock");
725
+ if (
726
+ trimNonEmpty(legacyGlobalState.vertexProjectId) ||
727
+ trimNonEmpty(legacyGlobalState.vertexRegion)
728
+ ) {
729
+ candidates.add("vertex");
730
+ }
731
+ if (trimNonEmpty(legacySecrets.clineApiKey)) candidates.add("cline");
732
+ if (trimNonEmpty(legacySecrets.ocaApiKey)) candidates.add("oca");
733
+ return candidates;
734
+ }
735
+
736
+ export function migrateLegacyProviderSettings(
737
+ options: MigrateLegacyProviderSettingsOptions,
738
+ ): MigrateLegacyProviderSettingsResult {
739
+ const existing = options.providerSettingsManager.read();
740
+ const legacyStorage = resolveLegacyStorage(options);
741
+ if (!legacyStorage) {
742
+ return {
743
+ migrated: false,
744
+ providerCount: Object.keys(existing.providers).length,
745
+ lastUsedProvider: existing.lastUsedProvider,
746
+ };
747
+ }
748
+
749
+ const { globalState, secrets } = legacyStorage;
750
+ const mode: LegacyMode = globalState.mode === "plan" ? "plan" : "act";
751
+ const candidates = collectCandidateProviderIds(globalState, secrets);
752
+ const next = emptyStoredProviderSettings();
753
+ next.providers = { ...existing.providers };
754
+ next.lastUsedProvider = existing.lastUsedProvider;
755
+ const now = new Date().toISOString();
756
+ let addedProviderCount = 0;
757
+ const modelsPath = join(
758
+ dirname(options.providerSettingsManager.getFilePath()),
759
+ "models.json",
760
+ );
761
+ const modelsState = readModelsFile(modelsPath);
762
+ let addedCustomProviderCount = 0;
763
+
764
+ for (const legacyProviderId of candidates) {
765
+ const providerId = resolveMigratedProviderId(legacyProviderId, globalState);
766
+ if (next.providers[providerId]) {
767
+ continue;
768
+ }
769
+ const settings = buildLegacyProviderSettings(
770
+ legacyProviderId,
771
+ globalState,
772
+ secrets,
773
+ mode,
774
+ );
775
+ if (!settings) {
776
+ continue;
777
+ }
778
+ next.providers[providerId] = {
779
+ settings,
780
+ updatedAt: now,
781
+ tokenSource: "migration",
782
+ };
783
+ addedProviderCount += 1;
784
+ const registration = resolveLegacyCustomProviderRegistration(
785
+ providerId,
786
+ settings,
787
+ );
788
+ if (registration && !modelsState.providers[providerId]) {
789
+ modelsState.providers[providerId] = registration;
790
+ addedCustomProviderCount += 1;
791
+ }
792
+ }
793
+
794
+ if (addedProviderCount === 0 && addedCustomProviderCount === 0) {
795
+ return {
796
+ migrated: false,
797
+ providerCount: Object.keys(existing.providers).length,
798
+ lastUsedProvider: existing.lastUsedProvider,
799
+ };
800
+ }
801
+
802
+ const preferredProvider = trimNonEmpty(
803
+ mode === "plan"
804
+ ? globalState.planModeApiProvider
805
+ : globalState.actModeApiProvider,
806
+ );
807
+ const migratedPreferredProvider = preferredProvider
808
+ ? resolveMigratedProviderId(preferredProvider, globalState)
809
+ : undefined;
810
+ next.lastUsedProvider =
811
+ existing.lastUsedProvider ??
812
+ (migratedPreferredProvider && next.providers[migratedPreferredProvider]
813
+ ? migratedPreferredProvider
814
+ : Object.keys(next.providers)[0]);
815
+
816
+ options.providerSettingsManager.write(next);
817
+ if (addedCustomProviderCount > 0) {
818
+ writeModelsFile(modelsPath, modelsState);
819
+ }
820
+
821
+ return {
822
+ migrated: addedProviderCount > 0 || addedCustomProviderCount > 0,
823
+ providerCount: Object.keys(next.providers).length,
824
+ lastUsedProvider: next.lastUsedProvider,
825
+ };
826
+ }