@clinebot/core 0.0.22 → 0.0.23

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,191 @@
1
+ import { mkdtempSync, rmSync, writeFileSync } from "node:fs";
2
+ import os from "node:os";
3
+ import path from "node:path";
4
+ import * as LlmsModels from "@clinebot/llms/models";
5
+ import { afterEach, describe, expect, it } from "vitest";
6
+ import { ProviderSettingsManager } from "./provider-settings-manager";
7
+
8
+ describe("ProviderSettingsManager", () => {
9
+ const tempDirs: string[] = [];
10
+
11
+ afterEach(() => {
12
+ LlmsModels.resetRegistry();
13
+ for (const dir of tempDirs.splice(0)) {
14
+ rmSync(dir, { recursive: true, force: true });
15
+ }
16
+ });
17
+
18
+ it("persists and restores provider settings", () => {
19
+ const tempDir = mkdtempSync(
20
+ path.join(os.tmpdir(), "core-provider-settings-"),
21
+ );
22
+ tempDirs.push(tempDir);
23
+ const filePath = path.join(tempDir, "provider-settings.json");
24
+ const manager = new ProviderSettingsManager({ filePath });
25
+
26
+ manager.saveProviderSettings(
27
+ {
28
+ provider: "anthropic",
29
+ model: "claude-sonnet-4-6",
30
+ apiKey: "test-key",
31
+ },
32
+ { setLastUsed: true },
33
+ );
34
+
35
+ const reloaded = new ProviderSettingsManager({ filePath });
36
+ expect(reloaded.getLastUsedProviderSettings()).toEqual({
37
+ provider: "anthropic",
38
+ model: "claude-sonnet-4-6",
39
+ apiKey: "test-key",
40
+ });
41
+ expect(reloaded.getProviderConfig("anthropic")?.providerId).toBe(
42
+ "anthropic",
43
+ );
44
+ expect(reloaded.getProviderConfig("anthropic")?.modelId).toBe(
45
+ "claude-sonnet-4-6",
46
+ );
47
+ expect(reloaded.read().providers.anthropic?.tokenSource).toBe("manual");
48
+ });
49
+
50
+ it("migrates legacy provider settings during manager construction", () => {
51
+ const tempDir = mkdtempSync(
52
+ path.join(os.tmpdir(), "core-provider-settings-"),
53
+ );
54
+ tempDirs.push(tempDir);
55
+ const filePath = path.join(tempDir, "settings", "providers.json");
56
+
57
+ writeFileSync(
58
+ path.join(tempDir, "globalState.json"),
59
+ JSON.stringify(
60
+ {
61
+ mode: "act",
62
+ actModeApiProvider: "anthropic",
63
+ actModeApiModelId: "claude-sonnet-4-6",
64
+ },
65
+ null,
66
+ 2,
67
+ ),
68
+ );
69
+ writeFileSync(
70
+ path.join(tempDir, "secrets.json"),
71
+ JSON.stringify({ apiKey: "legacy-key" }, null, 2),
72
+ );
73
+
74
+ const manager = new ProviderSettingsManager({ filePath, dataDir: tempDir });
75
+
76
+ expect(manager.getLastUsedProviderSettings()).toEqual({
77
+ provider: "anthropic",
78
+ model: "claude-sonnet-4-6",
79
+ apiKey: "legacy-key",
80
+ });
81
+ expect(manager.read().providers.anthropic?.tokenSource).toBe("migration");
82
+ });
83
+
84
+ it("registers migrated custom providers during manager construction", async () => {
85
+ const tempDir = mkdtempSync(
86
+ path.join(os.tmpdir(), "core-provider-settings-"),
87
+ );
88
+ tempDirs.push(tempDir);
89
+ const filePath = path.join(tempDir, "settings", "providers.json");
90
+
91
+ writeFileSync(
92
+ path.join(tempDir, "globalState.json"),
93
+ JSON.stringify(
94
+ {
95
+ mode: "act",
96
+ actModeApiProvider: "openai",
97
+ actModeOpenAiModelId: "gpt-oss-120b",
98
+ openAiBaseUrl: "https://gateway.example.invalid/v1",
99
+ },
100
+ null,
101
+ 2,
102
+ ),
103
+ );
104
+ writeFileSync(
105
+ path.join(tempDir, "secrets.json"),
106
+ JSON.stringify({ openAiApiKey: "legacy-key" }, null, 2),
107
+ );
108
+
109
+ const manager = new ProviderSettingsManager({ filePath, dataDir: tempDir });
110
+ const providers = await LlmsModels.getAllProviders();
111
+ const openAiProvider = providers.find(
112
+ (provider) => provider.id === "openai",
113
+ );
114
+
115
+ expect(manager.getProviderSettings("openai")).toEqual({
116
+ provider: "openai",
117
+ model: "gpt-oss-120b",
118
+ apiKey: "legacy-key",
119
+ baseUrl: "https://gateway.example.invalid/v1",
120
+ });
121
+ expect(openAiProvider).toMatchObject({
122
+ id: "openai",
123
+ baseUrl: "https://gateway.example.invalid/v1",
124
+ defaultModelId: "gpt-oss-120b",
125
+ });
126
+ });
127
+
128
+ it("tracks provider-specific settings while preserving last-used provider", () => {
129
+ const tempDir = mkdtempSync(
130
+ path.join(os.tmpdir(), "core-provider-settings-"),
131
+ );
132
+ tempDirs.push(tempDir);
133
+ const filePath = path.join(tempDir, "provider-settings.json");
134
+ const manager = new ProviderSettingsManager({ filePath });
135
+
136
+ manager.saveProviderSettings({
137
+ provider: "anthropic",
138
+ model: "claude-sonnet-4-6",
139
+ });
140
+ manager.saveProviderSettings(
141
+ {
142
+ provider: "openai-native",
143
+ model: "gpt-5",
144
+ },
145
+ { setLastUsed: false },
146
+ );
147
+
148
+ expect(manager.getProviderSettings("anthropic")?.model).toBe(
149
+ "claude-sonnet-4-6",
150
+ );
151
+ expect(manager.getProviderSettings("openai-native")?.model).toBe("gpt-5");
152
+ expect(manager.getLastUsedProviderSettings()?.provider).toBe("anthropic");
153
+ expect(manager.read().providers["openai-native"]?.tokenSource).toBe(
154
+ "manual",
155
+ );
156
+ });
157
+
158
+ it("allows overriding token source metadata", () => {
159
+ const tempDir = mkdtempSync(
160
+ path.join(os.tmpdir(), "core-provider-settings-"),
161
+ );
162
+ tempDirs.push(tempDir);
163
+ const filePath = path.join(tempDir, "provider-settings.json");
164
+ const manager = new ProviderSettingsManager({ filePath });
165
+
166
+ manager.saveProviderSettings(
167
+ {
168
+ provider: "openai-codex",
169
+ apiKey: "oauth-token",
170
+ },
171
+ { tokenSource: "oauth" },
172
+ );
173
+
174
+ expect(manager.read().providers["openai-codex"]?.tokenSource).toBe("oauth");
175
+ });
176
+
177
+ it("ignores invalid persisted JSON and falls back to empty state", () => {
178
+ const tempDir = mkdtempSync(
179
+ path.join(os.tmpdir(), "core-provider-settings-"),
180
+ );
181
+ tempDirs.push(tempDir);
182
+ const filePath = path.join(tempDir, "provider-settings.json");
183
+ writeFileSync(filePath, "{ not-json", "utf8");
184
+
185
+ const manager = new ProviderSettingsManager({ filePath });
186
+ expect(manager.read()).toEqual({
187
+ version: 1,
188
+ providers: {},
189
+ });
190
+ });
191
+ });
@@ -0,0 +1,152 @@
1
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
2
+ import { basename, dirname } from "node:path";
3
+ import { resolveProviderSettingsPath } from "@clinebot/shared/storage";
4
+ import { ensureCustomProvidersLoadedSync } from "../providers/local-provider-registry";
5
+ import {
6
+ emptyStoredProviderSettings,
7
+ type ProviderConfig,
8
+ type ProviderSettings,
9
+ ProviderSettingsSchema,
10
+ type ProviderTokenSource,
11
+ type StoredProviderSettings,
12
+ StoredProviderSettingsSchema,
13
+ toProviderConfig,
14
+ } from "../types/provider-settings";
15
+ import { migrateLegacyProviderSettings } from "./provider-settings-legacy-migration";
16
+
17
+ function nowIso(): string {
18
+ return new Date().toISOString();
19
+ }
20
+
21
+ export interface ProviderSettingsManagerOptions {
22
+ filePath?: string;
23
+ dataDir?: string;
24
+ }
25
+
26
+ export interface SaveProviderSettingsOptions {
27
+ setLastUsed?: boolean;
28
+ tokenSource?: ProviderTokenSource;
29
+ }
30
+
31
+ function inferLegacyDataDir(filePath: string): string | undefined {
32
+ if (basename(filePath) !== "providers.json") {
33
+ return undefined;
34
+ }
35
+ const settingsDir = dirname(filePath);
36
+ if (basename(settingsDir) !== "settings") {
37
+ return undefined;
38
+ }
39
+ return dirname(settingsDir);
40
+ }
41
+
42
+ export class ProviderSettingsManager {
43
+ private readonly filePath: string;
44
+ private readonly dataDir?: string;
45
+
46
+ constructor(options: ProviderSettingsManagerOptions = {}) {
47
+ this.filePath = options.filePath ?? resolveProviderSettingsPath();
48
+ this.dataDir = options.dataDir ?? inferLegacyDataDir(this.filePath);
49
+ if (this.dataDir || !options.filePath) {
50
+ migrateLegacyProviderSettings({
51
+ providerSettingsManager: this,
52
+ dataDir: this.dataDir,
53
+ });
54
+ }
55
+ ensureCustomProvidersLoadedSync(this);
56
+ }
57
+
58
+ getFilePath(): string {
59
+ return this.filePath;
60
+ }
61
+
62
+ read(): StoredProviderSettings {
63
+ if (!existsSync(this.filePath)) {
64
+ return emptyStoredProviderSettings();
65
+ }
66
+
67
+ try {
68
+ const raw = readFileSync(this.filePath, "utf8");
69
+ const parsed = JSON.parse(raw) as unknown;
70
+ const result = StoredProviderSettingsSchema.safeParse(parsed);
71
+ if (result.success) {
72
+ return result.data;
73
+ }
74
+ } catch {
75
+ // Invalid content falls back to a clean state.
76
+ }
77
+
78
+ return emptyStoredProviderSettings();
79
+ }
80
+
81
+ write(state: StoredProviderSettings): void {
82
+ const normalized = StoredProviderSettingsSchema.parse(state);
83
+ const dir = dirname(this.filePath);
84
+ if (!existsSync(dir)) {
85
+ mkdirSync(dir, { recursive: true });
86
+ }
87
+ writeFileSync(
88
+ this.filePath,
89
+ `${JSON.stringify(normalized, null, 2)}\n`,
90
+ "utf8",
91
+ );
92
+ }
93
+
94
+ saveProviderSettings(
95
+ settings: unknown,
96
+ options: SaveProviderSettingsOptions = {},
97
+ ): StoredProviderSettings {
98
+ const validatedSettings = ProviderSettingsSchema.parse(settings);
99
+ const previous = this.read();
100
+ const providerId = validatedSettings.provider;
101
+ const shouldSetLastUsed = options.setLastUsed !== false;
102
+ const previousEntry = previous.providers[providerId];
103
+ const tokenSource =
104
+ options.tokenSource ?? previousEntry?.tokenSource ?? "manual";
105
+ const next: StoredProviderSettings = {
106
+ ...previous,
107
+ providers: {
108
+ ...previous.providers,
109
+ [providerId]: {
110
+ settings: validatedSettings,
111
+ updatedAt: nowIso(),
112
+ tokenSource,
113
+ },
114
+ },
115
+ lastUsedProvider: shouldSetLastUsed
116
+ ? providerId
117
+ : previous.lastUsedProvider,
118
+ };
119
+ this.write(next);
120
+ return next;
121
+ }
122
+
123
+ getProviderSettings(providerId: string): ProviderSettings | undefined {
124
+ const state = this.read();
125
+ return state.providers[providerId]?.settings;
126
+ }
127
+
128
+ getLastUsedProviderSettings(): ProviderSettings | undefined {
129
+ const state = this.read();
130
+ const providerId = state.lastUsedProvider;
131
+ if (!providerId) {
132
+ return undefined;
133
+ }
134
+ return state.providers[providerId]?.settings;
135
+ }
136
+
137
+ getProviderConfig(providerId: string): ProviderConfig | undefined {
138
+ const settings = this.getProviderSettings(providerId);
139
+ if (!settings) {
140
+ return undefined;
141
+ }
142
+ return toProviderConfig(settings);
143
+ }
144
+
145
+ getLastUsedProviderConfig(): ProviderConfig | undefined {
146
+ const settings = this.getLastUsedProviderSettings();
147
+ if (!settings) {
148
+ return undefined;
149
+ }
150
+ return toProviderConfig(settings);
151
+ }
152
+ }
@@ -0,0 +1 @@
1
+ export type { SessionStore } from "../types/storage";
@@ -0,0 +1,275 @@
1
+ import { existsSync, mkdirSync } from "node:fs";
2
+ import { join } from "node:path";
3
+ import {
4
+ asBool,
5
+ asOptionalString,
6
+ asString,
7
+ ensureSessionSchema,
8
+ loadSqliteDb,
9
+ nowIso,
10
+ type SqliteDb,
11
+ toBoolInt,
12
+ } from "@clinebot/shared/db";
13
+ import { resolveSessionDataDir } from "@clinebot/shared/storage";
14
+ import type { SessionStatus } from "../types/common";
15
+ import type { SessionRecord } from "../types/sessions";
16
+ import type { SessionStore } from "../types/storage";
17
+
18
+ function defaultSessionsDir(): string {
19
+ return resolveSessionDataDir();
20
+ }
21
+
22
+ export interface SqliteSessionStoreOptions {
23
+ sessionsDir?: string;
24
+ }
25
+
26
+ export class SqliteSessionStore implements SessionStore {
27
+ private readonly sessionsDirPath: string;
28
+ private db: SqliteDb | undefined;
29
+
30
+ constructor(options: SqliteSessionStoreOptions = {}) {
31
+ this.sessionsDirPath = options.sessionsDir ?? defaultSessionsDir();
32
+ }
33
+
34
+ init(): void {
35
+ this.getRawDb();
36
+ }
37
+
38
+ ensureSessionsDir(): string {
39
+ if (!existsSync(this.sessionsDirPath)) {
40
+ mkdirSync(this.sessionsDirPath, { recursive: true });
41
+ }
42
+ return this.sessionsDirPath;
43
+ }
44
+
45
+ sessionDbPath(): string {
46
+ return join(this.ensureSessionsDir(), "sessions.db");
47
+ }
48
+
49
+ getRawDb(): SqliteDb {
50
+ if (this.db) {
51
+ return this.db;
52
+ }
53
+ const db = loadSqliteDb(this.sessionDbPath());
54
+ ensureSessionSchema(db, { includeLegacyMigrations: true });
55
+
56
+ this.db = db;
57
+ return db;
58
+ }
59
+
60
+ close(): void {
61
+ this.db?.close?.();
62
+ this.db = undefined;
63
+ }
64
+
65
+ run(sql: string, params: unknown[] = []): { changes?: number } {
66
+ return this.getRawDb()
67
+ .prepare(sql)
68
+ .run(...params);
69
+ }
70
+
71
+ queryOne<T>(sql: string, params: unknown[] = []): T | undefined {
72
+ const row = this.getRawDb()
73
+ .prepare(sql)
74
+ .get(...params);
75
+ return (row as T | null) ?? undefined;
76
+ }
77
+
78
+ queryAll<T>(sql: string, params: unknown[] = []): T[] {
79
+ return this.getRawDb()
80
+ .prepare(sql)
81
+ .all(...params) as T[];
82
+ }
83
+
84
+ create(record: SessionRecord): void {
85
+ const now = nowIso();
86
+ this.run(
87
+ `INSERT OR REPLACE INTO sessions (
88
+ session_id, source, pid, started_at, ended_at, exit_code, status, status_lock, interactive,
89
+ provider, model, cwd, workspace_root, team_name, enable_tools, enable_spawn, enable_teams,
90
+ parent_session_id, parent_agent_id, agent_id, conversation_id, is_subagent, prompt,
91
+ metadata_json, transcript_path, hook_path, messages_path, updated_at
92
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
93
+ [
94
+ record.sessionId,
95
+ record.source,
96
+ record.pid,
97
+ record.startedAt,
98
+ record.endedAt ?? null,
99
+ record.exitCode ?? null,
100
+ record.status,
101
+ 0,
102
+ toBoolInt(record.interactive),
103
+ record.provider,
104
+ record.model,
105
+ record.cwd,
106
+ record.workspaceRoot,
107
+ record.teamName ?? null,
108
+ toBoolInt(record.enableTools),
109
+ toBoolInt(record.enableSpawn),
110
+ toBoolInt(record.enableTeams),
111
+ record.parentSessionId ?? null,
112
+ record.parentAgentId ?? null,
113
+ record.agentId ?? null,
114
+ record.conversationId ?? null,
115
+ toBoolInt(record.isSubagent),
116
+ record.prompt ?? null,
117
+ record.metadata ? JSON.stringify(record.metadata) : null,
118
+ record.transcriptPath ?? "",
119
+ record.hookPath ?? "",
120
+ record.messagesPath ?? null,
121
+ now,
122
+ ],
123
+ );
124
+ }
125
+
126
+ update(record: Partial<SessionRecord> & { sessionId: string }): void {
127
+ const fields: string[] = [];
128
+ const params: unknown[] = [];
129
+ if (record.endedAt !== undefined) {
130
+ fields.push("ended_at = ?");
131
+ params.push(record.endedAt);
132
+ }
133
+ if (record.exitCode !== undefined) {
134
+ fields.push("exit_code = ?");
135
+ params.push(record.exitCode);
136
+ }
137
+ if (record.status !== undefined) {
138
+ fields.push("status = ?");
139
+ params.push(record.status);
140
+ }
141
+ if (record.prompt !== undefined) {
142
+ fields.push("prompt = ?");
143
+ params.push(record.prompt);
144
+ }
145
+ if (record.metadata !== undefined) {
146
+ fields.push("metadata_json = ?");
147
+ params.push(record.metadata ? JSON.stringify(record.metadata) : null);
148
+ }
149
+ if (record.parentSessionId !== undefined) {
150
+ fields.push("parent_session_id = ?");
151
+ params.push(record.parentSessionId);
152
+ }
153
+ if (record.parentAgentId !== undefined) {
154
+ fields.push("parent_agent_id = ?");
155
+ params.push(record.parentAgentId);
156
+ }
157
+ if (record.agentId !== undefined) {
158
+ fields.push("agent_id = ?");
159
+ params.push(record.agentId);
160
+ }
161
+ if (record.conversationId !== undefined) {
162
+ fields.push("conversation_id = ?");
163
+ params.push(record.conversationId);
164
+ }
165
+ if (fields.length === 0) {
166
+ return;
167
+ }
168
+ fields.push("updated_at = ?");
169
+ params.push(nowIso());
170
+ params.push(record.sessionId);
171
+ this.run(
172
+ `UPDATE sessions SET ${fields.join(", ")} WHERE session_id = ?`,
173
+ params,
174
+ );
175
+ }
176
+
177
+ updateStatus(
178
+ sessionId: string,
179
+ status: SessionStatus,
180
+ exitCode?: number | null,
181
+ ): void {
182
+ this.update({
183
+ sessionId,
184
+ status,
185
+ endedAt: status === "running" ? null : nowIso(),
186
+ exitCode:
187
+ status === "running"
188
+ ? null
189
+ : (exitCode ?? (status === "failed" ? 1 : 0)),
190
+ });
191
+ }
192
+
193
+ get(sessionId: string): SessionRecord | undefined {
194
+ const row = this.queryOne<Record<string, unknown>>(
195
+ `SELECT session_id, source, pid, started_at, ended_at, exit_code, status, interactive,
196
+ provider, model, cwd, workspace_root, team_name,
197
+ enable_tools, enable_spawn, enable_teams,
198
+ parent_session_id, parent_agent_id, agent_id, conversation_id, is_subagent,
199
+ prompt, metadata_json, transcript_path, hook_path, messages_path, updated_at
200
+ FROM sessions WHERE session_id = ?`,
201
+ [sessionId],
202
+ );
203
+ if (!row) {
204
+ return undefined;
205
+ }
206
+ return {
207
+ sessionId: asString(row.session_id),
208
+ source: asString(row.source) as SessionRecord["source"],
209
+ pid: Number(row.pid ?? 0),
210
+ startedAt: asString(row.started_at),
211
+ endedAt: (row.ended_at as string | null | undefined) ?? null,
212
+ exitCode: (row.exit_code as number | null | undefined) ?? null,
213
+ status: asString(row.status) as SessionRecord["status"],
214
+ interactive: asBool(row.interactive),
215
+ provider: asString(row.provider),
216
+ model: asString(row.model),
217
+ cwd: asString(row.cwd),
218
+ workspaceRoot: asString(row.workspace_root),
219
+ teamName: asOptionalString(row.team_name),
220
+ enableTools: asBool(row.enable_tools),
221
+ enableSpawn: asBool(row.enable_spawn),
222
+ enableTeams: asBool(row.enable_teams),
223
+ parentSessionId: asOptionalString(row.parent_session_id),
224
+ parentAgentId: asOptionalString(row.parent_agent_id),
225
+ agentId: asOptionalString(row.agent_id),
226
+ conversationId: asOptionalString(row.conversation_id),
227
+ isSubagent: asBool(row.is_subagent),
228
+ prompt: asOptionalString(row.prompt),
229
+ metadata: (() => {
230
+ const raw = asOptionalString(row.metadata_json);
231
+ if (!raw) {
232
+ return undefined;
233
+ }
234
+ try {
235
+ const parsed = JSON.parse(raw) as unknown;
236
+ if (parsed && typeof parsed === "object" && !Array.isArray(parsed)) {
237
+ return parsed as Record<string, unknown>;
238
+ }
239
+ } catch {
240
+ // Ignore malformed metadata payloads.
241
+ }
242
+ return undefined;
243
+ })(),
244
+ transcriptPath: asOptionalString(row.transcript_path),
245
+ hookPath: asOptionalString(row.hook_path),
246
+ messagesPath: asOptionalString(row.messages_path),
247
+ updatedAt: asOptionalString(row.updated_at) ?? nowIso(),
248
+ };
249
+ }
250
+
251
+ list(limit = 200): SessionRecord[] {
252
+ const rows = this.queryAll<Record<string, unknown>>(
253
+ `SELECT session_id FROM sessions ORDER BY started_at DESC LIMIT ?`,
254
+ [limit],
255
+ );
256
+ const result: SessionRecord[] = [];
257
+ for (const row of rows) {
258
+ const item = this.get(asString(row.session_id));
259
+ if (item) {
260
+ result.push(item);
261
+ }
262
+ }
263
+ return result;
264
+ }
265
+
266
+ delete(sessionId: string, cascade = false): boolean {
267
+ const changed =
268
+ this.run(`DELETE FROM sessions WHERE session_id = ?`, [sessionId])
269
+ .changes ?? 0;
270
+ if (cascade) {
271
+ this.run(`DELETE FROM sessions WHERE parent_session_id = ?`, [sessionId]);
272
+ }
273
+ return changed > 0;
274
+ }
275
+ }