@gatanot/qualia_core 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (297) hide show
  1. package/dist/agent/background.d.ts +6 -0
  2. package/dist/agent/background.d.ts.map +1 -0
  3. package/dist/agent/background.js +58 -0
  4. package/dist/agent/background.js.map +1 -0
  5. package/dist/agent/context-builder.d.ts +23 -0
  6. package/dist/agent/context-builder.d.ts.map +1 -0
  7. package/dist/agent/context-builder.js +102 -0
  8. package/dist/agent/context-builder.js.map +1 -0
  9. package/dist/agent/diary.d.ts +4 -0
  10. package/dist/agent/diary.d.ts.map +1 -0
  11. package/dist/agent/diary.js +67 -0
  12. package/dist/agent/diary.js.map +1 -0
  13. package/dist/agent/index.d.ts +22 -0
  14. package/dist/agent/index.d.ts.map +1 -0
  15. package/dist/agent/index.js +20 -0
  16. package/dist/agent/index.js.map +1 -0
  17. package/dist/agent/logger.d.ts +23 -0
  18. package/dist/agent/logger.d.ts.map +1 -0
  19. package/dist/agent/logger.js +70 -0
  20. package/dist/agent/logger.js.map +1 -0
  21. package/dist/agent/logger.test.d.ts +2 -0
  22. package/dist/agent/logger.test.d.ts.map +1 -0
  23. package/dist/agent/logger.test.js +28 -0
  24. package/dist/agent/logger.test.js.map +1 -0
  25. package/dist/agent/loop.d.ts +52 -0
  26. package/dist/agent/loop.d.ts.map +1 -0
  27. package/dist/agent/loop.js +557 -0
  28. package/dist/agent/loop.js.map +1 -0
  29. package/dist/agent/message-sanitizer.d.ts +13 -0
  30. package/dist/agent/message-sanitizer.d.ts.map +1 -0
  31. package/dist/agent/message-sanitizer.js +138 -0
  32. package/dist/agent/message-sanitizer.js.map +1 -0
  33. package/dist/agent/message-sanitizer.test.d.ts +2 -0
  34. package/dist/agent/message-sanitizer.test.d.ts.map +1 -0
  35. package/dist/agent/message-sanitizer.test.js +110 -0
  36. package/dist/agent/message-sanitizer.test.js.map +1 -0
  37. package/dist/agent/prompts.d.ts +15 -0
  38. package/dist/agent/prompts.d.ts.map +1 -0
  39. package/dist/agent/prompts.js +34 -0
  40. package/dist/agent/prompts.js.map +1 -0
  41. package/dist/agent/summarize-worker.d.ts +6 -0
  42. package/dist/agent/summarize-worker.d.ts.map +1 -0
  43. package/dist/agent/summarize-worker.js +62 -0
  44. package/dist/agent/summarize-worker.js.map +1 -0
  45. package/dist/agent/summarizer.d.ts +10 -0
  46. package/dist/agent/summarizer.d.ts.map +1 -0
  47. package/dist/agent/summarizer.js +133 -0
  48. package/dist/agent/summarizer.js.map +1 -0
  49. package/dist/agent/types.d.ts +135 -0
  50. package/dist/agent/types.d.ts.map +1 -0
  51. package/dist/agent/types.js +21 -0
  52. package/dist/agent/types.js.map +1 -0
  53. package/dist/ai/base.d.ts +19 -0
  54. package/dist/ai/base.d.ts.map +1 -0
  55. package/dist/ai/base.js +2 -0
  56. package/dist/ai/base.js.map +1 -0
  57. package/dist/ai/deepseek.d.ts +15 -0
  58. package/dist/ai/deepseek.d.ts.map +1 -0
  59. package/dist/ai/deepseek.js +24 -0
  60. package/dist/ai/deepseek.js.map +1 -0
  61. package/dist/ai/factory.d.ts +4 -0
  62. package/dist/ai/factory.d.ts.map +1 -0
  63. package/dist/ai/factory.js +53 -0
  64. package/dist/ai/factory.js.map +1 -0
  65. package/dist/ai/index.d.ts +43 -0
  66. package/dist/ai/index.d.ts.map +1 -0
  67. package/dist/ai/index.js +36 -0
  68. package/dist/ai/index.js.map +1 -0
  69. package/dist/ai/models.d.ts +13 -0
  70. package/dist/ai/models.d.ts.map +1 -0
  71. package/dist/ai/models.js +67 -0
  72. package/dist/ai/models.js.map +1 -0
  73. package/dist/ai/ollama.d.ts +13 -0
  74. package/dist/ai/ollama.d.ts.map +1 -0
  75. package/dist/ai/ollama.js +32 -0
  76. package/dist/ai/ollama.js.map +1 -0
  77. package/dist/ai/openai.d.ts +46 -0
  78. package/dist/ai/openai.d.ts.map +1 -0
  79. package/dist/ai/openai.js +90 -0
  80. package/dist/ai/openai.js.map +1 -0
  81. package/dist/ai/types.d.ts +154 -0
  82. package/dist/ai/types.d.ts.map +1 -0
  83. package/dist/ai/types.js +2 -0
  84. package/dist/ai/types.js.map +1 -0
  85. package/dist/ai/utils.d.ts +47 -0
  86. package/dist/ai/utils.d.ts.map +1 -0
  87. package/dist/ai/utils.js +145 -0
  88. package/dist/ai/utils.js.map +1 -0
  89. package/dist/ai/xiaomi.d.ts +15 -0
  90. package/dist/ai/xiaomi.d.ts.map +1 -0
  91. package/dist/ai/xiaomi.js +23 -0
  92. package/dist/ai/xiaomi.js.map +1 -0
  93. package/dist/chat-confirm.d.ts +4 -0
  94. package/dist/chat-confirm.d.ts.map +1 -0
  95. package/dist/chat-confirm.js +2 -0
  96. package/dist/chat-confirm.js.map +1 -0
  97. package/dist/chat-steering.d.ts +6 -0
  98. package/dist/chat-steering.d.ts.map +1 -0
  99. package/dist/chat-steering.js +2 -0
  100. package/dist/chat-steering.js.map +1 -0
  101. package/dist/concurrency/background-worker.d.ts +16 -0
  102. package/dist/concurrency/background-worker.d.ts.map +1 -0
  103. package/dist/concurrency/background-worker.js +62 -0
  104. package/dist/concurrency/background-worker.js.map +1 -0
  105. package/dist/concurrency/background-worker.test.d.ts +2 -0
  106. package/dist/concurrency/background-worker.test.d.ts.map +1 -0
  107. package/dist/concurrency/background-worker.test.js +55 -0
  108. package/dist/concurrency/background-worker.test.js.map +1 -0
  109. package/dist/concurrency/file-mutex.d.ts +8 -0
  110. package/dist/concurrency/file-mutex.d.ts.map +1 -0
  111. package/dist/concurrency/file-mutex.js +41 -0
  112. package/dist/concurrency/file-mutex.js.map +1 -0
  113. package/dist/concurrency/file-mutex.test.d.ts +2 -0
  114. package/dist/concurrency/file-mutex.test.d.ts.map +1 -0
  115. package/dist/concurrency/file-mutex.test.js +50 -0
  116. package/dist/concurrency/file-mutex.test.js.map +1 -0
  117. package/dist/concurrency/index.d.ts +5 -0
  118. package/dist/concurrency/index.d.ts.map +1 -0
  119. package/dist/concurrency/index.js +4 -0
  120. package/dist/concurrency/index.js.map +1 -0
  121. package/dist/concurrency/session-lock.d.ts +8 -0
  122. package/dist/concurrency/session-lock.d.ts.map +1 -0
  123. package/dist/concurrency/session-lock.js +41 -0
  124. package/dist/concurrency/session-lock.js.map +1 -0
  125. package/dist/concurrency/session-lock.test.d.ts +2 -0
  126. package/dist/concurrency/session-lock.test.d.ts.map +1 -0
  127. package/dist/concurrency/session-lock.test.js +51 -0
  128. package/dist/concurrency/session-lock.test.js.map +1 -0
  129. package/dist/config/index.d.ts +12 -0
  130. package/dist/config/index.d.ts.map +1 -0
  131. package/dist/config/index.js +10 -0
  132. package/dist/config/index.js.map +1 -0
  133. package/dist/config/store.d.ts +17 -0
  134. package/dist/config/store.d.ts.map +1 -0
  135. package/dist/config/store.js +214 -0
  136. package/dist/config/store.js.map +1 -0
  137. package/dist/config/types.d.ts +44 -0
  138. package/dist/config/types.d.ts.map +1 -0
  139. package/dist/config/types.js +2 -0
  140. package/dist/config/types.js.map +1 -0
  141. package/dist/gateway/adapters/email.d.ts +23 -0
  142. package/dist/gateway/adapters/email.d.ts.map +1 -0
  143. package/dist/gateway/adapters/email.js +145 -0
  144. package/dist/gateway/adapters/email.js.map +1 -0
  145. package/dist/gateway/adapters/telegram-sessions.d.ts +4 -0
  146. package/dist/gateway/adapters/telegram-sessions.d.ts.map +1 -0
  147. package/dist/gateway/adapters/telegram-sessions.js +39 -0
  148. package/dist/gateway/adapters/telegram-sessions.js.map +1 -0
  149. package/dist/gateway/adapters/telegram.d.ts +22 -0
  150. package/dist/gateway/adapters/telegram.d.ts.map +1 -0
  151. package/dist/gateway/adapters/telegram.js +131 -0
  152. package/dist/gateway/adapters/telegram.js.map +1 -0
  153. package/dist/gateway/dispatcher.d.ts +19 -0
  154. package/dist/gateway/dispatcher.d.ts.map +1 -0
  155. package/dist/gateway/dispatcher.js +81 -0
  156. package/dist/gateway/dispatcher.js.map +1 -0
  157. package/dist/gateway/inbound-handler.d.ts +3 -0
  158. package/dist/gateway/inbound-handler.d.ts.map +1 -0
  159. package/dist/gateway/inbound-handler.js +95 -0
  160. package/dist/gateway/inbound-handler.js.map +1 -0
  161. package/dist/gateway/index.d.ts +9 -0
  162. package/dist/gateway/index.d.ts.map +1 -0
  163. package/dist/gateway/index.js +6 -0
  164. package/dist/gateway/index.js.map +1 -0
  165. package/dist/gateway/lifecycle.d.ts +3 -0
  166. package/dist/gateway/lifecycle.d.ts.map +1 -0
  167. package/dist/gateway/lifecycle.js +38 -0
  168. package/dist/gateway/lifecycle.js.map +1 -0
  169. package/dist/gateway/types.d.ts +32 -0
  170. package/dist/gateway/types.d.ts.map +1 -0
  171. package/dist/gateway/types.js +2 -0
  172. package/dist/gateway/types.js.map +1 -0
  173. package/dist/index.d.ts +23 -0
  174. package/dist/index.d.ts.map +1 -0
  175. package/dist/index.js +14 -0
  176. package/dist/index.js.map +1 -0
  177. package/dist/markdown.d.ts +2 -0
  178. package/dist/markdown.d.ts.map +1 -0
  179. package/dist/markdown.js +94 -0
  180. package/dist/markdown.js.map +1 -0
  181. package/dist/paths.d.ts +5 -0
  182. package/dist/paths.d.ts.map +1 -0
  183. package/dist/paths.js +17 -0
  184. package/dist/paths.js.map +1 -0
  185. package/dist/storage/index.d.ts +10 -0
  186. package/dist/storage/index.d.ts.map +1 -0
  187. package/dist/storage/index.js +18 -0
  188. package/dist/storage/index.js.map +1 -0
  189. package/dist/storage/memory.d.ts +39 -0
  190. package/dist/storage/memory.d.ts.map +1 -0
  191. package/dist/storage/memory.js +252 -0
  192. package/dist/storage/memory.js.map +1 -0
  193. package/dist/storage/sqlite.d.ts +44 -0
  194. package/dist/storage/sqlite.d.ts.map +1 -0
  195. package/dist/storage/sqlite.js +275 -0
  196. package/dist/storage/sqlite.js.map +1 -0
  197. package/dist/storage/types.d.ts +134 -0
  198. package/dist/storage/types.d.ts.map +1 -0
  199. package/dist/storage/types.js +2 -0
  200. package/dist/storage/types.js.map +1 -0
  201. package/dist/storage/utils.d.ts +2 -0
  202. package/dist/storage/utils.d.ts.map +1 -0
  203. package/dist/storage/utils.js +8 -0
  204. package/dist/storage/utils.js.map +1 -0
  205. package/dist/task/executor.d.ts +3 -0
  206. package/dist/task/executor.d.ts.map +1 -0
  207. package/dist/task/executor.js +80 -0
  208. package/dist/task/executor.js.map +1 -0
  209. package/dist/task/index.d.ts +5 -0
  210. package/dist/task/index.d.ts.map +1 -0
  211. package/dist/task/index.js +4 -0
  212. package/dist/task/index.js.map +1 -0
  213. package/dist/task/scheduler.d.ts +5 -0
  214. package/dist/task/scheduler.d.ts.map +1 -0
  215. package/dist/task/scheduler.js +47 -0
  216. package/dist/task/scheduler.js.map +1 -0
  217. package/dist/task/store.d.ts +14 -0
  218. package/dist/task/store.d.ts.map +1 -0
  219. package/dist/task/store.js +156 -0
  220. package/dist/task/store.js.map +1 -0
  221. package/dist/task/types.d.ts +13 -0
  222. package/dist/task/types.d.ts.map +1 -0
  223. package/dist/task/types.js +2 -0
  224. package/dist/task/types.js.map +1 -0
  225. package/dist/tool/env.d.ts +20 -0
  226. package/dist/tool/env.d.ts.map +1 -0
  227. package/dist/tool/env.js +26 -0
  228. package/dist/tool/env.js.map +1 -0
  229. package/dist/tool/env.test.d.ts +2 -0
  230. package/dist/tool/env.test.d.ts.map +1 -0
  231. package/dist/tool/env.test.js +42 -0
  232. package/dist/tool/env.test.js.map +1 -0
  233. package/dist/tool/index.d.ts +42 -0
  234. package/dist/tool/index.d.ts.map +1 -0
  235. package/dist/tool/index.js +53 -0
  236. package/dist/tool/index.js.map +1 -0
  237. package/dist/tool/registry.d.ts +43 -0
  238. package/dist/tool/registry.d.ts.map +1 -0
  239. package/dist/tool/registry.js +79 -0
  240. package/dist/tool/registry.js.map +1 -0
  241. package/dist/tool/safeguard.d.ts +21 -0
  242. package/dist/tool/safeguard.d.ts.map +1 -0
  243. package/dist/tool/safeguard.js +212 -0
  244. package/dist/tool/safeguard.js.map +1 -0
  245. package/dist/tool/tools/delete-file.d.ts +10 -0
  246. package/dist/tool/tools/delete-file.d.ts.map +1 -0
  247. package/dist/tool/tools/delete-file.js +52 -0
  248. package/dist/tool/tools/delete-file.js.map +1 -0
  249. package/dist/tool/tools/edit.d.ts +10 -0
  250. package/dist/tool/tools/edit.d.ts.map +1 -0
  251. package/dist/tool/tools/edit.js +159 -0
  252. package/dist/tool/tools/edit.js.map +1 -0
  253. package/dist/tool/tools/exec.d.ts +12 -0
  254. package/dist/tool/tools/exec.d.ts.map +1 -0
  255. package/dist/tool/tools/exec.js +170 -0
  256. package/dist/tool/tools/exec.js.map +1 -0
  257. package/dist/tool/tools/file-utils.d.ts +13 -0
  258. package/dist/tool/tools/file-utils.d.ts.map +1 -0
  259. package/dist/tool/tools/file-utils.js +65 -0
  260. package/dist/tool/tools/file-utils.js.map +1 -0
  261. package/dist/tool/tools/read-file.d.ts +10 -0
  262. package/dist/tool/tools/read-file.d.ts.map +1 -0
  263. package/dist/tool/tools/read-file.js +116 -0
  264. package/dist/tool/tools/read-file.js.map +1 -0
  265. package/dist/tool/tools/read-memory.d.ts +3 -0
  266. package/dist/tool/tools/read-memory.d.ts.map +1 -0
  267. package/dist/tool/tools/read-memory.js +58 -0
  268. package/dist/tool/tools/read-memory.js.map +1 -0
  269. package/dist/tool/tools/read-tasks.d.ts +3 -0
  270. package/dist/tool/tools/read-tasks.d.ts.map +1 -0
  271. package/dist/tool/tools/read-tasks.js +29 -0
  272. package/dist/tool/tools/read-tasks.js.map +1 -0
  273. package/dist/tool/tools/schedule-task.d.ts +3 -0
  274. package/dist/tool/tools/schedule-task.d.ts.map +1 -0
  275. package/dist/tool/tools/schedule-task.js +50 -0
  276. package/dist/tool/tools/schedule-task.js.map +1 -0
  277. package/dist/tool/tools/search-history.d.ts +4 -0
  278. package/dist/tool/tools/search-history.d.ts.map +1 -0
  279. package/dist/tool/tools/search-history.js +58 -0
  280. package/dist/tool/tools/search-history.js.map +1 -0
  281. package/dist/tool/tools/web-search.d.ts +3 -0
  282. package/dist/tool/tools/web-search.d.ts.map +1 -0
  283. package/dist/tool/tools/web-search.js +89 -0
  284. package/dist/tool/tools/web-search.js.map +1 -0
  285. package/dist/tool/tools/write-file.d.ts +11 -0
  286. package/dist/tool/tools/write-file.d.ts.map +1 -0
  287. package/dist/tool/tools/write-file.js +79 -0
  288. package/dist/tool/tools/write-file.js.map +1 -0
  289. package/dist/tool/tools/write-memory.d.ts +3 -0
  290. package/dist/tool/tools/write-memory.d.ts.map +1 -0
  291. package/dist/tool/tools/write-memory.js +99 -0
  292. package/dist/tool/tools/write-memory.js.map +1 -0
  293. package/dist/tool/types.d.ts +63 -0
  294. package/dist/tool/types.d.ts.map +1 -0
  295. package/dist/tool/types.js +33 -0
  296. package/dist/tool/types.js.map +1 -0
  297. package/package.json +53 -0
