@skillfm/local 2.6.3 → 2.7.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 (294) hide show
  1. package/README.md +50 -1
  2. package/dist/_signers/aliyun-v3.d.ts +21 -0
  3. package/dist/_signers/aliyun-v3.d.ts.map +1 -0
  4. package/dist/_signers/aliyun-v3.js +47 -0
  5. package/dist/_signers/aliyun-v3.js.map +1 -0
  6. package/dist/_signers/canonical.d.ts +29 -0
  7. package/dist/_signers/canonical.d.ts.map +1 -0
  8. package/dist/_signers/canonical.js +59 -0
  9. package/dist/_signers/canonical.js.map +1 -0
  10. package/dist/_signers/index.d.ts +9 -0
  11. package/dist/_signers/index.d.ts.map +1 -0
  12. package/dist/_signers/index.js +6 -0
  13. package/dist/_signers/index.js.map +1 -0
  14. package/dist/_signers/tencent-tc3.d.ts +21 -0
  15. package/dist/_signers/tencent-tc3.d.ts.map +1 -0
  16. package/dist/_signers/tencent-tc3.js +48 -0
  17. package/dist/_signers/tencent-tc3.js.map +1 -0
  18. package/dist/_signers/volc-v4.d.ts +18 -0
  19. package/dist/_signers/volc-v4.d.ts.map +1 -0
  20. package/dist/_signers/volc-v4.js +43 -0
  21. package/dist/_signers/volc-v4.js.map +1 -0
  22. package/dist/asset-connectors/aliyun-ecs.d.ts +18 -0
  23. package/dist/asset-connectors/aliyun-ecs.d.ts.map +1 -0
  24. package/dist/asset-connectors/aliyun-ecs.js +99 -0
  25. package/dist/asset-connectors/aliyun-ecs.js.map +1 -0
  26. package/dist/asset-connectors/base.d.ts +15 -0
  27. package/dist/asset-connectors/base.d.ts.map +1 -0
  28. package/dist/asset-connectors/base.js +19 -0
  29. package/dist/asset-connectors/base.js.map +1 -0
  30. package/dist/asset-connectors/cloudflare.d.ts +18 -0
  31. package/dist/asset-connectors/cloudflare.d.ts.map +1 -0
  32. package/dist/asset-connectors/cloudflare.js +94 -0
  33. package/dist/asset-connectors/cloudflare.js.map +1 -0
  34. package/dist/asset-connectors/digitalocean.d.ts +18 -0
  35. package/dist/asset-connectors/digitalocean.d.ts.map +1 -0
  36. package/dist/asset-connectors/digitalocean.js +79 -0
  37. package/dist/asset-connectors/digitalocean.js.map +1 -0
  38. package/dist/asset-connectors/godaddy.d.ts +18 -0
  39. package/dist/asset-connectors/godaddy.d.ts.map +1 -0
  40. package/dist/asset-connectors/godaddy.js +62 -0
  41. package/dist/asset-connectors/godaddy.js.map +1 -0
  42. package/dist/asset-connectors/hetzner.d.ts +18 -0
  43. package/dist/asset-connectors/hetzner.d.ts.map +1 -0
  44. package/dist/asset-connectors/hetzner.js +52 -0
  45. package/dist/asset-connectors/hetzner.js.map +1 -0
  46. package/dist/asset-connectors/index.d.ts +18 -0
  47. package/dist/asset-connectors/index.d.ts.map +1 -0
  48. package/dist/asset-connectors/index.js +52 -0
  49. package/dist/asset-connectors/index.js.map +1 -0
  50. package/dist/asset-connectors/namecheap.d.ts +34 -0
  51. package/dist/asset-connectors/namecheap.d.ts.map +1 -0
  52. package/dist/asset-connectors/namecheap.js +178 -0
  53. package/dist/asset-connectors/namecheap.js.map +1 -0
  54. package/dist/asset-connectors/ssl-labs.d.ts +22 -0
  55. package/dist/asset-connectors/ssl-labs.d.ts.map +1 -0
  56. package/dist/asset-connectors/ssl-labs.js +79 -0
  57. package/dist/asset-connectors/ssl-labs.js.map +1 -0
  58. package/dist/asset-connectors/tencent-cvm.d.ts +18 -0
  59. package/dist/asset-connectors/tencent-cvm.d.ts.map +1 -0
  60. package/dist/asset-connectors/tencent-cvm.js +145 -0
  61. package/dist/asset-connectors/tencent-cvm.js.map +1 -0
  62. package/dist/asset-connectors/types.d.ts +45 -0
  63. package/dist/asset-connectors/types.d.ts.map +1 -0
  64. package/dist/asset-connectors/types.js +26 -0
  65. package/dist/asset-connectors/types.js.map +1 -0
  66. package/dist/asset-connectors/vultr.d.ts +18 -0
  67. package/dist/asset-connectors/vultr.d.ts.map +1 -0
  68. package/dist/asset-connectors/vultr.js +73 -0
  69. package/dist/asset-connectors/vultr.js.map +1 -0
  70. package/dist/connectors/anthropic.d.ts +10 -0
  71. package/dist/connectors/anthropic.d.ts.map +1 -0
  72. package/dist/connectors/anthropic.js +103 -0
  73. package/dist/connectors/anthropic.js.map +1 -0
  74. package/dist/connectors/base.d.ts +11 -0
  75. package/dist/connectors/base.d.ts.map +1 -0
  76. package/dist/connectors/base.js +19 -0
  77. package/dist/connectors/base.js.map +1 -0
  78. package/dist/connectors/deepseek.d.ts +14 -0
  79. package/dist/connectors/deepseek.d.ts.map +1 -0
  80. package/dist/connectors/deepseek.js +77 -0
  81. package/dist/connectors/deepseek.js.map +1 -0
  82. package/dist/connectors/doubao.d.ts +10 -0
  83. package/dist/connectors/doubao.d.ts.map +1 -0
  84. package/dist/connectors/doubao.js +64 -0
  85. package/dist/connectors/doubao.js.map +1 -0
  86. package/dist/connectors/index.d.ts +14 -0
  87. package/dist/connectors/index.d.ts.map +1 -0
  88. package/dist/connectors/index.js +42 -0
  89. package/dist/connectors/index.js.map +1 -0
  90. package/dist/connectors/kimi.d.ts +10 -0
  91. package/dist/connectors/kimi.d.ts.map +1 -0
  92. package/dist/connectors/kimi.js +67 -0
  93. package/dist/connectors/kimi.js.map +1 -0
  94. package/dist/connectors/openai.d.ts +10 -0
  95. package/dist/connectors/openai.d.ts.map +1 -0
  96. package/dist/connectors/openai.js +101 -0
  97. package/dist/connectors/openai.js.map +1 -0
  98. package/dist/connectors/qwen.d.ts +10 -0
  99. package/dist/connectors/qwen.d.ts.map +1 -0
  100. package/dist/connectors/qwen.js +66 -0
  101. package/dist/connectors/qwen.js.map +1 -0
  102. package/dist/connectors/types.d.ts +47 -0
  103. package/dist/connectors/types.d.ts.map +1 -0
  104. package/dist/connectors/types.js +24 -0
  105. package/dist/connectors/types.js.map +1 -0
  106. package/dist/guard/bin.js +0 -0
  107. package/dist/index.js +50 -1
  108. package/dist/index.js.map +1 -1
  109. package/dist/mcp/_vendored-contracts/beacon-mcp-tools-v1.d.ts +373 -0
  110. package/dist/mcp/_vendored-contracts/beacon-mcp-tools-v1.d.ts.map +1 -0
  111. package/dist/mcp/_vendored-contracts/beacon-mcp-tools-v1.js +128 -0
  112. package/dist/mcp/_vendored-contracts/beacon-mcp-tools-v1.js.map +1 -0
  113. package/dist/mcp/brain-client.d.ts +96 -0
  114. package/dist/mcp/brain-client.d.ts.map +1 -0
  115. package/dist/mcp/brain-client.js +214 -0
  116. package/dist/mcp/brain-client.js.map +1 -0
  117. package/dist/mcp/cache.d.ts +55 -0
  118. package/dist/mcp/cache.d.ts.map +1 -0
  119. package/dist/mcp/cache.js +109 -0
  120. package/dist/mcp/cache.js.map +1 -0
  121. package/dist/mcp/config.d.ts +29 -0
  122. package/dist/mcp/config.d.ts.map +1 -0
  123. package/dist/mcp/config.js +69 -0
  124. package/dist/mcp/config.js.map +1 -0
  125. package/dist/mcp/decision-engine.d.ts +54 -0
  126. package/dist/mcp/decision-engine.d.ts.map +1 -0
  127. package/dist/mcp/decision-engine.js +107 -0
  128. package/dist/mcp/decision-engine.js.map +1 -0
  129. package/dist/mcp/errors.d.ts +66 -0
  130. package/dist/mcp/errors.d.ts.map +1 -0
  131. package/dist/mcp/errors.js +127 -0
  132. package/dist/mcp/errors.js.map +1 -0
  133. package/dist/mcp/index.d.ts +40 -0
  134. package/dist/mcp/index.d.ts.map +1 -0
  135. package/dist/mcp/index.js +98 -0
  136. package/dist/mcp/index.js.map +1 -0
  137. package/dist/mcp/tools/_placeholder.d.ts +24 -0
  138. package/dist/mcp/tools/_placeholder.d.ts.map +1 -0
  139. package/dist/mcp/tools/_placeholder.js +29 -0
  140. package/dist/mcp/tools/_placeholder.js.map +1 -0
  141. package/dist/mcp/tools/avatar-market.d.ts +22 -0
  142. package/dist/mcp/tools/avatar-market.d.ts.map +1 -0
  143. package/dist/mcp/tools/avatar-market.js +33 -0
  144. package/dist/mcp/tools/avatar-market.js.map +1 -0
  145. package/dist/mcp/tools/check-expiry.d.ts +23 -0
  146. package/dist/mcp/tools/check-expiry.d.ts.map +1 -0
  147. package/dist/mcp/tools/check-expiry.js +29 -0
  148. package/dist/mcp/tools/check-expiry.js.map +1 -0
  149. package/dist/mcp/tools/check-usage.d.ts +25 -0
  150. package/dist/mcp/tools/check-usage.d.ts.map +1 -0
  151. package/dist/mcp/tools/check-usage.js +33 -0
  152. package/dist/mcp/tools/check-usage.js.map +1 -0
  153. package/dist/mcp/tools/get-monthly-picks.d.ts +25 -0
  154. package/dist/mcp/tools/get-monthly-picks.d.ts.map +1 -0
  155. package/dist/mcp/tools/get-monthly-picks.js +35 -0
  156. package/dist/mcp/tools/get-monthly-picks.js.map +1 -0
  157. package/dist/mcp/tools/index.d.ts +41 -0
  158. package/dist/mcp/tools/index.d.ts.map +1 -0
  159. package/dist/mcp/tools/index.js +139 -0
  160. package/dist/mcp/tools/index.js.map +1 -0
  161. package/dist/mcp/tools/memory-recent.d.ts +17 -0
  162. package/dist/mcp/tools/memory-recent.d.ts.map +1 -0
  163. package/dist/mcp/tools/memory-recent.js +46 -0
  164. package/dist/mcp/tools/memory-recent.js.map +1 -0
  165. package/dist/mcp/tools/memory-save.d.ts +15 -0
  166. package/dist/mcp/tools/memory-save.d.ts.map +1 -0
  167. package/dist/mcp/tools/memory-save.js +53 -0
  168. package/dist/mcp/tools/memory-save.js.map +1 -0
  169. package/dist/mcp/tools/memory-search.d.ts +24 -0
  170. package/dist/mcp/tools/memory-search.d.ts.map +1 -0
  171. package/dist/mcp/tools/memory-search.js +64 -0
  172. package/dist/mcp/tools/memory-search.js.map +1 -0
  173. package/dist/mcp/tools/record-acceptance.d.ts +22 -0
  174. package/dist/mcp/tools/record-acceptance.d.ts.map +1 -0
  175. package/dist/mcp/tools/record-acceptance.js +33 -0
  176. package/dist/mcp/tools/record-acceptance.js.map +1 -0
  177. package/dist/mcp/tools/record-rejection.d.ts +23 -0
  178. package/dist/mcp/tools/record-rejection.d.ts.map +1 -0
  179. package/dist/mcp/tools/record-rejection.js +34 -0
  180. package/dist/mcp/tools/record-rejection.js.map +1 -0
  181. package/dist/mcp/tools/suggest-route.d.ts +26 -0
  182. package/dist/mcp/tools/suggest-route.d.ts.map +1 -0
  183. package/dist/mcp/tools/suggest-route.js +56 -0
  184. package/dist/mcp/tools/suggest-route.js.map +1 -0
  185. package/dist/mcp/tools/user-preferences.d.ts +22 -0
  186. package/dist/mcp/tools/user-preferences.d.ts.map +1 -0
  187. package/dist/mcp/tools/user-preferences.js +63 -0
  188. package/dist/mcp/tools/user-preferences.js.map +1 -0
  189. package/dist/mcp-stdio/bin.js +0 -0
  190. package/dist/memory/index.d.ts +5 -0
  191. package/dist/memory/index.d.ts.map +1 -0
  192. package/dist/memory/index.js +4 -0
  193. package/dist/memory/index.js.map +1 -0
  194. package/dist/memory/mem0-client.d.ts +32 -0
  195. package/dist/memory/mem0-client.d.ts.map +1 -0
  196. package/dist/memory/mem0-client.js +161 -0
  197. package/dist/memory/mem0-client.js.map +1 -0
  198. package/dist/memory/types.d.ts +47 -0
  199. package/dist/memory/types.d.ts.map +1 -0
  200. package/dist/memory/types.js +20 -0
  201. package/dist/memory/types.js.map +1 -0
  202. package/dist/router/catalog.d.ts +4 -0
  203. package/dist/router/catalog.d.ts.map +1 -0
  204. package/dist/router/catalog.js +103 -0
  205. package/dist/router/catalog.js.map +1 -0
  206. package/dist/router/classifier.d.ts +5 -0
  207. package/dist/router/classifier.d.ts.map +1 -0
  208. package/dist/router/classifier.js +43 -0
  209. package/dist/router/classifier.js.map +1 -0
  210. package/dist/router/index.d.ts +7 -0
  211. package/dist/router/index.d.ts.map +1 -0
  212. package/dist/router/index.js +6 -0
  213. package/dist/router/index.js.map +1 -0
  214. package/dist/router/litellm-config.d.ts +34 -0
  215. package/dist/router/litellm-config.d.ts.map +1 -0
  216. package/dist/router/litellm-config.js +78 -0
  217. package/dist/router/litellm-config.js.map +1 -0
  218. package/dist/router/route.d.ts +3 -0
  219. package/dist/router/route.d.ts.map +1 -0
  220. package/dist/router/route.js +63 -0
  221. package/dist/router/route.js.map +1 -0
  222. package/dist/router/types.d.ts +36 -0
  223. package/dist/router/types.d.ts.map +1 -0
  224. package/dist/router/types.js +7 -0
  225. package/dist/router/types.js.map +1 -0
  226. package/dist/saas-connectors/base.d.ts +16 -0
  227. package/dist/saas-connectors/base.d.ts.map +1 -0
  228. package/dist/saas-connectors/base.js +16 -0
  229. package/dist/saas-connectors/base.js.map +1 -0
  230. package/dist/saas-connectors/brave.d.ts +19 -0
  231. package/dist/saas-connectors/brave.d.ts.map +1 -0
  232. package/dist/saas-connectors/brave.js +50 -0
  233. package/dist/saas-connectors/brave.js.map +1 -0
  234. package/dist/saas-connectors/fal.d.ts +15 -0
  235. package/dist/saas-connectors/fal.d.ts.map +1 -0
  236. package/dist/saas-connectors/fal.js +35 -0
  237. package/dist/saas-connectors/fal.js.map +1 -0
  238. package/dist/saas-connectors/index.d.ts +11 -0
  239. package/dist/saas-connectors/index.d.ts.map +1 -0
  240. package/dist/saas-connectors/index.js +32 -0
  241. package/dist/saas-connectors/index.js.map +1 -0
  242. package/dist/saas-connectors/replicate.d.ts +19 -0
  243. package/dist/saas-connectors/replicate.d.ts.map +1 -0
  244. package/dist/saas-connectors/replicate.js +43 -0
  245. package/dist/saas-connectors/replicate.js.map +1 -0
  246. package/dist/saas-connectors/serper.d.ts +19 -0
  247. package/dist/saas-connectors/serper.d.ts.map +1 -0
  248. package/dist/saas-connectors/serper.js +47 -0
  249. package/dist/saas-connectors/serper.js.map +1 -0
  250. package/dist/saas-connectors/types.d.ts +31 -0
  251. package/dist/saas-connectors/types.d.ts.map +1 -0
  252. package/dist/saas-connectors/types.js +6 -0
  253. package/dist/saas-connectors/types.js.map +1 -0
  254. package/dist/scheduler/index.d.ts +4 -0
  255. package/dist/scheduler/index.d.ts.map +1 -0
  256. package/dist/scheduler/index.js +4 -0
  257. package/dist/scheduler/index.js.map +1 -0
  258. package/dist/scheduler/poll.d.ts +4 -0
  259. package/dist/scheduler/poll.d.ts.map +1 -0
  260. package/dist/scheduler/poll.js +170 -0
  261. package/dist/scheduler/poll.js.map +1 -0
  262. package/dist/scheduler/scheduler.d.ts +18 -0
  263. package/dist/scheduler/scheduler.d.ts.map +1 -0
  264. package/dist/scheduler/scheduler.js +67 -0
  265. package/dist/scheduler/scheduler.js.map +1 -0
  266. package/dist/scheduler/types.d.ts +49 -0
  267. package/dist/scheduler/types.d.ts.map +1 -0
  268. package/dist/scheduler/types.js +14 -0
  269. package/dist/scheduler/types.js.map +1 -0
  270. package/dist/skill-installer/npm-installer.d.ts +11 -1
  271. package/dist/skill-installer/npm-installer.d.ts.map +1 -1
  272. package/dist/skill-installer/npm-installer.js +13 -3
  273. package/dist/skill-installer/npm-installer.js.map +1 -1
  274. package/dist/vault/cli.d.ts +2 -0
  275. package/dist/vault/cli.d.ts.map +1 -0
  276. package/dist/vault/cli.js +170 -0
  277. package/dist/vault/cli.js.map +1 -0
  278. package/dist/vault/crypto.d.ts +9 -0
  279. package/dist/vault/crypto.d.ts.map +1 -0
  280. package/dist/vault/crypto.js +54 -0
  281. package/dist/vault/crypto.js.map +1 -0
  282. package/dist/vault/index.d.ts +6 -0
  283. package/dist/vault/index.d.ts.map +1 -0
  284. package/dist/vault/index.js +5 -0
  285. package/dist/vault/index.js.map +1 -0
  286. package/dist/vault/types.d.ts +28 -0
  287. package/dist/vault/types.d.ts.map +1 -0
  288. package/dist/vault/types.js +22 -0
  289. package/dist/vault/types.js.map +1 -0
  290. package/dist/vault/vault.d.ts +34 -0
  291. package/dist/vault/vault.d.ts.map +1 -0
  292. package/dist/vault/vault.js +149 -0
  293. package/dist/vault/vault.js.map +1 -0
  294. package/package.json +17 -5
