@folterung/project-memory 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 (215) hide show
  1. package/.memignore.example +11 -0
  2. package/README.md +48 -0
  3. package/docker-compose.yml +17 -0
  4. package/package.json +36 -0
  5. package/packages/cli/bin/mem.js +6 -0
  6. package/packages/cli/coverage/lcov-report/base.css +224 -0
  7. package/packages/cli/coverage/lcov-report/block-navigation.js +87 -0
  8. package/packages/cli/coverage/lcov-report/chunking/chunker.ts.html +538 -0
  9. package/packages/cli/coverage/lcov-report/chunking/hash.ts.html +148 -0
  10. package/packages/cli/coverage/lcov-report/chunking/index.html +146 -0
  11. package/packages/cli/coverage/lcov-report/chunking/types.ts.html +214 -0
  12. package/packages/cli/coverage/lcov-report/config/index.html +131 -0
  13. package/packages/cli/coverage/lcov-report/config/load.ts.html +184 -0
  14. package/packages/cli/coverage/lcov-report/config/types.ts.html +232 -0
  15. package/packages/cli/coverage/lcov-report/embedding/index.html +116 -0
  16. package/packages/cli/coverage/lcov-report/embedding/stub.ts.html +181 -0
  17. package/packages/cli/coverage/lcov-report/favicon.png +0 -0
  18. package/packages/cli/coverage/lcov-report/index.html +161 -0
  19. package/packages/cli/coverage/lcov-report/prettify.css +1 -0
  20. package/packages/cli/coverage/lcov-report/prettify.js +2 -0
  21. package/packages/cli/coverage/lcov-report/scope/allowlist.ts.html +199 -0
  22. package/packages/cli/coverage/lcov-report/scope/ignore.ts.html +343 -0
  23. package/packages/cli/coverage/lcov-report/scope/index.html +131 -0
  24. package/packages/cli/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  25. package/packages/cli/coverage/lcov-report/sorter.js +210 -0
  26. package/packages/cli/coverage/lcov.info +669 -0
  27. package/packages/cli/coverage/tmp/coverage-25917-1770055893226-0.json +1 -0
  28. package/packages/cli/coverage/tmp/coverage-25918-1770055893272-0.json +1 -0
  29. package/packages/cli/coverage/tmp/coverage-25919-1770055893273-0.json +1 -0
  30. package/packages/cli/coverage/tmp/coverage-25920-1770055893271-0.json +1 -0
  31. package/packages/cli/coverage/tmp/coverage-25921-1770055893279-0.json +1 -0
  32. package/packages/cli/coverage/tmp/coverage-25922-1770055893272-0.json +1 -0
  33. package/packages/cli/coverage/tmp/coverage-25923-1770055893275-0.json +1 -0
  34. package/packages/cli/coverage/tmp/coverage-25924-1770055893294-0.json +1 -0
  35. package/packages/cli/coverage/tmp/coverage-25925-1770055893290-0.json +1 -0
  36. package/packages/cli/dist/chunking/chunker.d.ts +2 -0
  37. package/packages/cli/dist/chunking/chunker.js +142 -0
  38. package/packages/cli/dist/chunking/chunker.js.map +1 -0
  39. package/packages/cli/dist/chunking/chunker.test.d.ts +1 -0
  40. package/packages/cli/dist/chunking/chunker.test.js +50 -0
  41. package/packages/cli/dist/chunking/chunker.test.js.map +1 -0
  42. package/packages/cli/dist/chunking/hash.d.ts +3 -0
  43. package/packages/cli/dist/chunking/hash.js +17 -0
  44. package/packages/cli/dist/chunking/hash.js.map +1 -0
  45. package/packages/cli/dist/chunking/hash.test.d.ts +1 -0
  46. package/packages/cli/dist/chunking/hash.test.js +36 -0
  47. package/packages/cli/dist/chunking/hash.test.js.map +1 -0
  48. package/packages/cli/dist/chunking/types.d.ts +19 -0
  49. package/packages/cli/dist/chunking/types.js +24 -0
  50. package/packages/cli/dist/chunking/types.js.map +1 -0
  51. package/packages/cli/dist/chunking/types.test.d.ts +1 -0
  52. package/packages/cli/dist/chunking/types.test.js +25 -0
  53. package/packages/cli/dist/chunking/types.test.js.map +1 -0
  54. package/packages/cli/dist/cli/index.d.ts +1 -0
  55. package/packages/cli/dist/cli/index.js +67 -0
  56. package/packages/cli/dist/cli/index.js.map +1 -0
  57. package/packages/cli/dist/commands/deep-index.d.ts +1 -0
  58. package/packages/cli/dist/commands/deep-index.js +17 -0
  59. package/packages/cli/dist/commands/deep-index.js.map +1 -0
  60. package/packages/cli/dist/commands/doctor.d.ts +1 -0
  61. package/packages/cli/dist/commands/doctor.js +27 -0
  62. package/packages/cli/dist/commands/doctor.js.map +1 -0
  63. package/packages/cli/dist/commands/down.d.ts +1 -0
  64. package/packages/cli/dist/commands/down.js +13 -0
  65. package/packages/cli/dist/commands/down.js.map +1 -0
  66. package/packages/cli/dist/commands/explain.d.ts +1 -0
  67. package/packages/cli/dist/commands/explain.js +23 -0
  68. package/packages/cli/dist/commands/explain.js.map +1 -0
  69. package/packages/cli/dist/commands/init.d.ts +1 -0
  70. package/packages/cli/dist/commands/init.js +35 -0
  71. package/packages/cli/dist/commands/init.js.map +1 -0
  72. package/packages/cli/dist/commands/query.d.ts +1 -0
  73. package/packages/cli/dist/commands/query.js +44 -0
  74. package/packages/cli/dist/commands/query.js.map +1 -0
  75. package/packages/cli/dist/commands/reset.d.ts +3 -0
  76. package/packages/cli/dist/commands/reset.js +28 -0
  77. package/packages/cli/dist/commands/reset.js.map +1 -0
  78. package/packages/cli/dist/commands/scaffold.d.ts +1 -0
  79. package/packages/cli/dist/commands/scaffold.js +52 -0
  80. package/packages/cli/dist/commands/scaffold.js.map +1 -0
  81. package/packages/cli/dist/commands/update.d.ts +5 -0
  82. package/packages/cli/dist/commands/update.js +23 -0
  83. package/packages/cli/dist/commands/update.js.map +1 -0
  84. package/packages/cli/dist/commands/watch.d.ts +1 -0
  85. package/packages/cli/dist/commands/watch.js +46 -0
  86. package/packages/cli/dist/commands/watch.js.map +1 -0
  87. package/packages/cli/dist/config/defaults.d.ts +2 -0
  88. package/packages/cli/dist/config/defaults.js +47 -0
  89. package/packages/cli/dist/config/defaults.js.map +1 -0
  90. package/packages/cli/dist/config/load.d.ts +4 -0
  91. package/packages/cli/dist/config/load.js +29 -0
  92. package/packages/cli/dist/config/load.js.map +1 -0
  93. package/packages/cli/dist/config/load.test.d.ts +1 -0
  94. package/packages/cli/dist/config/load.test.js +51 -0
  95. package/packages/cli/dist/config/load.test.js.map +1 -0
  96. package/packages/cli/dist/config/types.d.ts +32 -0
  97. package/packages/cli/dist/config/types.js +24 -0
  98. package/packages/cli/dist/config/types.js.map +1 -0
  99. package/packages/cli/dist/config/types.test.d.ts +1 -0
  100. package/packages/cli/dist/config/types.test.js +31 -0
  101. package/packages/cli/dist/config/types.test.js.map +1 -0
  102. package/packages/cli/dist/docker/compose.d.ts +5 -0
  103. package/packages/cli/dist/docker/compose.js +75 -0
  104. package/packages/cli/dist/docker/compose.js.map +1 -0
  105. package/packages/cli/dist/embedding/stub.d.ts +6 -0
  106. package/packages/cli/dist/embedding/stub.js +29 -0
  107. package/packages/cli/dist/embedding/stub.js.map +1 -0
  108. package/packages/cli/dist/embedding/stub.test.d.ts +1 -0
  109. package/packages/cli/dist/embedding/stub.test.js +31 -0
  110. package/packages/cli/dist/embedding/stub.test.js.map +1 -0
  111. package/packages/cli/dist/phase1/pipeline.d.ts +1 -0
  112. package/packages/cli/dist/phase1/pipeline.js +145 -0
  113. package/packages/cli/dist/phase1/pipeline.js.map +1 -0
  114. package/packages/cli/dist/phase2/deep-index.d.ts +1 -0
  115. package/packages/cli/dist/phase2/deep-index.js +105 -0
  116. package/packages/cli/dist/phase2/deep-index.js.map +1 -0
  117. package/packages/cli/dist/qdrant/upsert.d.ts +14 -0
  118. package/packages/cli/dist/qdrant/upsert.js +30 -0
  119. package/packages/cli/dist/qdrant/upsert.js.map +1 -0
  120. package/packages/cli/dist/scope/allowlist.d.ts +11 -0
  121. package/packages/cli/dist/scope/allowlist.js +36 -0
  122. package/packages/cli/dist/scope/allowlist.js.map +1 -0
  123. package/packages/cli/dist/scope/allowlist.test.d.ts +1 -0
  124. package/packages/cli/dist/scope/allowlist.test.js +23 -0
  125. package/packages/cli/dist/scope/allowlist.test.js.map +1 -0
  126. package/packages/cli/dist/scope/ignore.d.ts +2 -0
  127. package/packages/cli/dist/scope/ignore.js +80 -0
  128. package/packages/cli/dist/scope/ignore.js.map +1 -0
  129. package/packages/cli/dist/scope/ignore.test.d.ts +1 -0
  130. package/packages/cli/dist/scope/ignore.test.js +71 -0
  131. package/packages/cli/dist/scope/ignore.test.js.map +1 -0
  132. package/packages/cli/package.json +46 -0
  133. package/packages/cli/src/chunking/chunker.test.ts +54 -0
  134. package/packages/cli/src/chunking/chunker.ts +151 -0
  135. package/packages/cli/src/chunking/hash.test.ts +41 -0
  136. package/packages/cli/src/chunking/hash.ts +21 -0
  137. package/packages/cli/src/chunking/types.test.ts +28 -0
  138. package/packages/cli/src/chunking/types.ts +43 -0
  139. package/packages/cli/src/cli/index.ts +79 -0
  140. package/packages/cli/src/commands/deep-index.ts +16 -0
  141. package/packages/cli/src/commands/doctor.ts +32 -0
  142. package/packages/cli/src/commands/down.ts +12 -0
  143. package/packages/cli/src/commands/explain.ts +22 -0
  144. package/packages/cli/src/commands/init.ts +37 -0
  145. package/packages/cli/src/commands/query.ts +49 -0
  146. package/packages/cli/src/commands/reset.ts +30 -0
  147. package/packages/cli/src/commands/scaffold.ts +55 -0
  148. package/packages/cli/src/commands/update.ts +22 -0
  149. package/packages/cli/src/commands/watch.ts +47 -0
  150. package/packages/cli/src/config/defaults.ts +49 -0
  151. package/packages/cli/src/config/load.test.ts +55 -0
  152. package/packages/cli/src/config/load.ts +33 -0
  153. package/packages/cli/src/config/types.test.ts +43 -0
  154. package/packages/cli/src/config/types.ts +49 -0
  155. package/packages/cli/src/docker/compose.ts +75 -0
  156. package/packages/cli/src/embedding/stub.test.ts +35 -0
  157. package/packages/cli/src/embedding/stub.ts +32 -0
  158. package/packages/cli/src/phase1/pipeline.ts +164 -0
  159. package/packages/cli/src/phase2/deep-index.ts +120 -0
  160. package/packages/cli/src/qdrant/upsert.ts +45 -0
  161. package/packages/cli/src/scope/allowlist.test.ts +25 -0
  162. package/packages/cli/src/scope/allowlist.ts +38 -0
  163. package/packages/cli/src/scope/ignore.test.ts +71 -0
  164. package/packages/cli/src/scope/ignore.ts +86 -0
  165. package/packages/cli/tsconfig.json +16 -0
  166. package/packages/server/coverage/lcov-report/base.css +224 -0
  167. package/packages/server/coverage/lcov-report/block-navigation.js +87 -0
  168. package/packages/server/coverage/lcov-report/favicon.png +0 -0
  169. package/packages/server/coverage/lcov-report/index.html +116 -0
  170. package/packages/server/coverage/lcov-report/prettify.css +1 -0
  171. package/packages/server/coverage/lcov-report/prettify.js +2 -0
  172. package/packages/server/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  173. package/packages/server/coverage/lcov-report/sorter.js +210 -0
  174. package/packages/server/coverage/lcov-report/stub.ts.html +184 -0
  175. package/packages/server/coverage/lcov.info +57 -0
  176. package/packages/server/coverage/tmp/coverage-26012-1770055894042-0.json +1 -0
  177. package/packages/server/coverage/tmp/coverage-26013-1770055894077-0.json +1 -0
  178. package/packages/server/dist/api/explain.d.ts +6 -0
  179. package/packages/server/dist/api/explain.js +16 -0
  180. package/packages/server/dist/api/explain.js.map +1 -0
  181. package/packages/server/dist/api/health.d.ts +7 -0
  182. package/packages/server/dist/api/health.js +43 -0
  183. package/packages/server/dist/api/health.js.map +1 -0
  184. package/packages/server/dist/api/refresh.d.ts +2 -0
  185. package/packages/server/dist/api/refresh.js +8 -0
  186. package/packages/server/dist/api/refresh.js.map +1 -0
  187. package/packages/server/dist/api/search.d.ts +28 -0
  188. package/packages/server/dist/api/search.js +36 -0
  189. package/packages/server/dist/api/search.js.map +1 -0
  190. package/packages/server/dist/api/system.d.ts +6 -0
  191. package/packages/server/dist/api/system.js +17 -0
  192. package/packages/server/dist/api/system.js.map +1 -0
  193. package/packages/server/dist/embedding/stub.d.ts +7 -0
  194. package/packages/server/dist/embedding/stub.js +30 -0
  195. package/packages/server/dist/embedding/stub.js.map +1 -0
  196. package/packages/server/dist/embedding/stub.test.d.ts +1 -0
  197. package/packages/server/dist/embedding/stub.test.js +25 -0
  198. package/packages/server/dist/embedding/stub.test.js.map +1 -0
  199. package/packages/server/dist/index.d.ts +1 -0
  200. package/packages/server/dist/index.js +26 -0
  201. package/packages/server/dist/index.js.map +1 -0
  202. package/packages/server/dist/qdrant/client.d.ts +24 -0
  203. package/packages/server/dist/qdrant/client.js +55 -0
  204. package/packages/server/dist/qdrant/client.js.map +1 -0
  205. package/packages/server/package.json +37 -0
  206. package/packages/server/src/api/explain.ts +24 -0
  207. package/packages/server/src/api/health.ts +50 -0
  208. package/packages/server/src/api/refresh.ts +9 -0
  209. package/packages/server/src/api/search.ts +57 -0
  210. package/packages/server/src/api/system.ts +25 -0
  211. package/packages/server/src/embedding/stub.test.ts +28 -0
  212. package/packages/server/src/embedding/stub.ts +33 -0
  213. package/packages/server/src/index.ts +29 -0
  214. package/packages/server/src/qdrant/client.ts +92 -0
  215. package/packages/server/tsconfig.json +16 -0
