hippo-memory 1.14.0 → 1.16.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 (106) hide show
  1. package/README.md +862 -861
  2. package/dist/audit.d.ts +1 -1
  3. package/dist/audit.d.ts.map +1 -1
  4. package/dist/audit.js.map +1 -1
  5. package/dist/cli.d.ts.map +1 -1
  6. package/dist/cli.js +1594 -229
  7. package/dist/cli.js.map +1 -1
  8. package/dist/customer-notes.d.ts +95 -0
  9. package/dist/customer-notes.d.ts.map +1 -0
  10. package/dist/customer-notes.js +296 -0
  11. package/dist/customer-notes.js.map +1 -0
  12. package/dist/db.d.ts.map +1 -1
  13. package/dist/db.js +1286 -472
  14. package/dist/db.js.map +1 -1
  15. package/dist/decisions.d.ts +91 -0
  16. package/dist/decisions.d.ts.map +1 -0
  17. package/dist/decisions.js +278 -0
  18. package/dist/decisions.js.map +1 -0
  19. package/dist/graph-extract.d.ts +39 -0
  20. package/dist/graph-extract.d.ts.map +1 -0
  21. package/dist/graph-extract.js +141 -0
  22. package/dist/graph-extract.js.map +1 -0
  23. package/dist/graph-recall.d.ts +41 -0
  24. package/dist/graph-recall.d.ts.map +1 -0
  25. package/dist/graph-recall.js +246 -0
  26. package/dist/graph-recall.js.map +1 -0
  27. package/dist/graph.d.ts +137 -0
  28. package/dist/graph.d.ts.map +1 -0
  29. package/dist/graph.js +433 -0
  30. package/dist/graph.js.map +1 -0
  31. package/dist/incidents.d.ts +100 -0
  32. package/dist/incidents.d.ts.map +1 -0
  33. package/dist/incidents.js +322 -0
  34. package/dist/incidents.js.map +1 -0
  35. package/dist/index.d.ts +1 -0
  36. package/dist/index.d.ts.map +1 -1
  37. package/dist/index.js +1 -0
  38. package/dist/index.js.map +1 -1
  39. package/dist/memory.d.ts +6 -0
  40. package/dist/memory.d.ts.map +1 -1
  41. package/dist/memory.js +6 -0
  42. package/dist/memory.js.map +1 -1
  43. package/dist/policies.d.ts +149 -0
  44. package/dist/policies.d.ts.map +1 -0
  45. package/dist/policies.js +380 -0
  46. package/dist/policies.js.map +1 -0
  47. package/dist/processes.d.ts +104 -0
  48. package/dist/processes.d.ts.map +1 -0
  49. package/dist/processes.js +330 -0
  50. package/dist/processes.js.map +1 -0
  51. package/dist/project-briefs.d.ts +126 -0
  52. package/dist/project-briefs.d.ts.map +1 -0
  53. package/dist/project-briefs.js +453 -0
  54. package/dist/project-briefs.js.map +1 -0
  55. package/dist/search.d.ts +7 -0
  56. package/dist/search.d.ts.map +1 -1
  57. package/dist/search.js.map +1 -1
  58. package/dist/server.d.ts.map +1 -1
  59. package/dist/server.js +1181 -8
  60. package/dist/server.js.map +1 -1
  61. package/dist/skills.d.ts +98 -0
  62. package/dist/skills.d.ts.map +1 -0
  63. package/dist/skills.js +339 -0
  64. package/dist/skills.js.map +1 -0
  65. package/dist/src/audit.js.map +1 -1
  66. package/dist/src/cli.js +1594 -229
  67. package/dist/src/cli.js.map +1 -1
  68. package/dist/src/customer-notes.js +296 -0
  69. package/dist/src/customer-notes.js.map +1 -0
  70. package/dist/src/db.js +1286 -472
  71. package/dist/src/db.js.map +1 -1
  72. package/dist/src/decisions.js +278 -0
  73. package/dist/src/decisions.js.map +1 -0
  74. package/dist/src/graph-extract.js +141 -0
  75. package/dist/src/graph-extract.js.map +1 -0
  76. package/dist/src/graph-recall.js +246 -0
  77. package/dist/src/graph-recall.js.map +1 -0
  78. package/dist/src/graph.js +433 -0
  79. package/dist/src/graph.js.map +1 -0
  80. package/dist/src/incidents.js +322 -0
  81. package/dist/src/incidents.js.map +1 -0
  82. package/dist/src/index.js +1 -0
  83. package/dist/src/index.js.map +1 -1
  84. package/dist/src/memory.js +6 -0
  85. package/dist/src/memory.js.map +1 -1
  86. package/dist/src/policies.js +380 -0
  87. package/dist/src/policies.js.map +1 -0
  88. package/dist/src/processes.js +330 -0
  89. package/dist/src/processes.js.map +1 -0
  90. package/dist/src/project-briefs.js +453 -0
  91. package/dist/src/project-briefs.js.map +1 -0
  92. package/dist/src/search.js.map +1 -1
  93. package/dist/src/server.js +1181 -8
  94. package/dist/src/server.js.map +1 -1
  95. package/dist/src/skills.js +339 -0
  96. package/dist/src/skills.js.map +1 -0
  97. package/dist/src/version.js +1 -1
  98. package/dist/src/version.js.map +1 -1
  99. package/dist/version.d.ts +1 -1
  100. package/dist/version.d.ts.map +1 -1
  101. package/dist/version.js +1 -1
  102. package/dist/version.js.map +1 -1
  103. package/extensions/openclaw-plugin/openclaw.plugin.json +46 -46
  104. package/extensions/openclaw-plugin/package.json +14 -14
  105. package/openclaw.plugin.json +45 -45
  106. package/package.json +75 -75
