clawvault 3.1.0 → 3.2.1
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 +451 -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-C7OK5WKP.js → chunk-2JQ3O2YL.js} +4 -4
- package/dist/{chunk-VR5NE7PZ.js → chunk-2RAZ4ZFE.js} +1 -1
- package/dist/{chunk-F2JEUD4J.js → chunk-4ITRXIVT.js} +5 -7
- package/dist/{chunk-GUKMRGM7.js → chunk-4OXMU5S2.js} +1 -1
- package/dist/chunk-5PJ4STIC.js +465 -0
- package/dist/{chunk-62YTUT6J.js → chunk-AZYOKJYC.js} +2 -2
- package/dist/chunk-BSJ6RIT7.js +447 -0
- package/dist/chunk-ECRZL5XR.js +50 -0
- package/dist/chunk-ERNE2FZ5.js +189 -0
- package/dist/{chunk-WAZ3NLWL.js → chunk-F55HGNU4.js} +0 -47
- package/dist/{chunk-VGLOTGAS.js → chunk-FAKNOB7Y.js} +2 -2
- package/dist/{chunk-QK3UCXWL.js → chunk-FHFUXL6G.js} +2 -2
- package/dist/chunk-GNJL4YGR.js +79 -0
- package/dist/chunk-HR4KN6S2.js +152 -0
- package/dist/{chunk-OZ7RIXTO.js → chunk-IIOU45CK.js} +1 -1
- package/dist/chunk-IJBFGPCS.js +33 -0
- package/dist/chunk-IVRIKYFE.js +520 -0
- package/dist/chunk-K7PNYS45.js +93 -0
- package/dist/chunk-MDIH26GC.js +183 -0
- package/dist/{chunk-LYHGEHXG.js → chunk-MFAWT5O5.js} +0 -1
- package/dist/{chunk-H34S76MB.js → chunk-MNPUYCHQ.js} +6 -6
- package/dist/chunk-NTOPJI7W.js +207 -0
- package/dist/{chunk-QBLMXKF2.js → chunk-OIWVQYQF.js} +1 -1
- package/dist/chunk-PG56HX5T.js +154 -0
- package/dist/{chunk-LNJA2UGL.js → chunk-PI4WMLMG.js} +7 -84
- package/dist/chunk-QMHPQYUV.js +363 -0
- package/dist/{chunk-H62BP7RI.js → chunk-QPDDIHXE.js} +209 -43
- 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-S5OJEGFG.js} +2 -2
- package/dist/chunk-SS4B7P7V.js +99 -0
- package/dist/chunk-TIGW564L.js +628 -0
- package/dist/chunk-U67V476Y.js +35 -0
- package/dist/{chunk-JY6FYXIT.js → chunk-UCQAOZHW.js} +6 -11
- package/dist/{chunk-ITPEXLHA.js → chunk-URXDAUVH.js} +24 -5
- package/dist/chunk-WIOLLGAD.js +190 -0
- package/dist/{chunk-3WRJEKN4.js → chunk-WJVWINEM.js} +72 -8
- package/dist/chunk-WMGIIABP.js +15 -0
- package/dist/{chunk-33UGEQRT.js → chunk-X3SPPUFG.js} +151 -64
- package/dist/{chunk-3NSBOUT3.js → chunk-Y3TIJEBP.js} +314 -79
- package/dist/chunk-Y6VJKXGL.js +373 -0
- package/dist/{chunk-LI4O6NVK.js → chunk-YDWHS4LJ.js} +49 -9
- package/dist/{chunk-U55BGUAU.js → chunk-YNIPYN4F.js} +5 -5
- package/dist/chunk-YXQCA6B7.js +226 -0
- package/dist/cli/index.js +26 -22
- 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 +7 -5
- package/dist/commands/doctor.d.ts +11 -7
- package/dist/commands/doctor.js +16 -14
- package/dist/commands/embed.js +5 -6
- package/dist/commands/entities.js +2 -2
- package/dist/commands/graph.js +3 -3
- package/dist/commands/inject.d.ts +1 -1
- package/dist/commands/inject.js +4 -5
- package/dist/commands/kanban.js +4 -4
- package/dist/commands/link.js +2 -2
- package/dist/commands/migrate-observations.js +4 -4
- package/dist/commands/observe.d.ts +0 -1
- package/dist/commands/observe.js +13 -12
- 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 +18 -17
- package/dist/commands/status.d.ts +2 -0
- package/dist/commands/status.js +40 -30
- 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/index.d.ts +334 -191
- package/dist/index.js +432 -108
- 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 +1 -1
- 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/plugin/index.d.ts +344 -28
- package/dist/plugin/index.js +3919 -227
- 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/{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 +1559 -0
- package/hooks/clawvault/handler.test.js +510 -0
- package/hooks/clawvault/openclaw.plugin.json +72 -0
- package/openclaw.plugin.json +235 -30
- package/package.json +20 -20
- 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/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
|
@@ -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
|
+
};
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import {
|
|
2
|
+
loadVaultQmdConfig
|
|
3
|
+
} from "./chunk-WIOLLGAD.js";
|
|
4
|
+
import {
|
|
5
|
+
QmdUnavailableError,
|
|
6
|
+
hasQmd,
|
|
7
|
+
qmdEmbed
|
|
8
|
+
} from "./chunk-5PJ4STIC.js";
|
|
9
|
+
import {
|
|
10
|
+
resolveVaultPath
|
|
11
|
+
} from "./chunk-GNJL4YGR.js";
|
|
12
|
+
|
|
13
|
+
// src/commands/embed.ts
|
|
14
|
+
async function embedCommand(options = {}) {
|
|
15
|
+
if (!hasQmd()) {
|
|
16
|
+
throw new QmdUnavailableError();
|
|
17
|
+
}
|
|
18
|
+
const vaultPath = resolveVaultPath({ explicitPath: options.vaultPath });
|
|
19
|
+
const qmdConfig = loadVaultQmdConfig(vaultPath);
|
|
20
|
+
const startedAt = (/* @__PURE__ */ new Date()).toISOString();
|
|
21
|
+
if (!options.quiet) {
|
|
22
|
+
console.log(
|
|
23
|
+
`Embedding pending documents for collection "${qmdConfig.qmdCollection}" (root: ${qmdConfig.qmdRoot})...`
|
|
24
|
+
);
|
|
25
|
+
}
|
|
26
|
+
qmdEmbed(qmdConfig.qmdCollection);
|
|
27
|
+
const finishedAt = (/* @__PURE__ */ new Date()).toISOString();
|
|
28
|
+
if (!options.quiet) {
|
|
29
|
+
console.log(`\u2713 Embedding complete for "${qmdConfig.qmdCollection}"`);
|
|
30
|
+
}
|
|
31
|
+
return {
|
|
32
|
+
vaultPath,
|
|
33
|
+
qmdCollection: qmdConfig.qmdCollection,
|
|
34
|
+
qmdRoot: qmdConfig.qmdRoot,
|
|
35
|
+
startedAt,
|
|
36
|
+
finishedAt
|
|
37
|
+
};
|
|
38
|
+
}
|
|
39
|
+
function registerEmbedCommand(program) {
|
|
40
|
+
program.command("embed").description("Run qmd embedding for pending vault documents").option("-v, --vault <path>", "Vault path").action(async (rawOptions) => {
|
|
41
|
+
await embedCommand({
|
|
42
|
+
vaultPath: rawOptions.vault
|
|
43
|
+
});
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
export {
|
|
48
|
+
embedCommand,
|
|
49
|
+
registerEmbedCommand
|
|
50
|
+
};
|
|
@@ -0,0 +1,189 @@
|
|
|
1
|
+
import {
|
|
2
|
+
create,
|
|
3
|
+
read,
|
|
4
|
+
update
|
|
5
|
+
} from "./chunk-HR4KN6S2.js";
|
|
6
|
+
import {
|
|
7
|
+
append,
|
|
8
|
+
currentOwner
|
|
9
|
+
} from "./chunk-K7PNYS45.js";
|
|
10
|
+
import {
|
|
11
|
+
THREAD_STATUS_TRANSITIONS
|
|
12
|
+
} from "./chunk-WMGIIABP.js";
|
|
13
|
+
import {
|
|
14
|
+
__export
|
|
15
|
+
} from "./chunk-U67V476Y.js";
|
|
16
|
+
|
|
17
|
+
// src/workgraph/thread.ts
|
|
18
|
+
var thread_exports = {};
|
|
19
|
+
__export(thread_exports, {
|
|
20
|
+
block: () => block,
|
|
21
|
+
cancel: () => cancel,
|
|
22
|
+
claim: () => claim,
|
|
23
|
+
createThread: () => createThread,
|
|
24
|
+
decompose: () => decompose,
|
|
25
|
+
done: () => done,
|
|
26
|
+
release: () => release,
|
|
27
|
+
unblock: () => unblock
|
|
28
|
+
});
|
|
29
|
+
function createThread(vaultPath, title, goal, actor, opts = {}) {
|
|
30
|
+
return create(vaultPath, "thread", {
|
|
31
|
+
title,
|
|
32
|
+
goal,
|
|
33
|
+
status: "open",
|
|
34
|
+
priority: opts.priority ?? "medium",
|
|
35
|
+
deps: opts.deps ?? [],
|
|
36
|
+
parent: opts.parent,
|
|
37
|
+
context_refs: opts.context_refs ?? [],
|
|
38
|
+
tags: opts.tags ?? []
|
|
39
|
+
}, `## Goal
|
|
40
|
+
|
|
41
|
+
${goal}
|
|
42
|
+
`, actor);
|
|
43
|
+
}
|
|
44
|
+
function claim(vaultPath, threadPath, actor) {
|
|
45
|
+
const thread = read(vaultPath, threadPath);
|
|
46
|
+
if (!thread) throw new Error(`Thread not found: ${threadPath}`);
|
|
47
|
+
const status = thread.fields.status;
|
|
48
|
+
if (status !== "open") {
|
|
49
|
+
throw new Error(`Cannot claim thread in "${status}" state. Only "open" threads can be claimed.`);
|
|
50
|
+
}
|
|
51
|
+
const owner = currentOwner(vaultPath, threadPath);
|
|
52
|
+
if (owner) {
|
|
53
|
+
throw new Error(`Thread already claimed by "${owner}". Wait for release or use a different thread.`);
|
|
54
|
+
}
|
|
55
|
+
append(vaultPath, actor, "claim", threadPath, "thread");
|
|
56
|
+
return update(vaultPath, threadPath, {
|
|
57
|
+
status: "active",
|
|
58
|
+
owner: actor
|
|
59
|
+
}, void 0, actor);
|
|
60
|
+
}
|
|
61
|
+
function release(vaultPath, threadPath, actor, reason) {
|
|
62
|
+
const thread = read(vaultPath, threadPath);
|
|
63
|
+
if (!thread) throw new Error(`Thread not found: ${threadPath}`);
|
|
64
|
+
assertOwner(vaultPath, threadPath, actor);
|
|
65
|
+
append(
|
|
66
|
+
vaultPath,
|
|
67
|
+
actor,
|
|
68
|
+
"release",
|
|
69
|
+
threadPath,
|
|
70
|
+
"thread",
|
|
71
|
+
reason ? { reason } : void 0
|
|
72
|
+
);
|
|
73
|
+
return update(vaultPath, threadPath, {
|
|
74
|
+
status: "open",
|
|
75
|
+
owner: null
|
|
76
|
+
}, void 0, actor);
|
|
77
|
+
}
|
|
78
|
+
function block(vaultPath, threadPath, actor, blockedBy, reason) {
|
|
79
|
+
const thread = read(vaultPath, threadPath);
|
|
80
|
+
if (!thread) throw new Error(`Thread not found: ${threadPath}`);
|
|
81
|
+
assertTransition(thread.fields.status, "blocked");
|
|
82
|
+
append(vaultPath, actor, "block", threadPath, "thread", {
|
|
83
|
+
blocked_by: blockedBy,
|
|
84
|
+
...reason ? { reason } : {}
|
|
85
|
+
});
|
|
86
|
+
const currentDeps = thread.fields.deps ?? [];
|
|
87
|
+
const updatedDeps = currentDeps.includes(blockedBy) ? currentDeps : [...currentDeps, blockedBy];
|
|
88
|
+
return update(vaultPath, threadPath, {
|
|
89
|
+
status: "blocked",
|
|
90
|
+
deps: updatedDeps
|
|
91
|
+
}, void 0, actor);
|
|
92
|
+
}
|
|
93
|
+
function unblock(vaultPath, threadPath, actor) {
|
|
94
|
+
const thread = read(vaultPath, threadPath);
|
|
95
|
+
if (!thread) throw new Error(`Thread not found: ${threadPath}`);
|
|
96
|
+
assertTransition(thread.fields.status, "active");
|
|
97
|
+
append(vaultPath, actor, "unblock", threadPath, "thread");
|
|
98
|
+
return update(vaultPath, threadPath, {
|
|
99
|
+
status: "active"
|
|
100
|
+
}, void 0, actor);
|
|
101
|
+
}
|
|
102
|
+
function done(vaultPath, threadPath, actor, output) {
|
|
103
|
+
const thread = read(vaultPath, threadPath);
|
|
104
|
+
if (!thread) throw new Error(`Thread not found: ${threadPath}`);
|
|
105
|
+
assertTransition(thread.fields.status, "done");
|
|
106
|
+
assertOwner(vaultPath, threadPath, actor);
|
|
107
|
+
append(
|
|
108
|
+
vaultPath,
|
|
109
|
+
actor,
|
|
110
|
+
"done",
|
|
111
|
+
threadPath,
|
|
112
|
+
"thread",
|
|
113
|
+
output ? { output } : void 0
|
|
114
|
+
);
|
|
115
|
+
const newBody = output ? `${thread.body}
|
|
116
|
+
|
|
117
|
+
## Output
|
|
118
|
+
|
|
119
|
+
${output}
|
|
120
|
+
` : thread.body;
|
|
121
|
+
return update(vaultPath, threadPath, {
|
|
122
|
+
status: "done"
|
|
123
|
+
}, newBody, actor);
|
|
124
|
+
}
|
|
125
|
+
function cancel(vaultPath, threadPath, actor, reason) {
|
|
126
|
+
const thread = read(vaultPath, threadPath);
|
|
127
|
+
if (!thread) throw new Error(`Thread not found: ${threadPath}`);
|
|
128
|
+
assertTransition(thread.fields.status, "cancelled");
|
|
129
|
+
append(
|
|
130
|
+
vaultPath,
|
|
131
|
+
actor,
|
|
132
|
+
"cancel",
|
|
133
|
+
threadPath,
|
|
134
|
+
"thread",
|
|
135
|
+
reason ? { reason } : void 0
|
|
136
|
+
);
|
|
137
|
+
return update(vaultPath, threadPath, {
|
|
138
|
+
status: "cancelled",
|
|
139
|
+
owner: null
|
|
140
|
+
}, void 0, actor);
|
|
141
|
+
}
|
|
142
|
+
function decompose(vaultPath, parentPath, subthreads, actor) {
|
|
143
|
+
const parent = read(vaultPath, parentPath);
|
|
144
|
+
if (!parent) throw new Error(`Thread not found: ${parentPath}`);
|
|
145
|
+
const created = [];
|
|
146
|
+
for (const sub of subthreads) {
|
|
147
|
+
const inst = createThread(vaultPath, sub.title, sub.goal, actor, {
|
|
148
|
+
parent: parentPath,
|
|
149
|
+
deps: sub.deps
|
|
150
|
+
});
|
|
151
|
+
created.push(inst);
|
|
152
|
+
}
|
|
153
|
+
const childRefs = created.map((c) => `[[${c.path}]]`);
|
|
154
|
+
const decomposeNote = `
|
|
155
|
+
|
|
156
|
+
## Sub-threads
|
|
157
|
+
|
|
158
|
+
${childRefs.map((r) => `- ${r}`).join("\n")}
|
|
159
|
+
`;
|
|
160
|
+
update(vaultPath, parentPath, {}, parent.body + decomposeNote, actor);
|
|
161
|
+
append(vaultPath, actor, "decompose", parentPath, "thread", {
|
|
162
|
+
children: created.map((c) => c.path)
|
|
163
|
+
});
|
|
164
|
+
return created;
|
|
165
|
+
}
|
|
166
|
+
function assertTransition(from, to) {
|
|
167
|
+
const allowed = THREAD_STATUS_TRANSITIONS[from];
|
|
168
|
+
if (!allowed?.includes(to)) {
|
|
169
|
+
throw new Error(`Invalid transition: "${from}" \u2192 "${to}". Allowed: ${allowed?.join(", ") ?? "none"}`);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
function assertOwner(vaultPath, threadPath, actor) {
|
|
173
|
+
const owner = currentOwner(vaultPath, threadPath);
|
|
174
|
+
if (owner && owner !== actor) {
|
|
175
|
+
throw new Error(`Thread is owned by "${owner}", not "${actor}". Only the owner can perform this action.`);
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
export {
|
|
180
|
+
createThread,
|
|
181
|
+
claim,
|
|
182
|
+
release,
|
|
183
|
+
block,
|
|
184
|
+
unblock,
|
|
185
|
+
done,
|
|
186
|
+
cancel,
|
|
187
|
+
decompose,
|
|
188
|
+
thread_exports
|
|
189
|
+
};
|