@@ -0,0 +1,6 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export declare function registerExplainRoutes(app: FastifyInstance, options: {
3
+ qdrantHost?: string;
4
+ qdrantPort?: number;
5
+ collection?: string;
6
+ }): void;
@@ -0,0 +1,16 @@
1
+ import { createQdrantClient } from "../qdrant/client.js";
2
+ import { embedText } from "../embedding/stub.js";
3
+ import { searchPoints } from "../qdrant/client.js";
4
+ export function registerExplainRoutes(app, options) {
5
+ const client = createQdrantClient(options.qdrantHost ?? "127.0.0.1", options.qdrantPort ?? 6333);
6
+ const collection = options.collection ?? "project_memory";
7
+ app.get("/explain/:symbol", async (request, reply) => {
8
+ const symbol = decodeURIComponent(request.params.symbol);
9
+ const vector = embedText(symbol);
10
+ const results = await searchPoints(client, collection, vector, 3);
11
+ const snippet = results[0]?.payload?.text ?? "";
12
+ const path = results[0]?.payload?.path;
13
+ await reply.send({ symbol, snippet, path, results: results.length });
14
+ });
15
+ }
16
+ //# sourceMappingURL=explain.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"explain.js","sourceRoot":"","sources":["../../src/api/explain.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAEnD,MAAM,UAAU,qBAAqB,CACnC,GAAoB,EACpB,OAA0E;IAE1E,MAAM,MAAM,GAAG,kBAAkB,CAC/B,OAAO,CAAC,UAAU,IAAI,WAAW,EACjC,OAAO,CAAC,UAAU,IAAI,IAAI,CAC3B,CAAC;IACF,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,IAAI,gBAAgB,CAAC;IAE1D,GAAG,CAAC,GAAG,CAAiC,kBAAkB,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QACnF,MAAM,MAAM,GAAG,kBAAkB,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACzD,MAAM,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;QACjC,MAAM,OAAO,GAAG,MAAM,YAAY,CAAC,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC;QAClE,MAAM,OAAO,GAAG,OAAO,CAAC,CAAC,CAAC,EAAE,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC;QAChD,MAAM,IAAI,GAAG,OAAO,CAAC,CAAC,CAAC,EAAE,OAAO,EAAE,IAAI,CAAC;QACvC,MAAM,KAAK,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;IACvE,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,7 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export declare function registerHealthRoutes(app: FastifyInstance, options: {
3
+ cwd: string;
4
+ qdrantHost?: string;
5
+ qdrantPort?: number;
6
+ collection?: string;
7
+ }): void;
@@ -0,0 +1,43 @@
1
+ import { readFileSync, existsSync } from "node:fs";
2
+ import { join } from "node:path";
3
+ import { createQdrantClient } from "../qdrant/client.js";
4
+ const DEEP_INDEX_STATE_FILE = ".mem/state/deep-index.json";
5
+ function getDeepIndexProgress(cwd) {
6
+ const path = join(cwd, DEEP_INDEX_STATE_FILE);
7
+ if (!existsSync(path))
8
+ return null;
9
+ try {
10
+ const raw = readFileSync(path, "utf8");
11
+ const data = JSON.parse(raw);
12
+ const embedded = data.embedded ?? (Array.isArray(data.embeddedIds) ? data.embeddedIds.length : 0);
13
+ const total = data.total ?? 0;
14
+ const percent = total > 0 ? Math.round((embedded / total) * 100) : 0;
15
+ return { percent, embedded, total };
16
+ }
17
+ catch {
18
+ return null;
19
+ }
20
+ }
21
+ export function registerHealthRoutes(app, options) {
22
+ const client = createQdrantClient(options.qdrantHost ?? "127.0.0.1", options.qdrantPort ?? 6333);
23
+ const collection = options.collection ?? "project_memory";
24
+ app.get("/health", async (_request, reply) => {
25
+ let qdrant = false;
26
+ try {
27
+ await client.getCollection(collection);
28
+ qdrant = true;
29
+ }
30
+ catch {
31
+ // ignore
32
+ }
33
+ const deepIndexProgress = getDeepIndexProgress(options.cwd);
34
+ const indexVersion = deepIndexProgress ? `phase1+${deepIndexProgress.embedded}` : "phase1";
35
+ await reply.send({
36
+ status: qdrant ? "ok" : "degraded",
37
+ qdrant,
38
+ indexVersion,
39
+ deepIndexProgress: deepIndexProgress ?? undefined,
40
+ });
41
+ });
42
+ }
43
+ //# sourceMappingURL=health.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"health.js","sourceRoot":"","sources":["../../src/api/health.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACnD,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AAEzD,MAAM,qBAAqB,GAAG,4BAA4B,CAAC;AAE3D,SAAS,oBAAoB,CAAC,GAAW;IACvC,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,EAAE,qBAAqB,CAAC,CAAC;IAC9C,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;QAAE,OAAO,IAAI,CAAC;IACnC,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QACvC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAkE,CAAC;QAC9F,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAClG,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,IAAI,CAAC,CAAC;QAC9B,MAAM,OAAO,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,QAAQ,GAAG,KAAK,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACrE,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;IACtC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,MAAM,UAAU,oBAAoB,CAClC,GAAoB,EACpB,OAAuF;IAEvF,MAAM,MAAM,GAAG,kBAAkB,CAC/B,OAAO,CAAC,UAAU,IAAI,WAAW,EACjC,OAAO,CAAC,UAAU,IAAI,IAAI,CAC3B,CAAC;IACF,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,IAAI,gBAAgB,CAAC;IAE1D,GAAG,CAAC,GAAG,CAAC,SAAS,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE;QAC3C,IAAI,MAAM,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC;YACH,MAAM,MAAM,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;YACvC,MAAM,GAAG,IAAI,CAAC;QAChB,CAAC;QAAC,MAAM,CAAC;YACP,SAAS;QACX,CAAC;QACD,MAAM,iBAAiB,GAAG,oBAAoB,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAC5D,MAAM,YAAY,GAAG,iBAAiB,CAAC,CAAC,CAAC,UAAU,iBAAiB,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC;QAC3F,MAAM,KAAK,CAAC,IAAI,CAAC;YACf,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,UAAU;YAClC,MAAM;YACN,YAAY;YACZ,iBAAiB,EAAE,iBAAiB,IAAI,SAAS;SAClD,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,2 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export declare function registerRefreshRoutes(app: FastifyInstance): void;
@@ -0,0 +1,8 @@
1
+ export function registerRefreshRoutes(app) {
2
+ app.post("/refresh", async (_request, reply) => {
3
+ await reply.send({
4
+ message: "Refresh triggered. Run mem scaffold or mem update from CLI to re-index.",
5
+ });
6
+ });
7
+ }
8
+ //# sourceMappingURL=refresh.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"refresh.js","sourceRoot":"","sources":["../../src/api/refresh.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,qBAAqB,CAAC,GAAoB;IACxD,GAAG,CAAC,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE;QAC7C,MAAM,KAAK,CAAC,IAAI,CAAC;YACf,OAAO,EAAE,yEAAyE;SACnF,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,28 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export interface SearchBody {
3
+ query: string;
4
+ limit?: number;
5
+ }
6
+ export interface SearchResponse {
7
+ systemMaps: Array<{
8
+ id: string;
9
+ text: string;
10
+ path?: string;
11
+ }>;
12
+ moduleSummaries: Array<{
13
+ id: string;
14
+ text: string;
15
+ path?: string;
16
+ }>;
17
+ chunks: Array<{
18
+ id: string;
19
+ text: string;
20
+ path?: string;
21
+ score?: number;
22
+ }>;
23
+ }
24
+ export declare function registerSearchRoutes(app: FastifyInstance, options: {
25
+ qdrantHost?: string;
26
+ qdrantPort?: number;
27
+ collection?: string;
28
+ }): void;
@@ -0,0 +1,36 @@
1
+ import { createQdrantClient } from "../qdrant/client.js";
2
+ import { embedText } from "../embedding/stub.js";
3
+ import { searchPoints, searchPointsKinds } from "../qdrant/client.js";
4
+ export function registerSearchRoutes(app, options) {
5
+ const client = createQdrantClient(options.qdrantHost ?? "127.0.0.1", options.qdrantPort ?? 6333);
6
+ const collection = options.collection ?? "project_memory";
7
+ app.post("/search", async (request, reply) => {
8
+ const { query, limit = 20 } = request.body ?? {};
9
+ const vector = embedText(query ?? "");
10
+ const [systemMaps, moduleSummaries, chunks] = await Promise.all([
11
+ searchPoints(client, collection, vector, 5, "system_map"),
12
+ searchPoints(client, collection, vector, 10, "module_summary"),
13
+ searchPointsKinds(client, collection, vector, limit, ["chunk", "entrypoint"]),
14
+ ]);
15
+ const response = {
16
+ systemMaps: systemMaps.map((r) => ({
17
+ id: r.id,
18
+ text: r.payload.text ?? "",
19
+ path: r.payload.path,
20
+ })),
21
+ moduleSummaries: moduleSummaries.map((r) => ({
22
+ id: r.id,
23
+ text: r.payload.text ?? "",
24
+ path: r.payload.path,
25
+ })),
26
+ chunks: chunks.map((r) => ({
27
+ id: r.id,
28
+ text: r.payload.text ?? "",
29
+ path: r.payload.path,
30
+ score: r.score,
31
+ })),
32
+ };
33
+ await reply.send(response);
34
+ });
35
+ }
36
+ //# sourceMappingURL=search.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search.js","sourceRoot":"","sources":["../../src/api/search.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAatE,MAAM,UAAU,oBAAoB,CAClC,GAAoB,EACpB,OAA0E;IAE1E,MAAM,MAAM,GAAG,kBAAkB,CAC/B,OAAO,CAAC,UAAU,IAAI,WAAW,EACjC,OAAO,CAAC,UAAU,IAAI,IAAI,CAC3B,CAAC;IACF,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,IAAI,gBAAgB,CAAC;IAE1D,GAAG,CAAC,IAAI,CAAuB,SAAS,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QACjE,MAAM,EAAE,KAAK,EAAE,KAAK,GAAG,EAAE,EAAE,GAAG,OAAO,CAAC,IAAI,IAAI,EAAE,CAAC;QACjD,MAAM,MAAM,GAAG,SAAS,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC;QAEtC,MAAM,CAAC,UAAU,EAAE,eAAe,EAAE,MAAM,CAAC,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;YAC9D,YAAY,CAAC,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,CAAC,EAAE,YAAY,CAAC;YACzD,YAAY,CAAC,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,EAAE,EAAE,gBAAgB,CAAC;YAC9D,iBAAiB,CAAC,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;SAC9E,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAmB;YAC/B,UAAU,EAAE,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACjC,EAAE,EAAE,CAAC,CAAC,EAAE;gBACR,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,IAAI,EAAE;gBAC1B,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI;aACrB,CAAC,CAAC;YACH,eAAe,EAAE,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBAC3C,EAAE,EAAE,CAAC,CAAC,EAAE;gBACR,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,IAAI,EAAE;gBAC1B,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI;aACrB,CAAC,CAAC;YACH,MAAM,EAAE,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACzB,EAAE,EAAE,CAAC,CAAC,EAAE;gBACR,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,IAAI,EAAE;gBAC1B,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI;gBACpB,KAAK,EAAE,CAAC,CAAC,KAAK;aACf,CAAC,CAAC;SACJ,CAAC;QACF,MAAM,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC7B,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,6 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export declare function registerSystemRoutes(app: FastifyInstance, options: {
3
+ qdrantHost?: string;
4
+ qdrantPort?: number;
5
+ collection?: string;
6
+ }): void;
@@ -0,0 +1,17 @@
1
+ import { createQdrantClient } from "../qdrant/client.js";
2
+ import { searchPoints } from "../qdrant/client.js";
3
+ export function registerSystemRoutes(app, options) {
4
+ const client = createQdrantClient(options.qdrantHost ?? "127.0.0.1", options.qdrantPort ?? 6333);
5
+ const collection = options.collection ?? "project_memory";
6
+ app.get("/system/overview", async (_request, reply) => {
7
+ const vector = new Array(384).fill(0).map((_, i) => Math.sin(i * 0.1) * 0.5 + 0.5);
8
+ const results = await searchPoints(client, collection, vector, 10, "system_map");
9
+ const overview = results.map((r) => ({
10
+ id: r.id,
11
+ text: r.payload.text ?? "",
12
+ path: r.payload.path,
13
+ }));
14
+ await reply.send({ overview });
15
+ });
16
+ }
17
+ //# sourceMappingURL=system.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"system.js","sourceRoot":"","sources":["../../src/api/system.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAEnD,MAAM,UAAU,oBAAoB,CAClC,GAAoB,EACpB,OAA0E;IAE1E,MAAM,MAAM,GAAG,kBAAkB,CAC/B,OAAO,CAAC,UAAU,IAAI,WAAW,EACjC,OAAO,CAAC,UAAU,IAAI,IAAI,CAC3B,CAAC;IACF,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,IAAI,gBAAgB,CAAC;IAE1D,GAAG,CAAC,GAAG,CAAC,kBAAkB,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE;QACpD,MAAM,MAAM,GAAG,IAAI,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;QACnF,MAAM,OAAO,GAAG,MAAM,YAAY,CAAC,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,EAAE,EAAE,YAAY,CAAC,CAAC;QACjF,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACnC,EAAE,EAAE,CAAC,CAAC,EAAE;YACR,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,IAAI,EAAE;YAC1B,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI;SACrB,CAAC,CAAC,CAAC;QACJ,MAAM,KAAK,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Stub embedder: deterministic pseudo-vector from text (same text -> same vector).
3
+ * V1 local default; replace with real model (e.g. transformers.js) later.
4
+ */
5
+ export declare function embedText(text: string): number[];
6
+ export declare function embedBatch(texts: string[]): number[][];
7
+ export declare function getVectorSize(): number;
@@ -0,0 +1,30 @@
1
+ const VECTOR_SIZE = 384;
2
+ function simpleHash(str) {
3
+ let h = 0;
4
+ for (let i = 0; i < str.length; i++) {
5
+ h = (h * 31 + str.charCodeAt(i)) >>> 0;
6
+ }
7
+ return h;
8
+ }
9
+ /**
10
+ * Stub embedder: deterministic pseudo-vector from text (same text -> same vector).
11
+ * V1 local default; replace with real model (e.g. transformers.js) later.
12
+ */
13
+ export function embedText(text) {
14
+ const vec = [];
15
+ const words = text.split(/\s+/).filter(Boolean);
16
+ for (let i = 0; i < VECTOR_SIZE; i++) {
17
+ const seed = words[i % words.length]
18
+ ? simpleHash(words[i % words.length]) + i * 31
19
+ : i;
20
+ vec.push(Math.sin(seed) * 0.5 + 0.5);
21
+ }
22
+ return vec;
23
+ }
24
+ export function embedBatch(texts) {
25
+ return texts.map(embedText);
26
+ }
27
+ export function getVectorSize() {
28
+ return VECTOR_SIZE;
29
+ }
30
+ //# sourceMappingURL=stub.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stub.js","sourceRoot":"","sources":["../../src/embedding/stub.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,GAAG,GAAG,CAAC;AAExB,SAAS,UAAU,CAAC,GAAW;IAC7B,IAAI,CAAC,GAAG,CAAC,CAAC;IACV,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACpC,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;IACzC,CAAC;IACD,OAAO,CAAC,CAAC;AACX,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,SAAS,CAAC,IAAY;IACpC,MAAM,GAAG,GAAa,EAAE,CAAC;IACzB,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IAChD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,WAAW,EAAE,CAAC,EAAE,EAAE,CAAC;QACrC,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,GAAG,KAAK,CAAC,MAAM,CAAC;YAClC,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,GAAG,EAAE;YAC9C,CAAC,CAAC,CAAC,CAAC;QACN,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;IACvC,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,KAAe;IACxC,OAAO,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;AAC9B,CAAC;AAED,MAAM,UAAU,aAAa;IAC3B,OAAO,WAAW,CAAC;AACrB,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,25 @@
1
+ import { strict as assert } from "node:assert";
2
+ import { describe, it } from "node:test";
3
+ import { embedText, embedBatch, getVectorSize } from "./stub.js";
4
+ describe("embedding stub (server)", () => {
5
+ it("getVectorSize returns 384", () => {
6
+ assert.strictEqual(getVectorSize(), 384);
7
+ });
8
+ it("embedText returns array of length 384", () => {
9
+ const vec = embedText("hello world");
10
+ assert.strictEqual(vec.length, 384);
11
+ assert.ok(vec.every((x) => typeof x === "number" && x >= 0 && x <= 1));
12
+ });
13
+ it("embedText is deterministic for same input", () => {
14
+ const a = embedText("same text");
15
+ const b = embedText("same text");
16
+ assert.deepStrictEqual(a, b);
17
+ });
18
+ it("embedBatch returns one vector per text", () => {
19
+ const batch = embedBatch(["a", "b"]);
20
+ assert.strictEqual(batch.length, 2);
21
+ assert.strictEqual(batch[0].length, 384);
22
+ assert.strictEqual(batch[1].length, 384);
23
+ });
24
+ });
25
+ //# sourceMappingURL=stub.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stub.test.js","sourceRoot":"","sources":["../../src/embedding/stub.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,IAAI,MAAM,EAAE,MAAM,aAAa,CAAC;AAC/C,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,WAAW,CAAC;AACzC,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAEjE,QAAQ,CAAC,yBAAyB,EAAE,GAAG,EAAE;IACvC,EAAE,CAAC,2BAA2B,EAAE,GAAG,EAAE;QACnC,MAAM,CAAC,WAAW,CAAC,aAAa,EAAE,EAAE,GAAG,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;QAC/C,MAAM,GAAG,GAAG,SAAS,CAAC,aAAa,CAAC,CAAC;QACrC,MAAM,CAAC,WAAW,CAAC,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QACpC,MAAM,CAAC,EAAE,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,QAAQ,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACzE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;QACnD,MAAM,CAAC,GAAG,SAAS,CAAC,WAAW,CAAC,CAAC;QACjC,MAAM,CAAC,GAAG,SAAS,CAAC,WAAW,CAAC,CAAC;QACjC,MAAM,CAAC,eAAe,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAC/B,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wCAAwC,EAAE,GAAG,EAAE;QAChD,MAAM,KAAK,GAAG,UAAU,CAAC,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC,CAAC;QACrC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QACpC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QACzC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,26 @@
1
+ import Fastify from "fastify";
2
+ import { registerHealthRoutes } from "./api/health.js";
3
+ import { registerSearchRoutes } from "./api/search.js";
4
+ import { registerExplainRoutes } from "./api/explain.js";
5
+ import { registerSystemRoutes } from "./api/system.js";
6
+ import { registerRefreshRoutes } from "./api/refresh.js";
7
+ const PORT = Number(process.env.MEM_SERVER_PORT) || 31415;
8
+ const CWD = process.env.MEM_CWD || process.cwd();
9
+ const QDRANT_HOST = process.env.MEM_QDRANT_HOST || "127.0.0.1";
10
+ const QDRANT_PORT = Number(process.env.MEM_QDRANT_PORT) || 6333;
11
+ const COLLECTION = process.env.MEM_COLLECTION || "project_memory";
12
+ async function main() {
13
+ const app = Fastify({ logger: true });
14
+ registerHealthRoutes(app, { cwd: CWD, qdrantHost: QDRANT_HOST, qdrantPort: QDRANT_PORT, collection: COLLECTION });
15
+ registerSearchRoutes(app, { qdrantHost: QDRANT_HOST, qdrantPort: QDRANT_PORT, collection: COLLECTION });
16
+ registerExplainRoutes(app, { qdrantHost: QDRANT_HOST, qdrantPort: QDRANT_PORT, collection: COLLECTION });
17
+ registerSystemRoutes(app, { qdrantHost: QDRANT_HOST, qdrantPort: QDRANT_PORT, collection: COLLECTION });
18
+ registerRefreshRoutes(app);
19
+ await app.listen({ port: PORT, host: "127.0.0.1" });
20
+ console.log(`Memory server at http://127.0.0.1:${PORT}`);
21
+ }
22
+ main().catch((err) => {
23
+ console.error(err);
24
+ process.exit(1);
25
+ });
26
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,SAAS,CAAC;AAC9B,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AACvD,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AACvD,OAAO,EAAE,qBAAqB,EAAE,MAAM,kBAAkB,CAAC;AACzD,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AACvD,OAAO,EAAE,qBAAqB,EAAE,MAAM,kBAAkB,CAAC;AAEzD,MAAM,IAAI,GAAG,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,IAAI,KAAK,CAAC;AAC1D,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;AACjD,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,CAAC,eAAe,IAAI,WAAW,CAAC;AAC/D,MAAM,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,IAAI,IAAI,CAAC;AAChE,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,gBAAgB,CAAC;AAElE,KAAK,UAAU,IAAI;IACjB,MAAM,GAAG,GAAG,OAAO,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;IACtC,oBAAoB,CAAC,GAAG,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC,CAAC;IAClH,oBAAoB,CAAC,GAAG,EAAE,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC,CAAC;IACxG,qBAAqB,CAAC,GAAG,EAAE,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC,CAAC;IACzG,oBAAoB,CAAC,GAAG,EAAE,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,WAAW,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC,CAAC;IACxG,qBAAqB,CAAC,GAAG,CAAC,CAAC;IAE3B,MAAM,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC;IACpD,OAAO,CAAC,GAAG,CAAC,qCAAqC,IAAI,EAAE,CAAC,CAAC;AAC3D,CAAC;AAED,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;IACnB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACnB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
@@ -0,0 +1,24 @@
1
+ import { QdrantClient } from "@qdrant/js-client-rest";
2
+ export declare function createQdrantClient(host?: string, port?: number): QdrantClient;
3
+ export declare function ensureCollection(client: QdrantClient, collectionName?: string): Promise<void>;
4
+ export interface PointPayload {
5
+ kind: string;
6
+ path?: string;
7
+ text: string;
8
+ [key: string]: unknown;
9
+ }
10
+ export declare function upsertPoints(client: QdrantClient, collectionName: string, points: {
11
+ id: string;
12
+ vector: number[];
13
+ payload: PointPayload;
14
+ }[]): Promise<void>;
15
+ export declare function searchPoints(client: QdrantClient, collectionName: string, vector: number[], limit?: number, filterKind?: string): Promise<{
16
+ id: string;
17
+ score: number;
18
+ payload: PointPayload;
19
+ }[]>;
20
+ export declare function searchPointsKinds(client: QdrantClient, collectionName: string, vector: number[], limit: number, kinds: string[]): Promise<{
21
+ id: string;
22
+ score: number;
23
+ payload: PointPayload;
24
+ }[]>;
@@ -0,0 +1,55 @@
1
+ import { QdrantClient } from "@qdrant/js-client-rest";
2
+ import { getVectorSize } from "../embedding/stub.js";
3
+ const DEFAULT_COLLECTION = "project_memory";
4
+ export function createQdrantClient(host = "127.0.0.1", port = 6333) {
5
+ return new QdrantClient({ url: `http://${host}:${port}` });
6
+ }
7
+ export async function ensureCollection(client, collectionName = DEFAULT_COLLECTION) {
8
+ const size = getVectorSize();
9
+ try {
10
+ await client.getCollection(collectionName);
11
+ }
12
+ catch {
13
+ await client.createCollection(collectionName, {
14
+ vectors: { size, distance: "Cosine" },
15
+ });
16
+ }
17
+ }
18
+ export async function upsertPoints(client, collectionName, points) {
19
+ if (points.length === 0)
20
+ return;
21
+ await client.upsert(collectionName, {
22
+ wait: true,
23
+ points: points.map((p) => ({
24
+ id: p.id,
25
+ vector: p.vector,
26
+ payload: p.payload,
27
+ })),
28
+ });
29
+ }
30
+ export async function searchPoints(client, collectionName, vector, limit = 10, filterKind) {
31
+ const filter = filterKind
32
+ ? { must: [{ key: "kind", match: { value: filterKind } }] }
33
+ : undefined;
34
+ return doSearch(client, collectionName, vector, limit, filter);
35
+ }
36
+ export async function searchPointsKinds(client, collectionName, vector, limit, kinds) {
37
+ const filter = kinds.length > 0
38
+ ? { should: kinds.map((k) => ({ key: "kind", match: { value: k } })) }
39
+ : undefined;
40
+ return doSearch(client, collectionName, vector, limit, filter);
41
+ }
42
+ async function doSearch(client, collectionName, vector, limit, filter) {
43
+ const result = await client.search(collectionName, {
44
+ vector,
45
+ limit,
46
+ with_payload: true,
47
+ filter,
48
+ });
49
+ return result.map((r) => ({
50
+ id: r.id ?? String(r.id),
51
+ score: r.score ?? 0,
52
+ payload: (r.payload ?? {}),
53
+ }));
54
+ }
55
+ //# sourceMappingURL=client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client.js","sourceRoot":"","sources":["../../src/qdrant/client.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,37 @@
1
+ {
2
+ "name": "@project-memory/server",
3
+ "version": "0.1.0",
4
+ "description": "Project Memory HTTP API — health, search, explain",
5
+ "type": "module",
6
+ "main": "dist/index.js",
7
+ "scripts": {
8
+ "build": "tsc",
9
+ "start": "node dist/index.js",
10
+ "dev": "node --watch dist/index.js",
11
+ "test": "npm run build && c8 node --test dist/",
12
+ "test:no-coverage": "npm run build && node --test dist/"
13
+ },
14
+ "c8": {
15
+ "exclude": ["**/*.test.js", "**/index.js"],
16
+ "include": ["dist/**/*.js"],
17
+ "check-coverage": true,
18
+ "lines": 85,
19
+ "functions": 85,
20
+ "branches": 85,
21
+ "statements": 85,
22
+ "reporter": ["text", "lcov"],
23
+ "reports-dir": "coverage"
24
+ },
25
+ "dependencies": {
26
+ "fastify": "^4.24.3",
27
+ "@qdrant/js-client-rest": "^1.9.0"
28
+ },
29
+ "devDependencies": {
30
+ "@types/node": "^20.10.0",
31
+ "c8": "^9.1.0",
32
+ "typescript": "^5.3.0"
33
+ },
34
+ "engines": {
35
+ "node": ">=18"
36
+ }
37
+ }
@@ -0,0 +1,24 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ import { createQdrantClient } from "../qdrant/client.js";
3
+ import { embedText } from "../embedding/stub.js";
4
+ import { searchPoints } from "../qdrant/client.js";
5
+
6
+ export function registerExplainRoutes(
7
+ app: FastifyInstance,
8
+ options: { qdrantHost?: string; qdrantPort?: number; collection?: string }
9
+ ): void {
10
+ const client = createQdrantClient(
11
+ options.qdrantHost ?? "127.0.0.1",
12
+ options.qdrantPort ?? 6333
13
+ );
14
+ const collection = options.collection ?? "project_memory";
15
+
16
+ app.get<{ Params: { symbol: string } }>("/explain/:symbol", async (request, reply) => {
17
+ const symbol = decodeURIComponent(request.params.symbol);
18
+ const vector = embedText(symbol);
19
+ const results = await searchPoints(client, collection, vector, 3);
20
+ const snippet = results[0]?.payload?.text ?? "";
21
+ const path = results[0]?.payload?.path;
22
+ await reply.send({ symbol, snippet, path, results: results.length });
23
+ });
24
+ }
@@ -0,0 +1,50 @@
1
+ import { readFileSync, existsSync } from "node:fs";
2
+ import { join } from "node:path";
3
+ import type { FastifyInstance } from "fastify";
4
+ import { createQdrantClient } from "../qdrant/client.js";
5
+
6
+ const DEEP_INDEX_STATE_FILE = ".mem/state/deep-index.json";
7
+
8
+ function getDeepIndexProgress(cwd: string): { percent: number; embedded: number; total: number } | null {
9
+ const path = join(cwd, DEEP_INDEX_STATE_FILE);
10
+ if (!existsSync(path)) return null;
11
+ try {
12
+ const raw = readFileSync(path, "utf8");
13
+ const data = JSON.parse(raw) as { embedded?: number; total?: number; embeddedIds?: string[] };
14
+ const embedded = data.embedded ?? (Array.isArray(data.embeddedIds) ? data.embeddedIds.length : 0);
15
+ const total = data.total ?? 0;
16
+ const percent = total > 0 ? Math.round((embedded / total) * 100) : 0;
17
+ return { percent, embedded, total };
18
+ } catch {
19
+ return null;
20
+ }
21
+ }
22
+
23
+ export function registerHealthRoutes(
24
+ app: FastifyInstance,
25
+ options: { cwd: string; qdrantHost?: string; qdrantPort?: number; collection?: string }
26
+ ): void {
27
+ const client = createQdrantClient(
28
+ options.qdrantHost ?? "127.0.0.1",
29
+ options.qdrantPort ?? 6333
30
+ );
31
+ const collection = options.collection ?? "project_memory";
32
+
33
+ app.get("/health", async (_request, reply) => {
34
+ let qdrant = false;
35
+ try {
36
+ await client.getCollection(collection);
37
+ qdrant = true;
38
+ } catch {
39
+ // ignore
40
+ }
41
+ const deepIndexProgress = getDeepIndexProgress(options.cwd);
42
+ const indexVersion = deepIndexProgress ? `phase1+${deepIndexProgress.embedded}` : "phase1";
43
+ await reply.send({
44
+ status: qdrant ? "ok" : "degraded",
45
+ qdrant,
46
+ indexVersion,
47
+ deepIndexProgress: deepIndexProgress ?? undefined,
48
+ });
49
+ });
50
+ }
@@ -0,0 +1,9 @@
1
+ import type { FastifyInstance } from "fastify";
2
+
3
+ export function registerRefreshRoutes(app: FastifyInstance): void {
4
+ app.post("/refresh", async (_request, reply) => {
5
+ await reply.send({
6
+ message: "Refresh triggered. Run mem scaffold or mem update from CLI to re-index.",
7
+ });
8
+ });
9
+ }
@@ -0,0 +1,57 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ import { createQdrantClient } from "../qdrant/client.js";
3
+ import { embedText } from "../embedding/stub.js";
4
+ import { searchPoints, searchPointsKinds } from "../qdrant/client.js";
5
+
6
+ export interface SearchBody {
7
+ query: string;
8
+ limit?: number;
9
+ }
10
+
11
+ export interface SearchResponse {
12
+ systemMaps: Array<{ id: string; text: string; path?: string }>;
13
+ moduleSummaries: Array<{ id: string; text: string; path?: string }>;
14
+ chunks: Array<{ id: string; text: string; path?: string; score?: number }>;
15
+ }
16
+
17
+ export function registerSearchRoutes(
18
+ app: FastifyInstance,
19
+ options: { qdrantHost?: string; qdrantPort?: number; collection?: string }
20
+ ): void {
21
+ const client = createQdrantClient(
22
+ options.qdrantHost ?? "127.0.0.1",
23
+ options.qdrantPort ?? 6333
24
+ );
25
+ const collection = options.collection ?? "project_memory";
26
+
27
+ app.post<{ Body: SearchBody }>("/search", async (request, reply) => {
28
+ const { query, limit = 20 } = request.body ?? {};
29
+ const vector = embedText(query ?? "");
30
+
31
+ const [systemMaps, moduleSummaries, chunks] = await Promise.all([
32
+ searchPoints(client, collection, vector, 5, "system_map"),
33
+ searchPoints(client, collection, vector, 10, "module_summary"),
34
+ searchPointsKinds(client, collection, vector, limit, ["chunk", "entrypoint"]),
35
+ ]);
36
+
37
+ const response: SearchResponse = {
38
+ systemMaps: systemMaps.map((r) => ({
39
+ id: r.id,
40
+ text: r.payload.text ?? "",
41
+ path: r.payload.path,
42
+ })),
43
+ moduleSummaries: moduleSummaries.map((r) => ({
44
+ id: r.id,
45
+ text: r.payload.text ?? "",
46
+ path: r.payload.path,
47
+ })),
48
+ chunks: chunks.map((r) => ({
49
+ id: r.id,
50
+ text: r.payload.text ?? "",
51
+ path: r.payload.path,
52
+ score: r.score,
53
+ })),
54
+ };
55
+ await reply.send(response);
56
+ });
57
+ }
@@ -0,0 +1,25 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ import { createQdrantClient } from "../qdrant/client.js";
3
+ import { searchPoints } from "../qdrant/client.js";
4
+
5
+ export function registerSystemRoutes(
6
+ app: FastifyInstance,
7
+ options: { qdrantHost?: string; qdrantPort?: number; collection?: string }
8
+ ): void {
9
+ const client = createQdrantClient(
10
+ options.qdrantHost ?? "127.0.0.1",
11
+ options.qdrantPort ?? 6333
12
+ );
13
+ const collection = options.collection ?? "project_memory";
14
+
15
+ app.get("/system/overview", async (_request, reply) => {
16
+ const vector = new Array(384).fill(0).map((_, i) => Math.sin(i * 0.1) * 0.5 + 0.5);
17
+ const results = await searchPoints(client, collection, vector, 10, "system_map");
18
+ const overview = results.map((r) => ({
19
+ id: r.id,
20
+ text: r.payload.text ?? "",
21
+ path: r.payload.path,
22
+ }));
23
+ await reply.send({ overview });
24
+ });
25
+ }