@testrelic/mcp 2.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 (204) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +198 -0
  3. package/dist/.tsbuildinfo +1 -0
  4. package/dist/cache/blob.d.ts +22 -0
  5. package/dist/cache/blob.d.ts.map +1 -0
  6. package/dist/cache/blob.js +92 -0
  7. package/dist/cache/blob.js.map +1 -0
  8. package/dist/cache/diff-reader.d.ts +29 -0
  9. package/dist/cache/diff-reader.d.ts.map +1 -0
  10. package/dist/cache/diff-reader.js +34 -0
  11. package/dist/cache/diff-reader.js.map +1 -0
  12. package/dist/cache/index.d.ts +57 -0
  13. package/dist/cache/index.d.ts.map +1 -0
  14. package/dist/cache/index.js +99 -0
  15. package/dist/cache/index.js.map +1 -0
  16. package/dist/cache/key.d.ts +15 -0
  17. package/dist/cache/key.d.ts.map +1 -0
  18. package/dist/cache/key.js +32 -0
  19. package/dist/cache/key.js.map +1 -0
  20. package/dist/cache/lru.d.ts +21 -0
  21. package/dist/cache/lru.d.ts.map +1 -0
  22. package/dist/cache/lru.js +32 -0
  23. package/dist/cache/lru.js.map +1 -0
  24. package/dist/cache/sqlite.d.ts +22 -0
  25. package/dist/cache/sqlite.d.ts.map +1 -0
  26. package/dist/cache/sqlite.js +102 -0
  27. package/dist/cache/sqlite.js.map +1 -0
  28. package/dist/cache/vector.d.ts +42 -0
  29. package/dist/cache/vector.d.ts.map +1 -0
  30. package/dist/cache/vector.js +187 -0
  31. package/dist/cache/vector.js.map +1 -0
  32. package/dist/cli.d.ts +3 -0
  33. package/dist/cli.d.ts.map +1 -0
  34. package/dist/cli.js +205 -0
  35. package/dist/cli.js.map +1 -0
  36. package/dist/clients/amplitude.d.ts +24 -0
  37. package/dist/clients/amplitude.d.ts.map +1 -0
  38. package/dist/clients/amplitude.js +15 -0
  39. package/dist/clients/amplitude.js.map +1 -0
  40. package/dist/clients/clickhouse.d.ts +10 -0
  41. package/dist/clients/clickhouse.d.ts.map +1 -0
  42. package/dist/clients/clickhouse.js +8 -0
  43. package/dist/clients/clickhouse.js.map +1 -0
  44. package/dist/clients/cloud.d.ts +347 -0
  45. package/dist/clients/cloud.d.ts.map +1 -0
  46. package/dist/clients/cloud.js +402 -0
  47. package/dist/clients/cloud.js.map +1 -0
  48. package/dist/clients/http.d.ts +40 -0
  49. package/dist/clients/http.d.ts.map +1 -0
  50. package/dist/clients/http.js +67 -0
  51. package/dist/clients/http.js.map +1 -0
  52. package/dist/clients/index.d.ts +38 -0
  53. package/dist/clients/index.d.ts.map +1 -0
  54. package/dist/clients/index.js +24 -0
  55. package/dist/clients/index.js.map +1 -0
  56. package/dist/clients/jira.d.ts +16 -0
  57. package/dist/clients/jira.d.ts.map +1 -0
  58. package/dist/clients/jira.js +11 -0
  59. package/dist/clients/jira.js.map +1 -0
  60. package/dist/clients/loki.d.ts +7 -0
  61. package/dist/clients/loki.d.ts.map +1 -0
  62. package/dist/clients/loki.js +8 -0
  63. package/dist/clients/loki.js.map +1 -0
  64. package/dist/clients/retry.d.ts +28 -0
  65. package/dist/clients/retry.d.ts.map +1 -0
  66. package/dist/clients/retry.js +79 -0
  67. package/dist/clients/retry.js.map +1 -0
  68. package/dist/clients/testrelic.d.ts +90 -0
  69. package/dist/clients/testrelic.d.ts.map +1 -0
  70. package/dist/clients/testrelic.js +68 -0
  71. package/dist/clients/testrelic.js.map +1 -0
  72. package/dist/config.d.ts +216 -0
  73. package/dist/config.d.ts.map +1 -0
  74. package/dist/config.js +233 -0
  75. package/dist/config.js.map +1 -0
  76. package/dist/context/code-map.d.ts +35 -0
  77. package/dist/context/code-map.d.ts.map +1 -0
  78. package/dist/context/code-map.js +187 -0
  79. package/dist/context/code-map.js.map +1 -0
  80. package/dist/context/correlator.d.ts +32 -0
  81. package/dist/context/correlator.d.ts.map +1 -0
  82. package/dist/context/correlator.js +106 -0
  83. package/dist/context/correlator.js.map +1 -0
  84. package/dist/context/coverage-map.d.ts +25 -0
  85. package/dist/context/coverage-map.d.ts.map +1 -0
  86. package/dist/context/coverage-map.js +44 -0
  87. package/dist/context/coverage-map.js.map +1 -0
  88. package/dist/context/index.d.ts +20 -0
  89. package/dist/context/index.d.ts.map +1 -0
  90. package/dist/context/index.js +18 -0
  91. package/dist/context/index.js.map +1 -0
  92. package/dist/context/journey-graph.d.ts +22 -0
  93. package/dist/context/journey-graph.d.ts.map +1 -0
  94. package/dist/context/journey-graph.js +38 -0
  95. package/dist/context/journey-graph.js.map +1 -0
  96. package/dist/context/signal-map.d.ts +26 -0
  97. package/dist/context/signal-map.d.ts.map +1 -0
  98. package/dist/context/signal-map.js +30 -0
  99. package/dist/context/signal-map.js.map +1 -0
  100. package/dist/elicit/ask.d.ts +28 -0
  101. package/dist/elicit/ask.d.ts.map +1 -0
  102. package/dist/elicit/ask.js +31 -0
  103. package/dist/elicit/ask.js.map +1 -0
  104. package/dist/elicit/zod-to-json.d.ts +7 -0
  105. package/dist/elicit/zod-to-json.d.ts.map +1 -0
  106. package/dist/elicit/zod-to-json.js +37 -0
  107. package/dist/elicit/zod-to-json.js.map +1 -0
  108. package/dist/errors.d.ts +59 -0
  109. package/dist/errors.d.ts.map +1 -0
  110. package/dist/errors.js +122 -0
  111. package/dist/errors.js.map +1 -0
  112. package/dist/index.d.ts +43 -0
  113. package/dist/index.d.ts.map +1 -0
  114. package/dist/index.js +127 -0
  115. package/dist/index.js.map +1 -0
  116. package/dist/logger.d.ts +6 -0
  117. package/dist/logger.d.ts.map +1 -0
  118. package/dist/logger.js +29 -0
  119. package/dist/logger.js.map +1 -0
  120. package/dist/prompts/index.d.ts +8 -0
  121. package/dist/prompts/index.d.ts.map +1 -0
  122. package/dist/prompts/index.js +86 -0
  123. package/dist/prompts/index.js.map +1 -0
  124. package/dist/registry/index.d.ts +70 -0
  125. package/dist/registry/index.d.ts.map +1 -0
  126. package/dist/registry/index.js +88 -0
  127. package/dist/registry/index.js.map +1 -0
  128. package/dist/registry/project.d.ts +20 -0
  129. package/dist/registry/project.d.ts.map +1 -0
  130. package/dist/registry/project.js +44 -0
  131. package/dist/registry/project.js.map +1 -0
  132. package/dist/resources/index.d.ts +13 -0
  133. package/dist/resources/index.d.ts.map +1 -0
  134. package/dist/resources/index.js +76 -0
  135. package/dist/resources/index.js.map +1 -0
  136. package/dist/sampling/bridge.d.ts +35 -0
  137. package/dist/sampling/bridge.d.ts.map +1 -0
  138. package/dist/sampling/bridge.js +39 -0
  139. package/dist/sampling/bridge.js.map +1 -0
  140. package/dist/telemetry/metrics.d.ts +29 -0
  141. package/dist/telemetry/metrics.d.ts.map +1 -0
  142. package/dist/telemetry/metrics.js +46 -0
  143. package/dist/telemetry/metrics.js.map +1 -0
  144. package/dist/telemetry/tokens.d.ts +16 -0
  145. package/dist/telemetry/tokens.d.ts.map +1 -0
  146. package/dist/telemetry/tokens.js +40 -0
  147. package/dist/telemetry/tokens.js.map +1 -0
  148. package/dist/tools/core/index.d.ts +8 -0
  149. package/dist/tools/core/index.d.ts.map +1 -0
  150. package/dist/tools/core/index.js +219 -0
  151. package/dist/tools/core/index.js.map +1 -0
  152. package/dist/tools/coverage/index.d.ts +9 -0
  153. package/dist/tools/coverage/index.d.ts.map +1 -0
  154. package/dist/tools/coverage/index.js +247 -0
  155. package/dist/tools/coverage/index.js.map +1 -0
  156. package/dist/tools/creation/index.d.ts +9 -0
  157. package/dist/tools/creation/index.d.ts.map +1 -0
  158. package/dist/tools/creation/index.js +357 -0
  159. package/dist/tools/creation/index.js.map +1 -0
  160. package/dist/tools/creation/templates.d.ts +17 -0
  161. package/dist/tools/creation/templates.d.ts.map +1 -0
  162. package/dist/tools/creation/templates.js +63 -0
  163. package/dist/tools/creation/templates.js.map +1 -0
  164. package/dist/tools/devtools/index.d.ts +9 -0
  165. package/dist/tools/devtools/index.d.ts.map +1 -0
  166. package/dist/tools/devtools/index.js +106 -0
  167. package/dist/tools/devtools/index.js.map +1 -0
  168. package/dist/tools/healing/index.d.ts +10 -0
  169. package/dist/tools/healing/index.d.ts.map +1 -0
  170. package/dist/tools/healing/index.js +190 -0
  171. package/dist/tools/healing/index.js.map +1 -0
  172. package/dist/tools/impact/index.d.ts +8 -0
  173. package/dist/tools/impact/index.d.ts.map +1 -0
  174. package/dist/tools/impact/index.js +215 -0
  175. package/dist/tools/impact/index.js.map +1 -0
  176. package/dist/tools/index.d.ts +9 -0
  177. package/dist/tools/index.d.ts.map +1 -0
  178. package/dist/tools/index.js +29 -0
  179. package/dist/tools/index.js.map +1 -0
  180. package/dist/tools/signals/index.d.ts +9 -0
  181. package/dist/tools/signals/index.d.ts.map +1 -0
  182. package/dist/tools/signals/index.js +100 -0
  183. package/dist/tools/signals/index.js.map +1 -0
  184. package/dist/tools/triage/index.d.ts +9 -0
  185. package/dist/tools/triage/index.d.ts.map +1 -0
  186. package/dist/tools/triage/index.js +366 -0
  187. package/dist/tools/triage/index.js.map +1 -0
  188. package/dist/transport/http.d.ts +16 -0
  189. package/dist/transport/http.d.ts.map +1 -0
  190. package/dist/transport/http.js +110 -0
  191. package/dist/transport/http.js.map +1 -0
  192. package/dist/transport/stdio.d.ts +6 -0
  193. package/dist/transport/stdio.d.ts.map +1 -0
  194. package/dist/transport/stdio.js +19 -0
  195. package/dist/transport/stdio.js.map +1 -0
  196. package/dist/types/index.d.ts +268 -0
  197. package/dist/types/index.d.ts.map +1 -0
  198. package/dist/types/index.js +6 -0
  199. package/dist/types/index.js.map +1 -0
  200. package/dist/version.d.ts +7 -0
  201. package/dist/version.d.ts.map +1 -0
  202. package/dist/version.js +7 -0
  203. package/dist/version.js.map +1 -0
  204. package/package.json +84 -0
