@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.
- package/.memignore.example +11 -0
- package/README.md +48 -0
- package/docker-compose.yml +17 -0
- package/package.json +36 -0
- package/packages/cli/bin/mem.js +6 -0
- package/packages/cli/coverage/lcov-report/base.css +224 -0
- package/packages/cli/coverage/lcov-report/block-navigation.js +87 -0
- package/packages/cli/coverage/lcov-report/chunking/chunker.ts.html +538 -0
- package/packages/cli/coverage/lcov-report/chunking/hash.ts.html +148 -0
- package/packages/cli/coverage/lcov-report/chunking/index.html +146 -0
- package/packages/cli/coverage/lcov-report/chunking/types.ts.html +214 -0
- package/packages/cli/coverage/lcov-report/config/index.html +131 -0
- package/packages/cli/coverage/lcov-report/config/load.ts.html +184 -0
- package/packages/cli/coverage/lcov-report/config/types.ts.html +232 -0
- package/packages/cli/coverage/lcov-report/embedding/index.html +116 -0
- package/packages/cli/coverage/lcov-report/embedding/stub.ts.html +181 -0
- package/packages/cli/coverage/lcov-report/favicon.png +0 -0
- package/packages/cli/coverage/lcov-report/index.html +161 -0
- package/packages/cli/coverage/lcov-report/prettify.css +1 -0
- package/packages/cli/coverage/lcov-report/prettify.js +2 -0
- package/packages/cli/coverage/lcov-report/scope/allowlist.ts.html +199 -0
- package/packages/cli/coverage/lcov-report/scope/ignore.ts.html +343 -0
- package/packages/cli/coverage/lcov-report/scope/index.html +131 -0
- package/packages/cli/coverage/lcov-report/sort-arrow-sprite.png +0 -0
- package/packages/cli/coverage/lcov-report/sorter.js +210 -0
- package/packages/cli/coverage/lcov.info +669 -0
- package/packages/cli/coverage/tmp/coverage-25917-1770055893226-0.json +1 -0
- package/packages/cli/coverage/tmp/coverage-25918-1770055893272-0.json +1 -0
- package/packages/cli/coverage/tmp/coverage-25919-1770055893273-0.json +1 -0
- package/packages/cli/coverage/tmp/coverage-25920-1770055893271-0.json +1 -0
- package/packages/cli/coverage/tmp/coverage-25921-1770055893279-0.json +1 -0
- package/packages/cli/coverage/tmp/coverage-25922-1770055893272-0.json +1 -0
- package/packages/cli/coverage/tmp/coverage-25923-1770055893275-0.json +1 -0
- package/packages/cli/coverage/tmp/coverage-25924-1770055893294-0.json +1 -0
- package/packages/cli/coverage/tmp/coverage-25925-1770055893290-0.json +1 -0
- package/packages/cli/dist/chunking/chunker.d.ts +2 -0
- package/packages/cli/dist/chunking/chunker.js +142 -0
- package/packages/cli/dist/chunking/chunker.js.map +1 -0
- package/packages/cli/dist/chunking/chunker.test.d.ts +1 -0
- package/packages/cli/dist/chunking/chunker.test.js +50 -0
- package/packages/cli/dist/chunking/chunker.test.js.map +1 -0
- package/packages/cli/dist/chunking/hash.d.ts +3 -0
- package/packages/cli/dist/chunking/hash.js +17 -0
- package/packages/cli/dist/chunking/hash.js.map +1 -0
- package/packages/cli/dist/chunking/hash.test.d.ts +1 -0
- package/packages/cli/dist/chunking/hash.test.js +36 -0
- package/packages/cli/dist/chunking/hash.test.js.map +1 -0
- package/packages/cli/dist/chunking/types.d.ts +19 -0
- package/packages/cli/dist/chunking/types.js +24 -0
- package/packages/cli/dist/chunking/types.js.map +1 -0
- package/packages/cli/dist/chunking/types.test.d.ts +1 -0
- package/packages/cli/dist/chunking/types.test.js +25 -0
- package/packages/cli/dist/chunking/types.test.js.map +1 -0
- package/packages/cli/dist/cli/index.d.ts +1 -0
- package/packages/cli/dist/cli/index.js +67 -0
- package/packages/cli/dist/cli/index.js.map +1 -0
- package/packages/cli/dist/commands/deep-index.d.ts +1 -0
- package/packages/cli/dist/commands/deep-index.js +17 -0
- package/packages/cli/dist/commands/deep-index.js.map +1 -0
- package/packages/cli/dist/commands/doctor.d.ts +1 -0
- package/packages/cli/dist/commands/doctor.js +27 -0
- package/packages/cli/dist/commands/doctor.js.map +1 -0
- package/packages/cli/dist/commands/down.d.ts +1 -0
- package/packages/cli/dist/commands/down.js +13 -0
- package/packages/cli/dist/commands/down.js.map +1 -0
- package/packages/cli/dist/commands/explain.d.ts +1 -0
- package/packages/cli/dist/commands/explain.js +23 -0
- package/packages/cli/dist/commands/explain.js.map +1 -0
- package/packages/cli/dist/commands/init.d.ts +1 -0
- package/packages/cli/dist/commands/init.js +35 -0
- package/packages/cli/dist/commands/init.js.map +1 -0
- package/packages/cli/dist/commands/query.d.ts +1 -0
- package/packages/cli/dist/commands/query.js +44 -0
- package/packages/cli/dist/commands/query.js.map +1 -0
- package/packages/cli/dist/commands/reset.d.ts +3 -0
- package/packages/cli/dist/commands/reset.js +28 -0
- package/packages/cli/dist/commands/reset.js.map +1 -0
- package/packages/cli/dist/commands/scaffold.d.ts +1 -0
- package/packages/cli/dist/commands/scaffold.js +52 -0
- package/packages/cli/dist/commands/scaffold.js.map +1 -0
- package/packages/cli/dist/commands/update.d.ts +5 -0
- package/packages/cli/dist/commands/update.js +23 -0
- package/packages/cli/dist/commands/update.js.map +1 -0
- package/packages/cli/dist/commands/watch.d.ts +1 -0
- package/packages/cli/dist/commands/watch.js +46 -0
- package/packages/cli/dist/commands/watch.js.map +1 -0
- package/packages/cli/dist/config/defaults.d.ts +2 -0
- package/packages/cli/dist/config/defaults.js +47 -0
- package/packages/cli/dist/config/defaults.js.map +1 -0
- package/packages/cli/dist/config/load.d.ts +4 -0
- package/packages/cli/dist/config/load.js +29 -0
- package/packages/cli/dist/config/load.js.map +1 -0
- package/packages/cli/dist/config/load.test.d.ts +1 -0
- package/packages/cli/dist/config/load.test.js +51 -0
- package/packages/cli/dist/config/load.test.js.map +1 -0
- package/packages/cli/dist/config/types.d.ts +32 -0
- package/packages/cli/dist/config/types.js +24 -0
- package/packages/cli/dist/config/types.js.map +1 -0
- package/packages/cli/dist/config/types.test.d.ts +1 -0
- package/packages/cli/dist/config/types.test.js +31 -0
- package/packages/cli/dist/config/types.test.js.map +1 -0
- package/packages/cli/dist/docker/compose.d.ts +5 -0
- package/packages/cli/dist/docker/compose.js +75 -0
- package/packages/cli/dist/docker/compose.js.map +1 -0
- package/packages/cli/dist/embedding/stub.d.ts +6 -0
- package/packages/cli/dist/embedding/stub.js +29 -0
- package/packages/cli/dist/embedding/stub.js.map +1 -0
- package/packages/cli/dist/embedding/stub.test.d.ts +1 -0
- package/packages/cli/dist/embedding/stub.test.js +31 -0
- package/packages/cli/dist/embedding/stub.test.js.map +1 -0
- package/packages/cli/dist/phase1/pipeline.d.ts +1 -0
- package/packages/cli/dist/phase1/pipeline.js +145 -0
- package/packages/cli/dist/phase1/pipeline.js.map +1 -0
- package/packages/cli/dist/phase2/deep-index.d.ts +1 -0
- package/packages/cli/dist/phase2/deep-index.js +105 -0
- package/packages/cli/dist/phase2/deep-index.js.map +1 -0
- package/packages/cli/dist/qdrant/upsert.d.ts +14 -0
- package/packages/cli/dist/qdrant/upsert.js +30 -0
- package/packages/cli/dist/qdrant/upsert.js.map +1 -0
- package/packages/cli/dist/scope/allowlist.d.ts +11 -0
- package/packages/cli/dist/scope/allowlist.js +36 -0
- package/packages/cli/dist/scope/allowlist.js.map +1 -0
- package/packages/cli/dist/scope/allowlist.test.d.ts +1 -0
- package/packages/cli/dist/scope/allowlist.test.js +23 -0
- package/packages/cli/dist/scope/allowlist.test.js.map +1 -0
- package/packages/cli/dist/scope/ignore.d.ts +2 -0
- package/packages/cli/dist/scope/ignore.js +80 -0
- package/packages/cli/dist/scope/ignore.js.map +1 -0
- package/packages/cli/dist/scope/ignore.test.d.ts +1 -0
- package/packages/cli/dist/scope/ignore.test.js +71 -0
- package/packages/cli/dist/scope/ignore.test.js.map +1 -0
- package/packages/cli/package.json +46 -0
- package/packages/cli/src/chunking/chunker.test.ts +54 -0
- package/packages/cli/src/chunking/chunker.ts +151 -0
- package/packages/cli/src/chunking/hash.test.ts +41 -0
- package/packages/cli/src/chunking/hash.ts +21 -0
- package/packages/cli/src/chunking/types.test.ts +28 -0
- package/packages/cli/src/chunking/types.ts +43 -0
- package/packages/cli/src/cli/index.ts +79 -0
- package/packages/cli/src/commands/deep-index.ts +16 -0
- package/packages/cli/src/commands/doctor.ts +32 -0
- package/packages/cli/src/commands/down.ts +12 -0
- package/packages/cli/src/commands/explain.ts +22 -0
- package/packages/cli/src/commands/init.ts +37 -0
- package/packages/cli/src/commands/query.ts +49 -0
- package/packages/cli/src/commands/reset.ts +30 -0
- package/packages/cli/src/commands/scaffold.ts +55 -0
- package/packages/cli/src/commands/update.ts +22 -0
- package/packages/cli/src/commands/watch.ts +47 -0
- package/packages/cli/src/config/defaults.ts +49 -0
- package/packages/cli/src/config/load.test.ts +55 -0
- package/packages/cli/src/config/load.ts +33 -0
- package/packages/cli/src/config/types.test.ts +43 -0
- package/packages/cli/src/config/types.ts +49 -0
- package/packages/cli/src/docker/compose.ts +75 -0
- package/packages/cli/src/embedding/stub.test.ts +35 -0
- package/packages/cli/src/embedding/stub.ts +32 -0
- package/packages/cli/src/phase1/pipeline.ts +164 -0
- package/packages/cli/src/phase2/deep-index.ts +120 -0
- package/packages/cli/src/qdrant/upsert.ts +45 -0
- package/packages/cli/src/scope/allowlist.test.ts +25 -0
- package/packages/cli/src/scope/allowlist.ts +38 -0
- package/packages/cli/src/scope/ignore.test.ts +71 -0
- package/packages/cli/src/scope/ignore.ts +86 -0
- package/packages/cli/tsconfig.json +16 -0
- package/packages/server/coverage/lcov-report/base.css +224 -0
- package/packages/server/coverage/lcov-report/block-navigation.js +87 -0
- package/packages/server/coverage/lcov-report/favicon.png +0 -0
- package/packages/server/coverage/lcov-report/index.html +116 -0
- package/packages/server/coverage/lcov-report/prettify.css +1 -0
- package/packages/server/coverage/lcov-report/prettify.js +2 -0
- package/packages/server/coverage/lcov-report/sort-arrow-sprite.png +0 -0
- package/packages/server/coverage/lcov-report/sorter.js +210 -0
- package/packages/server/coverage/lcov-report/stub.ts.html +184 -0
- package/packages/server/coverage/lcov.info +57 -0
- package/packages/server/coverage/tmp/coverage-26012-1770055894042-0.json +1 -0
- package/packages/server/coverage/tmp/coverage-26013-1770055894077-0.json +1 -0
- package/packages/server/dist/api/explain.d.ts +6 -0
- package/packages/server/dist/api/explain.js +16 -0
- package/packages/server/dist/api/explain.js.map +1 -0
- package/packages/server/dist/api/health.d.ts +7 -0
- package/packages/server/dist/api/health.js +43 -0
- package/packages/server/dist/api/health.js.map +1 -0
- package/packages/server/dist/api/refresh.d.ts +2 -0
- package/packages/server/dist/api/refresh.js +8 -0
- package/packages/server/dist/api/refresh.js.map +1 -0
- package/packages/server/dist/api/search.d.ts +28 -0
- package/packages/server/dist/api/search.js +36 -0
- package/packages/server/dist/api/search.js.map +1 -0
- package/packages/server/dist/api/system.d.ts +6 -0
- package/packages/server/dist/api/system.js +17 -0
- package/packages/server/dist/api/system.js.map +1 -0
- package/packages/server/dist/embedding/stub.d.ts +7 -0
- package/packages/server/dist/embedding/stub.js +30 -0
- package/packages/server/dist/embedding/stub.js.map +1 -0
- package/packages/server/dist/embedding/stub.test.d.ts +1 -0
- package/packages/server/dist/embedding/stub.test.js +25 -0
- package/packages/server/dist/embedding/stub.test.js.map +1 -0
- package/packages/server/dist/index.d.ts +1 -0
- package/packages/server/dist/index.js +26 -0
- package/packages/server/dist/index.js.map +1 -0
- package/packages/server/dist/qdrant/client.d.ts +24 -0
- package/packages/server/dist/qdrant/client.js +55 -0
- package/packages/server/dist/qdrant/client.js.map +1 -0
- package/packages/server/package.json +37 -0
- package/packages/server/src/api/explain.ts +24 -0
- package/packages/server/src/api/health.ts +50 -0
- package/packages/server/src/api/refresh.ts +9 -0
- package/packages/server/src/api/search.ts +57 -0
- package/packages/server/src/api/system.ts +25 -0
- package/packages/server/src/embedding/stub.test.ts +28 -0
- package/packages/server/src/embedding/stub.ts +33 -0
- package/packages/server/src/index.ts +29 -0
- package/packages/server/src/qdrant/client.ts +92 -0
- package/packages/server/tsconfig.json +16 -0
|
@@ -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,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 @@
|
|
|
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,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
|
+
}
|