@@ -0,0 +1,246 @@
1
+ /**
2
+ * E3.2 multi-hop graph recall (docs/plans/2026-06-02-e3.2-multihop-recall.md).
3
+ *
4
+ * READ-ONLY consumer of the E3 graph substrate (entities/relations built by E3.1,
5
+ * guarded by E3.3). Given the lexical recall seeds, walk the relations graph up to N
6
+ * hops and surface the memories of reached entities that the lexical search did not
7
+ * already return.
8
+ *
9
+ * Relation-type-AGNOSTIC: it walks whatever edges exist. Today the graph holds only
10
+ * `supersedes` edges (so a 1-hop walk surfaces a supersession-linked predecessor/
11
+ * successor a lexical search may miss); the moment E3.1 emits cross-object edges
12
+ * (owns/depends-on/blocked-by/references) the SAME traversal lights up cross-entity
13
+ * multi-hop with zero rework here.
14
+ *
15
+ * Design points (the first two were forced by the verify-stage benchmark, the rest by
16
+ * codex review — all root-cause, not patches):
17
+ * 1. Graph-reached memories are loaded DIRECTLY by id (tenant-scoped PK fetch), NOT
18
+ * intersected with the recall handler's candidate set — that set is lexically
19
+ * prefiltered (loadSearchRows filters by query tokens), so intersecting would exclude
20
+ * exactly the lexically-orthogonal neighbours graph recall exists to surface. We
21
+ * re-apply the recall HARD filters to the directly-loaded rows: the FULL bi-temporal
22
+ * as-of rule (valid_from <= asOf AND, for a superseded row, successor.valid_from >
23
+ * asOf — same as cmdRecall) and the default superseded-drop. SCOPE is intentionally
24
+ * NOT re-applied: the CLI caller (cmdRecall) does not hard-filter scope either (it
25
+ * only soft-boosts it). Rows are tenant-scoped + archived-excluded (loadEntriesByIds).
26
+ * 2. A graph hit inherits its origin seed's relevance (minus a per-hop discount) and is
27
+ * placed adjacent to that seed; budget selection is by score so a high-value hit wins
28
+ * a token slot over a noise distractor. Dead-last appending made the feature do
29
+ * nothing at realistic budgets.
30
+ * 3. BOTH the local and global stores are expanded — a global seed's entities/relations
31
+ * live under the global root, so graph recall must traverse each seed in the store its
32
+ * graph lives in (codex review).
33
+ * 4. By-id loads are chunked at 500 (loadEntriesByIds caps at 500/call), so a high-fanout
34
+ * traversal (--hops 3 --max-neighbors 200 -> up to 600 ids) loses none (codex review).
35
+ *
36
+ * No graph writes (only SELECTs via graph.ts read helpers + store reads), so the E3.3
37
+ * check-graph-writes lint permits this module living outside graph.ts.
38
+ */
39
+ import { loadEntriesByIds } from './store.js';
40
+ import { estimateTokens } from './search.js';
41
+ import { loadEntitiesByMemoryId, loadEntitiesByIds, loadNeighborRelations, } from './graph.js';
42
+ /** Hard cap on `--hops` (a higher value just walks more of a finite graph; this bounds
43
+ * worst-case work and keeps the flag honest). */
44
+ export const MAX_HOPS = 3;
45
+ /** Default per-hop fanout cap (bounds blow-up on a future dense graph). */
46
+ export const DEFAULT_MAX_NEIGHBORS = 25;
47
+ /** Per-hop relevance discount: a graph hit inherits its origin seed's score, scaled down
48
+ * 1% per hop, so it ranks just below its seed and orders by hop-distance. */
49
+ const HOP_DISCOUNT = 0.01;
50
+ /** loadEntriesByIds caps each call at 500 ids; chunk to lose none on high fanout. */
51
+ const LOAD_CHUNK = 500;
52
+ /** Load memories by id in <=500-id chunks (loadEntriesByIds caps each call at 500). */
53
+ function loadByIdsChunked(root, tenantId, ids) {
54
+ if (ids.length === 0)
55
+ return [];
56
+ const out = [];
57
+ for (let i = 0; i < ids.length; i += LOAD_CHUNK) {
58
+ out.push(...loadEntriesByIds(root, ids.slice(i, i + LOAD_CHUNK), tenantId));
59
+ }
60
+ return out;
61
+ }
62
+ /**
63
+ * Traverse one store's graph from the seeds present in it and accumulate new graph hits
64
+ * into `hitsByOrigin`. Mutates `seenMemoryIds` so a memory is surfaced at most once across
65
+ * stores. Pure reads.
66
+ */
67
+ function produceHitsForRoot(root, baseResults, baseScoreByMemId, seenMemoryIds, hitsByOrigin, opts) {
68
+ const { hops, maxNeighbors, tenantId, includeSuperseded, asOfDate } = opts;
69
+ // Seeds = graph entities (in THIS store) whose source memory is a base result.
70
+ const seedEntities = loadEntitiesByMemoryId(root, tenantId, baseResults.map((r) => r.entry.id));
71
+ if (seedEntities.length === 0)
72
+ return;
73
+ // BFS, both directions, up to `hops`. `visited` prevents re-expansion (cycle-safe).
74
+ // `originMemByEntityId` propagates the base-result memory id each reached node descends
75
+ // from (for adjacency placement + score inheritance).
76
+ const visitedEntityIds = new Set(seedEntities.map((e) => e.id));
77
+ const reached = new Map();
78
+ const originMemByEntityId = new Map();
79
+ for (const se of seedEntities)
80
+ originMemByEntityId.set(se.id, se.memoryId);
81
+ let frontier = seedEntities.map((e) => e.id);
82
+ for (let depth = 1; depth <= hops && frontier.length > 0; depth++) {
83
+ const frontierSet = new Set(frontier);
84
+ const rels = loadNeighborRelations(root, tenantId, frontier, {
85
+ limit: Math.max(maxNeighbors, maxNeighbors * frontier.length),
86
+ });
87
+ const nextFrontier = [];
88
+ for (const rel of rels) {
89
+ const fromIn = frontierSet.has(rel.fromEntityId);
90
+ const toIn = frontierSet.has(rel.toEntityId);
91
+ let neighborId;
92
+ let reacherId;
93
+ let direction;
94
+ if (fromIn && !toIn) {
95
+ neighborId = rel.toEntityId;
96
+ reacherId = rel.fromEntityId;
97
+ direction = 'to';
98
+ }
99
+ else if (toIn && !fromIn) {
100
+ neighborId = rel.fromEntityId;
101
+ reacherId = rel.toEntityId;
102
+ direction = 'from';
103
+ }
104
+ else
105
+ continue;
106
+ if (visitedEntityIds.has(neighborId))
107
+ continue;
108
+ visitedEntityIds.add(neighborId);
109
+ reached.set(neighborId, { hops: depth, relType: rel.relType, direction });
110
+ const origin = originMemByEntityId.get(reacherId);
111
+ if (origin !== undefined)
112
+ originMemByEntityId.set(neighborId, origin);
113
+ nextFrontier.push(neighborId);
114
+ if (nextFrontier.length >= maxNeighbors)
115
+ break; // per-hop fanout cap
116
+ }
117
+ frontier = nextFrontier;
118
+ }
119
+ if (reached.size === 0)
120
+ return;
121
+ // Reached entities -> source memory ids -> load DIRECTLY by id (chunked), not lexical.
122
+ const reachedEntities = loadEntitiesByIds(root, tenantId, [...reached.keys()]);
123
+ const needLoad = [...new Set(reachedEntities.map((e) => e.memoryId).filter((id) => !seenMemoryIds.has(id)))];
124
+ const loadedById = new Map(loadByIdsChunked(root, tenantId, needLoad).map((m) => [m.id, m]));
125
+ // For the bi-temporal as-of rule on a superseded reached row we need its successor's
126
+ // valid_from. Batch-load the successors referenced by the loaded rows.
127
+ let successorValidFrom = new Map();
128
+ if (asOfDate) {
129
+ const succIds = [...new Set([...loadedById.values()].map((m) => m.superseded_by).filter((id) => !!id))];
130
+ successorValidFrom = new Map(loadByIdsChunked(root, tenantId, succIds).map((m) => [m.id, m.valid_from]));
131
+ }
132
+ for (const ent of reachedEntities) {
133
+ const mem = loadedById.get(ent.memoryId);
134
+ if (!mem)
135
+ continue; // not found / wrong tenant / already in base
136
+ if (seenMemoryIds.has(mem.id))
137
+ continue; // another reached entity already added it
138
+ const via = reached.get(ent.id);
139
+ // A node reached as the `to` endpoint of a `supersedes` edge IS the superseded
140
+ // (older) version — the graph is the authoritative signal (the memory mirror's
141
+ // `superseded_by` is NOT set by `hippo decide`, only the decisions table is). By
142
+ // default recall shows current truth, so drop it unless --include-superseded; the
143
+ // `from` endpoint (the newer successor) is always kept.
144
+ const isSupersededEndpoint = via.relType === 'supersedes' && via.direction === 'to';
145
+ if (asOfDate) {
146
+ if (new Date(mem.valid_from) > asOfDate)
147
+ continue; // not yet valid at asOf
148
+ if (mem.superseded_by) {
149
+ const succVf = successorValidFrom.get(mem.superseded_by);
150
+ // Visible only while its successor was NOT yet valid at asOf (matches cmdRecall).
151
+ if (succVf && new Date(succVf) <= asOfDate)
152
+ continue;
153
+ }
154
+ }
155
+ else if (!includeSuperseded && (mem.superseded_by || isSupersededEndpoint)) {
156
+ continue; // default recall drops superseded
157
+ }
158
+ const origin = originMemByEntityId.get(ent.id) ?? baseResults[0].entry.id;
159
+ const originScore = baseScoreByMemId.get(origin) ?? baseResults[baseResults.length - 1].score;
160
+ seenMemoryIds.add(mem.id);
161
+ const hit = {
162
+ entry: mem,
163
+ score: originScore * (1 - HOP_DISCOUNT * via.hops),
164
+ bm25: 0, cosine: 0,
165
+ tokens: estimateTokens(mem.content),
166
+ graphVia: via,
167
+ };
168
+ if (!hitsByOrigin.has(origin))
169
+ hitsByOrigin.set(origin, []);
170
+ hitsByOrigin.get(origin).push(hit);
171
+ }
172
+ }
173
+ /**
174
+ * Augment `baseResults` with memories reached by walking the graph `hops` edges out from
175
+ * the seed results' entities (across the local AND global stores). Each graph hit is
176
+ * inserted directly after the base result it descends from, scored just below that seed;
177
+ * the base list's own order is preserved. Token-budget-bounded; deduped against the base
178
+ * set and across stores.
179
+ *
180
+ * No-op (returns `baseResults` unchanged) when `hops <= 0`, there are no base results, the
181
+ * graph is empty, no seed maps to an entity, or nothing new survives the filters/budget.
182
+ */
183
+ export function graphExpandRecall(baseResults, opts) {
184
+ const { hops, hippoRoot, globalRoot, tenantId } = opts;
185
+ if (hops <= 0 || baseResults.length === 0)
186
+ return baseResults;
187
+ const maxNeighbors = opts.maxNeighbors ?? DEFAULT_MAX_NEIGHBORS;
188
+ const budget = opts.budget ?? 4000;
189
+ const includeSuperseded = opts.includeSuperseded ?? false;
190
+ const asOfDate = opts.asOf ? new Date(opts.asOf) : null;
191
+ const minResults = opts.minResults ?? 1;
192
+ const baseScoreByMemId = new Map(baseResults.map((r) => [r.entry.id, r.score]));
193
+ const seenMemoryIds = new Set(baseResults.map((r) => r.entry.id));
194
+ const hitsByOrigin = new Map();
195
+ // Expand against each distinct store the seeds may live in (local + global).
196
+ const roots = globalRoot && globalRoot !== hippoRoot ? [hippoRoot, globalRoot] : [hippoRoot];
197
+ for (const root of roots) {
198
+ produceHitsForRoot(root, baseResults, baseScoreByMemId, seenMemoryIds, hitsByOrigin, {
199
+ hops, maxNeighbors, tenantId, includeSuperseded, asOfDate,
200
+ });
201
+ }
202
+ if (hitsByOrigin.size === 0)
203
+ return baseResults;
204
+ // Closer hops first within each origin group, then by inherited score.
205
+ for (const hits of hitsByOrigin.values()) {
206
+ hits.sort((a, b) => a.graphVia.hops - b.graphVia.hops || b.score - a.score);
207
+ }
208
+ const allHits = [...hitsByOrigin.values()].flat();
209
+ // Budget SELECTION by score (not by position): a high-value graph hit (it inherits its
210
+ // origin seed's relevance) must be able to win a token slot over a low-score lexical
211
+ // distractor — otherwise a tight --budget keeps the noise and drops the memory --hops
212
+ // surfaced. The greedy pack uses `continue` (not `break`), so a hit's origin seed is
213
+ // NOT guaranteed kept just because the hit is; the DISPLAY loop guards that (a hit is
214
+ // emitted ONLY under a kept seed, never orphaned). At least one result is always kept.
215
+ // (NOTE: at a tight budget a new graph hit can displace a weakly-scored base result;
216
+ // aggregate recall stays >= baseline, the displaced item is the lowest-value one.)
217
+ // Protect the top --min-results base rows from eviction (graph expansion must not
218
+ // violate the recall min-results floor; codex P2). They are kept regardless of budget;
219
+ // baseResults is score-ordered, so slice(0, N) is the top N.
220
+ const protectedCount = Math.min(Math.max(minResults, 1), baseResults.length);
221
+ const keep = new Set(baseResults.slice(0, protectedCount));
222
+ let usedTokens = [...keep].reduce((s, r) => s + r.tokens, 0);
223
+ for (const r of [...baseResults.slice(protectedCount), ...allHits].sort((a, b) => b.score - a.score)) {
224
+ if (usedTokens + r.tokens > budget)
225
+ continue;
226
+ usedTokens += r.tokens;
227
+ keep.add(r);
228
+ }
229
+ // DISPLAY order: base order preserved (it may be MMR-diversified); each kept new hit
230
+ // placed directly after the seed it descends from. Hits emit ONLY under a kept seed.
231
+ const merged = [];
232
+ let emittedHit = false;
233
+ for (const r of baseResults) {
234
+ if (!keep.has(r))
235
+ continue;
236
+ merged.push(r);
237
+ for (const hit of hitsByOrigin.get(r.entry.id) ?? []) {
238
+ if (keep.has(hit)) {
239
+ merged.push(hit);
240
+ emittedHit = true;
241
+ }
242
+ }
243
+ }
244
+ return emittedHit ? merged : baseResults; // nothing new survived budget -> original base
245
+ }
246
+ //# sourceMappingURL=graph-recall.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graph-recall.js","sourceRoot":"","sources":["../../src/graph-recall.ts"],"names":[],"mappings":"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"}