@@ -0,0 +1,22 @@
1
+ /**
2
+ * L4: filesystem blob store. Keys by content SHA256 so duplicate payloads
3
+ * share one file. LRU-evicted by total size on disk.
4
+ */
5
+ export interface BlobLayerOptions {
6
+ cacheDir: string;
7
+ maxBytes?: number;
8
+ }
9
+ export declare class BlobLayer {
10
+ private readonly dir;
11
+ private readonly maxBytes;
12
+ constructor(opts: BlobLayerOptions);
13
+ write(content: string | Buffer): string;
14
+ read(sha: string): Buffer | null;
15
+ readText(sha: string): string | null;
16
+ has(sha: string): boolean;
17
+ private pathFor;
18
+ private maybeEvict;
19
+ private initDirsSync;
20
+ ensureShards(): void;
21
+ }
22
+ //# sourceMappingURL=blob.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"blob.d.ts","sourceRoot":"","sources":["../../src/cache/blob.ts"],"names":[],"mappings":"AAKA;;;GAGG;AAEH,MAAM,WAAW,gBAAgB;IAC/B,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,qBAAa,SAAS;IACpB,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAS;IAC7B,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAS;gBAEtB,IAAI,EAAE,gBAAgB;IAM3B,KAAK,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM;IAWvC,IAAI,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAMhC,QAAQ,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAKpC,GAAG,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAIhC,OAAO,CAAC,OAAO;IAIf,OAAO,CAAC,UAAU;IAqClB,OAAO,CAAC,YAAY;IAOb,YAAY,IAAI,IAAI;CAG5B"}
@@ -0,0 +1,92 @@
1
+ import { createHash } from "node:crypto";
2
+ import { existsSync, mkdirSync, readFileSync, readdirSync, statSync, unlinkSync, writeFileSync } from "node:fs";
3
+ import { join } from "node:path";
4
+ import { getLogger } from "../logger.js";
5
+ export class BlobLayer {
6
+ dir;
7
+ maxBytes;
8
+ constructor(opts) {
9
+ this.dir = join(opts.cacheDir, "blobs");
10
+ this.maxBytes = opts.maxBytes ?? 500 * 1024 * 1024;
11
+ mkdirSync(this.dir, { recursive: true });
12
+ }
13
+ write(content) {
14
+ const buf = typeof content === "string" ? Buffer.from(content, "utf-8") : content;
15
+ const sha = createHash("sha256").update(buf).digest("hex");
16
+ const p = this.pathFor(sha);
17
+ if (!existsSync(p)) {
18
+ writeFileSync(p, buf);
19
+ this.maybeEvict();
20
+ }
21
+ return sha;
22
+ }
23
+ read(sha) {
24
+ const p = this.pathFor(sha);
25
+ if (!existsSync(p))
26
+ return null;
27
+ return readFileSync(p);
28
+ }
29
+ readText(sha) {
30
+ const b = this.read(sha);
31
+ return b ? b.toString("utf-8") : null;
32
+ }
33
+ has(sha) {
34
+ return existsSync(this.pathFor(sha));
35
+ }
36
+ pathFor(sha) {
37
+ return join(this.dir, sha.slice(0, 2), `${sha}.blob`);
38
+ }
39
+ maybeEvict() {
40
+ try {
41
+ const files = [];
42
+ for (const sub of readdirSync(this.dir)) {
43
+ const subDir = join(this.dir, sub);
44
+ try {
45
+ if (!statSync(subDir).isDirectory())
46
+ continue;
47
+ }
48
+ catch {
49
+ continue;
50
+ }
51
+ for (const f of readdirSync(subDir)) {
52
+ const p = join(subDir, f);
53
+ try {
54
+ const st = statSync(p);
55
+ files.push({ path: p, size: st.size, atime: st.atimeMs });
56
+ }
57
+ catch {
58
+ // skip missing files
59
+ }
60
+ }
61
+ }
62
+ let total = files.reduce((a, f) => a + f.size, 0);
63
+ if (total <= this.maxBytes)
64
+ return;
65
+ files.sort((a, b) => a.atime - b.atime);
66
+ for (const f of files) {
67
+ if (total <= this.maxBytes * 0.9)
68
+ break;
69
+ try {
70
+ unlinkSync(f.path);
71
+ total -= f.size;
72
+ }
73
+ catch {
74
+ // ignore eviction errors
75
+ }
76
+ }
77
+ }
78
+ catch (err) {
79
+ getLogger().debug({ err }, "blob eviction failed");
80
+ }
81
+ }
82
+ initDirsSync() {
83
+ // Make sure shard directories exist on first use
84
+ for (let i = 0; i < 256; i++) {
85
+ mkdirSync(join(this.dir, i.toString(16).padStart(2, "0")), { recursive: true });
86
+ }
87
+ }
88
+ ensureShards() {
89
+ this.initDirsSync();
90
+ }
91
+ }
92
+ //# sourceMappingURL=blob.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"blob.js","sourceRoot":"","sources":["../../src/cache/blob.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,29 @@
1
+ /**
2
+ * 3-state reader: returns either the full payload, an "unchanged" stub, or a
3
+ * unified diff against the previously returned state. Keyed by caller-supplied
4
+ * resource id.
5
+ *
6
+ * Token savings: 80–95% on repeats of large, mostly-stable payloads
7
+ * (e.g. coverage reports, repo manifests, journey catalogs).
8
+ */
9
+ export type ReadState = {
10
+ state: "full";
11
+ content: string;
12
+ fingerprint: string;
13
+ } | {
14
+ state: "unchanged";
15
+ fingerprint: string;
16
+ } | {
17
+ state: "diff";
18
+ content: string;
19
+ fingerprint: string;
20
+ previousFingerprint: string;
21
+ };
22
+ export declare class DiffReader {
23
+ private readonly snapshots;
24
+ read(resourceId: string, content: string): ReadState;
25
+ forget(resourceId: string): void;
26
+ clear(): void;
27
+ size(): number;
28
+ }
29
+ //# sourceMappingURL=diff-reader.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"diff-reader.d.ts","sourceRoot":"","sources":["../../src/cache/diff-reader.ts"],"names":[],"mappings":"AAGA;;;;;;;GAOG;AAEH,MAAM,MAAM,SAAS,GACjB;IAAE,KAAK,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,WAAW,EAAE,MAAM,CAAA;CAAE,GACvD;IAAE,KAAK,EAAE,WAAW,CAAC;IAAC,WAAW,EAAE,MAAM,CAAA;CAAE,GAC3C;IAAE,KAAK,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,WAAW,EAAE,MAAM,CAAC;IAAC,mBAAmB,EAAE,MAAM,CAAA;CAAE,CAAC;AAQzF,qBAAa,UAAU;IACrB,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAqC;IAExD,IAAI,CAAC,UAAU,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,SAAS;IAuBpD,MAAM,CAAC,UAAU,EAAE,MAAM,GAAG,IAAI;IAIhC,KAAK,IAAI,IAAI;IAIb,IAAI,IAAI,MAAM;CAGtB"}
@@ -0,0 +1,34 @@
1
+ import { createPatch } from "diff";
2
+ import { simHash } from "./key.js";
3
+ export class DiffReader {
4
+ snapshots = new Map();
5
+ read(resourceId, content) {
6
+ const fingerprint = simHash(content);
7
+ const prev = this.snapshots.get(resourceId);
8
+ if (!prev) {
9
+ this.snapshots.set(resourceId, { fingerprint, content, ts: Date.now() });
10
+ return { state: "full", content, fingerprint };
11
+ }
12
+ if (prev.fingerprint === fingerprint) {
13
+ return { state: "unchanged", fingerprint };
14
+ }
15
+ const patch = createPatch(resourceId, prev.content, content, "previous", "current");
16
+ this.snapshots.set(resourceId, { fingerprint, content, ts: Date.now() });
17
+ return {
18
+ state: "diff",
19
+ content: patch,
20
+ fingerprint,
21
+ previousFingerprint: prev.fingerprint,
22
+ };
23
+ }
24
+ forget(resourceId) {
25
+ this.snapshots.delete(resourceId);
26
+ }
27
+ clear() {
28
+ this.snapshots.clear();
29
+ }
30
+ size() {
31
+ return this.snapshots.size;
32
+ }
33
+ }
34
+ //# sourceMappingURL=diff-reader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"diff-reader.js","sourceRoot":"","sources":["../../src/cache/diff-reader.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,MAAM,CAAC;AACnC,OAAO,EAAE,OAAO,EAAE,MAAM,UAAU,CAAC;AAsBnC,MAAM,OAAO,UAAU;IACJ,SAAS,GAAG,IAAI,GAAG,EAA0B,CAAC;IAExD,IAAI,CAAC,UAAkB,EAAE,OAAe;QAC7C,MAAM,WAAW,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC;QACrC,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAE5C,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,UAAU,EAAE,EAAE,WAAW,EAAE,OAAO,EAAE,EAAE,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;YACzE,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,WAAW,EAAE,CAAC;QACjD,CAAC;QAED,IAAI,IAAI,CAAC,WAAW,KAAK,WAAW,EAAE,CAAC;YACrC,OAAO,EAAE,KAAK,EAAE,WAAW,EAAE,WAAW,EAAE,CAAC;QAC7C,CAAC;QAED,MAAM,KAAK,GAAG,WAAW,CAAC,UAAU,EAAE,IAAI,CAAC,OAAO,EAAE,OAAO,EAAE,UAAU,EAAE,SAAS,CAAC,CAAC;QACpF,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,UAAU,EAAE,EAAE,WAAW,EAAE,OAAO,EAAE,EAAE,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;QACzE,OAAO;YACL,KAAK,EAAE,MAAM;YACb,OAAO,EAAE,KAAK;YACd,WAAW;YACX,mBAAmB,EAAE,IAAI,CAAC,WAAW;SACtC,CAAC;IACJ,CAAC;IAEM,MAAM,CAAC,UAAkB;QAC9B,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;IACpC,CAAC;IAEM,KAAK;QACV,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,CAAC;IACzB,CAAC;IAEM,IAAI;QACT,OAAO,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;IAC7B,CAAC;CACF"}
@@ -0,0 +1,57 @@
1
+ import type { ResolvedConfig } from "../config.js";
2
+ import { BlobLayer } from "./blob.js";
3
+ import { DiffReader } from "./diff-reader.js";
4
+ import { LruLayer } from "./lru.js";
5
+ import { SqliteLayer } from "./sqlite.js";
6
+ import { VectorStore } from "./vector.js";
7
+ /**
8
+ * Unified cache facade. Tools talk to this; it in turn decides which tier to
9
+ * consult. For the token-reduction strategy:
10
+ * 1. L1 LRU (ms) — burst repeats
11
+ * 2. L2 SQLite (ms) — warm hits across sessions
12
+ * 3. L3 HNSW vector (ms) — semantic neighbors
13
+ * 4. L4 blob (ms) — large artifacts referenced by `cache_key`
14
+ */
15
+ export interface CacheLookup<T> {
16
+ value: T;
17
+ layer: "L1" | "L2" | "L3" | "L4";
18
+ }
19
+ export interface CacheStats {
20
+ l1Hits: number;
21
+ l1Misses: number;
22
+ l2Hits: number;
23
+ l2Misses: number;
24
+ l3Hits: number;
25
+ l3Misses: number;
26
+ l4Hits: number;
27
+ l4Misses: number;
28
+ lruSize: number;
29
+ vectorSize: number;
30
+ }
31
+ export declare class CacheManager {
32
+ private readonly config;
33
+ readonly lru: LruLayer;
34
+ readonly sqlite: SqliteLayer;
35
+ readonly vector: VectorStore;
36
+ readonly blob: BlobLayer;
37
+ readonly diff: DiffReader;
38
+ readonly stats: CacheStats;
39
+ constructor(config: ResolvedConfig);
40
+ init(): Promise<void>;
41
+ key(tool: string, input: unknown, schemaVersion?: string): string;
42
+ get<T>(key: string): CacheLookup<T> | undefined;
43
+ set<T>(key: string, value: T, opts?: {
44
+ ttlSeconds?: number;
45
+ namespace?: string;
46
+ }): void;
47
+ invalidateNamespace(namespace: string): void;
48
+ close(): Promise<void>;
49
+ snapshot(): CacheStats;
50
+ }
51
+ export { cacheKey, simHash } from "./key.js";
52
+ export { LruLayer } from "./lru.js";
53
+ export { SqliteLayer } from "./sqlite.js";
54
+ export { VectorStore } from "./vector.js";
55
+ export { BlobLayer } from "./blob.js";
56
+ export { DiffReader, type ReadState } from "./diff-reader.js";
57
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/cache/index.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AACnD,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AACtC,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAE9C,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAE1C;;;;;;;GAOG;AAEH,MAAM,WAAW,WAAW,CAAC,CAAC;IAC5B,KAAK,EAAE,CAAC,CAAC;IACT,KAAK,EAAE,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI,CAAC;CAClC;AAED,MAAM,WAAW,UAAU;IACzB,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;CACpB;AAED,qBAAa,YAAY;IAmBX,OAAO,CAAC,QAAQ,CAAC,MAAM;IAlBnC,SAAgB,GAAG,WAA+C;IAClE,SAAgB,MAAM,EAAE,WAAW,CAAC;IACpC,SAAgB,MAAM,EAAE,WAAW,CAAC;IACpC,SAAgB,IAAI,EAAE,SAAS,CAAC;IAChC,SAAgB,IAAI,aAAoB;IACxC,SAAgB,KAAK,EAAE,UAAU,CAW/B;gBAE2B,MAAM,EAAE,cAAc;IAMtC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAe3B,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,aAAa,SAAO,GAAG,MAAM;IAI/D,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,GAAG,WAAW,CAAC,CAAC,CAAC,GAAG,SAAS;IAkB/C,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,EAAE,IAAI,CAAC,EAAE;QAAE,UAAU,CAAC,EAAE,MAAM,CAAC;QAAC,SAAS,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI;IAKvF,mBAAmB,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAKtC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAS5B,QAAQ,IAAI,UAAU;CAO9B;AAED,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,UAAU,CAAC;AAC7C,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AACtC,OAAO,EAAE,UAAU,EAAE,KAAK,SAAS,EAAE,MAAM,kBAAkB,CAAC"}
@@ -0,0 +1,99 @@
1
+ import { rmSync } from "node:fs";
2
+ import { join } from "node:path";
3
+ import { getLogger } from "../logger.js";
4
+ import { BlobLayer } from "./blob.js";
5
+ import { DiffReader } from "./diff-reader.js";
6
+ import { cacheKey } from "./key.js";
7
+ import { LruLayer } from "./lru.js";
8
+ import { SqliteLayer } from "./sqlite.js";
9
+ import { VectorStore } from "./vector.js";
10
+ export class CacheManager {
11
+ config;
12
+ lru = new LruLayer({ max: 2_000, ttlMs: 60_000 });
13
+ sqlite;
14
+ vector;
15
+ blob;
16
+ diff = new DiffReader();
17
+ stats = {
18
+ l1Hits: 0,
19
+ l1Misses: 0,
20
+ l2Hits: 0,
21
+ l2Misses: 0,
22
+ l3Hits: 0,
23
+ l3Misses: 0,
24
+ l4Hits: 0,
25
+ l4Misses: 0,
26
+ lruSize: 0,
27
+ vectorSize: 0,
28
+ };
29
+ constructor(config) {
30
+ this.config = config;
31
+ this.sqlite = new SqliteLayer({ path: join(config.cacheDir, "store.sqlite") });
32
+ this.vector = new VectorStore({ cacheDir: config.cacheDir });
33
+ this.blob = new BlobLayer({ cacheDir: config.cacheDir });
34
+ }
35
+ async init() {
36
+ if (this.config.isolated) {
37
+ // Best-effort wipe. Windows can hold the dir open when concurrent
38
+ // runs share a parent; a failure just means we append, not replace.
39
+ try {
40
+ rmSync(this.config.cacheDir, { recursive: true, force: true, maxRetries: 2, retryDelay: 50 });
41
+ }
42
+ catch (err) {
43
+ getLogger().debug({ err }, "isolated cacheDir wipe skipped (likely in use)");
44
+ }
45
+ }
46
+ this.blob.ensureShards();
47
+ await this.sqlite.init();
48
+ await this.vector.init();
49
+ }
50
+ key(tool, input, schemaVersion = "v1") {
51
+ return cacheKey(tool, input, schemaVersion);
52
+ }
53
+ get(key) {
54
+ const l1 = this.lru.get(key);
55
+ if (l1 !== undefined) {
56
+ this.stats.l1Hits++;
57
+ return { value: l1, layer: "L1" };
58
+ }
59
+ this.stats.l1Misses++;
60
+ const l2 = this.sqlite.get(key);
61
+ if (l2 !== undefined) {
62
+ this.stats.l2Hits++;
63
+ this.lru.set(key, l2);
64
+ return { value: l2, layer: "L2" };
65
+ }
66
+ this.stats.l2Misses++;
67
+ return undefined;
68
+ }
69
+ set(key, value, opts) {
70
+ this.lru.set(key, value);
71
+ this.sqlite.set(key, value, opts);
72
+ }
73
+ invalidateNamespace(namespace) {
74
+ this.sqlite.invalidateNamespace(namespace);
75
+ this.lru.clear();
76
+ }
77
+ async close() {
78
+ this.stats.lruSize = this.lru.size();
79
+ this.stats.vectorSize = this.vector.size();
80
+ if (this.config.saveSession) {
81
+ this.vector.persist();
82
+ }
83
+ this.sqlite.close();
84
+ }
85
+ snapshot() {
86
+ return {
87
+ ...this.stats,
88
+ lruSize: this.lru.size(),
89
+ vectorSize: this.vector.size(),
90
+ };
91
+ }
92
+ }
93
+ export { cacheKey, simHash } from "./key.js";
94
+ export { LruLayer } from "./lru.js";
95
+ export { SqliteLayer } from "./sqlite.js";
96
+ export { VectorStore } from "./vector.js";
97
+ export { BlobLayer } from "./blob.js";
98
+ export { DiffReader } from "./diff-reader.js";
99
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cache/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAEzC,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AACtC,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC9C,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AA6B1C,MAAM,OAAO,YAAY;IAmBM;IAlBb,GAAG,GAAG,IAAI,QAAQ,CAAC,EAAE,GAAG,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;IAClD,MAAM,CAAc;IACpB,MAAM,CAAc;IACpB,IAAI,CAAY;IAChB,IAAI,GAAG,IAAI,UAAU,EAAE,CAAC;IACxB,KAAK,GAAe;QAClC,MAAM,EAAE,CAAC;QACT,QAAQ,EAAE,CAAC;QACX,MAAM,EAAE,CAAC;QACT,QAAQ,EAAE,CAAC;QACX,MAAM,EAAE,CAAC;QACT,QAAQ,EAAE,CAAC;QACX,MAAM,EAAE,CAAC;QACT,QAAQ,EAAE,CAAC;QACX,OAAO,EAAE,CAAC;QACV,UAAU,EAAE,CAAC;KACd,CAAC;IAEF,YAA6B,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;QACjD,IAAI,CAAC,MAAM,GAAG,IAAI,WAAW,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,cAAc,CAAC,EAAE,CAAC,CAAC;QAC/E,IAAI,CAAC,MAAM,GAAG,IAAI,WAAW,CAAC,EAAE,QAAQ,EAAE,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC7D,IAAI,CAAC,IAAI,GAAG,IAAI,SAAS,CAAC,EAAE,QAAQ,EAAE,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;IAC3D,CAAC;IAEM,KAAK,CAAC,IAAI;QACf,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;YACzB,kEAAkE;YAClE,oEAAoE;YACpE,IAAI,CAAC;gBACH,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC,EAAE,UAAU,EAAE,EAAE,EAAE,CAAC,CAAC;YAChG,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,SAAS,EAAE,CAAC,KAAK,CAAC,EAAE,GAAG,EAAE,EAAE,gDAAgD,CAAC,CAAC;YAC/E,CAAC;QACH,CAAC;QACD,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC;QACzB,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QACzB,MAAM,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;IAC3B,CAAC;IAEM,GAAG,CAAC,IAAY,EAAE,KAAc,EAAE,aAAa,GAAG,IAAI;QAC3D,OAAO,QAAQ,CAAC,IAAI,EAAE,KAAK,EAAE,aAAa,CAAC,CAAC;IAC9C,CAAC;IAEM,GAAG,CAAI,GAAW;QACvB,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAI,GAAG,CAAC,CAAC;QAChC,IAAI,EAAE,KAAK,SAAS,EAAE,CAAC;YACrB,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACpB,OAAO,EAAE,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;QACpC,CAAC;QACD,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC;QAEtB,MAAM,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAI,GAAG,CAAC,CAAC;QACnC,IAAI,EAAE,KAAK,SAAS,EAAE,CAAC;YACrB,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACpB,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;YACtB,OAAO,EAAE,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;QACpC,CAAC;QACD,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC;QACtB,OAAO,SAAS,CAAC;IACnB,CAAC;IAEM,GAAG,CAAI,GAAW,EAAE,KAAQ,EAAE,IAAkD;QACrF,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QACzB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;IACpC,CAAC;IAEM,mBAAmB,CAAC,SAAiB;QAC1C,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC;QAC3C,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;IACnB,CAAC;IAEM,KAAK,CAAC,KAAK;QAChB,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;QACrC,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QAC3C,IAAI,IAAI,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC;YAC5B,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;QACxB,CAAC;QACD,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;IACtB,CAAC;IAEM,QAAQ;QACb,OAAO;YACL,GAAG,IAAI,CAAC,KAAK;YACb,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE;YACxB,UAAU,EAAE,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE;SAC/B,CAAC;IACJ,CAAC;CACF;AAED,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,UAAU,CAAC;AAC7C,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AACtC,OAAO,EAAE,UAAU,EAAkB,MAAM,kBAAkB,CAAC"}
@@ -0,0 +1,15 @@
1
+ /**
2
+ * Stable cache key derivation.
3
+ *
4
+ * Shape: sha256(tool + "::" + schemaVersion + "::" + canonicalJSON(input))
5
+ *
6
+ * Canonicalization ensures `{a:1,b:2}` and `{b:2,a:1}` yield the same key.
7
+ */
8
+ export declare function canonicalJSON(value: unknown): string;
9
+ export declare function cacheKey(tool: string, input: unknown, schemaVersion?: string): string;
10
+ /**
11
+ * SimHash-style stable fingerprint of a text blob. Used by the 3-state diff reader
12
+ * to detect "unchanged" on re-reads cheaply.
13
+ */
14
+ export declare function simHash(text: string): string;
15
+ //# sourceMappingURL=key.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"key.d.ts","sourceRoot":"","sources":["../../src/cache/key.ts"],"names":[],"mappings":"AAEA;;;;;;GAMG;AAEH,wBAAgB,aAAa,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM,CAEpD;AAWD,wBAAgB,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,aAAa,SAAO,GAAG,MAAM,CAGnF;AAED;;;GAGG;AACH,wBAAgB,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAE5C"}
@@ -0,0 +1,32 @@
1
+ import { createHash } from "node:crypto";
2
+ /**
3
+ * Stable cache key derivation.
4
+ *
5
+ * Shape: sha256(tool + "::" + schemaVersion + "::" + canonicalJSON(input))
6
+ *
7
+ * Canonicalization ensures `{a:1,b:2}` and `{b:2,a:1}` yield the same key.
8
+ */
9
+ export function canonicalJSON(value) {
10
+ return JSON.stringify(sortKeys(value));
11
+ }
12
+ function sortKeys(value) {
13
+ if (Array.isArray(value))
14
+ return value.map(sortKeys);
15
+ if (value !== null && typeof value === "object") {
16
+ const entries = Object.entries(value).sort(([a], [b]) => a.localeCompare(b));
17
+ return Object.fromEntries(entries.map(([k, v]) => [k, sortKeys(v)]));
18
+ }
19
+ return value;
20
+ }
21
+ export function cacheKey(tool, input, schemaVersion = "v1") {
22
+ const payload = `${tool}::${schemaVersion}::${canonicalJSON(input)}`;
23
+ return createHash("sha256").update(payload).digest("hex");
24
+ }
25
+ /**
26
+ * SimHash-style stable fingerprint of a text blob. Used by the 3-state diff reader
27
+ * to detect "unchanged" on re-reads cheaply.
28
+ */
29
+ export function simHash(text) {
30
+ return createHash("sha256").update(text).digest("hex").slice(0, 16);
31
+ }
32
+ //# sourceMappingURL=key.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"key.js","sourceRoot":"","sources":["../../src/cache/key.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAEzC;;;;;;GAMG;AAEH,MAAM,UAAU,aAAa,CAAC,KAAc;IAC1C,OAAO,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;AACzC,CAAC;AAED,SAAS,QAAQ,CAAC,KAAc;IAC9B,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;QAAE,OAAO,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IACrD,IAAI,KAAK,KAAK,IAAI,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAChD,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,KAAgC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;QACxG,OAAO,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACvE,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,MAAM,UAAU,QAAQ,CAAC,IAAY,EAAE,KAAc,EAAE,aAAa,GAAG,IAAI;IACzE,MAAM,OAAO,GAAG,GAAG,IAAI,KAAK,aAAa,KAAK,aAAa,CAAC,KAAK,CAAC,EAAE,CAAC;IACrE,OAAO,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;AAC5D,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,OAAO,CAAC,IAAY;IAClC,OAAO,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;AACtE,CAAC"}
@@ -0,0 +1,21 @@
1
+ export interface LruEntry<T> {
2
+ value: T;
3
+ storedAt: number;
4
+ }
5
+ /**
6
+ * L1: in-process LRU. Short TTL, sized by count.
7
+ */
8
+ export declare class LruLayer {
9
+ private cache;
10
+ constructor(opts?: {
11
+ max?: number;
12
+ ttlMs?: number;
13
+ });
14
+ get<T>(key: string): T | undefined;
15
+ set<T>(key: string, value: T): void;
16
+ has(key: string): boolean;
17
+ delete(key: string): boolean;
18
+ clear(): void;
19
+ size(): number;
20
+ }
21
+ //# sourceMappingURL=lru.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lru.d.ts","sourceRoot":"","sources":["../../src/cache/lru.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,QAAQ,CAAC,CAAC;IACzB,KAAK,EAAE,CAAC,CAAC;IACT,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED;;GAEG;AACH,qBAAa,QAAQ;IACnB,OAAO,CAAC,KAAK,CAAsC;gBAEvC,IAAI,GAAE;QAAE,GAAG,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAA;KAAO;IAOhD,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,GAAG,CAAC,GAAG,SAAS;IAIlC,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,GAAG,IAAI;IAInC,GAAG,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAIzB,MAAM,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAI5B,KAAK,IAAI,IAAI;IAIb,IAAI,IAAI,MAAM;CAGtB"}
@@ -0,0 +1,32 @@
1
+ import { LRUCache } from "lru-cache";
2
+ /**
3
+ * L1: in-process LRU. Short TTL, sized by count.
4
+ */
5
+ export class LruLayer {
6
+ cache;
7
+ constructor(opts = {}) {
8
+ this.cache = new LRUCache({
9
+ max: opts.max ?? 1_000,
10
+ ttl: opts.ttlMs ?? 60_000,
11
+ });
12
+ }
13
+ get(key) {
14
+ return this.cache.get(key)?.value;
15
+ }
16
+ set(key, value) {
17
+ this.cache.set(key, { value, storedAt: Date.now() });
18
+ }
19
+ has(key) {
20
+ return this.cache.has(key);
21
+ }
22
+ delete(key) {
23
+ return this.cache.delete(key);
24
+ }
25
+ clear() {
26
+ this.cache.clear();
27
+ }
28
+ size() {
29
+ return this.cache.size;
30
+ }
31
+ }
32
+ //# sourceMappingURL=lru.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lru.js","sourceRoot":"","sources":["../../src/cache/lru.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AAOrC;;GAEG;AACH,MAAM,OAAO,QAAQ;IACX,KAAK,CAAsC;IAEnD,YAAY,OAAyC,EAAE;QACrD,IAAI,CAAC,KAAK,GAAG,IAAI,QAAQ,CAAC;YACxB,GAAG,EAAE,IAAI,CAAC,GAAG,IAAI,KAAK;YACtB,GAAG,EAAE,IAAI,CAAC,KAAK,IAAI,MAAM;SAC1B,CAAC,CAAC;IACL,CAAC;IAEM,GAAG,CAAI,GAAW;QACvB,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,KAAsB,CAAC;IACrD,CAAC;IAEM,GAAG,CAAI,GAAW,EAAE,KAAQ;QACjC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IACvD,CAAC;IAEM,GAAG,CAAC,GAAW;QACpB,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IAC7B,CAAC;IAEM,MAAM,CAAC,GAAW;QACvB,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;IAChC,CAAC;IAEM,KAAK;QACV,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;IACrB,CAAC;IAEM,IAAI;QACT,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC;IACzB,CAAC;CACF"}
@@ -0,0 +1,22 @@
1
+ export interface SqliteLayerOptions {
2
+ path: string;
3
+ /** Default TTL in seconds. Individual writes can override. */
4
+ defaultTtlSeconds?: number;
5
+ }
6
+ export declare class SqliteLayer {
7
+ private readonly opts;
8
+ private db;
9
+ private fallback;
10
+ private readonly defaultTtlSeconds;
11
+ constructor(opts: SqliteLayerOptions);
12
+ init(): Promise<void>;
13
+ get<T>(key: string): T | undefined;
14
+ set<T>(key: string, value: T, opts?: {
15
+ ttlSeconds?: number;
16
+ namespace?: string;
17
+ }): void;
18
+ invalidateNamespace(namespace: string): void;
19
+ clear(): void;
20
+ close(): void;
21
+ }
22
+ //# sourceMappingURL=sqlite.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sqlite.d.ts","sourceRoot":"","sources":["../../src/cache/sqlite.ts"],"names":[],"mappings":"AAsBA,MAAM,WAAW,kBAAkB;IACjC,IAAI,EAAE,MAAM,CAAC;IACb,8DAA8D;IAC9D,iBAAiB,CAAC,EAAE,MAAM,CAAC;CAC5B;AAED,qBAAa,WAAW;IAKV,OAAO,CAAC,QAAQ,CAAC,IAAI;IAJjC,OAAO,CAAC,EAAE,CAA6B;IACvC,OAAO,CAAC,QAAQ,CAA2D;IAC3E,OAAO,CAAC,QAAQ,CAAC,iBAAiB,CAAS;gBAEd,IAAI,EAAE,kBAAkB;IAIxC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IA8B3B,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,GAAG,CAAC,GAAG,SAAS;IAsBlC,GAAG,CAAC,CAAC,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,EAAE,IAAI,CAAC,EAAE;QAAE,UAAU,CAAC,EAAE,MAAM,CAAC;QAAC,SAAS,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI;IAevF,mBAAmB,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAS5C,KAAK,IAAI,IAAI;IAQb,KAAK,IAAI,IAAI;CAOrB"}
@@ -0,0 +1,102 @@
1
+ import { mkdirSync } from "node:fs";
2
+ import { dirname } from "node:path";
3
+ import { getLogger } from "../logger.js";
4
+ export class SqliteLayer {
5
+ opts;
6
+ db = null;
7
+ fallback = new Map();
8
+ defaultTtlSeconds;
9
+ constructor(opts) {
10
+ this.opts = opts;
11
+ this.defaultTtlSeconds = opts.defaultTtlSeconds ?? 3_600;
12
+ }
13
+ async init() {
14
+ try {
15
+ // Dynamic import — optional dependency. Not typed because it's
16
+ // optional; we wrap with a guard and fall back cleanly.
17
+ // eslint-disable-next-line @typescript-eslint/ban-ts-comment
18
+ // @ts-ignore optional native dep
19
+ const mod = await import("better-sqlite3").catch(() => null);
20
+ if (!mod || !mod.default) {
21
+ getLogger().warn("better-sqlite3 not available — L2 cache falls back to in-memory map");
22
+ return;
23
+ }
24
+ mkdirSync(dirname(this.opts.path), { recursive: true });
25
+ const Database = mod.default;
26
+ this.db = new Database(this.opts.path);
27
+ this.db.exec(`
28
+ CREATE TABLE IF NOT EXISTS kv (
29
+ k TEXT PRIMARY KEY,
30
+ v TEXT NOT NULL,
31
+ expires_at INTEGER NOT NULL,
32
+ namespace TEXT
33
+ );
34
+ CREATE INDEX IF NOT EXISTS idx_kv_namespace ON kv(namespace);
35
+ CREATE INDEX IF NOT EXISTS idx_kv_expires ON kv(expires_at);
36
+ `);
37
+ }
38
+ catch (err) {
39
+ getLogger().warn({ err }, "SQLite init failed; using in-memory fallback");
40
+ this.db = null;
41
+ }
42
+ }
43
+ get(key) {
44
+ const now = Date.now();
45
+ if (this.db) {
46
+ const row = this.db.prepare("SELECT v, expires_at FROM kv WHERE k = ?").get(key);
47
+ if (!row)
48
+ return undefined;
49
+ if (row.expires_at < now) {
50
+ this.db.prepare("DELETE FROM kv WHERE k = ?").run(key);
51
+ return undefined;
52
+ }
53
+ return JSON.parse(row.v);
54
+ }
55
+ const fb = this.fallback.get(key);
56
+ if (!fb)
57
+ return undefined;
58
+ if (fb.expiresAt < now) {
59
+ this.fallback.delete(key);
60
+ return undefined;
61
+ }
62
+ return JSON.parse(fb.value);
63
+ }
64
+ set(key, value, opts) {
65
+ const ttl = opts?.ttlSeconds ?? this.defaultTtlSeconds;
66
+ const expiresAt = Date.now() + ttl * 1000;
67
+ const serialized = JSON.stringify(value);
68
+ if (this.db) {
69
+ this.db
70
+ .prepare("INSERT INTO kv (k, v, expires_at, namespace) VALUES (?, ?, ?, ?) ON CONFLICT(k) DO UPDATE SET v = excluded.v, expires_at = excluded.expires_at")
71
+ .run(key, serialized, expiresAt, opts?.namespace ?? null);
72
+ }
73
+ else {
74
+ this.fallback.set(key, { value: serialized, expiresAt });
75
+ }
76
+ }
77
+ invalidateNamespace(namespace) {
78
+ if (this.db) {
79
+ this.db.prepare("DELETE FROM kv WHERE namespace = ?").run(namespace);
80
+ }
81
+ else {
82
+ // No namespace tracking in fallback — nuke all.
83
+ this.fallback.clear();
84
+ }
85
+ }
86
+ clear() {
87
+ if (this.db) {
88
+ this.db.exec("DELETE FROM kv");
89
+ }
90
+ else {
91
+ this.fallback.clear();
92
+ }
93
+ }
94
+ close() {
95
+ if (this.db) {
96
+ this.db.close();
97
+ this.db = null;
98
+ }
99
+ this.fallback.clear();
100
+ }
101
+ }
102
+ //# sourceMappingURL=sqlite.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sqlite.js","sourceRoot":"","sources":["../../src/cache/sqlite.ts"],"names":[],"mappings":"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"}