@@ -0,0 +1,161 @@
1
+ // sdk/skillfm-local/src/memory/mem0-client.ts
2
+ //
3
+ // Mem0 self-host REST client (Day 5 B3)
4
+ // 默认 endpoint: http://localhost:11000 (用户跑 `docker run -p 11000:8000 mem0ai/mem0` 启)
5
+ // Doc: https://docs.mem0.ai/api-reference (自托管时 endpoint 跟 cloud 一致, 仅 baseUrl 不同)
6
+ //
7
+ // REST API:
8
+ // POST /v1/memories — add memory
9
+ // GET /v1/memories — list memories
10
+ // POST /v1/search — semantic search
11
+ // DELETE /v1/memories/<id>
12
+ import { MemoryError, MEMORY_ERROR_CODES, } from './types.js';
13
+ const DEFAULT_BASE = 'http://localhost:11000';
14
+ const DEFAULT_TIMEOUT = 10_000;
15
+ export class Mem0Client {
16
+ base_url;
17
+ default_user_id;
18
+ api_key;
19
+ timeout_ms;
20
+ constructor(cfg) {
21
+ this.base_url = (cfg?.base_url ?? DEFAULT_BASE).replace(/\/+$/, '');
22
+ this.default_user_id = cfg?.default_user_id;
23
+ this.api_key = cfg?.api_key;
24
+ this.timeout_ms = cfg?.timeout_ms ?? DEFAULT_TIMEOUT;
25
+ }
26
+ headers(extra) {
27
+ const h = {
28
+ 'content-type': 'application/json',
29
+ ...extra,
30
+ };
31
+ if (this.api_key)
32
+ h['authorization'] = `Token ${this.api_key}`;
33
+ return h;
34
+ }
35
+ async fetchWithTimeout(url, init) {
36
+ const ctrl = new AbortController();
37
+ const t = setTimeout(() => ctrl.abort(), this.timeout_ms);
38
+ try {
39
+ return await fetch(url, { ...init, signal: ctrl.signal });
40
+ }
41
+ catch (e) {
42
+ throw new MemoryError(MEMORY_ERROR_CODES.BACKEND_UNREACHABLE, `Mem0 backend ${this.base_url}: ${e instanceof Error ? e.message : String(e)}`);
43
+ }
44
+ finally {
45
+ clearTimeout(t);
46
+ }
47
+ }
48
+ resolveUserId(opts) {
49
+ const uid = opts?.user_id ?? this.default_user_id;
50
+ if (!uid) {
51
+ throw new MemoryError(MEMORY_ERROR_CODES.INVALID_INPUT, 'user_id required (set default_user_id in config or pass per-call)');
52
+ }
53
+ return uid;
54
+ }
55
+ toItem(raw) {
56
+ return {
57
+ id: raw.id,
58
+ text: raw.memory ?? raw.text ?? '',
59
+ metadata: (raw.metadata ?? {}),
60
+ created_at: raw.created_at ?? new Date().toISOString(),
61
+ updated_at: raw.updated_at ?? raw.created_at ?? new Date().toISOString(),
62
+ };
63
+ }
64
+ async put(text, metadata) {
65
+ if (!text || text.length === 0) {
66
+ throw new MemoryError(MEMORY_ERROR_CODES.INVALID_INPUT, 'text required');
67
+ }
68
+ const user_id = this.resolveUserId();
69
+ const body = JSON.stringify({
70
+ messages: [{ role: 'user', content: text }],
71
+ user_id,
72
+ metadata: metadata ?? {},
73
+ });
74
+ const r = await this.fetchWithTimeout(`${this.base_url}/v1/memories/`, {
75
+ method: 'POST',
76
+ headers: this.headers(),
77
+ body,
78
+ });
79
+ if (!r.ok) {
80
+ throw new MemoryError(r.status === 401 || r.status === 403 ? MEMORY_ERROR_CODES.AUTH_FAILED : MEMORY_ERROR_CODES.BACKEND_UNREACHABLE, `Mem0 put HTTP ${r.status}`);
81
+ }
82
+ const data = (await r.json());
83
+ const result = data.result ?? data.results?.[0];
84
+ if (!result?.id) {
85
+ throw new MemoryError(MEMORY_ERROR_CODES.BACKEND_UNREACHABLE, 'Mem0 put: no id in response');
86
+ }
87
+ return result.id;
88
+ }
89
+ async search(query, opts) {
90
+ if (!query || query.length === 0) {
91
+ throw new MemoryError(MEMORY_ERROR_CODES.INVALID_INPUT, 'query required');
92
+ }
93
+ const user_id = this.resolveUserId(opts);
94
+ const body = JSON.stringify({
95
+ query,
96
+ user_id,
97
+ limit: opts?.k ?? 5,
98
+ });
99
+ const r = await this.fetchWithTimeout(`${this.base_url}/v1/memories/search/`, {
100
+ method: 'POST',
101
+ headers: this.headers(),
102
+ body,
103
+ });
104
+ if (!r.ok) {
105
+ throw new MemoryError(MEMORY_ERROR_CODES.BACKEND_UNREACHABLE, `Mem0 search HTTP ${r.status}`);
106
+ }
107
+ const data = (await r.json());
108
+ const results = data.results ?? [];
109
+ return results
110
+ .map((raw) => ({
111
+ item: this.toItem(raw),
112
+ score: raw.score ?? 0,
113
+ }))
114
+ .filter((sr) => sr.score >= (opts?.min_score ?? 0));
115
+ }
116
+ async recent(limit, user_id) {
117
+ const uid = this.resolveUserId({ user_id });
118
+ const params = new URLSearchParams({ user_id: uid, limit: String(limit ?? 20) });
119
+ const r = await this.fetchWithTimeout(`${this.base_url}/v1/memories/?${params}`, {
120
+ method: 'GET',
121
+ headers: this.headers(),
122
+ });
123
+ if (!r.ok) {
124
+ throw new MemoryError(MEMORY_ERROR_CODES.BACKEND_UNREACHABLE, `Mem0 list HTTP ${r.status}`);
125
+ }
126
+ const data = (await r.json());
127
+ return (data.results ?? []).map((raw) => this.toItem(raw));
128
+ }
129
+ async delete(id) {
130
+ const r = await this.fetchWithTimeout(`${this.base_url}/v1/memories/${encodeURIComponent(id)}/`, {
131
+ method: 'DELETE',
132
+ headers: this.headers(),
133
+ });
134
+ if (r.status === 404)
135
+ return false;
136
+ if (!r.ok)
137
+ throw new MemoryError(MEMORY_ERROR_CODES.BACKEND_UNREACHABLE, `Mem0 delete HTTP ${r.status}`);
138
+ return true;
139
+ }
140
+ async health() {
141
+ try {
142
+ // Mem0 没有专门 /health, 用 list 0 条做轻量 ping
143
+ const r = await this.fetchWithTimeout(`${this.base_url}/v1/memories/?user_id=__health__&limit=1`, {
144
+ method: 'GET',
145
+ headers: this.headers(),
146
+ });
147
+ if (r.ok || r.status === 404) {
148
+ return { ok: true, backend: `mem0:${this.base_url}` };
149
+ }
150
+ return { ok: false, error: `HTTP ${r.status}`, backend: `mem0:${this.base_url}` };
151
+ }
152
+ catch (e) {
153
+ return {
154
+ ok: false,
155
+ error: e instanceof Error ? e.message : String(e),
156
+ backend: `mem0:${this.base_url}`,
157
+ };
158
+ }
159
+ }
160
+ }
161
+ //# sourceMappingURL=mem0-client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mem0-client.js","sourceRoot":"","sources":["../../src/memory/mem0-client.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,47 @@
1
+ export type MemoryId = string;
2
+ export interface MemoryMetadata {
3
+ source?: 'claude-code' | 'cursor' | 'openclaw' | 'manual';
4
+ session_id?: string;
5
+ user_tags?: string[];
6
+ privacy?: 'private' | 'shared';
7
+ [key: string]: unknown;
8
+ }
9
+ export interface MemoryItem {
10
+ id: MemoryId;
11
+ text: string;
12
+ metadata: MemoryMetadata;
13
+ created_at: string;
14
+ updated_at: string;
15
+ }
16
+ export interface MemorySearchResult {
17
+ item: MemoryItem;
18
+ score: number;
19
+ }
20
+ export interface SearchOptions {
21
+ k?: number;
22
+ min_score?: number;
23
+ privacy?: 'private' | 'shared' | 'all';
24
+ user_id?: string;
25
+ }
26
+ export interface MemoryStore {
27
+ put(text: string, metadata?: MemoryMetadata): Promise<MemoryId>;
28
+ search(query: string, opts?: SearchOptions): Promise<MemorySearchResult[]>;
29
+ recent(limit?: number, user_id?: string): Promise<MemoryItem[]>;
30
+ delete(id: MemoryId): Promise<boolean>;
31
+ health(): Promise<{
32
+ ok: boolean;
33
+ error?: string;
34
+ backend: string;
35
+ }>;
36
+ }
37
+ export declare class MemoryError extends Error {
38
+ code: string;
39
+ constructor(code: string, message: string);
40
+ }
41
+ export declare const MEMORY_ERROR_CODES: {
42
+ readonly BACKEND_UNREACHABLE: "MEMORY_BACKEND_UNREACHABLE";
43
+ readonly AUTH_FAILED: "MEMORY_AUTH_FAILED";
44
+ readonly NOT_FOUND: "MEMORY_NOT_FOUND";
45
+ readonly INVALID_INPUT: "MEMORY_INVALID_INPUT";
46
+ };
47
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/memory/types.ts"],"names":[],"mappings":"AAMA,MAAM,MAAM,QAAQ,GAAG,MAAM,CAAC;AAE9B,MAAM,WAAW,cAAc;IAC7B,MAAM,CAAC,EAAE,aAAa,GAAG,QAAQ,GAAG,UAAU,GAAG,QAAQ,CAAC;IAC1D,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,SAAS,CAAC,EAAE,MAAM,EAAE,CAAC;IACrB,OAAO,CAAC,EAAE,SAAS,GAAG,QAAQ,CAAC;IAC/B,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,UAAU;IACzB,EAAE,EAAE,QAAQ,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,cAAc,CAAC;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,kBAAkB;IACjC,IAAI,EAAE,UAAU,CAAC;IACjB,KAAK,EAAE,MAAM,CAAC;CACf;AAED,MAAM,WAAW,aAAa;IAC5B,CAAC,CAAC,EAAE,MAAM,CAAC;IACX,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,SAAS,GAAG,QAAQ,GAAG,KAAK,CAAC;IACvC,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,WAAW;IAC1B,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,cAAc,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;IAChE,MAAM,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC,CAAC;IAC3E,MAAM,CAAC,KAAK,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC,CAAC;IAChE,MAAM,CAAC,EAAE,EAAE,QAAQ,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;IACvC,MAAM,IAAI,OAAO,CAAC;QAAE,EAAE,EAAE,OAAO,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;CACrE;AAED,qBAAa,WAAY,SAAQ,KAAK;IACjB,IAAI,EAAE,MAAM;gBAAZ,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM;CAIjD;AAED,eAAO,MAAM,kBAAkB;;;;;CAKrB,CAAC"}
@@ -0,0 +1,20 @@
1
+ // sdk/skillfm-local/src/memory/types.ts
2
+ //
3
+ // Memory 模块类型 (Day 5 B3)
4
+ // 路径 A — Mem0 self-host 薄封装 (Apache 2.0 直接用, 0 法律风险)
5
+ // 替代 docs/prd/MEMORY-SPEC-FROM-CLAUDE-MEM.md 路径 B (clean room rewrite)
6
+ export class MemoryError extends Error {
7
+ code;
8
+ constructor(code, message) {
9
+ super(message);
10
+ this.code = code;
11
+ this.name = 'MemoryError';
12
+ }
13
+ }
14
+ export const MEMORY_ERROR_CODES = {
15
+ BACKEND_UNREACHABLE: 'MEMORY_BACKEND_UNREACHABLE',
16
+ AUTH_FAILED: 'MEMORY_AUTH_FAILED',
17
+ NOT_FOUND: 'MEMORY_NOT_FOUND',
18
+ INVALID_INPUT: 'MEMORY_INVALID_INPUT',
19
+ };
20
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/memory/types.ts"],"names":[],"mappings":"AAAA,wCAAwC;AACxC,EAAE;AACF,yBAAyB;AACzB,qDAAqD;AACrD,uEAAuE;AAwCvE,MAAM,OAAO,WAAY,SAAQ,KAAK;IACjB;IAAnB,YAAmB,IAAY,EAAE,OAAe;QAC9C,KAAK,CAAC,OAAO,CAAC,CAAC;QADE,SAAI,GAAJ,IAAI,CAAQ;QAE7B,IAAI,CAAC,IAAI,GAAG,aAAa,CAAC;IAC5B,CAAC;CACF;AAED,MAAM,CAAC,MAAM,kBAAkB,GAAG;IAChC,mBAAmB,EAAE,4BAA4B;IACjD,WAAW,EAAE,oBAAoB;IACjC,SAAS,EAAE,kBAAkB;IAC7B,aAAa,EAAE,sBAAsB;CAC7B,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { ModelSpec } from './types.js';
2
+ export declare const MODEL_CATALOG: ModelSpec[];
3
+ export declare function getModelSpec(id: string): ModelSpec | undefined;
4
+ //# sourceMappingURL=catalog.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"catalog.d.ts","sourceRoot":"","sources":["../../src/router/catalog.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAE5C,eAAO,MAAM,aAAa,EAAE,SAAS,EA8FpC,CAAC;AAEF,wBAAgB,YAAY,CAAC,EAAE,EAAE,MAAM,GAAG,SAAS,GAAG,SAAS,CAE9D"}
@@ -0,0 +1,103 @@
1
+ // sdk/skillfm-local/src/router/catalog.ts
2
+ //
3
+ // Model 目录 (2026-04 公开价目, 仅近似)
4
+ // Day 2 用作 router 决策的成本表
5
+ export const MODEL_CATALOG = [
6
+ // Anthropic
7
+ {
8
+ id: 'claude-haiku-4-5',
9
+ provider: 'anthropic',
10
+ tier: 'cheap',
11
+ price_per_1m_input: 0.8,
12
+ price_per_1m_output: 4,
13
+ multimodal: true,
14
+ context_window: 200000,
15
+ },
16
+ {
17
+ id: 'claude-sonnet-4-6',
18
+ provider: 'anthropic',
19
+ tier: 'standard',
20
+ price_per_1m_input: 3,
21
+ price_per_1m_output: 15,
22
+ multimodal: true,
23
+ context_window: 200000,
24
+ },
25
+ {
26
+ id: 'claude-opus-4-7',
27
+ provider: 'anthropic',
28
+ tier: 'smart',
29
+ price_per_1m_input: 15,
30
+ price_per_1m_output: 75,
31
+ multimodal: true,
32
+ context_window: 200000,
33
+ },
34
+ // OpenAI
35
+ {
36
+ id: 'gpt-4o-mini',
37
+ provider: 'openai',
38
+ tier: 'cheap',
39
+ price_per_1m_input: 0.15,
40
+ price_per_1m_output: 0.6,
41
+ multimodal: true,
42
+ context_window: 128000,
43
+ },
44
+ {
45
+ id: 'gpt-4o',
46
+ provider: 'openai',
47
+ tier: 'standard',
48
+ price_per_1m_input: 2.5,
49
+ price_per_1m_output: 10,
50
+ multimodal: true,
51
+ context_window: 128000,
52
+ },
53
+ {
54
+ id: 'o1',
55
+ provider: 'openai',
56
+ tier: 'smart',
57
+ price_per_1m_input: 15,
58
+ price_per_1m_output: 60,
59
+ multimodal: false,
60
+ context_window: 200000,
61
+ },
62
+ // 国产 (cheaper baseline, 国内 BYOK)
63
+ {
64
+ id: 'deepseek-chat',
65
+ provider: 'deepseek',
66
+ tier: 'cheap',
67
+ price_per_1m_input: 0.14,
68
+ price_per_1m_output: 0.28,
69
+ multimodal: false,
70
+ context_window: 64000,
71
+ },
72
+ {
73
+ id: 'kimi-k2',
74
+ provider: 'kimi',
75
+ tier: 'standard',
76
+ price_per_1m_input: 1.2,
77
+ price_per_1m_output: 12,
78
+ multimodal: false,
79
+ context_window: 128000,
80
+ },
81
+ {
82
+ id: 'qwen-max',
83
+ provider: 'qwen',
84
+ tier: 'standard',
85
+ price_per_1m_input: 1.4,
86
+ price_per_1m_output: 5.6,
87
+ multimodal: true,
88
+ context_window: 32000,
89
+ },
90
+ {
91
+ id: 'doubao-pro-256k',
92
+ provider: 'doubao',
93
+ tier: 'standard',
94
+ price_per_1m_input: 0.7,
95
+ price_per_1m_output: 1.4,
96
+ multimodal: false,
97
+ context_window: 256000,
98
+ },
99
+ ];
100
+ export function getModelSpec(id) {
101
+ return MODEL_CATALOG.find((m) => m.id === id);
102
+ }
103
+ //# sourceMappingURL=catalog.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"catalog.js","sourceRoot":"","sources":["../../src/router/catalog.ts"],"names":[],"mappings":"AAAA,0CAA0C;AAC1C,EAAE;AACF,+BAA+B;AAC/B,yBAAyB;AAIzB,MAAM,CAAC,MAAM,aAAa,GAAgB;IACxC,YAAY;IACZ;QACE,EAAE,EAAE,kBAAkB;QACtB,QAAQ,EAAE,WAAW;QACrB,IAAI,EAAE,OAAO;QACb,kBAAkB,EAAE,GAAG;QACvB,mBAAmB,EAAE,CAAC;QACtB,UAAU,EAAE,IAAI;QAChB,cAAc,EAAE,MAAM;KACvB;IACD;QACE,EAAE,EAAE,mBAAmB;QACvB,QAAQ,EAAE,WAAW;QACrB,IAAI,EAAE,UAAU;QAChB,kBAAkB,EAAE,CAAC;QACrB,mBAAmB,EAAE,EAAE;QACvB,UAAU,EAAE,IAAI;QAChB,cAAc,EAAE,MAAM;KACvB;IACD;QACE,EAAE,EAAE,iBAAiB;QACrB,QAAQ,EAAE,WAAW;QACrB,IAAI,EAAE,OAAO;QACb,kBAAkB,EAAE,EAAE;QACtB,mBAAmB,EAAE,EAAE;QACvB,UAAU,EAAE,IAAI;QAChB,cAAc,EAAE,MAAM;KACvB;IACD,SAAS;IACT;QACE,EAAE,EAAE,aAAa;QACjB,QAAQ,EAAE,QAAQ;QAClB,IAAI,EAAE,OAAO;QACb,kBAAkB,EAAE,IAAI;QACxB,mBAAmB,EAAE,GAAG;QACxB,UAAU,EAAE,IAAI;QAChB,cAAc,EAAE,MAAM;KACvB;IACD;QACE,EAAE,EAAE,QAAQ;QACZ,QAAQ,EAAE,QAAQ;QAClB,IAAI,EAAE,UAAU;QAChB,kBAAkB,EAAE,GAAG;QACvB,mBAAmB,EAAE,EAAE;QACvB,UAAU,EAAE,IAAI;QAChB,cAAc,EAAE,MAAM;KACvB;IACD;QACE,EAAE,EAAE,IAAI;QACR,QAAQ,EAAE,QAAQ;QAClB,IAAI,EAAE,OAAO;QACb,kBAAkB,EAAE,EAAE;QACtB,mBAAmB,EAAE,EAAE;QACvB,UAAU,EAAE,KAAK;QACjB,cAAc,EAAE,MAAM;KACvB;IACD,iCAAiC;IACjC;QACE,EAAE,EAAE,eAAe;QACnB,QAAQ,EAAE,UAAU;QACpB,IAAI,EAAE,OAAO;QACb,kBAAkB,EAAE,IAAI;QACxB,mBAAmB,EAAE,IAAI;QACzB,UAAU,EAAE,KAAK;QACjB,cAAc,EAAE,KAAK;KACtB;IACD;QACE,EAAE,EAAE,SAAS;QACb,QAAQ,EAAE,MAAM;QAChB,IAAI,EAAE,UAAU;QAChB,kBAAkB,EAAE,GAAG;QACvB,mBAAmB,EAAE,EAAE;QACvB,UAAU,EAAE,KAAK;QACjB,cAAc,EAAE,MAAM;KACvB;IACD;QACE,EAAE,EAAE,UAAU;QACd,QAAQ,EAAE,MAAM;QAChB,IAAI,EAAE,UAAU;QAChB,kBAAkB,EAAE,GAAG;QACvB,mBAAmB,EAAE,GAAG;QACxB,UAAU,EAAE,IAAI;QAChB,cAAc,EAAE,KAAK;KACtB;IACD;QACE,EAAE,EAAE,iBAAiB;QACrB,QAAQ,EAAE,QAAQ;QAClB,IAAI,EAAE,UAAU;QAChB,kBAAkB,EAAE,GAAG;QACvB,mBAAmB,EAAE,GAAG;QACxB,UAAU,EAAE,KAAK;QACjB,cAAc,EAAE,MAAM;KACvB;CACF,CAAC;AAEF,MAAM,UAAU,YAAY,CAAC,EAAU;IACrC,OAAO,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;AAChD,CAAC"}
@@ -0,0 +1,5 @@
1
+ import type { Complexity, RouteRequest } from './types.js';
2
+ export declare function classify(req: RouteRequest): Complexity;
3
+ /** 从原始 message[] 估算 token 数 (粗估: 1 token ≈ 4 字符) */
4
+ export declare function estimateTokens(text: string): number;
5
+ //# sourceMappingURL=classifier.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"classifier.d.ts","sourceRoot":"","sources":["../../src/router/classifier.ts"],"names":[],"mappings":"AAaA,OAAO,KAAK,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAY3D,wBAAgB,QAAQ,CAAC,GAAG,EAAE,YAAY,GAAG,UAAU,CAiBtD;AAED,oDAAoD;AACpD,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAEnD"}
@@ -0,0 +1,43 @@
1
+ // sdk/skillfm-local/src/router/classifier.ts
2
+ //
3
+ // Heuristic complexity classifier (Day 2)
4
+ // Day 3+ 接 Phi-4 onnx 本地跑 / 或 brain 端 small classifier
5
+ //
6
+ // 当前 heuristic 规则:
7
+ // - 含图片 → multimodal
8
+ // - 关键词 "debug" / "analyze" / "design" / "architect" / "reason" → high
9
+ // - 关键词 "summarize" / "translate" / "rephrase" / "extract" → low
10
+ // - input_tokens < 500 → low
11
+ // - input_tokens > 4000 → high
12
+ // - 其他 → medium
13
+ const HIGH_KEYWORDS = [
14
+ 'debug', 'analyze', 'design', 'architect', 'reason', 'plan', 'compare',
15
+ '推理', '设计', '架构', '调试', '分析', '规划', '比较', '复杂',
16
+ ];
17
+ const LOW_KEYWORDS = [
18
+ 'summarize', 'translate', 'rephrase', 'extract', 'classify',
19
+ '总结', '翻译', '改写', '抽取', '分类',
20
+ ];
21
+ export function classify(req) {
22
+ if (req.has_image)
23
+ return 'multimodal';
24
+ const hint = (req.hint ?? '').toLowerCase();
25
+ for (const kw of HIGH_KEYWORDS) {
26
+ if (hint.includes(kw))
27
+ return 'high';
28
+ }
29
+ for (const kw of LOW_KEYWORDS) {
30
+ if (hint.includes(kw))
31
+ return 'low';
32
+ }
33
+ if (req.estimated_input_tokens < 500)
34
+ return 'low';
35
+ if (req.estimated_input_tokens > 4000)
36
+ return 'high';
37
+ return 'medium';
38
+ }
39
+ /** 从原始 message[] 估算 token 数 (粗估: 1 token ≈ 4 字符) */
40
+ export function estimateTokens(text) {
41
+ return Math.ceil(text.length / 4);
42
+ }
43
+ //# sourceMappingURL=classifier.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"classifier.js","sourceRoot":"","sources":["../../src/router/classifier.ts"],"names":[],"mappings":"AAAA,6CAA6C;AAC7C,EAAE;AACF,0CAA0C;AAC1C,uDAAuD;AACvD,EAAE;AACF,mBAAmB;AACnB,uBAAuB;AACvB,yEAAyE;AACzE,mEAAmE;AACnE,+BAA+B;AAC/B,iCAAiC;AACjC,kBAAkB;AAIlB,MAAM,aAAa,GAAG;IACpB,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,EAAE,SAAS;IACtE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI;CAC/C,CAAC;AAEF,MAAM,YAAY,GAAG;IACnB,WAAW,EAAE,WAAW,EAAE,UAAU,EAAE,SAAS,EAAE,UAAU;IAC3D,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI;CAC7B,CAAC;AAEF,MAAM,UAAU,QAAQ,CAAC,GAAiB;IACxC,IAAI,GAAG,CAAC,SAAS;QAAE,OAAO,YAAY,CAAC;IAEvC,MAAM,IAAI,GAAG,CAAC,GAAG,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;IAE5C,KAAK,MAAM,EAAE,IAAI,aAAa,EAAE,CAAC;QAC/B,IAAI,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;YAAE,OAAO,MAAM,CAAC;IACvC,CAAC;IAED,KAAK,MAAM,EAAE,IAAI,YAAY,EAAE,CAAC;QAC9B,IAAI,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;YAAE,OAAO,KAAK,CAAC;IACtC,CAAC;IAED,IAAI,GAAG,CAAC,sBAAsB,GAAG,GAAG;QAAE,OAAO,KAAK,CAAC;IACnD,IAAI,GAAG,CAAC,sBAAsB,GAAG,IAAI;QAAE,OAAO,MAAM,CAAC;IAErD,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,oDAAoD;AACpD,MAAM,UAAU,cAAc,CAAC,IAAY;IACzC,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;AACpC,CAAC"}
@@ -0,0 +1,7 @@
1
+ export type { Complexity, ModelTier, ModelId, ModelSpec, RouteRequest, RouteDecision } from './types.js';
2
+ export { MODEL_CATALOG, getModelSpec } from './catalog.js';
3
+ export { classify, estimateTokens } from './classifier.js';
4
+ export { route } from './route.js';
5
+ export { generateLiteLLMConfig, configToYaml } from './litellm-config.js';
6
+ export type { LiteLLMConfigOptions } from './litellm-config.js';
7
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/router/index.ts"],"names":[],"mappings":"AAEA,YAAY,EAAE,UAAU,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,YAAY,CAAC;AACzG,OAAO,EAAE,aAAa,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAC3D,OAAO,EAAE,QAAQ,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAC3D,OAAO,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AACnC,OAAO,EAAE,qBAAqB,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAC1E,YAAY,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC"}
@@ -0,0 +1,6 @@
1
+ // sdk/skillfm-local/src/router/index.ts
2
+ export { MODEL_CATALOG, getModelSpec } from './catalog.js';
3
+ export { classify, estimateTokens } from './classifier.js';
4
+ export { route } from './route.js';
5
+ export { generateLiteLLMConfig, configToYaml } from './litellm-config.js';
6
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/router/index.ts"],"names":[],"mappings":"AAAA,wCAAwC;AAGxC,OAAO,EAAE,aAAa,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAC3D,OAAO,EAAE,QAAQ,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAC3D,OAAO,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AACnC,OAAO,EAAE,qBAAqB,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC"}
@@ -0,0 +1,34 @@
1
+ import type { ModelSpec } from './types.js';
2
+ export interface LiteLLMConfigOptions {
3
+ /** 用户配置的 BYOK key, 按 provider 分 */
4
+ byok_keys: Partial<{
5
+ anthropic: string;
6
+ openai: string;
7
+ deepseek: string;
8
+ kimi: string;
9
+ qwen: string;
10
+ doubao: string;
11
+ }>;
12
+ /** 排除哪些 provider (e.g. 中国用户排除 OpenAI) */
13
+ exclude_providers?: ReadonlyArray<ModelSpec['provider']>;
14
+ }
15
+ interface LiteLLMModelEntry {
16
+ model_name: string;
17
+ litellm_params: {
18
+ model: string;
19
+ api_key: string;
20
+ api_base?: string;
21
+ };
22
+ }
23
+ interface LiteLLMConfig {
24
+ model_list: LiteLLMModelEntry[];
25
+ router_settings: {
26
+ routing_strategy: string;
27
+ model_group_alias: Record<string, string[]>;
28
+ };
29
+ }
30
+ export declare function generateLiteLLMConfig(opts: LiteLLMConfigOptions): LiteLLMConfig;
31
+ /** YAML 序列化 (简单版,避免引入 yaml 依赖) */
32
+ export declare function configToYaml(cfg: LiteLLMConfig): string;
33
+ export {};
34
+ //# sourceMappingURL=litellm-config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"litellm-config.d.ts","sourceRoot":"","sources":["../../src/router/litellm-config.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAG5C,MAAM,WAAW,oBAAoB;IACnC,mCAAmC;IACnC,SAAS,EAAE,OAAO,CAAC;QACjB,SAAS,EAAE,MAAM,CAAC;QAClB,MAAM,EAAE,MAAM,CAAC;QACf,QAAQ,EAAE,MAAM,CAAC;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,MAAM,CAAC;QACb,MAAM,EAAE,MAAM,CAAC;KAChB,CAAC,CAAC;IACH,yCAAyC;IACzC,iBAAiB,CAAC,EAAE,aAAa,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC,CAAC;CAC1D;AAED,UAAU,iBAAiB;IACzB,UAAU,EAAE,MAAM,CAAC;IACnB,cAAc,EAAE;QACd,KAAK,EAAE,MAAM,CAAC;QACd,OAAO,EAAE,MAAM,CAAC;QAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;KACnB,CAAC;CACH;AAED,UAAU,aAAa;IACrB,UAAU,EAAE,iBAAiB,EAAE,CAAC;IAChC,eAAe,EAAE;QACf,gBAAgB,EAAE,MAAM,CAAC;QACzB,iBAAiB,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;KAC7C,CAAC;CACH;AAoBD,wBAAgB,qBAAqB,CAAC,IAAI,EAAE,oBAAoB,GAAG,aAAa,CAgC/E;AAED,kCAAkC;AAClC,wBAAgB,YAAY,CAAC,GAAG,EAAE,aAAa,GAAG,MAAM,CAmBvD"}
@@ -0,0 +1,78 @@
1
+ // sdk/skillfm-local/src/router/litellm-config.ts
2
+ //
3
+ // 生成 LiteLLM proxy YAML 配置 (Day 2)
4
+ // 从 sidecar 的 BYOK key 集合 + 用户偏好生成
5
+ // LiteLLM doc: https://docs.litellm.ai/docs/proxy/configs
6
+ import { MODEL_CATALOG } from './catalog.js';
7
+ const PROVIDER_API_BASE = {
8
+ anthropic: undefined, // 走默认
9
+ openai: undefined,
10
+ deepseek: 'https://api.deepseek.com',
11
+ kimi: 'https://api.moonshot.cn/v1',
12
+ qwen: 'https://dashscope.aliyuncs.com/compatible-mode/v1',
13
+ doubao: 'https://ark.cn-beijing.volces.com/api/v3',
14
+ };
15
+ const PROVIDER_LITELLM_PREFIX = {
16
+ anthropic: 'anthropic/',
17
+ openai: 'openai/',
18
+ deepseek: 'openai/', // DeepSeek 兼容 OpenAI schema
19
+ kimi: 'openai/',
20
+ qwen: 'openai/',
21
+ doubao: 'openai/',
22
+ };
23
+ export function generateLiteLLMConfig(opts) {
24
+ const exclude = new Set(opts.exclude_providers ?? []);
25
+ const model_list = [];
26
+ const tier_groups = { cheap: [], standard: [], smart: [], multimodal: [] };
27
+ for (const m of MODEL_CATALOG) {
28
+ if (exclude.has(m.provider))
29
+ continue;
30
+ const key = opts.byok_keys[m.provider];
31
+ if (!key)
32
+ continue;
33
+ const litellm_model = `${PROVIDER_LITELLM_PREFIX[m.provider]}${m.id}`;
34
+ const entry = {
35
+ model_name: m.id,
36
+ litellm_params: {
37
+ model: litellm_model,
38
+ api_key: key,
39
+ },
40
+ };
41
+ const base = PROVIDER_API_BASE[m.provider];
42
+ if (base)
43
+ entry.litellm_params.api_base = base;
44
+ model_list.push(entry);
45
+ tier_groups[m.tier].push(m.id);
46
+ if (m.multimodal)
47
+ tier_groups.multimodal.push(m.id);
48
+ }
49
+ return {
50
+ model_list,
51
+ router_settings: {
52
+ routing_strategy: 'simple-shuffle', // Day 3+ 升级为 cost-aware-routing
53
+ model_group_alias: tier_groups,
54
+ },
55
+ };
56
+ }
57
+ /** YAML 序列化 (简单版,避免引入 yaml 依赖) */
58
+ export function configToYaml(cfg) {
59
+ const lines = ['model_list:'];
60
+ for (const m of cfg.model_list) {
61
+ lines.push(` - model_name: ${m.model_name}`);
62
+ lines.push(` litellm_params:`);
63
+ lines.push(` model: ${m.litellm_params.model}`);
64
+ lines.push(` api_key: ${m.litellm_params.api_key}`);
65
+ if (m.litellm_params.api_base) {
66
+ lines.push(` api_base: ${m.litellm_params.api_base}`);
67
+ }
68
+ }
69
+ lines.push('');
70
+ lines.push('router_settings:');
71
+ lines.push(` routing_strategy: ${cfg.router_settings.routing_strategy}`);
72
+ lines.push(' model_group_alias:');
73
+ for (const [group, ids] of Object.entries(cfg.router_settings.model_group_alias)) {
74
+ lines.push(` ${group}: [${ids.map((i) => `"${i}"`).join(', ')}]`);
75
+ }
76
+ return lines.join('\n') + '\n';
77
+ }
78
+ //# sourceMappingURL=litellm-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"litellm-config.js","sourceRoot":"","sources":["../../src/router/litellm-config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { RouteDecision, RouteRequest } from './types.js';
2
+ export declare function route(req: RouteRequest): RouteDecision;
3
+ //# sourceMappingURL=route.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"route.d.ts","sourceRoot":"","sources":["../../src/router/route.ts"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAA6C,aAAa,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAuBzG,wBAAgB,KAAK,CAAC,GAAG,EAAE,YAAY,GAAG,aAAa,CAyCtD"}
@@ -0,0 +1,63 @@
1
+ // sdk/skillfm-local/src/router/route.ts
2
+ //
3
+ // 路由决策器 (Day 2)
4
+ // 接收 RouteRequest → 用 classifier 判 complexity → 选最低成本满足条件的 model
5
+ import { MODEL_CATALOG } from './catalog.js';
6
+ import { classify } from './classifier.js';
7
+ const COMPLEXITY_TO_TIER = {
8
+ low: 'cheap',
9
+ medium: 'standard',
10
+ high: 'smart',
11
+ multimodal: 'multimodal',
12
+ };
13
+ function tierMatches(spec, target) {
14
+ if (target === 'multimodal')
15
+ return spec.multimodal;
16
+ // smart 用户也能用 standard 兜底, 但 standard 用户不能用 cheap (成本可控)
17
+ if (target === 'smart')
18
+ return spec.tier === 'smart' || spec.tier === 'standard';
19
+ if (target === 'standard')
20
+ return spec.tier === 'standard' || spec.tier === 'cheap';
21
+ return spec.tier === 'cheap';
22
+ }
23
+ function estimateCost(spec, input_tokens) {
24
+ // 假设 output ≈ input * 0.5
25
+ const out = input_tokens * 0.5;
26
+ return (input_tokens / 1_000_000) * spec.price_per_1m_input + (out / 1_000_000) * spec.price_per_1m_output;
27
+ }
28
+ export function route(req) {
29
+ const complexity = classify(req);
30
+ const tier_target = req.prefer_tier ?? COMPLEXITY_TO_TIER[complexity];
31
+ // 候选集
32
+ let candidates = MODEL_CATALOG.filter((m) => tierMatches(m, tier_target));
33
+ // context window 必须 ≥ input
34
+ candidates = candidates.filter((m) => m.context_window >= req.estimated_input_tokens);
35
+ // multimodal 必须支持
36
+ if (req.has_image) {
37
+ candidates = candidates.filter((m) => m.multimodal);
38
+ }
39
+ // 排除 provider
40
+ if (req.exclude_providers?.length) {
41
+ candidates = candidates.filter((m) => !req.exclude_providers.includes(m.provider));
42
+ }
43
+ if (candidates.length === 0) {
44
+ // 兜底: 所有 model 中支持 context + multimodal 的, 选最便宜
45
+ candidates = MODEL_CATALOG.filter((m) => m.context_window >= req.estimated_input_tokens && (!req.has_image || m.multimodal));
46
+ }
47
+ if (candidates.length === 0) {
48
+ // 极端兜底: 最大 context model
49
+ candidates = [...MODEL_CATALOG].sort((a, b) => b.context_window - a.context_window).slice(0, 1);
50
+ }
51
+ // 选成本最低
52
+ candidates.sort((a, b) => estimateCost(a, req.estimated_input_tokens) - estimateCost(b, req.estimated_input_tokens));
53
+ const winner = candidates[0];
54
+ const winnerCost = estimateCost(winner, req.estimated_input_tokens);
55
+ return {
56
+ model_id: winner.id,
57
+ reason: `complexity=${complexity}, tier=${tier_target}, cheapest match`,
58
+ complexity,
59
+ estimated_cost_usd: winnerCost,
60
+ alternatives: candidates.slice(1, 4).map((c) => c.id),
61
+ };
62
+ }
63
+ //# sourceMappingURL=route.js.map