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
package/dist/plugin/observe.cjs
DELETED
|
@@ -1,631 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __defProp = Object.defineProperty;
|
|
3
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
-
var __export = (target, all) => {
|
|
7
|
-
for (var name in all)
|
|
8
|
-
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
-
};
|
|
10
|
-
var __copyProps = (to, from, except, desc) => {
|
|
11
|
-
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
-
for (let key of __getOwnPropNames(from))
|
|
13
|
-
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
-
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
-
}
|
|
16
|
-
return to;
|
|
17
|
-
};
|
|
18
|
-
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
-
|
|
20
|
-
// src/plugin/observe.ts
|
|
21
|
-
var observe_exports = {};
|
|
22
|
-
__export(observe_exports, {
|
|
23
|
-
detectCategory: () => detectCategory,
|
|
24
|
-
extractObservations: () => extractObservations,
|
|
25
|
-
extractPreferences: () => extractPreferences,
|
|
26
|
-
extractSearchTerms: () => extractSearchTerms,
|
|
27
|
-
isObservable: () => isObservable,
|
|
28
|
-
processMessageForObservations: () => processMessageForObservations
|
|
29
|
-
});
|
|
30
|
-
module.exports = __toCommonJS(observe_exports);
|
|
31
|
-
|
|
32
|
-
// src/plugin/templates.ts
|
|
33
|
-
var import_node_fs = require("fs");
|
|
34
|
-
var import_node_path = require("path");
|
|
35
|
-
var DEFAULT_SCHEMAS = [
|
|
36
|
-
{
|
|
37
|
-
primitive: "memory_event",
|
|
38
|
-
description: "General memory event for observations",
|
|
39
|
-
fields: {
|
|
40
|
-
type: { type: "string", required: true, default: "memory_event" },
|
|
41
|
-
status: { type: "string", required: true, default: "recorded", enum: ["recorded", "superseded", "corrected"] },
|
|
42
|
-
created: { type: "datetime", required: true, default: "{{datetime}}" },
|
|
43
|
-
observed_at: { type: "datetime", required: true },
|
|
44
|
-
source: { type: "string", required: true, enum: ["openclaw", "claude-code", "replay", "manual-correction"] },
|
|
45
|
-
summary: { type: "string", required: true },
|
|
46
|
-
confidence: { type: "number" },
|
|
47
|
-
importance: { type: "number" }
|
|
48
|
-
},
|
|
49
|
-
keywords: ["preference", "like", "hate", "want", "need", "always", "never", "remember", "note"]
|
|
50
|
-
},
|
|
51
|
-
{
|
|
52
|
-
primitive: "person",
|
|
53
|
-
description: "People and relationship notes",
|
|
54
|
-
fields: {
|
|
55
|
-
title: { type: "string", required: true, default: "{{title}}" },
|
|
56
|
-
date: { type: "date", required: true, default: "{{date}}" },
|
|
57
|
-
type: { type: "string", required: true, default: "person" },
|
|
58
|
-
relationship: { type: "string", default: "contact" }
|
|
59
|
-
},
|
|
60
|
-
keywords: ["person", "contact", "colleague", "friend", "works at", "lives in", "email", "phone", "name is"]
|
|
61
|
-
},
|
|
62
|
-
{
|
|
63
|
-
primitive: "decision",
|
|
64
|
-
description: "Decision records",
|
|
65
|
-
fields: {
|
|
66
|
-
title: { type: "string", required: true, default: "{{title}}" },
|
|
67
|
-
date: { type: "date", required: true, default: "{{date}}" },
|
|
68
|
-
type: { type: "string", required: true, default: "decision" },
|
|
69
|
-
status: { type: "string", default: "decided", enum: ["proposed", "decided", "superseded"] }
|
|
70
|
-
},
|
|
71
|
-
keywords: ["decided", "decision", "chose", "will use", "go with", "ship", "approved", "rejected"]
|
|
72
|
-
},
|
|
73
|
-
{
|
|
74
|
-
primitive: "task",
|
|
75
|
-
description: "Task primitives",
|
|
76
|
-
fields: {
|
|
77
|
-
status: { type: "string", required: true, default: "open", enum: ["open", "in-progress", "blocked", "done"] },
|
|
78
|
-
created: { type: "datetime", required: true, default: "{{datetime}}" },
|
|
79
|
-
updated: { type: "datetime", required: true, default: "{{datetime}}" },
|
|
80
|
-
priority: { type: "string", enum: ["critical", "high", "medium", "low"] },
|
|
81
|
-
due: { type: "date" }
|
|
82
|
-
},
|
|
83
|
-
keywords: ["task", "todo", "need to", "should", "must", "deadline", "due", "by tomorrow", "by tonight"]
|
|
84
|
-
},
|
|
85
|
-
{
|
|
86
|
-
primitive: "project",
|
|
87
|
-
description: "Project definition documents",
|
|
88
|
-
fields: {
|
|
89
|
-
type: { type: "string", required: true, default: "project" },
|
|
90
|
-
status: { type: "string", required: true, default: "active", enum: ["active", "paused", "completed", "archived"] },
|
|
91
|
-
created: { type: "datetime", required: true, default: "{{datetime}}" },
|
|
92
|
-
updated: { type: "datetime", required: true, default: "{{datetime}}" }
|
|
93
|
-
},
|
|
94
|
-
keywords: ["project", "initiative", "working on", "building", "developing"]
|
|
95
|
-
},
|
|
96
|
-
{
|
|
97
|
-
primitive: "lesson",
|
|
98
|
-
description: "Lessons learned",
|
|
99
|
-
fields: {
|
|
100
|
-
title: { type: "string", required: true, default: "{{title}}" },
|
|
101
|
-
date: { type: "date", required: true, default: "{{date}}" },
|
|
102
|
-
type: { type: "string", required: true, default: "lesson" }
|
|
103
|
-
},
|
|
104
|
-
keywords: ["learned", "lesson", "insight", "realized", "discovered", "found out"]
|
|
105
|
-
}
|
|
106
|
-
];
|
|
107
|
-
function parseYamlFrontmatter(content) {
|
|
108
|
-
const match = content.match(/^---\n([\s\S]*?)\n---\n?([\s\S]*)$/);
|
|
109
|
-
if (!match) return null;
|
|
110
|
-
const yamlContent = match[1];
|
|
111
|
-
const body = match[2];
|
|
112
|
-
try {
|
|
113
|
-
const frontmatter = parseSimpleYaml(yamlContent);
|
|
114
|
-
return { frontmatter, body };
|
|
115
|
-
} catch {
|
|
116
|
-
return null;
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
function parseSimpleYaml(yaml) {
|
|
120
|
-
const result = {};
|
|
121
|
-
const lines = yaml.split("\n");
|
|
122
|
-
let currentKey = "";
|
|
123
|
-
let currentIndent = 0;
|
|
124
|
-
let nestedObject = null;
|
|
125
|
-
let nestedKey = "";
|
|
126
|
-
for (const line of lines) {
|
|
127
|
-
if (!line.trim() || line.trim().startsWith("#")) continue;
|
|
128
|
-
const indent = line.search(/\S/);
|
|
129
|
-
const trimmed = line.trim();
|
|
130
|
-
if (trimmed.startsWith("- ")) {
|
|
131
|
-
const value = trimmed.slice(2).trim();
|
|
132
|
-
if (nestedObject && nestedKey) {
|
|
133
|
-
const arr = nestedObject[nestedKey];
|
|
134
|
-
if (Array.isArray(arr)) {
|
|
135
|
-
arr.push(parseYamlValue(value));
|
|
136
|
-
}
|
|
137
|
-
} else if (currentKey && result[currentKey]) {
|
|
138
|
-
const arr = result[currentKey];
|
|
139
|
-
if (Array.isArray(arr)) {
|
|
140
|
-
arr.push(parseYamlValue(value));
|
|
141
|
-
}
|
|
142
|
-
}
|
|
143
|
-
continue;
|
|
144
|
-
}
|
|
145
|
-
const colonIndex = trimmed.indexOf(":");
|
|
146
|
-
if (colonIndex === -1) continue;
|
|
147
|
-
const key = trimmed.slice(0, colonIndex).trim();
|
|
148
|
-
const valueStr = trimmed.slice(colonIndex + 1).trim();
|
|
149
|
-
if (indent === 0) {
|
|
150
|
-
if (valueStr === "" || valueStr === "|" || valueStr === ">") {
|
|
151
|
-
if (key === "fields") {
|
|
152
|
-
result[key] = {};
|
|
153
|
-
nestedObject = result[key];
|
|
154
|
-
nestedKey = "";
|
|
155
|
-
} else {
|
|
156
|
-
result[key] = {};
|
|
157
|
-
nestedObject = null;
|
|
158
|
-
}
|
|
159
|
-
} else {
|
|
160
|
-
result[key] = parseYamlValue(valueStr);
|
|
161
|
-
nestedObject = null;
|
|
162
|
-
}
|
|
163
|
-
currentKey = key;
|
|
164
|
-
currentIndent = indent;
|
|
165
|
-
} else if (nestedObject && indent > 0) {
|
|
166
|
-
if (valueStr === "" || valueStr === "|" || valueStr === ">") {
|
|
167
|
-
nestedObject[key] = {};
|
|
168
|
-
nestedKey = key;
|
|
169
|
-
} else if (nestedKey && indent > 2) {
|
|
170
|
-
const fieldObj = nestedObject[nestedKey];
|
|
171
|
-
if (fieldObj) {
|
|
172
|
-
if (key === "enum") {
|
|
173
|
-
fieldObj[key] = [];
|
|
174
|
-
} else {
|
|
175
|
-
fieldObj[key] = parseYamlValue(valueStr);
|
|
176
|
-
}
|
|
177
|
-
}
|
|
178
|
-
} else {
|
|
179
|
-
nestedObject[key] = parseYamlValue(valueStr);
|
|
180
|
-
nestedKey = key;
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
}
|
|
184
|
-
return result;
|
|
185
|
-
}
|
|
186
|
-
function parseYamlValue(value) {
|
|
187
|
-
if (value === "" || value === "null" || value === "~") return null;
|
|
188
|
-
if (value === "true") return true;
|
|
189
|
-
if (value === "false") return false;
|
|
190
|
-
if (/^-?\d+$/.test(value)) return parseInt(value, 10);
|
|
191
|
-
if (/^-?\d+\.\d+$/.test(value)) return parseFloat(value);
|
|
192
|
-
if (value.startsWith('"') && value.endsWith('"') || value.startsWith("'") && value.endsWith("'")) {
|
|
193
|
-
return value.slice(1, -1);
|
|
194
|
-
}
|
|
195
|
-
return value;
|
|
196
|
-
}
|
|
197
|
-
var registry = null;
|
|
198
|
-
function getTemplateRegistry() {
|
|
199
|
-
if (!registry) {
|
|
200
|
-
registry = {
|
|
201
|
-
schemas: /* @__PURE__ */ new Map(),
|
|
202
|
-
keywordIndex: /* @__PURE__ */ new Map(),
|
|
203
|
-
initialized: false
|
|
204
|
-
};
|
|
205
|
-
}
|
|
206
|
-
return registry;
|
|
207
|
-
}
|
|
208
|
-
function initializeTemplateRegistry(templatesDir) {
|
|
209
|
-
const reg = getTemplateRegistry();
|
|
210
|
-
if (reg.initialized) {
|
|
211
|
-
return reg;
|
|
212
|
-
}
|
|
213
|
-
const dirsToTry = templatesDir ? [templatesDir] : [
|
|
214
|
-
(0, import_node_path.join)(process.cwd(), "templates"),
|
|
215
|
-
(0, import_node_path.join)(process.cwd(), "..", "..", "templates"),
|
|
216
|
-
(0, import_node_path.join)(process.env.HOME ?? ".", "clawvault", "templates"),
|
|
217
|
-
(0, import_node_path.join)(process.env.HOME ?? ".", ".clawvault", "templates")
|
|
218
|
-
];
|
|
219
|
-
let loaded = false;
|
|
220
|
-
for (const dir of dirsToTry) {
|
|
221
|
-
if ((0, import_node_fs.existsSync)(dir)) {
|
|
222
|
-
try {
|
|
223
|
-
loadTemplatesFromDirectory(dir, reg);
|
|
224
|
-
loaded = true;
|
|
225
|
-
break;
|
|
226
|
-
} catch {
|
|
227
|
-
}
|
|
228
|
-
}
|
|
229
|
-
}
|
|
230
|
-
if (!loaded || reg.schemas.size === 0) {
|
|
231
|
-
loadDefaultSchemas(reg);
|
|
232
|
-
}
|
|
233
|
-
buildKeywordIndex(reg);
|
|
234
|
-
reg.initialized = true;
|
|
235
|
-
return reg;
|
|
236
|
-
}
|
|
237
|
-
function loadTemplatesFromDirectory(dir, reg) {
|
|
238
|
-
const files = (0, import_node_fs.readdirSync)(dir).filter((f) => f.endsWith(".md"));
|
|
239
|
-
for (const file of files) {
|
|
240
|
-
const filePath = (0, import_node_path.join)(dir, file);
|
|
241
|
-
const content = (0, import_node_fs.readFileSync)(filePath, "utf-8");
|
|
242
|
-
const parsed = parseYamlFrontmatter(content);
|
|
243
|
-
if (!parsed?.frontmatter?.primitive) continue;
|
|
244
|
-
const schema = convertFrontmatterToSchema(parsed.frontmatter, parsed.body);
|
|
245
|
-
if (schema) {
|
|
246
|
-
reg.schemas.set(schema.primitive, schema);
|
|
247
|
-
}
|
|
248
|
-
}
|
|
249
|
-
}
|
|
250
|
-
function convertFrontmatterToSchema(fm, body) {
|
|
251
|
-
const primitive = fm.primitive;
|
|
252
|
-
if (!primitive) return null;
|
|
253
|
-
const fields = {};
|
|
254
|
-
const fmFields = fm.fields;
|
|
255
|
-
if (fmFields) {
|
|
256
|
-
for (const [fieldName, fieldDef] of Object.entries(fmFields)) {
|
|
257
|
-
if (typeof fieldDef === "object" && fieldDef !== null) {
|
|
258
|
-
const def = fieldDef;
|
|
259
|
-
fields[fieldName] = {
|
|
260
|
-
type: def.type || "string",
|
|
261
|
-
required: def.required,
|
|
262
|
-
default: def.default,
|
|
263
|
-
enum: def.enum,
|
|
264
|
-
description: def.description
|
|
265
|
-
};
|
|
266
|
-
}
|
|
267
|
-
}
|
|
268
|
-
}
|
|
269
|
-
const keywords = extractKeywordsFromSchema(primitive, fm.description, fields);
|
|
270
|
-
return {
|
|
271
|
-
primitive,
|
|
272
|
-
description: fm.description,
|
|
273
|
-
fields,
|
|
274
|
-
bodyTemplate: body,
|
|
275
|
-
keywords
|
|
276
|
-
};
|
|
277
|
-
}
|
|
278
|
-
function extractKeywordsFromSchema(primitive, description, fields) {
|
|
279
|
-
const keywords = [primitive];
|
|
280
|
-
keywords.push(primitive.replace(/-/g, " "));
|
|
281
|
-
keywords.push(primitive.replace(/_/g, " "));
|
|
282
|
-
switch (primitive) {
|
|
283
|
-
case "memory_event":
|
|
284
|
-
keywords.push("preference", "like", "hate", "want", "need", "always", "never", "remember", "note");
|
|
285
|
-
break;
|
|
286
|
-
case "person":
|
|
287
|
-
keywords.push("person", "contact", "colleague", "friend", "works at", "lives in", "email", "phone", "name is");
|
|
288
|
-
break;
|
|
289
|
-
case "decision":
|
|
290
|
-
keywords.push("decided", "decision", "chose", "will use", "go with", "ship", "approved", "rejected");
|
|
291
|
-
break;
|
|
292
|
-
case "task":
|
|
293
|
-
keywords.push("task", "todo", "need to", "should", "must", "deadline", "due", "by tomorrow", "by tonight");
|
|
294
|
-
break;
|
|
295
|
-
case "project":
|
|
296
|
-
keywords.push("project", "initiative", "working on", "building", "developing");
|
|
297
|
-
break;
|
|
298
|
-
case "lesson":
|
|
299
|
-
keywords.push("learned", "lesson", "insight", "realized", "discovered", "found out");
|
|
300
|
-
break;
|
|
301
|
-
case "trigger":
|
|
302
|
-
keywords.push("trigger", "schedule", "cron", "automated", "recurring");
|
|
303
|
-
break;
|
|
304
|
-
case "run":
|
|
305
|
-
keywords.push("run", "execution", "job", "started", "finished", "failed");
|
|
306
|
-
break;
|
|
307
|
-
case "checkpoint":
|
|
308
|
-
keywords.push("checkpoint", "snapshot", "state", "progress");
|
|
309
|
-
break;
|
|
310
|
-
case "handoff":
|
|
311
|
-
keywords.push("handoff", "transition", "context", "resume");
|
|
312
|
-
break;
|
|
313
|
-
case "daily-note":
|
|
314
|
-
case "daily":
|
|
315
|
-
keywords.push("daily", "today", "journal", "log");
|
|
316
|
-
break;
|
|
317
|
-
case "party":
|
|
318
|
-
keywords.push("party", "agent", "human", "runtime", "service");
|
|
319
|
-
break;
|
|
320
|
-
case "workspace":
|
|
321
|
-
keywords.push("workspace", "shared", "collaboration");
|
|
322
|
-
break;
|
|
323
|
-
}
|
|
324
|
-
if (fields.status?.enum) {
|
|
325
|
-
keywords.push(...fields.status.enum);
|
|
326
|
-
}
|
|
327
|
-
return [...new Set(keywords)];
|
|
328
|
-
}
|
|
329
|
-
function loadDefaultSchemas(reg) {
|
|
330
|
-
for (const schema of DEFAULT_SCHEMAS) {
|
|
331
|
-
reg.schemas.set(schema.primitive, schema);
|
|
332
|
-
}
|
|
333
|
-
}
|
|
334
|
-
function buildKeywordIndex(reg) {
|
|
335
|
-
reg.keywordIndex.clear();
|
|
336
|
-
for (const [primitive, schema] of reg.schemas) {
|
|
337
|
-
const keywords = schema.keywords ?? [primitive];
|
|
338
|
-
for (const keyword of keywords) {
|
|
339
|
-
const lower = keyword.toLowerCase();
|
|
340
|
-
const existing = reg.keywordIndex.get(lower) ?? [];
|
|
341
|
-
if (!existing.includes(primitive)) {
|
|
342
|
-
existing.push(primitive);
|
|
343
|
-
}
|
|
344
|
-
reg.keywordIndex.set(lower, existing);
|
|
345
|
-
}
|
|
346
|
-
}
|
|
347
|
-
}
|
|
348
|
-
function classifyText(text) {
|
|
349
|
-
const reg = getTemplateRegistry();
|
|
350
|
-
if (!reg.initialized) {
|
|
351
|
-
initializeTemplateRegistry();
|
|
352
|
-
}
|
|
353
|
-
const lower = text.toLowerCase();
|
|
354
|
-
const scores = /* @__PURE__ */ new Map();
|
|
355
|
-
for (const [keyword, primitives] of reg.keywordIndex) {
|
|
356
|
-
if (lower.includes(keyword)) {
|
|
357
|
-
for (const primitive of primitives) {
|
|
358
|
-
const existing = scores.get(primitive) ?? { score: 0, keywords: [] };
|
|
359
|
-
existing.score += getKeywordWeight(keyword, primitive);
|
|
360
|
-
existing.keywords.push(keyword);
|
|
361
|
-
scores.set(primitive, existing);
|
|
362
|
-
}
|
|
363
|
-
}
|
|
364
|
-
}
|
|
365
|
-
applyPatternScoring(lower, scores);
|
|
366
|
-
let bestPrimitive = "memory_event";
|
|
367
|
-
let bestScore = 0;
|
|
368
|
-
let bestKeywords = [];
|
|
369
|
-
for (const [primitive, data] of scores) {
|
|
370
|
-
if (data.score > bestScore) {
|
|
371
|
-
bestScore = data.score;
|
|
372
|
-
bestPrimitive = primitive;
|
|
373
|
-
bestKeywords = data.keywords;
|
|
374
|
-
}
|
|
375
|
-
}
|
|
376
|
-
const confidence = Math.min(1, bestScore / 5);
|
|
377
|
-
return {
|
|
378
|
-
primitiveType: bestPrimitive,
|
|
379
|
-
confidence,
|
|
380
|
-
matchedKeywords: [...new Set(bestKeywords)]
|
|
381
|
-
};
|
|
382
|
-
}
|
|
383
|
-
function getKeywordWeight(keyword, primitive) {
|
|
384
|
-
if (keyword === primitive || keyword === primitive.replace(/-/g, " ")) {
|
|
385
|
-
return 3;
|
|
386
|
-
}
|
|
387
|
-
const strongIndicators = {
|
|
388
|
-
person: ["works at", "lives in", "email", "phone", "name is"],
|
|
389
|
-
decision: ["decided", "chose", "will use", "go with"],
|
|
390
|
-
task: ["deadline", "due", "by tomorrow", "by tonight"],
|
|
391
|
-
memory_event: ["preference", "remember", "note"]
|
|
392
|
-
};
|
|
393
|
-
if (strongIndicators[primitive]?.includes(keyword)) {
|
|
394
|
-
return 2;
|
|
395
|
-
}
|
|
396
|
-
return 1;
|
|
397
|
-
}
|
|
398
|
-
function applyPatternScoring(text, scores) {
|
|
399
|
-
if (/\b(my .+ is|his .+ is|her .+ is|their .+ is)\b/i.test(text)) {
|
|
400
|
-
const existing = scores.get("person") ?? { score: 0, keywords: [] };
|
|
401
|
-
existing.score += 2;
|
|
402
|
-
existing.keywords.push("possessive pattern");
|
|
403
|
-
scores.set("person", existing);
|
|
404
|
-
}
|
|
405
|
-
if (/[\w.-]+@[\w.-]+\.\w+|\+\d{10,}/.test(text)) {
|
|
406
|
-
const existing = scores.get("person") ?? { score: 0, keywords: [] };
|
|
407
|
-
existing.score += 3;
|
|
408
|
-
existing.keywords.push("contact info");
|
|
409
|
-
scores.set("person", existing);
|
|
410
|
-
}
|
|
411
|
-
if (/\b(i prefer|i like|i hate|i love|i want|i need|i always|i never|don't like|dont like)\b/i.test(text)) {
|
|
412
|
-
const existing = scores.get("memory_event") ?? { score: 0, keywords: [] };
|
|
413
|
-
existing.score += 3;
|
|
414
|
-
existing.keywords.push("preference pattern");
|
|
415
|
-
scores.set("memory_event", existing);
|
|
416
|
-
}
|
|
417
|
-
if (/\b(we decided|let's go with|we're going|i chose|we'll use|ship it|do it)\b/i.test(text)) {
|
|
418
|
-
const existing = scores.get("decision") ?? { score: 0, keywords: [] };
|
|
419
|
-
existing.score += 3;
|
|
420
|
-
existing.keywords.push("decision pattern");
|
|
421
|
-
scores.set("decision", existing);
|
|
422
|
-
}
|
|
423
|
-
if (/\b(by tonight|by tomorrow|deadline|due date|by end of|ship by|ready by)\b/i.test(text)) {
|
|
424
|
-
const existing = scores.get("task") ?? { score: 0, keywords: [] };
|
|
425
|
-
existing.score += 2;
|
|
426
|
-
existing.keywords.push("deadline pattern");
|
|
427
|
-
scores.set("task", existing);
|
|
428
|
-
}
|
|
429
|
-
}
|
|
430
|
-
|
|
431
|
-
// src/plugin/observe.ts
|
|
432
|
-
function isObservable(text) {
|
|
433
|
-
if (!text || text.length < 20 || text.length > 5e3) return false;
|
|
434
|
-
if (text.includes("<relevant-memories>")) return false;
|
|
435
|
-
if (text.startsWith("[System")) return false;
|
|
436
|
-
if (text.includes("HEARTBEAT")) return false;
|
|
437
|
-
if (text.startsWith("NO_REPLY")) return false;
|
|
438
|
-
if (text.startsWith("{") && text.includes('"')) return false;
|
|
439
|
-
const markdownDensity = (text.match(/[#*`\-|>]/g) || []).length / text.length;
|
|
440
|
-
if (markdownDensity > 0.15) return false;
|
|
441
|
-
return true;
|
|
442
|
-
}
|
|
443
|
-
var OBSERVATION_PATTERNS = [
|
|
444
|
-
// Preferences
|
|
445
|
-
{ pattern: /\b(i prefer|i like|i hate|i love|i want|i need|i always|i never|don't like|dont like)\b/i, weight: 2 },
|
|
446
|
-
// Decisions
|
|
447
|
-
{ pattern: /\b(we decided|let's go with|we're going|i chose|we'll use|ship it|do it|go with)\b/i, weight: 2 },
|
|
448
|
-
// Facts about people/things
|
|
449
|
-
{ pattern: /\b(my .+ is|his .+ is|her .+ is|their .+ is|works at|lives in|born in)\b/i, weight: 1.5 },
|
|
450
|
-
// Contact info
|
|
451
|
-
{ pattern: /[\w.-]+@[\w.-]+\.\w+|\+\d{10,}/i, weight: 2 },
|
|
452
|
-
// Explicit memory request
|
|
453
|
-
{ pattern: /\b(remember|don't forget|keep in mind|note that|important:)\b/i, weight: 2.5 },
|
|
454
|
-
// Deadlines/dates
|
|
455
|
-
{ pattern: /\b(by tonight|by tomorrow|deadline|due date|by end of|ship by|ready by)\b/i, weight: 1.5 },
|
|
456
|
-
// Lessons learned
|
|
457
|
-
{ pattern: /\b(i learned|we learned|lesson|realized|discovered|found out)\b/i, weight: 1.5 },
|
|
458
|
-
// Tasks
|
|
459
|
-
{ pattern: /\b(need to|should|must|have to|todo|task)\b/i, weight: 1 },
|
|
460
|
-
// Projects
|
|
461
|
-
{ pattern: /\b(working on|building|developing|project|initiative)\b/i, weight: 1 }
|
|
462
|
-
];
|
|
463
|
-
function extractObservations(text) {
|
|
464
|
-
const observations = [];
|
|
465
|
-
const sentences = splitIntoSentences(text);
|
|
466
|
-
const now = /* @__PURE__ */ new Date();
|
|
467
|
-
for (const sentence of sentences) {
|
|
468
|
-
if (sentence.length < 15) continue;
|
|
469
|
-
let totalWeight = 0;
|
|
470
|
-
for (const { pattern, weight } of OBSERVATION_PATTERNS) {
|
|
471
|
-
if (pattern.test(sentence)) {
|
|
472
|
-
totalWeight += weight;
|
|
473
|
-
}
|
|
474
|
-
}
|
|
475
|
-
if (totalWeight < 1) continue;
|
|
476
|
-
const classification = classifyText(sentence);
|
|
477
|
-
const category = deriveCategoryFromPrimitive(classification.primitiveType, sentence);
|
|
478
|
-
const tags = generateTags(classification, sentence);
|
|
479
|
-
observations.push({
|
|
480
|
-
text: sentence.trim(),
|
|
481
|
-
primitiveType: classification.primitiveType,
|
|
482
|
-
confidence: classification.confidence,
|
|
483
|
-
matchedKeywords: classification.matchedKeywords,
|
|
484
|
-
category,
|
|
485
|
-
tags,
|
|
486
|
-
extractedAt: now
|
|
487
|
-
});
|
|
488
|
-
}
|
|
489
|
-
return observations;
|
|
490
|
-
}
|
|
491
|
-
function splitIntoSentences(text) {
|
|
492
|
-
const raw = text.split(/(?<=[.!?\n])\s+/);
|
|
493
|
-
const sentences = [];
|
|
494
|
-
for (const s of raw) {
|
|
495
|
-
const trimmed = s.trim();
|
|
496
|
-
if (trimmed.length > 0) {
|
|
497
|
-
sentences.push(trimmed);
|
|
498
|
-
}
|
|
499
|
-
}
|
|
500
|
-
return sentences;
|
|
501
|
-
}
|
|
502
|
-
function deriveCategoryFromPrimitive(primitiveType, text) {
|
|
503
|
-
const lower = text.toLowerCase();
|
|
504
|
-
if (primitiveType === "memory_event") {
|
|
505
|
-
if (/prefer|like|love|hate|want|need|always|never/i.test(lower)) {
|
|
506
|
-
return "preference";
|
|
507
|
-
}
|
|
508
|
-
if (/remember|don't forget|keep in mind|note that/i.test(lower)) {
|
|
509
|
-
return "note";
|
|
510
|
-
}
|
|
511
|
-
return "fact";
|
|
512
|
-
}
|
|
513
|
-
const categoryMap = {
|
|
514
|
-
person: "entity",
|
|
515
|
-
decision: "decision",
|
|
516
|
-
task: "task",
|
|
517
|
-
project: "project",
|
|
518
|
-
lesson: "lesson",
|
|
519
|
-
trigger: "automation",
|
|
520
|
-
run: "execution",
|
|
521
|
-
checkpoint: "checkpoint",
|
|
522
|
-
handoff: "handoff",
|
|
523
|
-
"daily-note": "daily",
|
|
524
|
-
daily: "daily",
|
|
525
|
-
party: "entity",
|
|
526
|
-
workspace: "workspace"
|
|
527
|
-
};
|
|
528
|
-
return categoryMap[primitiveType] ?? "fact";
|
|
529
|
-
}
|
|
530
|
-
function generateTags(classification, text) {
|
|
531
|
-
const tags = [classification.primitiveType];
|
|
532
|
-
const lower = text.toLowerCase();
|
|
533
|
-
if (/prefer|like|love/i.test(lower)) tags.push("positive");
|
|
534
|
-
if (/hate|dislike|don't like/i.test(lower)) tags.push("negative");
|
|
535
|
-
if (/deadline|due|by tomorrow|by tonight/i.test(lower)) tags.push("time-sensitive");
|
|
536
|
-
if (/important|critical|urgent/i.test(lower)) tags.push("high-priority");
|
|
537
|
-
if (/email|phone|contact/i.test(lower)) tags.push("contact-info");
|
|
538
|
-
if (/decided|chose|approved/i.test(lower)) tags.push("finalized");
|
|
539
|
-
if (/proposed|considering|might/i.test(lower)) tags.push("tentative");
|
|
540
|
-
return [...new Set(tags)];
|
|
541
|
-
}
|
|
542
|
-
function processMessageForObservations(content, options = {}) {
|
|
543
|
-
if (!isObservable(content)) {
|
|
544
|
-
return {
|
|
545
|
-
observations: [],
|
|
546
|
-
skipped: 1,
|
|
547
|
-
reason: "Content not observable"
|
|
548
|
-
};
|
|
549
|
-
}
|
|
550
|
-
const observations = extractObservations(content);
|
|
551
|
-
const maxObservations = 5;
|
|
552
|
-
const limited = observations.slice(0, maxObservations);
|
|
553
|
-
const skipped = observations.length - limited.length;
|
|
554
|
-
return {
|
|
555
|
-
observations: limited,
|
|
556
|
-
skipped,
|
|
557
|
-
reason: skipped > 0 ? `Limited to ${maxObservations} observations` : void 0
|
|
558
|
-
};
|
|
559
|
-
}
|
|
560
|
-
function detectCategory(text) {
|
|
561
|
-
const classification = classifyText(text);
|
|
562
|
-
return deriveCategoryFromPrimitive(classification.primitiveType, text);
|
|
563
|
-
}
|
|
564
|
-
function extractPreferences(text) {
|
|
565
|
-
const preferences = [];
|
|
566
|
-
const patterns = [
|
|
567
|
-
{ regex: /i (?:really )?(?:like|love|enjoy|prefer)\s+(.+?)(?:\.|,|$)/gi, sentiment: "positive" },
|
|
568
|
-
{ regex: /i (?:don't|do not|hate|dislike)\s+(.+?)(?:\.|,|$)/gi, sentiment: "negative" },
|
|
569
|
-
{ regex: /my favorite\s+(.+?)\s+is\s+(.+?)(?:\.|,|$)/gi, sentiment: "positive", hasCategory: true },
|
|
570
|
-
{ regex: /i prefer\s+(.+?)\s+over\s+(.+?)(?:\.|,|$)/gi, sentiment: "positive" }
|
|
571
|
-
];
|
|
572
|
-
for (const { regex, sentiment, hasCategory } of patterns) {
|
|
573
|
-
let match;
|
|
574
|
-
regex.lastIndex = 0;
|
|
575
|
-
while ((match = regex.exec(text)) !== null) {
|
|
576
|
-
if (hasCategory && match[2]) {
|
|
577
|
-
preferences.push({
|
|
578
|
-
category: normalizeCategory(match[1]),
|
|
579
|
-
item: match[2].trim(),
|
|
580
|
-
sentiment,
|
|
581
|
-
confidence: 0.8,
|
|
582
|
-
source: text.slice(0, 100)
|
|
583
|
-
});
|
|
584
|
-
} else if (match[1]) {
|
|
585
|
-
const item = match[1].trim();
|
|
586
|
-
preferences.push({
|
|
587
|
-
category: inferCategory(item),
|
|
588
|
-
item,
|
|
589
|
-
sentiment,
|
|
590
|
-
confidence: 0.7,
|
|
591
|
-
source: text.slice(0, 100)
|
|
592
|
-
});
|
|
593
|
-
}
|
|
594
|
-
}
|
|
595
|
-
}
|
|
596
|
-
return preferences;
|
|
597
|
-
}
|
|
598
|
-
function normalizeCategory(category) {
|
|
599
|
-
return category.toLowerCase().trim().replace(/\s+/g, "_");
|
|
600
|
-
}
|
|
601
|
-
function inferCategory(item) {
|
|
602
|
-
const itemLower = item.toLowerCase();
|
|
603
|
-
const categoryKeywords = {
|
|
604
|
-
food: ["pizza", "coffee", "tea", "food", "restaurant", "cuisine", "dish", "meal", "drink"],
|
|
605
|
-
technology: ["programming", "code", "software", "app", "technology", "framework", "language", "tool"],
|
|
606
|
-
entertainment: ["movie", "music", "book", "game", "show", "series", "film", "song", "album"],
|
|
607
|
-
work: ["meeting", "project", "task", "deadline", "team", "client"],
|
|
608
|
-
communication: ["email", "slack", "chat", "call", "message"]
|
|
609
|
-
};
|
|
610
|
-
for (const [category, keywords] of Object.entries(categoryKeywords)) {
|
|
611
|
-
if (keywords.some((k) => itemLower.includes(k))) {
|
|
612
|
-
return category;
|
|
613
|
-
}
|
|
614
|
-
}
|
|
615
|
-
return "general";
|
|
616
|
-
}
|
|
617
|
-
function extractSearchTerms(input) {
|
|
618
|
-
const noise = /\b(hey|hi|hello|um|uh|like|just|so|well|you know|i mean|basically|actually|really|very|pretty|quite|how does it feel|how do you|can you|could you|would you|do you|what do you think|tell me about)\b/gi;
|
|
619
|
-
let cleaned = input.replace(noise, " ").replace(/\s+/g, " ").trim();
|
|
620
|
-
if (cleaned.length < 5) cleaned = input.trim();
|
|
621
|
-
return cleaned;
|
|
622
|
-
}
|
|
623
|
-
// Annotate the CommonJS export names for ESM import in node:
|
|
624
|
-
0 && (module.exports = {
|
|
625
|
-
detectCategory,
|
|
626
|
-
extractObservations,
|
|
627
|
-
extractPreferences,
|
|
628
|
-
extractSearchTerms,
|
|
629
|
-
isObservable,
|
|
630
|
-
processMessageForObservations
|
|
631
|
-
});
|
|
@@ -1,39 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Dynamic Observation Engine for ClawVault
|
|
3
|
-
*
|
|
4
|
-
* Replaces hardcoded categories with template-driven classification.
|
|
5
|
-
* Extracts observations from text and classifies them against the
|
|
6
|
-
* template registry using keyword/heuristic matching (no LLM calls).
|
|
7
|
-
*/
|
|
8
|
-
interface Observation {
|
|
9
|
-
text: string;
|
|
10
|
-
primitiveType: string;
|
|
11
|
-
confidence: number;
|
|
12
|
-
matchedKeywords: string[];
|
|
13
|
-
category: string;
|
|
14
|
-
tags: string[];
|
|
15
|
-
extractedAt: Date;
|
|
16
|
-
}
|
|
17
|
-
interface ObservationResult {
|
|
18
|
-
observations: Observation[];
|
|
19
|
-
skipped: number;
|
|
20
|
-
reason?: string;
|
|
21
|
-
}
|
|
22
|
-
declare function isObservable(text: string): boolean;
|
|
23
|
-
declare function extractObservations(text: string): Observation[];
|
|
24
|
-
declare function processMessageForObservations(content: string, options?: {
|
|
25
|
-
from?: string;
|
|
26
|
-
sessionId?: string;
|
|
27
|
-
}): ObservationResult;
|
|
28
|
-
declare function detectCategory(text: string): string;
|
|
29
|
-
interface ExtractedPreference {
|
|
30
|
-
category: string;
|
|
31
|
-
item: string;
|
|
32
|
-
sentiment: 'positive' | 'negative' | 'neutral';
|
|
33
|
-
confidence: number;
|
|
34
|
-
source: string;
|
|
35
|
-
}
|
|
36
|
-
declare function extractPreferences(text: string): ExtractedPreference[];
|
|
37
|
-
declare function extractSearchTerms(input: string): string;
|
|
38
|
-
|
|
39
|
-
export { type ExtractedPreference, type Observation, type ObservationResult, detectCategory, extractObservations, extractPreferences, extractSearchTerms, isObservable, processMessageForObservations };
|