@@ -0,0 +1,55 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { BackgroundWorker } from './background-worker.js';
3
+ describe('BackgroundWorker', () => {
4
+ it('calls scheduled task', async () => {
5
+ const worker = new BackgroundWorker();
6
+ const fn = vi.fn().mockResolvedValue(undefined);
7
+ worker.schedule('test', 10, fn);
8
+ worker.start();
9
+ await vi.waitFor(() => expect(fn).toHaveBeenCalled(), { timeout: 200 });
10
+ worker.stop();
11
+ });
12
+ it('does not re-enter if previous task still running', async () => {
13
+ const worker = new BackgroundWorker();
14
+ let resolve;
15
+ const promise = new Promise((r) => { resolve = r; });
16
+ let callCount = 0;
17
+ const fn = vi.fn().mockImplementation(async () => {
18
+ callCount++;
19
+ await promise;
20
+ });
21
+ worker.schedule('test', 5, fn);
22
+ worker.start();
23
+ await vi.waitFor(() => expect(callCount).toBeGreaterThanOrEqual(1), { timeout: 100 });
24
+ // Task should still be running, no re-entry
25
+ expect(callCount).toBe(1);
26
+ resolve();
27
+ await new Promise((r) => setTimeout(r, 20));
28
+ worker.stop();
29
+ });
30
+ it('isolates errors between tasks', async () => {
31
+ const worker = new BackgroundWorker();
32
+ const failFn = vi.fn().mockRejectedValue(new Error('boom'));
33
+ const okFn = vi.fn().mockResolvedValue(undefined);
34
+ worker.schedule('fail', 10, failFn);
35
+ worker.schedule('ok', 10, okFn);
36
+ worker.start();
37
+ await vi.waitFor(() => {
38
+ expect(failFn).toHaveBeenCalled();
39
+ expect(okFn).toHaveBeenCalled();
40
+ }, { timeout: 200 });
41
+ worker.stop();
42
+ });
43
+ it('stop prevents further execution', async () => {
44
+ const worker = new BackgroundWorker();
45
+ const fn = vi.fn().mockResolvedValue(undefined);
46
+ worker.schedule('test', 10, fn);
47
+ worker.start();
48
+ await vi.waitFor(() => expect(fn).toHaveBeenCalled(), { timeout: 200 });
49
+ const countAfterFirst = fn.mock.calls.length;
50
+ worker.stop();
51
+ await new Promise((r) => setTimeout(r, 50));
52
+ expect(fn).toHaveBeenCalledTimes(countAfterFirst);
53
+ });
54
+ });
55
+ //# sourceMappingURL=background-worker.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"background-worker.test.js","sourceRoot":"","sources":["../../src/concurrency/background-worker.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export declare class FileMutex {
2
+ private locks;
3
+ run<T>(key: string, fn: () => Promise<T>): Promise<T>;
4
+ private acquire;
5
+ private release;
6
+ }
7
+ export declare const fileMutex: FileMutex;
8
+ //# sourceMappingURL=file-mutex.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mutex.d.ts","sourceRoot":"","sources":["../../src/concurrency/file-mutex.ts"],"names":[],"mappings":"AAEA,qBAAa,SAAS;IACrB,OAAO,CAAC,KAAK,CAA6D;IAEpE,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAS3D,OAAO,CAAC,OAAO;IAiBf,OAAO,CAAC,OAAO;CAYf;AAED,eAAO,MAAM,SAAS,WAAkB,CAAC"}
@@ -0,0 +1,41 @@
1
+ export class FileMutex {
2
+ locks = new Map();
3
+ async run(key, fn) {
4
+ await this.acquire(key);
5
+ try {
6
+ return await fn();
7
+ }
8
+ finally {
9
+ this.release(key);
10
+ }
11
+ }
12
+ acquire(key) {
13
+ let entry = this.locks.get(key);
14
+ if (!entry) {
15
+ entry = { locked: false, waiters: [] };
16
+ this.locks.set(key, entry);
17
+ }
18
+ if (!entry.locked) {
19
+ entry.locked = true;
20
+ return Promise.resolve();
21
+ }
22
+ return new Promise((resolve, reject) => {
23
+ entry.waiters.push({ resolve, reject });
24
+ });
25
+ }
26
+ release(key) {
27
+ const entry = this.locks.get(key);
28
+ if (!entry)
29
+ return;
30
+ const next = entry.waiters.shift();
31
+ if (next) {
32
+ next.resolve();
33
+ }
34
+ else {
35
+ entry.locked = false;
36
+ this.locks.delete(key);
37
+ }
38
+ }
39
+ }
40
+ export const fileMutex = new FileMutex();
41
+ //# sourceMappingURL=file-mutex.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mutex.js","sourceRoot":"","sources":["../../src/concurrency/file-mutex.ts"],"names":[],"mappings":"AAEA,MAAM,OAAO,SAAS;IACb,KAAK,GAAG,IAAI,GAAG,EAAkD,CAAC;IAE1E,KAAK,CAAC,GAAG,CAAI,GAAW,EAAE,EAAoB;QAC7C,MAAM,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QACxB,IAAI,CAAC;YACJ,OAAO,MAAM,EAAE,EAAE,CAAC;QACnB,CAAC;gBAAS,CAAC;YACV,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QACnB,CAAC;IACF,CAAC;IAEO,OAAO,CAAC,GAAW;QAC1B,IAAI,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAChC,IAAI,CAAC,KAAK,EAAE,CAAC;YACZ,KAAK,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;YACvC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QAC5B,CAAC;QAED,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACnB,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC;YACpB,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;QAC1B,CAAC;QAED,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACtC,KAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC;QAC1C,CAAC,CAAC,CAAC;IACJ,CAAC;IAEO,OAAO,CAAC,GAAW;QAC1B,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAClC,IAAI,CAAC,KAAK;YAAE,OAAO;QAEnB,MAAM,IAAI,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QACnC,IAAI,IAAI,EAAE,CAAC;YACV,IAAI,CAAC,OAAO,EAAE,CAAC;QAChB,CAAC;aAAM,CAAC;YACP,KAAK,CAAC,MAAM,GAAG,KAAK,CAAC;YACrB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACxB,CAAC;IACF,CAAC;CACD;AAED,MAAM,CAAC,MAAM,SAAS,GAAG,IAAI,SAAS,EAAE,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=file-mutex.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mutex.test.d.ts","sourceRoot":"","sources":["../../src/concurrency/file-mutex.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,50 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { FileMutex } from './file-mutex.js';
3
+ describe('FileMutex', () => {
4
+ it('executes a single task', async () => {
5
+ const mutex = new FileMutex();
6
+ const result = await mutex.run('key1', async () => 42);
7
+ expect(result).toBe(42);
8
+ });
9
+ it('serializes concurrent tasks on same key', async () => {
10
+ const mutex = new FileMutex();
11
+ const order = [];
12
+ await Promise.all([
13
+ mutex.run('k', async () => { order.push(1); return 1; }),
14
+ mutex.run('k', async () => { order.push(2); return 2; }),
15
+ mutex.run('k', async () => { order.push(3); return 3; }),
16
+ ]);
17
+ expect(order).toEqual([1, 2, 3]);
18
+ });
19
+ it('runs different keys concurrently', async () => {
20
+ const mutex = new FileMutex();
21
+ const order = [];
22
+ await Promise.all([
23
+ mutex.run('a', async () => { order.push(1); return 1; }),
24
+ mutex.run('b', async () => { order.push(2); return 2; }),
25
+ ]);
26
+ expect(order).toContain(1);
27
+ expect(order).toContain(2);
28
+ });
29
+ it('releases lock on error and allows next task', async () => {
30
+ const mutex = new FileMutex();
31
+ await expect(mutex.run('k', async () => { throw new Error('fail'); })).rejects.toThrow('fail');
32
+ const result = await mutex.run('k', async () => 42);
33
+ expect(result).toBe(42);
34
+ });
35
+ it('supports N concurrent waiters', async () => {
36
+ const mutex = new FileMutex();
37
+ let running = 0;
38
+ let maxRunning = 0;
39
+ const tasks = Array.from({ length: 5 }, (_, i) => mutex.run('k', async () => {
40
+ running++;
41
+ maxRunning = Math.max(maxRunning, running);
42
+ await new Promise((r) => setTimeout(r, 5));
43
+ running--;
44
+ return i;
45
+ }));
46
+ await Promise.all(tasks);
47
+ expect(maxRunning).toBe(1);
48
+ });
49
+ });
50
+ //# sourceMappingURL=file-mutex.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"file-mutex.test.js","sourceRoot":"","sources":["../../src/concurrency/file-mutex.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export { FileMutex, fileMutex } from './file-mutex.js';
2
+ export { BackgroundWorker } from './background-worker.js';
3
+ export type { WorkerTask } from './background-worker.js';
4
+ export { SessionLock, sessionLock } from './session-lock.js';
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/concurrency/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AACvD,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC1D,YAAY,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC"}
@@ -0,0 +1,4 @@
1
+ export { FileMutex, fileMutex } from './file-mutex.js';
2
+ export { BackgroundWorker } from './background-worker.js';
3
+ export { SessionLock, sessionLock } from './session-lock.js';
4
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/concurrency/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AACvD,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAE1D,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC"}
@@ -0,0 +1,8 @@
1
+ export declare class SessionLock {
2
+ private sessions;
3
+ private createRelease;
4
+ acquire(sessionId: string): Promise<() => void>;
5
+ private release;
6
+ }
7
+ export declare const sessionLock: SessionLock;
8
+ //# sourceMappingURL=session-lock.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-lock.d.ts","sourceRoot":"","sources":["../../src/concurrency/session-lock.ts"],"names":[],"mappings":"AAKA,qBAAa,WAAW;IACvB,OAAO,CAAC,QAAQ,CAAoE;IAEpF,OAAO,CAAC,aAAa;IASf,OAAO,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,CAAC;IAiBrD,OAAO,CAAC,OAAO;CAYf;AAED,eAAO,MAAM,WAAW,aAAoB,CAAC"}
@@ -0,0 +1,41 @@
1
+ export class SessionLock {
2
+ sessions = new Map();
3
+ createRelease(sessionId) {
4
+ let released = false;
5
+ return () => {
6
+ if (released)
7
+ throw new Error(`SessionLock: session "${sessionId}" already released`);
8
+ released = true;
9
+ this.release(sessionId);
10
+ };
11
+ }
12
+ async acquire(sessionId) {
13
+ let entry = this.sessions.get(sessionId);
14
+ if (!entry) {
15
+ entry = { locked: false, waiters: [] };
16
+ this.sessions.set(sessionId, entry);
17
+ }
18
+ if (!entry.locked) {
19
+ entry.locked = true;
20
+ return this.createRelease(sessionId);
21
+ }
22
+ return new Promise((resolve, reject) => {
23
+ entry.waiters.push({ resolve, reject });
24
+ });
25
+ }
26
+ release(sessionId) {
27
+ const entry = this.sessions.get(sessionId);
28
+ if (!entry)
29
+ return;
30
+ const next = entry.waiters.shift();
31
+ if (next) {
32
+ next.resolve(this.createRelease(sessionId));
33
+ }
34
+ else {
35
+ entry.locked = false;
36
+ this.sessions.delete(sessionId);
37
+ }
38
+ }
39
+ }
40
+ export const sessionLock = new SessionLock();
41
+ //# sourceMappingURL=session-lock.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-lock.js","sourceRoot":"","sources":["../../src/concurrency/session-lock.ts"],"names":[],"mappings":"AAKA,MAAM,OAAO,WAAW;IACf,QAAQ,GAAG,IAAI,GAAG,EAAyD,CAAC;IAE5E,aAAa,CAAC,SAAiB;QACtC,IAAI,QAAQ,GAAG,KAAK,CAAC;QACrB,OAAO,GAAG,EAAE;YACX,IAAI,QAAQ;gBAAE,MAAM,IAAI,KAAK,CAAC,yBAAyB,SAAS,oBAAoB,CAAC,CAAC;YACtF,QAAQ,GAAG,IAAI,CAAC;YAChB,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QACzB,CAAC,CAAC;IACH,CAAC;IAED,KAAK,CAAC,OAAO,CAAC,SAAiB;QAC9B,IAAI,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACzC,IAAI,CAAC,KAAK,EAAE,CAAC;YACZ,KAAK,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;YACvC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;QACrC,CAAC;QAED,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACnB,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC;YACpB,OAAO,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;QACtC,CAAC;QAED,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACtC,KAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC;QAC1C,CAAC,CAAC,CAAC;IACJ,CAAC;IAEO,OAAO,CAAC,SAAiB;QAChC,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC3C,IAAI,CAAC,KAAK;YAAE,OAAO;QAEnB,MAAM,IAAI,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QACnC,IAAI,IAAI,EAAE,CAAC;YACV,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC,CAAC;QAC7C,CAAC;aAAM,CAAC;YACP,KAAK,CAAC,MAAM,GAAG,KAAK,CAAC;YACrB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QACjC,CAAC;IACF,CAAC;CACD;AAED,MAAM,CAAC,MAAM,WAAW,GAAG,IAAI,WAAW,EAAE,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=session-lock.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-lock.test.d.ts","sourceRoot":"","sources":["../../src/concurrency/session-lock.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,51 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { SessionLock } from './session-lock.js';
3
+ describe('SessionLock', () => {
4
+ it('acquires and releases a session', async () => {
5
+ const lock = new SessionLock();
6
+ const release = await lock.acquire('s1');
7
+ expect(typeof release).toBe('function');
8
+ release();
9
+ });
10
+ it('serializes concurrent acquires on same session', async () => {
11
+ const lock = new SessionLock();
12
+ const order = [];
13
+ await Promise.all([
14
+ (async () => {
15
+ const release = await lock.acquire('s1');
16
+ order.push(1);
17
+ release();
18
+ })(),
19
+ (async () => {
20
+ const release = await lock.acquire('s1');
21
+ order.push(2);
22
+ release();
23
+ })(),
24
+ ]);
25
+ expect(order).toEqual([1, 2]);
26
+ });
27
+ it('throws on double release', async () => {
28
+ const lock = new SessionLock();
29
+ const release = await lock.acquire('s1');
30
+ release();
31
+ expect(() => release()).toThrow('already released');
32
+ });
33
+ it('allows concurrent acquires on different sessions', async () => {
34
+ const lock = new SessionLock();
35
+ const results = [];
36
+ await Promise.all([
37
+ (async () => {
38
+ const r = await lock.acquire('a');
39
+ results.push('a');
40
+ r();
41
+ })(),
42
+ (async () => {
43
+ const r = await lock.acquire('b');
44
+ results.push('b');
45
+ r();
46
+ })(),
47
+ ]);
48
+ expect(results).toHaveLength(2);
49
+ });
50
+ });
51
+ //# sourceMappingURL=session-lock.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-lock.test.js","sourceRoot":"","sources":["../../src/concurrency/session-lock.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,QAAQ,CAAC,aAAa,EAAE,GAAG,EAAE;IAC5B,EAAE,CAAC,iCAAiC,EAAE,KAAK,IAAI,EAAE;QAChD,MAAM,IAAI,GAAG,IAAI,WAAW,EAAE,CAAC;QAC/B,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACzC,MAAM,CAAC,OAAO,OAAO,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACxC,OAAO,EAAE,CAAC;IACX,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,KAAK,IAAI,EAAE;QAC/D,MAAM,IAAI,GAAG,IAAI,WAAW,EAAE,CAAC;QAC/B,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,MAAM,OAAO,CAAC,GAAG,CAAC;YACjB,CAAC,KAAK,IAAI,EAAE;gBACX,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;gBACzC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;gBACd,OAAO,EAAE,CAAC;YACX,CAAC,CAAC,EAAE;YACJ,CAAC,KAAK,IAAI,EAAE;gBACX,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;gBACzC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;gBACd,OAAO,EAAE,CAAC;YACX,CAAC,CAAC,EAAE;SACJ,CAAC,CAAC;QAEH,MAAM,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;IAC/B,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0BAA0B,EAAE,KAAK,IAAI,EAAE;QACzC,MAAM,IAAI,GAAG,IAAI,WAAW,EAAE,CAAC;QAC/B,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACzC,OAAO,EAAE,CAAC;QACV,MAAM,CAAC,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC;IACrD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kDAAkD,EAAE,KAAK,IAAI,EAAE;QACjE,MAAM,IAAI,GAAG,IAAI,WAAW,EAAE,CAAC;QAC/B,MAAM,OAAO,GAAa,EAAE,CAAC;QAE7B,MAAM,OAAO,CAAC,GAAG,CAAC;YACjB,CAAC,KAAK,IAAI,EAAE;gBACX,MAAM,CAAC,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBAClC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAClB,CAAC,EAAE,CAAC;YACL,CAAC,CAAC,EAAE;YACJ,CAAC,KAAK,IAAI,EAAE;gBACX,MAAM,CAAC,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBAClC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAClB,CAAC,EAAE,CAAC;YACL,CAAC,CAAC,EAAE;SACJ,CAAC,CAAC;QAEH,MAAM,CAAC,OAAO,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;AACJ,CAAC,CAAC,CAAC"}
@@ -0,0 +1,12 @@
1
+ /**
2
+ * Config 模块
3
+ *
4
+ * 统一管理应用配置。配置来源为 data/config.json(JSON 文件),
5
+ * 通过 readConfig / writeConfig 读写,设置页面调用 API 路由间接操作。
6
+ *
7
+ * @module config
8
+ */
9
+ export type { AppConfig, ProviderConfig } from './types.js';
10
+ export type { ModelDef } from '../ai/models.js';
11
+ export { readConfig, writeConfig, addProvider, removeProvider, setActiveModel, setReasoningEffort, getProviderForModel, getAllAvailableModels, getFirstProvider, getActiveModel, getContextWindow } from './store.js';
12
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/config/index.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,YAAY,EAAE,SAAS,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC5D,YAAY,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAChD,OAAO,EACN,UAAU,EACV,WAAW,EACX,WAAW,EACX,cAAc,EACd,cAAc,EACd,kBAAkB,EAClB,mBAAmB,EACnB,qBAAqB,EACrB,gBAAgB,EAChB,cAAc,EACd,gBAAgB,EAChB,MAAM,YAAY,CAAC"}
@@ -0,0 +1,10 @@
1
+ /**
2
+ * Config 模块
3
+ *
4
+ * 统一管理应用配置。配置来源为 data/config.json(JSON 文件),
5
+ * 通过 readConfig / writeConfig 读写,设置页面调用 API 路由间接操作。
6
+ *
7
+ * @module config
8
+ */
9
+ export { readConfig, writeConfig, addProvider, removeProvider, setActiveModel, setReasoningEffort, getProviderForModel, getAllAvailableModels, getFirstProvider, getActiveModel, getContextWindow } from './store.js';
10
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/config/index.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAIH,OAAO,EACN,UAAU,EACV,WAAW,EACX,WAAW,EACX,cAAc,EACd,cAAc,EACd,kBAAkB,EAClB,mBAAmB,EACnB,qBAAqB,EACrB,gBAAgB,EAChB,cAAc,EACd,gBAAgB,EAChB,MAAM,YAAY,CAAC"}
@@ -0,0 +1,17 @@
1
+ import type { AppConfig, ProviderConfig } from './types.js';
2
+ import type { ModelDef } from '../ai/models.js';
3
+ export declare function readConfig(): AppConfig;
4
+ export declare function writeConfig(config: AppConfig): void;
5
+ export declare function addProvider(provider: ProviderConfig): AppConfig;
6
+ export declare function removeProvider(name: string): AppConfig;
7
+ export declare function setActiveModel(modelId: string): AppConfig;
8
+ export declare function setReasoningEffort(value: string | null): AppConfig;
9
+ export declare function getProviderForModel(modelId: string): ProviderConfig | undefined;
10
+ export declare function getAllAvailableModels(): {
11
+ model: ModelDef;
12
+ providerName: string;
13
+ }[];
14
+ export declare function getFirstProvider(): ProviderConfig | undefined;
15
+ export declare function getActiveModel(): ModelDef | undefined;
16
+ export declare function getContextWindow(): number;
17
+ //# sourceMappingURL=store.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"store.d.ts","sourceRoot":"","sources":["../../src/config/store.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,SAAS,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC5D,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAuDhD,wBAAgB,UAAU,IAAI,SAAS,CA4CtC;AAED,wBAAgB,WAAW,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAOnD;AAED,wBAAgB,WAAW,CAAC,QAAQ,EAAE,cAAc,GAAG,SAAS,CAiB/D;AAED,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,SAAS,CAYtD;AAED,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,SAAS,CAkBzD;AAED,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,GAAG,SAAS,CAclE;AAED,wBAAgB,mBAAmB,CAAC,OAAO,EAAE,MAAM,GAAG,cAAc,GAAG,SAAS,CAW/E;AAED,wBAAgB,qBAAqB,IAAI;IAAE,KAAK,EAAE,QAAQ,CAAC;IAAC,YAAY,EAAE,MAAM,CAAA;CAAE,EAAE,CASnF;AAED,wBAAgB,gBAAgB,IAAI,cAAc,GAAG,SAAS,CAE7D;AAED,wBAAgB,cAAc,IAAI,QAAQ,GAAG,SAAS,CAmBrD;AAED,wBAAgB,gBAAgB,IAAI,MAAM,CAEzC"}
@@ -0,0 +1,214 @@
1
+ import { readFileSync, writeFileSync, existsSync, mkdirSync, renameSync } from 'node:fs';
2
+ import { getDefaultModels } from '../ai/models.js';
3
+ import { DEFAULT_SYSTEM_PROMPT } from '../agent/index.js';
4
+ import { getConfigPath, getDataDir } from '../paths.js';
5
+ const DEFAULT_CONTEXT_WINDOW = 1_048_576;
6
+ function ensureDataDir() {
7
+ const dir = getDataDir();
8
+ if (!existsSync(dir))
9
+ mkdirSync(dir, { recursive: true });
10
+ }
11
+ const defaultConfig = {
12
+ providers: [],
13
+ activeModel: '',
14
+ storageEnabled: false,
15
+ systemPrompt: DEFAULT_SYSTEM_PROMPT,
16
+ customBrandIcon: false,
17
+ autoSummarize: true,
18
+ summaryMode: 'idle',
19
+ summaryIdleHours: 8,
20
+ summaryScheduleHour: 2,
21
+ summaryIntervalMin: 30,
22
+ compressionMode: 'auto',
23
+ compressionThreshold: 256_000,
24
+ searchEnabled: false,
25
+ searchProvider: 'searxng',
26
+ searxngURL: 'http://localhost:8080',
27
+ tavilyApiKey: '',
28
+ emailNotifications: false,
29
+ emailSmtpHost: '',
30
+ emailSmtpPort: 465,
31
+ emailSmtpSecure: true,
32
+ emailSmtpUser: '',
33
+ emailSmtpPass: '',
34
+ emailFrom: '',
35
+ emailTo: '',
36
+ telegramEnabled: false,
37
+ telegramBotToken: '',
38
+ telegramAllowedUsers: ''
39
+ };
40
+ function normalizeProvider(p) {
41
+ return {
42
+ type: p.type || 'openai',
43
+ name: p.name || '',
44
+ apiKey: p.apiKey || '',
45
+ baseURL: p.baseURL || '',
46
+ reasoningEffort: p.reasoningEffort,
47
+ timeout: p.timeout,
48
+ maxRetries: p.maxRetries,
49
+ ollamaURL: p.ollamaURL || 'http://localhost:11434'
50
+ };
51
+ }
52
+ export function readConfig() {
53
+ const path = getConfigPath();
54
+ if (!existsSync(path)) {
55
+ return structuredClone(defaultConfig);
56
+ }
57
+ try {
58
+ const raw = readFileSync(path, 'utf-8');
59
+ const parsed = JSON.parse(raw);
60
+ return {
61
+ providers: Array.isArray(parsed.providers)
62
+ ? parsed.providers.map((p) => normalizeProvider(p))
63
+ : [],
64
+ activeModel: parsed.activeModel || '',
65
+ storageEnabled: parsed.storageEnabled === true,
66
+ systemPrompt: parsed.systemPrompt || DEFAULT_SYSTEM_PROMPT,
67
+ customBrandIcon: parsed.customBrandIcon === true,
68
+ autoSummarize: parsed.autoSummarize !== false,
69
+ summaryMode: parsed.summaryMode === 'scheduled' ? 'scheduled' : 'idle',
70
+ summaryIdleHours: typeof parsed.summaryIdleHours === 'number' ? parsed.summaryIdleHours : 8,
71
+ summaryScheduleHour: typeof parsed.summaryScheduleHour === 'number' ? parsed.summaryScheduleHour : 2,
72
+ summaryIntervalMin: typeof parsed.summaryIntervalMin === 'number' ? parsed.summaryIntervalMin : 30,
73
+ compressionMode: parsed.compressionMode === 'custom' ? 'custom' : 'auto',
74
+ compressionThreshold: typeof parsed.compressionThreshold === 'number' ? parsed.compressionThreshold : 256_000,
75
+ searchEnabled: parsed.searchEnabled === true,
76
+ searchProvider: (parsed.searchProvider === 'searxng' || parsed.searchProvider === 'tavily') ? parsed.searchProvider : 'searxng',
77
+ searxngURL: typeof parsed.searxngURL === 'string' ? parsed.searxngURL : 'http://localhost:8080',
78
+ tavilyApiKey: typeof parsed.tavilyApiKey === 'string' ? parsed.tavilyApiKey : '',
79
+ emailNotifications: parsed.emailNotifications === true,
80
+ emailSmtpHost: typeof parsed.emailSmtpHost === 'string' ? parsed.emailSmtpHost : '',
81
+ emailSmtpPort: typeof parsed.emailSmtpPort === 'number' ? parsed.emailSmtpPort : 465,
82
+ emailSmtpSecure: parsed.emailSmtpSecure !== false,
83
+ emailSmtpUser: typeof parsed.emailSmtpUser === 'string' ? parsed.emailSmtpUser : '',
84
+ emailSmtpPass: typeof parsed.emailSmtpPass === 'string' ? parsed.emailSmtpPass : '',
85
+ emailFrom: typeof parsed.emailFrom === 'string' ? parsed.emailFrom : '',
86
+ emailTo: typeof parsed.emailTo === 'string' ? parsed.emailTo : '',
87
+ telegramEnabled: parsed.telegramEnabled === true,
88
+ telegramBotToken: typeof parsed.telegramBotToken === 'string' ? parsed.telegramBotToken : '',
89
+ telegramAllowedUsers: typeof parsed.telegramAllowedUsers === 'string' ? parsed.telegramAllowedUsers : ''
90
+ };
91
+ }
92
+ catch {
93
+ return structuredClone(defaultConfig);
94
+ }
95
+ }
96
+ export function writeConfig(config) {
97
+ const path = getConfigPath();
98
+ ensureDataDir();
99
+ const tmpPath = path + '.tmp';
100
+ writeFileSync(tmpPath, JSON.stringify(config, null, '\t'), 'utf-8');
101
+ renameSync(tmpPath, path);
102
+ }
103
+ export function addProvider(provider) {
104
+ const config = readConfig();
105
+ const existing = config.providers.findIndex((p) => p.name === provider.name);
106
+ if (existing !== -1) {
107
+ config.providers[existing] = provider;
108
+ }
109
+ else {
110
+ config.providers.push(provider);
111
+ }
112
+ if (!config.activeModel) {
113
+ const models = getDefaultModels(provider.type);
114
+ config.activeModel = models[0]?.id || '';
115
+ }
116
+ writeConfig(config);
117
+ return config;
118
+ }
119
+ export function removeProvider(name) {
120
+ const config = readConfig();
121
+ config.providers = config.providers.filter((p) => p.name !== name);
122
+ const provider = getProviderForModel(config.activeModel);
123
+ if (!provider) {
124
+ const first = config.providers[0];
125
+ config.activeModel = first ? (getDefaultModels(first.type)[0]?.id || '') : '';
126
+ }
127
+ writeConfig(config);
128
+ return config;
129
+ }
130
+ export function setActiveModel(modelId) {
131
+ const config = readConfig();
132
+ config.activeModel = modelId;
133
+ for (const provider of config.providers) {
134
+ const models = getDefaultModels(provider.type);
135
+ if (models.length > 0) {
136
+ const model = models.find((m) => m.id === modelId);
137
+ if (model && !model.supportsReasoning) {
138
+ provider.reasoningEffort = undefined;
139
+ }
140
+ }
141
+ else if (provider.type === 'ollama') {
142
+ provider.reasoningEffort = undefined;
143
+ }
144
+ }
145
+ writeConfig(config);
146
+ return config;
147
+ }
148
+ export function setReasoningEffort(value) {
149
+ const config = readConfig();
150
+ const provider = config.providers.find((p) => {
151
+ const models = getDefaultModels(p.type);
152
+ if (models.length > 0) {
153
+ return models.some((m) => m.id === config.activeModel);
154
+ }
155
+ return p.type === 'ollama' && !!config.activeModel;
156
+ });
157
+ if (provider) {
158
+ provider.reasoningEffort = value || undefined;
159
+ writeConfig(config);
160
+ }
161
+ return config;
162
+ }
163
+ export function getProviderForModel(modelId) {
164
+ const config = readConfig();
165
+ for (const provider of config.providers) {
166
+ const models = getDefaultModels(provider.type);
167
+ if (models.some((m) => m.id === modelId)) {
168
+ return provider;
169
+ }
170
+ }
171
+ const ollama = config.providers.find((p) => p.type === 'ollama');
172
+ if (ollama)
173
+ return ollama;
174
+ return undefined;
175
+ }
176
+ export function getAllAvailableModels() {
177
+ const config = readConfig();
178
+ const result = [];
179
+ for (const provider of config.providers) {
180
+ for (const model of getDefaultModels(provider.type)) {
181
+ result.push({ model, providerName: provider.name });
182
+ }
183
+ }
184
+ return result;
185
+ }
186
+ export function getFirstProvider() {
187
+ return readConfig().providers.find((p) => p.apiKey);
188
+ }
189
+ export function getActiveModel() {
190
+ const config = readConfig();
191
+ if (!config.activeModel)
192
+ return undefined;
193
+ for (const provider of config.providers) {
194
+ const models = getDefaultModels(provider.type);
195
+ const model = models.find((m) => m.id === config.activeModel);
196
+ if (model)
197
+ return model;
198
+ }
199
+ if (config.providers.some((p) => p.type === 'ollama')) {
200
+ return {
201
+ id: config.activeModel,
202
+ name: config.activeModel,
203
+ contextWindow: 128_000,
204
+ supportsReasoning: false,
205
+ reasoningEffortValues: [],
206
+ supportsVision: false
207
+ };
208
+ }
209
+ return undefined;
210
+ }
211
+ export function getContextWindow() {
212
+ return getActiveModel()?.contextWindow || DEFAULT_CONTEXT_WINDOW;
213
+ }
214
+ //# sourceMappingURL=store.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"store.js","sourceRoot":"","sources":["../../src/config/store.ts"],"names":[],"mappings":"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"}