clawvault 3.1.0 → 3.2.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/README.md +422 -141
- package/bin/clawvault.js +10 -2
- package/bin/command-registration.test.js +3 -1
- package/bin/command-runtime.js +9 -1
- package/bin/register-core-commands.js +23 -28
- package/bin/register-maintenance-commands.js +39 -3
- package/bin/register-query-commands.js +58 -29
- package/bin/register-tailscale-commands.js +106 -0
- package/bin/register-task-commands.js +18 -1
- package/bin/register-task-commands.test.js +16 -0
- package/bin/register-vault-operations-commands.js +29 -1
- package/bin/register-workgraph-commands.js +1368 -0
- package/dashboard/lib/graph-diff.js +104 -0
- package/dashboard/lib/graph-diff.test.js +75 -0
- package/dashboard/lib/vault-parser.js +556 -0
- package/dashboard/lib/vault-parser.test.js +254 -0
- package/dashboard/public/app.js +796 -0
- package/dashboard/public/index.html +52 -0
- package/dashboard/public/styles.css +221 -0
- package/dashboard/server.js +374 -0
- package/dist/{chunk-F2JEUD4J.js → chunk-23YDQ3QU.js} +6 -8
- package/dist/{chunk-C7OK5WKP.js → chunk-2JQ3O2YL.js} +4 -4
- package/dist/{chunk-VR5NE7PZ.js → chunk-2RAZ4ZFE.js} +1 -1
- package/dist/chunk-2ZDO52B4.js +52 -0
- package/dist/{chunk-ZZA73MFY.js → chunk-33DOSHTA.js} +176 -36
- package/dist/chunk-33VSQP4J.js +37 -0
- package/dist/chunk-4BQTQMJP.js +93 -0
- package/dist/{chunk-GUKMRGM7.js → chunk-4OXMU5S2.js} +1 -1
- package/dist/{chunk-62YTUT6J.js → chunk-4PY655YM.js} +15 -3
- package/dist/chunk-6FH3IULF.js +352 -0
- package/dist/{chunk-3NSBOUT3.js → chunk-77Q5CSPJ.js} +404 -80
- package/dist/{chunk-4VQTUVH7.js → chunk-7YZWHM36.js} +52 -26
- package/dist/chunk-BSJ6RIT7.js +447 -0
- package/dist/chunk-BUEW6IIK.js +364 -0
- package/dist/{chunk-LI4O6NVK.js → chunk-CLJTREDS.js} +74 -14
- package/dist/chunk-EK6S23ZB.js +469 -0
- package/dist/{chunk-LNJA2UGL.js → chunk-ESFLMDRB.js} +9 -86
- package/dist/{chunk-H34S76MB.js → chunk-ESVS6K2B.js} +6 -6
- package/dist/{chunk-WAZ3NLWL.js → chunk-F55HGNU4.js} +0 -47
- package/dist/{chunk-QK3UCXWL.js → chunk-FHFUXL6G.js} +2 -2
- package/dist/{chunk-H62BP7RI.js → chunk-GAOWA7GR.js} +212 -46
- package/dist/chunk-GGA32J2R.js +784 -0
- package/dist/chunk-GNJL4YGR.js +79 -0
- package/dist/chunk-IVRIKYFE.js +520 -0
- package/dist/chunk-MDIH26GC.js +183 -0
- package/dist/{chunk-LYHGEHXG.js → chunk-MFAWT5O5.js} +0 -1
- package/dist/chunk-MM6QGW3P.js +207 -0
- package/dist/{chunk-P5EPF6MB.js → chunk-MW5C6ZQA.js} +110 -13
- package/dist/chunk-NCKFNBHJ.js +257 -0
- package/dist/{chunk-QBLMXKF2.js → chunk-OIWVQYQF.js} +1 -1
- package/dist/{chunk-42MXU7A6.js → chunk-P62WHA27.js} +58 -47
- package/dist/chunk-PBACDKKP.js +66 -0
- package/dist/{chunk-VGLOTGAS.js → chunk-QSHD36LH.js} +2 -2
- package/dist/{chunk-OZ7RIXTO.js → chunk-QSRRMEYM.js} +2 -2
- package/dist/chunk-QVEERJSP.js +152 -0
- package/dist/{chunk-N2AXRYLC.js → chunk-QWQ3TIKS.js} +1 -1
- package/dist/{chunk-3DHXQHYG.js → chunk-R2MIW5G7.js} +1 -1
- package/dist/{chunk-SJSFRIYS.js → chunk-SLXOR3CC.js} +2 -2
- package/dist/chunk-SS4B7P7V.js +99 -0
- package/dist/{chunk-JY6FYXIT.js → chunk-STCQGCEQ.js} +6 -11
- package/dist/chunk-TIGW564L.js +628 -0
- package/dist/chunk-U4O6C46S.js +154 -0
- package/dist/{chunk-ITPEXLHA.js → chunk-URXDAUVH.js} +24 -5
- package/dist/chunk-VSL7KY3M.js +189 -0
- package/dist/{chunk-U55BGUAU.js → chunk-W4SPAEE7.js} +6 -6
- package/dist/chunk-WMGIIABP.js +15 -0
- package/dist/{chunk-33UGEQRT.js → chunk-X3SPPUFG.js} +151 -64
- package/dist/chunk-Y6VJKXGL.js +373 -0
- package/dist/{chunk-3WRJEKN4.js → chunk-ZN54U2OZ.js} +123 -10
- package/dist/cli/index.js +34 -24
- package/dist/commands/archive.js +3 -3
- package/dist/commands/backlog.js +3 -3
- package/dist/commands/blocked.js +3 -3
- package/dist/commands/canvas.d.ts +15 -0
- package/dist/commands/canvas.js +200 -0
- package/dist/commands/checkpoint.js +2 -2
- package/dist/commands/compat.js +2 -2
- package/dist/commands/context.js +8 -6
- package/dist/commands/doctor.d.ts +11 -7
- package/dist/commands/doctor.js +18 -16
- package/dist/commands/embed.js +5 -6
- package/dist/commands/entities.js +2 -2
- package/dist/commands/graph.js +4 -4
- package/dist/commands/inject.d.ts +1 -1
- package/dist/commands/inject.js +5 -6
- package/dist/commands/kanban.js +4 -4
- package/dist/commands/link.js +5 -5
- package/dist/commands/migrate-observations.js +4 -4
- package/dist/commands/observe.d.ts +0 -1
- package/dist/commands/observe.js +14 -13
- package/dist/commands/project.js +5 -5
- package/dist/commands/rebuild-embeddings.d.ts +21 -0
- package/dist/commands/rebuild-embeddings.js +91 -0
- package/dist/commands/rebuild.js +12 -11
- package/dist/commands/recover.js +3 -3
- package/dist/commands/reflect.js +6 -7
- package/dist/commands/repair-session.js +1 -1
- package/dist/commands/replay.js +14 -14
- package/dist/commands/session-recap.js +1 -1
- package/dist/commands/setup.d.ts +2 -89
- package/dist/commands/setup.js +3 -21
- package/dist/commands/shell-init.js +1 -1
- package/dist/commands/sleep.d.ts +1 -1
- package/dist/commands/sleep.js +20 -19
- package/dist/commands/status.d.ts +2 -0
- package/dist/commands/status.js +57 -35
- package/dist/commands/sync-bd.d.ts +10 -0
- package/dist/commands/sync-bd.js +10 -0
- package/dist/commands/tailscale.d.ts +52 -0
- package/dist/commands/tailscale.js +26 -0
- package/dist/commands/task.js +4 -4
- package/dist/commands/template.js +2 -2
- package/dist/commands/wake.d.ts +1 -1
- package/dist/commands/wake.js +11 -10
- package/dist/commands/workgraph.d.ts +124 -0
- package/dist/commands/workgraph.js +38 -0
- package/dist/index.d.ts +341 -191
- package/dist/index.js +446 -116
- package/dist/{inject-Bzi5E-By.d.ts → inject-DYUrDqQO.d.ts} +3 -3
- package/dist/ledger-B7g7jhqG.d.ts +44 -0
- package/dist/lib/auto-linker.js +2 -2
- package/dist/lib/canvas-layout.d.ts +115 -0
- package/dist/lib/canvas-layout.js +35 -0
- package/dist/lib/config.d.ts +27 -3
- package/dist/lib/config.js +4 -2
- package/dist/lib/entity-index.js +1 -1
- package/dist/lib/project-utils.js +4 -4
- package/dist/lib/session-repair.js +1 -1
- package/dist/lib/session-utils.js +1 -1
- package/dist/lib/tailscale.d.ts +225 -0
- package/dist/lib/tailscale.js +50 -0
- package/dist/lib/task-utils.js +3 -3
- package/dist/lib/template-engine.js +1 -1
- package/dist/lib/webdav.d.ts +109 -0
- package/dist/lib/webdav.js +35 -0
- package/dist/onnxruntime_binding-5QEF3SUC.node +0 -0
- package/dist/onnxruntime_binding-BKPKNEGC.node +0 -0
- package/dist/onnxruntime_binding-FMOXGIUT.node +0 -0
- package/dist/onnxruntime_binding-OI2KMXC5.node +0 -0
- package/dist/onnxruntime_binding-UX44MLAZ.node +0 -0
- package/dist/onnxruntime_binding-Y2W7N7WY.node +0 -0
- package/dist/openclaw-plugin.d.ts +8 -0
- package/dist/openclaw-plugin.js +14 -0
- package/dist/registry-BR4326o0.d.ts +30 -0
- package/dist/store-CA-6sKCJ.d.ts +34 -0
- package/dist/thread-B9LhXNU0.d.ts +41 -0
- package/dist/transformers.node-A2ZRORSQ.js +46775 -0
- package/dist/{types-Y2_Um2Ls.d.ts → types-BbWJoC1c.d.ts} +1 -44
- package/dist/workgraph/index.d.ts +5 -0
- package/dist/workgraph/index.js +23 -0
- package/dist/workgraph/ledger.d.ts +2 -0
- package/dist/workgraph/ledger.js +25 -0
- package/dist/workgraph/registry.d.ts +2 -0
- package/dist/workgraph/registry.js +19 -0
- package/dist/workgraph/store.d.ts +2 -0
- package/dist/workgraph/store.js +25 -0
- package/dist/workgraph/thread.d.ts +2 -0
- package/dist/workgraph/thread.js +25 -0
- package/dist/workgraph/types.d.ts +54 -0
- package/dist/workgraph/types.js +7 -0
- package/hooks/clawvault/HOOK.md +113 -0
- package/hooks/clawvault/handler.js +1561 -0
- package/hooks/clawvault/handler.test.js +510 -0
- package/hooks/clawvault/openclaw.plugin.json +72 -0
- package/openclaw.plugin.json +65 -38
- package/package.json +25 -22
- package/dist/chunk-3RG5ZIWI.js +0 -10
- package/dist/chunk-3ZIH425O.js +0 -871
- package/dist/chunk-6U6MK36V.js +0 -205
- package/dist/chunk-CMB7UL7C.js +0 -327
- package/dist/chunk-D2H45LON.js +0 -1074
- package/dist/chunk-E7MFQB6D.js +0 -163
- package/dist/chunk-GQSLDZTS.js +0 -560
- package/dist/chunk-MFM6K7PU.js +0 -374
- package/dist/chunk-MXSSG3QU.js +0 -42
- package/dist/chunk-OCGVIN3L.js +0 -88
- package/dist/chunk-PAH27GSN.js +0 -108
- package/dist/chunk-YCUNCH2I.js +0 -78
- package/dist/cli/index.cjs +0 -8584
- package/dist/cli/index.d.cts +0 -5
- package/dist/commands/archive.cjs +0 -287
- package/dist/commands/archive.d.cts +0 -11
- package/dist/commands/backlog.cjs +0 -721
- package/dist/commands/backlog.d.cts +0 -53
- package/dist/commands/blocked.cjs +0 -204
- package/dist/commands/blocked.d.cts +0 -26
- package/dist/commands/checkpoint.cjs +0 -244
- package/dist/commands/checkpoint.d.cts +0 -41
- package/dist/commands/compat.cjs +0 -294
- package/dist/commands/compat.d.cts +0 -28
- package/dist/commands/context.cjs +0 -2990
- package/dist/commands/context.d.cts +0 -2
- package/dist/commands/doctor.cjs +0 -2986
- package/dist/commands/doctor.d.cts +0 -21
- package/dist/commands/embed.cjs +0 -232
- package/dist/commands/embed.d.cts +0 -17
- package/dist/commands/entities.cjs +0 -141
- package/dist/commands/entities.d.cts +0 -7
- package/dist/commands/graph.cjs +0 -501
- package/dist/commands/graph.d.cts +0 -21
- package/dist/commands/inject.cjs +0 -1636
- package/dist/commands/inject.d.cts +0 -2
- package/dist/commands/kanban.cjs +0 -884
- package/dist/commands/kanban.d.cts +0 -63
- package/dist/commands/link.cjs +0 -965
- package/dist/commands/link.d.cts +0 -11
- package/dist/commands/migrate-observations.cjs +0 -362
- package/dist/commands/migrate-observations.d.cts +0 -19
- package/dist/commands/observe.cjs +0 -4099
- package/dist/commands/observe.d.cts +0 -23
- package/dist/commands/project.cjs +0 -1341
- package/dist/commands/project.d.cts +0 -85
- package/dist/commands/rebuild.cjs +0 -3136
- package/dist/commands/rebuild.d.cts +0 -11
- package/dist/commands/recover.cjs +0 -361
- package/dist/commands/recover.d.cts +0 -38
- package/dist/commands/reflect.cjs +0 -1008
- package/dist/commands/reflect.d.cts +0 -11
- package/dist/commands/repair-session.cjs +0 -457
- package/dist/commands/repair-session.d.cts +0 -38
- package/dist/commands/replay.cjs +0 -4103
- package/dist/commands/replay.d.cts +0 -16
- package/dist/commands/session-recap.cjs +0 -353
- package/dist/commands/session-recap.d.cts +0 -27
- package/dist/commands/setup.cjs +0 -1278
- package/dist/commands/setup.d.cts +0 -99
- package/dist/commands/shell-init.cjs +0 -75
- package/dist/commands/shell-init.d.cts +0 -7
- package/dist/commands/sleep.cjs +0 -6029
- package/dist/commands/sleep.d.cts +0 -36
- package/dist/commands/status.cjs +0 -2737
- package/dist/commands/status.d.cts +0 -52
- package/dist/commands/task.cjs +0 -1236
- package/dist/commands/task.d.cts +0 -97
- package/dist/commands/template.cjs +0 -457
- package/dist/commands/template.d.cts +0 -36
- package/dist/commands/wake.cjs +0 -2627
- package/dist/commands/wake.d.cts +0 -22
- package/dist/context-BUGaWpyL.d.cts +0 -46
- package/dist/index.cjs +0 -12373
- package/dist/index.d.cts +0 -854
- package/dist/inject-Bzi5E-By.d.cts +0 -137
- package/dist/lib/auto-linker.cjs +0 -176
- package/dist/lib/auto-linker.d.cts +0 -26
- package/dist/lib/config.cjs +0 -78
- package/dist/lib/config.d.cts +0 -11
- package/dist/lib/entity-index.cjs +0 -84
- package/dist/lib/entity-index.d.cts +0 -26
- package/dist/lib/project-utils.cjs +0 -864
- package/dist/lib/project-utils.d.cts +0 -97
- package/dist/lib/session-repair.cjs +0 -239
- package/dist/lib/session-repair.d.cts +0 -110
- package/dist/lib/session-utils.cjs +0 -209
- package/dist/lib/session-utils.d.cts +0 -63
- package/dist/lib/task-utils.cjs +0 -1137
- package/dist/lib/task-utils.d.cts +0 -208
- package/dist/lib/template-engine.cjs +0 -47
- package/dist/lib/template-engine.d.cts +0 -11
- package/dist/plugin/index.cjs +0 -1907
- package/dist/plugin/index.d.cts +0 -36
- package/dist/plugin/index.d.ts +0 -36
- package/dist/plugin/index.js +0 -572
- package/dist/plugin/inject.cjs +0 -356
- package/dist/plugin/inject.d.cts +0 -54
- package/dist/plugin/inject.d.ts +0 -54
- package/dist/plugin/inject.js +0 -17
- package/dist/plugin/observe.cjs +0 -631
- package/dist/plugin/observe.d.cts +0 -39
- package/dist/plugin/observe.d.ts +0 -39
- package/dist/plugin/observe.js +0 -18
- package/dist/plugin/templates.cjs +0 -593
- package/dist/plugin/templates.d.cts +0 -52
- package/dist/plugin/templates.d.ts +0 -52
- package/dist/plugin/templates.js +0 -25
- package/dist/plugin/types.cjs +0 -18
- package/dist/plugin/types.d.cts +0 -209
- package/dist/plugin/types.d.ts +0 -209
- package/dist/plugin/types.js +0 -0
- package/dist/plugin/vault.cjs +0 -927
- package/dist/plugin/vault.d.cts +0 -68
- package/dist/plugin/vault.d.ts +0 -68
- package/dist/plugin/vault.js +0 -22
- package/dist/types-Y2_Um2Ls.d.cts +0 -205
- package/templates/memory-event.md +0 -67
- package/templates/party.md +0 -63
- package/templates/primitive-registry.yaml +0 -551
- package/templates/run.md +0 -68
- package/templates/trigger.md +0 -68
- package/templates/workspace.md +0 -50
|
@@ -1,13 +1,16 @@
|
|
|
1
1
|
import {
|
|
2
2
|
buildEntityIndex
|
|
3
3
|
} from "./chunk-J7ZWCI2C.js";
|
|
4
|
+
import {
|
|
5
|
+
extractRawWikiLinks,
|
|
6
|
+
normalizeWikiLinkTarget
|
|
7
|
+
} from "./chunk-33DOSHTA.js";
|
|
4
8
|
|
|
5
9
|
// src/lib/backlinks.ts
|
|
6
10
|
import * as fs from "fs";
|
|
7
11
|
import * as path from "path";
|
|
8
12
|
var CLAWVAULT_DIR = ".clawvault";
|
|
9
13
|
var BACKLINKS_FILE = "backlinks.json";
|
|
10
|
-
var WIKI_LINK_REGEX = /\[\[([^\]]+)\]\]/g;
|
|
11
14
|
function ensureClawvaultDir(vaultPath) {
|
|
12
15
|
const dir = path.join(vaultPath, CLAWVAULT_DIR);
|
|
13
16
|
if (!fs.existsSync(dir)) {
|
|
@@ -20,29 +23,47 @@ function toVaultId(vaultPath, filePath) {
|
|
|
20
23
|
return relative2.split(path.sep).join("/");
|
|
21
24
|
}
|
|
22
25
|
function normalizeLinkTarget(raw) {
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
const
|
|
29
|
-
if (
|
|
30
|
-
|
|
31
|
-
}
|
|
32
|
-
if (target.startsWith("#")) return "";
|
|
33
|
-
const hashIndex = target.indexOf("#");
|
|
34
|
-
if (hashIndex !== -1) {
|
|
35
|
-
target = target.slice(0, hashIndex);
|
|
26
|
+
return normalizeWikiLinkTarget(raw);
|
|
27
|
+
}
|
|
28
|
+
function normalizeLookupCandidate(value) {
|
|
29
|
+
const normalized = normalizeLinkTarget(value);
|
|
30
|
+
if (!normalized) return "";
|
|
31
|
+
const resolved = path.posix.normalize(normalized).replace(/^\/+/, "");
|
|
32
|
+
if (!resolved || resolved === "." || resolved.startsWith("../")) {
|
|
33
|
+
return "";
|
|
36
34
|
}
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
35
|
+
return resolved;
|
|
36
|
+
}
|
|
37
|
+
function buildLookupCandidates(target, sourceId) {
|
|
38
|
+
const candidates = [];
|
|
39
|
+
const sourceDir = path.posix.dirname(sourceId);
|
|
40
|
+
const hasSourceDir = sourceDir !== ".";
|
|
41
|
+
const isRelativeTarget = target.startsWith("./") || target.startsWith("../");
|
|
42
|
+
const addCandidate = (candidate) => {
|
|
43
|
+
const normalized = normalizeLookupCandidate(candidate);
|
|
44
|
+
if (!normalized || candidates.includes(normalized)) return;
|
|
45
|
+
candidates.push(normalized);
|
|
46
|
+
};
|
|
47
|
+
if (isRelativeTarget) {
|
|
48
|
+
if (hasSourceDir) {
|
|
49
|
+
addCandidate(path.posix.join(sourceDir, target));
|
|
50
|
+
} else {
|
|
51
|
+
addCandidate(target);
|
|
52
|
+
}
|
|
53
|
+
if (target.startsWith("./")) {
|
|
54
|
+
addCandidate(target.slice(2));
|
|
55
|
+
}
|
|
56
|
+
return candidates;
|
|
41
57
|
}
|
|
42
|
-
if (target.
|
|
43
|
-
|
|
58
|
+
if (!target.includes("/")) {
|
|
59
|
+
if (hasSourceDir) {
|
|
60
|
+
addCandidate(`${sourceDir}/${target}`);
|
|
61
|
+
}
|
|
62
|
+
addCandidate(target);
|
|
63
|
+
return candidates;
|
|
44
64
|
}
|
|
45
|
-
|
|
65
|
+
addCandidate(target);
|
|
66
|
+
return candidates;
|
|
46
67
|
}
|
|
47
68
|
function listMarkdownFiles(vaultPath) {
|
|
48
69
|
const files = [];
|
|
@@ -75,11 +96,16 @@ function buildKnownIds(vaultPath, files) {
|
|
|
75
96
|
}
|
|
76
97
|
return { ids, idsLower };
|
|
77
98
|
}
|
|
78
|
-
function resolveTarget(target, known, entityIndex) {
|
|
99
|
+
function resolveTarget(target, sourceId, known, entityIndex) {
|
|
79
100
|
if (!target) return null;
|
|
80
|
-
|
|
101
|
+
for (const candidate of buildLookupCandidates(target, sourceId)) {
|
|
102
|
+
if (known.ids.has(candidate)) return candidate;
|
|
103
|
+
const lowerCandidate = candidate.toLowerCase();
|
|
104
|
+
if (known.idsLower.has(lowerCandidate)) {
|
|
105
|
+
return known.idsLower.get(lowerCandidate);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
81
108
|
const lower = target.toLowerCase();
|
|
82
|
-
if (known.idsLower.has(lower)) return known.idsLower.get(lower);
|
|
83
109
|
if (entityIndex?.entries.has(lower)) return entityIndex.entries.get(lower);
|
|
84
110
|
return null;
|
|
85
111
|
}
|
|
@@ -93,12 +119,12 @@ function scanVaultLinks(vaultPath, options = {}) {
|
|
|
93
119
|
for (const file of files) {
|
|
94
120
|
const sourceId = toVaultId(vaultPath, file);
|
|
95
121
|
const content = fs.readFileSync(file, "utf-8");
|
|
96
|
-
const matches = content
|
|
122
|
+
const matches = extractRawWikiLinks(content);
|
|
97
123
|
linkCount += matches.length;
|
|
98
124
|
for (const match of matches) {
|
|
99
125
|
const target = normalizeLinkTarget(match);
|
|
100
126
|
if (!target) continue;
|
|
101
|
-
const resolved = resolveTarget(target, known, entityIndex);
|
|
127
|
+
const resolved = resolveTarget(target, sourceId, known, entityIndex);
|
|
102
128
|
if (!resolved) {
|
|
103
129
|
orphans.push({ source: sourceId, target });
|
|
104
130
|
continue;
|
|
@@ -0,0 +1,447 @@
|
|
|
1
|
+
// src/lib/fact-extractor.ts
|
|
2
|
+
function normalizeEntity(name) {
|
|
3
|
+
return name.toLowerCase().replace(/[^a-z0-9\s]/g, "").replace(/\s+/g, " ").trim();
|
|
4
|
+
}
|
|
5
|
+
function factId(entity, relation, value) {
|
|
6
|
+
const key = `${normalizeEntity(entity)}::${relation.toLowerCase()}::${value.toLowerCase().trim()}`;
|
|
7
|
+
let hash = 0;
|
|
8
|
+
for (let i = 0; i < key.length; i++) {
|
|
9
|
+
const char = key.charCodeAt(i);
|
|
10
|
+
hash = (hash << 5) - hash + char;
|
|
11
|
+
hash = hash & hash;
|
|
12
|
+
}
|
|
13
|
+
return Math.abs(hash).toString(36);
|
|
14
|
+
}
|
|
15
|
+
var PREFERENCE_PATTERNS = [
|
|
16
|
+
{
|
|
17
|
+
// "I prefer X" / "I like X" / "I love X" / "I enjoy X"
|
|
18
|
+
pattern: /\b(?:i|user|they)\s+(?:prefer|like|love|enjoy|want|favor)s?\s+(.+?)(?:\.|,|$)/i,
|
|
19
|
+
extract: (m) => ({
|
|
20
|
+
entity: "user",
|
|
21
|
+
relation: "prefers",
|
|
22
|
+
value: m[1].trim(),
|
|
23
|
+
category: "preference"
|
|
24
|
+
})
|
|
25
|
+
},
|
|
26
|
+
{
|
|
27
|
+
// "my favorite X is Y"
|
|
28
|
+
pattern: /\bmy\s+(?:favorite|favourite|preferred)\s+(\w+)\s+(?:is|are)\s+(.+?)(?:\.|,|$)/i,
|
|
29
|
+
extract: (m) => ({
|
|
30
|
+
entity: "user",
|
|
31
|
+
relation: `favorite_${m[1].toLowerCase()}`,
|
|
32
|
+
value: m[2].trim(),
|
|
33
|
+
category: "preference"
|
|
34
|
+
})
|
|
35
|
+
},
|
|
36
|
+
{
|
|
37
|
+
// "I don't like X" / "I hate X" / "I dislike X"
|
|
38
|
+
pattern: /\b(?:i|user)\s+(?:don'?t\s+like|hate|dislike|avoid)s?\s+(.+?)(?:\.|,|$)/i,
|
|
39
|
+
extract: (m) => ({
|
|
40
|
+
entity: "user",
|
|
41
|
+
relation: "dislikes",
|
|
42
|
+
value: m[1].trim(),
|
|
43
|
+
category: "preference"
|
|
44
|
+
})
|
|
45
|
+
},
|
|
46
|
+
{
|
|
47
|
+
// "I'm allergic to X" / "I have an allergy to X"
|
|
48
|
+
pattern: /\b(?:i'?m|i\s+am|i\s+have)\s+(?:an?\s+)?allerg(?:ic|y)\s+(?:to\s+)?(.+?)(?:\.|,|$)/i,
|
|
49
|
+
extract: (m) => ({
|
|
50
|
+
entity: "user",
|
|
51
|
+
relation: "allergic_to",
|
|
52
|
+
value: m[1].trim(),
|
|
53
|
+
category: "preference"
|
|
54
|
+
})
|
|
55
|
+
}
|
|
56
|
+
];
|
|
57
|
+
var FACT_PATTERNS = [
|
|
58
|
+
{
|
|
59
|
+
// "X works at Y" / "X is employed at Y"
|
|
60
|
+
pattern: /\b(\w+(?:\s+\w+)?)\s+(?:works?\s+(?:at|for)|is\s+employed\s+(?:at|by))\s+(.+?)(?:\.|,|$)/i,
|
|
61
|
+
extract: (m) => ({
|
|
62
|
+
entity: m[1].trim(),
|
|
63
|
+
relation: "works_at",
|
|
64
|
+
value: m[2].trim(),
|
|
65
|
+
category: "fact"
|
|
66
|
+
})
|
|
67
|
+
},
|
|
68
|
+
{
|
|
69
|
+
// "X lives in Y" / "X moved to Y"
|
|
70
|
+
pattern: /\b(\w+(?:\s+\w+)?)\s+(?:live[sd]?\s+in|moved?\s+to|relocated?\s+to)\s+(.+?)(?:\.|,|$)/i,
|
|
71
|
+
extract: (m) => ({
|
|
72
|
+
entity: m[1].trim(),
|
|
73
|
+
relation: "lives_in",
|
|
74
|
+
value: m[2].trim(),
|
|
75
|
+
category: "fact"
|
|
76
|
+
})
|
|
77
|
+
},
|
|
78
|
+
{
|
|
79
|
+
// "X is Y years old" / "X's age is Y"
|
|
80
|
+
pattern: /\b(\w+(?:\s+\w+)?)\s+(?:is|turned)\s+(\d+)\s+years?\s+old/i,
|
|
81
|
+
extract: (m) => ({
|
|
82
|
+
entity: m[1].trim(),
|
|
83
|
+
relation: "age",
|
|
84
|
+
value: m[2],
|
|
85
|
+
category: "fact"
|
|
86
|
+
})
|
|
87
|
+
},
|
|
88
|
+
{
|
|
89
|
+
// "X bought Y" / "X purchased Y"
|
|
90
|
+
pattern: /\b(\w+(?:\s+\w+)?)\s+(?:bought|purchased|got|acquired)\s+(?:a\s+|an\s+|the\s+)?(.+?)(?:\s+for\s+\$?([\d,.]+))?(?:\.|,|$)/i,
|
|
91
|
+
extract: (m) => ({
|
|
92
|
+
entity: m[1].trim(),
|
|
93
|
+
relation: "bought",
|
|
94
|
+
value: m[3] ? `${m[2].trim()} ($${m[3]})` : m[2].trim(),
|
|
95
|
+
category: "event"
|
|
96
|
+
})
|
|
97
|
+
},
|
|
98
|
+
{
|
|
99
|
+
// "X spent $Y on Z"
|
|
100
|
+
pattern: /\b(\w+(?:\s+\w+)?)\s+spent\s+\$?([\d,.]+)\s+on\s+(.+?)(?:\.|,|$)/i,
|
|
101
|
+
extract: (m) => ({
|
|
102
|
+
entity: m[1].trim(),
|
|
103
|
+
relation: "spent_on",
|
|
104
|
+
value: `$${m[2]} on ${m[3].trim()}`,
|
|
105
|
+
category: "event"
|
|
106
|
+
})
|
|
107
|
+
}
|
|
108
|
+
];
|
|
109
|
+
var DECISION_PATTERNS = [
|
|
110
|
+
{
|
|
111
|
+
// "decided to X" / "we decided X"
|
|
112
|
+
pattern: /\b(?:i|we|user)\s+decided\s+(?:to\s+)?(.+?)(?:\.|,|$)/i,
|
|
113
|
+
extract: (m) => ({
|
|
114
|
+
entity: "user",
|
|
115
|
+
relation: "decided",
|
|
116
|
+
value: m[1].trim(),
|
|
117
|
+
category: "decision"
|
|
118
|
+
})
|
|
119
|
+
},
|
|
120
|
+
{
|
|
121
|
+
// "chose X over Y"
|
|
122
|
+
pattern: /\b(?:i|we|user)\s+chose\s+(.+?)\s+over\s+(.+?)(?:\.|,|$)/i,
|
|
123
|
+
extract: (m) => ({
|
|
124
|
+
entity: "user",
|
|
125
|
+
relation: "chose",
|
|
126
|
+
value: `${m[1].trim()} (over ${m[2].trim()})`,
|
|
127
|
+
category: "decision"
|
|
128
|
+
})
|
|
129
|
+
}
|
|
130
|
+
];
|
|
131
|
+
var ALL_PATTERNS = [...PREFERENCE_PATTERNS, ...FACT_PATTERNS, ...DECISION_PATTERNS];
|
|
132
|
+
function extractFactsRuleBased(text, source, timestamp) {
|
|
133
|
+
const facts = [];
|
|
134
|
+
const now = timestamp || (/* @__PURE__ */ new Date()).toISOString();
|
|
135
|
+
const sentences = text.split(/[.!?\n]+/).filter((s) => s.trim().length > 5);
|
|
136
|
+
for (const sentence of sentences) {
|
|
137
|
+
const trimmed = sentence.trim();
|
|
138
|
+
for (const rule of ALL_PATTERNS) {
|
|
139
|
+
const match = trimmed.match(rule.pattern);
|
|
140
|
+
if (match) {
|
|
141
|
+
const extracted = rule.extract(match);
|
|
142
|
+
if (extracted && extracted.value.length > 1 && extracted.value.length < 200) {
|
|
143
|
+
facts.push({
|
|
144
|
+
id: factId(extracted.entity, extracted.relation, extracted.value),
|
|
145
|
+
entity: extracted.entity,
|
|
146
|
+
entityNorm: normalizeEntity(extracted.entity),
|
|
147
|
+
relation: extracted.relation,
|
|
148
|
+
value: extracted.value,
|
|
149
|
+
validFrom: now,
|
|
150
|
+
validUntil: null,
|
|
151
|
+
confidence: 0.7,
|
|
152
|
+
// Rule-based gets moderate confidence
|
|
153
|
+
category: extracted.category,
|
|
154
|
+
source,
|
|
155
|
+
rawText: trimmed
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
return facts;
|
|
162
|
+
}
|
|
163
|
+
var EXTRACTION_PROMPT = `Extract structured facts from the following text. Return ONLY a JSON array of objects with these fields:
|
|
164
|
+
- entity: the subject (person, place, thing, or "user" for the speaker/first person)
|
|
165
|
+
- relation: the relationship type (see examples below)
|
|
166
|
+
- value: the object of the relation
|
|
167
|
+
- category: one of "preference", "fact", "decision", "entity", "event"
|
|
168
|
+
- confidence: 0.0 to 1.0
|
|
169
|
+
|
|
170
|
+
PREFERENCE EXTRACTION (critical \u2014 extract ALL of these):
|
|
171
|
+
- Likes, dislikes, preferences, favorites: "prefers", "likes", "dislikes", "favorite"
|
|
172
|
+
- Food/dietary: "allergic_to", "dietary_restriction", "favorite_food", "dislikes_food"
|
|
173
|
+
- Habits/routines: "habit", "routine", "schedule"
|
|
174
|
+
- Communication style: "prefers_communication", "timezone", "language"
|
|
175
|
+
- Tools/tech: "uses_tool", "prefers_editor", "prefers_language"
|
|
176
|
+
|
|
177
|
+
TEMPORAL FACTS (include dates when present):
|
|
178
|
+
- Include specific dates, times, relative references ("last Tuesday" = resolve if possible)
|
|
179
|
+
- Events: "happened_on", "started_on", "ended_on", "deadline"
|
|
180
|
+
- Use ISO format for dates when possible
|
|
181
|
+
|
|
182
|
+
OTHER RELATIONS:
|
|
183
|
+
- Identity: "works_at", "lives_in", "age", "role", "email", "phone"
|
|
184
|
+
- Actions: "bought", "spent_on", "created", "visited", "completed"
|
|
185
|
+
- Decisions: "decided", "chose", "rejected", "approved"
|
|
186
|
+
- Knowledge: "knows_about", "studied", "expertise"
|
|
187
|
+
|
|
188
|
+
Examples:
|
|
189
|
+
|
|
190
|
+
Input: "I really love Thai food, especially pad thai. I'm allergic to shellfish though."
|
|
191
|
+
Output: [
|
|
192
|
+
{"entity": "user", "relation": "favorite_food", "value": "Thai food, especially pad thai", "category": "preference", "confidence": 0.95},
|
|
193
|
+
{"entity": "user", "relation": "allergic_to", "value": "shellfish", "category": "preference", "confidence": 0.99}
|
|
194
|
+
]
|
|
195
|
+
|
|
196
|
+
Input: "We decided on Tuesday to use PostgreSQL for the new project. John will lead the backend team."
|
|
197
|
+
Output: [
|
|
198
|
+
{"entity": "team", "relation": "decided", "value": "use PostgreSQL for the new project", "category": "decision", "confidence": 0.95},
|
|
199
|
+
{"entity": "John", "relation": "role", "value": "backend team lead", "category": "fact", "confidence": 0.9}
|
|
200
|
+
]
|
|
201
|
+
|
|
202
|
+
Input: "My morning routine is: wake up at 6am, coffee, then gym. I prefer working out before work."
|
|
203
|
+
Output: [
|
|
204
|
+
{"entity": "user", "relation": "routine", "value": "wake up at 6am, coffee, then gym", "category": "preference", "confidence": 0.9},
|
|
205
|
+
{"entity": "user", "relation": "prefers", "value": "working out before work", "category": "preference", "confidence": 0.9}
|
|
206
|
+
]
|
|
207
|
+
|
|
208
|
+
Rules:
|
|
209
|
+
- Extract ALL facts, preferences, decisions, and events \u2014 err on the side of extracting more
|
|
210
|
+
- For preferences, use "user" as entity unless a specific person is named
|
|
211
|
+
- For monetary amounts, include the currency symbol
|
|
212
|
+
- Be precise \u2014 only extract what is explicitly stated or strongly implied
|
|
213
|
+
- Return empty array [] if no extractable facts found
|
|
214
|
+
|
|
215
|
+
Text:
|
|
216
|
+
`;
|
|
217
|
+
async function extractFactsLlm(text, source, timestamp, llmFn) {
|
|
218
|
+
if (!llmFn) {
|
|
219
|
+
return extractFactsRuleBased(text, source, timestamp);
|
|
220
|
+
}
|
|
221
|
+
const now = timestamp || (/* @__PURE__ */ new Date()).toISOString();
|
|
222
|
+
try {
|
|
223
|
+
const response = await llmFn(EXTRACTION_PROMPT + text);
|
|
224
|
+
const jsonMatch = response.match(/\[[\s\S]*?\]/);
|
|
225
|
+
if (!jsonMatch) {
|
|
226
|
+
return extractFactsRuleBased(text, source, timestamp);
|
|
227
|
+
}
|
|
228
|
+
const parsed = JSON.parse(jsonMatch[0]);
|
|
229
|
+
return parsed.map((f) => ({
|
|
230
|
+
id: factId(f.entity, f.relation, f.value),
|
|
231
|
+
entity: f.entity,
|
|
232
|
+
entityNorm: normalizeEntity(f.entity),
|
|
233
|
+
relation: f.relation,
|
|
234
|
+
value: f.value,
|
|
235
|
+
validFrom: now,
|
|
236
|
+
validUntil: null,
|
|
237
|
+
confidence: Math.min(1, Math.max(0, f.confidence || 0.8)),
|
|
238
|
+
category: f.category || "fact",
|
|
239
|
+
source,
|
|
240
|
+
rawText: text.substring(0, 500)
|
|
241
|
+
}));
|
|
242
|
+
} catch {
|
|
243
|
+
return extractFactsRuleBased(text, source, timestamp);
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
// src/lib/fact-store.ts
|
|
248
|
+
import * as fs from "fs";
|
|
249
|
+
import * as path from "path";
|
|
250
|
+
var FactStore = class {
|
|
251
|
+
facts = /* @__PURE__ */ new Map();
|
|
252
|
+
byEntity = /* @__PURE__ */ new Map();
|
|
253
|
+
byRelation = /* @__PURE__ */ new Map();
|
|
254
|
+
byCategory = /* @__PURE__ */ new Map();
|
|
255
|
+
factsPath;
|
|
256
|
+
dirty = false;
|
|
257
|
+
constructor(vaultPath) {
|
|
258
|
+
this.factsPath = path.join(vaultPath, ".clawvault", "facts.jsonl");
|
|
259
|
+
}
|
|
260
|
+
/** Load facts from disk */
|
|
261
|
+
load() {
|
|
262
|
+
this.facts.clear();
|
|
263
|
+
this.byEntity.clear();
|
|
264
|
+
this.byRelation.clear();
|
|
265
|
+
this.byCategory.clear();
|
|
266
|
+
if (!fs.existsSync(this.factsPath)) return;
|
|
267
|
+
const lines = fs.readFileSync(this.factsPath, "utf-8").split("\n").filter((l) => l.trim());
|
|
268
|
+
for (const line of lines) {
|
|
269
|
+
try {
|
|
270
|
+
const fact = JSON.parse(line);
|
|
271
|
+
this.indexFact(fact);
|
|
272
|
+
} catch {
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
/** Add facts with conflict resolution. Returns number of conflicts resolved. */
|
|
277
|
+
addFacts(newFacts) {
|
|
278
|
+
let conflicts = 0;
|
|
279
|
+
for (const fact of newFacts) {
|
|
280
|
+
const existing = this.findConflict(fact);
|
|
281
|
+
if (existing) {
|
|
282
|
+
existing.validUntil = fact.validFrom;
|
|
283
|
+
conflicts++;
|
|
284
|
+
}
|
|
285
|
+
this.indexFact(fact);
|
|
286
|
+
this.dirty = true;
|
|
287
|
+
}
|
|
288
|
+
return conflicts;
|
|
289
|
+
}
|
|
290
|
+
/** Find an existing fact that conflicts with the new one */
|
|
291
|
+
findConflict(newFact) {
|
|
292
|
+
const entityFacts = this.byEntity.get(newFact.entityNorm);
|
|
293
|
+
if (!entityFacts) return null;
|
|
294
|
+
for (const id of entityFacts) {
|
|
295
|
+
const existing = this.facts.get(id);
|
|
296
|
+
if (!existing || existing.validUntil) continue;
|
|
297
|
+
if (existing.relation === newFact.relation) {
|
|
298
|
+
if (this.isSimilarValue(existing.value, newFact.value)) {
|
|
299
|
+
return existing;
|
|
300
|
+
}
|
|
301
|
+
if (this.isExclusiveRelation(newFact.relation)) {
|
|
302
|
+
return existing;
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
return null;
|
|
307
|
+
}
|
|
308
|
+
/** Check if two values are similar enough to be considered the same fact */
|
|
309
|
+
isSimilarValue(a, b) {
|
|
310
|
+
const na = a.toLowerCase().trim();
|
|
311
|
+
const nb = b.toLowerCase().trim();
|
|
312
|
+
if (na === nb) return true;
|
|
313
|
+
if (na.includes(nb) || nb.includes(na)) return true;
|
|
314
|
+
return false;
|
|
315
|
+
}
|
|
316
|
+
/** Relations where only one value can be active at a time */
|
|
317
|
+
isExclusiveRelation(relation) {
|
|
318
|
+
const exclusive = /* @__PURE__ */ new Set([
|
|
319
|
+
"lives_in",
|
|
320
|
+
"works_at",
|
|
321
|
+
"age",
|
|
322
|
+
"favorite_color",
|
|
323
|
+
"favorite_food",
|
|
324
|
+
"favorite_restaurant",
|
|
325
|
+
"favorite_movie",
|
|
326
|
+
"favorite_book",
|
|
327
|
+
"favorite_music",
|
|
328
|
+
"favorite_sport",
|
|
329
|
+
"job_title",
|
|
330
|
+
"employer",
|
|
331
|
+
"marital_status",
|
|
332
|
+
"city",
|
|
333
|
+
"country"
|
|
334
|
+
]);
|
|
335
|
+
return exclusive.has(relation);
|
|
336
|
+
}
|
|
337
|
+
/** Index a fact in all lookup maps */
|
|
338
|
+
indexFact(fact) {
|
|
339
|
+
this.facts.set(fact.id, fact);
|
|
340
|
+
if (!this.byEntity.has(fact.entityNorm)) {
|
|
341
|
+
this.byEntity.set(fact.entityNorm, /* @__PURE__ */ new Set());
|
|
342
|
+
}
|
|
343
|
+
this.byEntity.get(fact.entityNorm).add(fact.id);
|
|
344
|
+
if (!this.byRelation.has(fact.relation)) {
|
|
345
|
+
this.byRelation.set(fact.relation, /* @__PURE__ */ new Set());
|
|
346
|
+
}
|
|
347
|
+
this.byRelation.get(fact.relation).add(fact.id);
|
|
348
|
+
if (!this.byCategory.has(fact.category)) {
|
|
349
|
+
this.byCategory.set(fact.category, /* @__PURE__ */ new Set());
|
|
350
|
+
}
|
|
351
|
+
this.byCategory.get(fact.category).add(fact.id);
|
|
352
|
+
}
|
|
353
|
+
/** Save facts to disk (full rewrite for consistency) */
|
|
354
|
+
save() {
|
|
355
|
+
if (!this.dirty && fs.existsSync(this.factsPath)) return;
|
|
356
|
+
const dir = path.dirname(this.factsPath);
|
|
357
|
+
if (!fs.existsSync(dir)) {
|
|
358
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
359
|
+
}
|
|
360
|
+
const lines = Array.from(this.facts.values()).map((f) => JSON.stringify(f)).join("\n");
|
|
361
|
+
fs.writeFileSync(this.factsPath, lines + "\n", "utf-8");
|
|
362
|
+
this.dirty = false;
|
|
363
|
+
}
|
|
364
|
+
/** Append new facts to disk (faster than full rewrite) */
|
|
365
|
+
append(facts) {
|
|
366
|
+
const dir = path.dirname(this.factsPath);
|
|
367
|
+
if (!fs.existsSync(dir)) {
|
|
368
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
369
|
+
}
|
|
370
|
+
const lines = facts.map((f) => JSON.stringify(f)).join("\n");
|
|
371
|
+
fs.appendFileSync(this.factsPath, lines + "\n", "utf-8");
|
|
372
|
+
}
|
|
373
|
+
// ─── Query methods ──────────────────────────────────────────────────────
|
|
374
|
+
/** Get all active facts for an entity */
|
|
375
|
+
getEntityFacts(entity) {
|
|
376
|
+
const norm = normalizeEntity(entity);
|
|
377
|
+
const ids = this.byEntity.get(norm);
|
|
378
|
+
if (!ids) return [];
|
|
379
|
+
return Array.from(ids).map((id) => this.facts.get(id)).filter((f) => f && !f.validUntil);
|
|
380
|
+
}
|
|
381
|
+
/** Get all active facts for a relation */
|
|
382
|
+
getRelationFacts(relation) {
|
|
383
|
+
const ids = this.byRelation.get(relation);
|
|
384
|
+
if (!ids) return [];
|
|
385
|
+
return Array.from(ids).map((id) => this.facts.get(id)).filter((f) => f && !f.validUntil);
|
|
386
|
+
}
|
|
387
|
+
/** Get all active facts in a category */
|
|
388
|
+
getCategoryFacts(category) {
|
|
389
|
+
const ids = this.byCategory.get(category);
|
|
390
|
+
if (!ids) return [];
|
|
391
|
+
return Array.from(ids).map((id) => this.facts.get(id)).filter((f) => f && !f.validUntil);
|
|
392
|
+
}
|
|
393
|
+
/** Get all active preferences */
|
|
394
|
+
getPreferences() {
|
|
395
|
+
return this.getCategoryFacts("preference");
|
|
396
|
+
}
|
|
397
|
+
/** Search facts by text query (simple keyword match) */
|
|
398
|
+
searchFacts(query) {
|
|
399
|
+
const terms = query.toLowerCase().split(/\s+/);
|
|
400
|
+
const results = [];
|
|
401
|
+
for (const fact of this.facts.values()) {
|
|
402
|
+
if (fact.validUntil) continue;
|
|
403
|
+
const text = `${fact.entity} ${fact.relation} ${fact.value} ${fact.rawText}`.toLowerCase();
|
|
404
|
+
const matches = terms.filter((t) => text.includes(t)).length;
|
|
405
|
+
if (matches >= Math.ceil(terms.length * 0.5)) {
|
|
406
|
+
results.push(fact);
|
|
407
|
+
}
|
|
408
|
+
}
|
|
409
|
+
return results;
|
|
410
|
+
}
|
|
411
|
+
/** Get facts valid at a specific time */
|
|
412
|
+
getFactsAt(timestamp) {
|
|
413
|
+
const t = new Date(timestamp).getTime();
|
|
414
|
+
const results = [];
|
|
415
|
+
for (const fact of this.facts.values()) {
|
|
416
|
+
const from = new Date(fact.validFrom).getTime();
|
|
417
|
+
const until = fact.validUntil ? new Date(fact.validUntil).getTime() : Infinity;
|
|
418
|
+
if (t >= from && t < until) {
|
|
419
|
+
results.push(fact);
|
|
420
|
+
}
|
|
421
|
+
}
|
|
422
|
+
return results;
|
|
423
|
+
}
|
|
424
|
+
/** Get stats */
|
|
425
|
+
stats() {
|
|
426
|
+
const active = Array.from(this.facts.values()).filter((f) => !f.validUntil);
|
|
427
|
+
return {
|
|
428
|
+
totalFacts: this.facts.size,
|
|
429
|
+
activeFacts: active.length,
|
|
430
|
+
supersededFacts: this.facts.size - active.length,
|
|
431
|
+
entities: this.byEntity.size,
|
|
432
|
+
relations: this.byRelation.size
|
|
433
|
+
};
|
|
434
|
+
}
|
|
435
|
+
/** Get all facts (for testing/debugging) */
|
|
436
|
+
getAllFacts() {
|
|
437
|
+
return Array.from(this.facts.values());
|
|
438
|
+
}
|
|
439
|
+
};
|
|
440
|
+
|
|
441
|
+
export {
|
|
442
|
+
normalizeEntity,
|
|
443
|
+
factId,
|
|
444
|
+
extractFactsRuleBased,
|
|
445
|
+
extractFactsLlm,
|
|
446
|
+
FactStore
|
|
447
|
+
};
|