@wipcomputer/memory-crystal 0.7.34-alpha.1 → 0.7.34-alpha.3
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/LICENSE +1 -1
- package/dist/bridge.js +64 -7
- package/dist/bulk-copy.js +67 -16
- package/dist/cc-hook.js +2291 -47
- package/dist/cc-poller.js +1967 -70
- package/dist/cli.js +4538 -139
- package/dist/core.js +1789 -6
- package/dist/crypto.js +153 -14
- package/dist/crystal-serve.js +64 -12
- package/dist/doctor.js +517 -52
- package/dist/dream-weaver.js +1755 -7
- package/dist/file-sync.js +407 -9
- package/dist/installer.js +840 -145
- package/dist/ldm.js +231 -16
- package/dist/mcp-server.js +1882 -17
- package/dist/migrate.js +1707 -11
- package/dist/mirror-sync.js +2052 -34
- package/dist/openclaw.js +1970 -69
- package/dist/pair.js +112 -16
- package/dist/poller.js +2275 -80
- package/dist/role.js +159 -7
- package/dist/staging.js +235 -10
- package/dist/summarize.js +142 -5
- package/package.json +3 -3
package/dist/pair.js
CHANGED
|
@@ -1,15 +1,111 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
} from "
|
|
8
|
-
|
|
1
|
+
// src/pair.ts
|
|
2
|
+
import { existsSync as existsSync3, writeFileSync as writeFileSync2, mkdirSync as mkdirSync2, chmodSync as chmodSync2 } from "fs";
|
|
3
|
+
import { dirname as dirname2 } from "path";
|
|
4
|
+
|
|
5
|
+
// src/crypto.ts
|
|
6
|
+
import { readFileSync as readFileSync2, existsSync as existsSync2 } from "fs";
|
|
7
|
+
import { createCipheriv, createDecipheriv, createHmac, randomBytes, hkdfSync } from "crypto";
|
|
8
|
+
|
|
9
|
+
// src/ldm.ts
|
|
10
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync, copyFileSync, chmodSync, readdirSync } from "fs";
|
|
11
|
+
import { join, dirname } from "path";
|
|
12
|
+
import { execSync } from "child_process";
|
|
13
|
+
import { fileURLToPath } from "url";
|
|
14
|
+
var HOME = process.env.HOME || "";
|
|
15
|
+
var LDM_ROOT = join(HOME, ".ldm");
|
|
16
|
+
function loadAgentConfig(id) {
|
|
17
|
+
const cfgPath = join(LDM_ROOT, "agents", id, "config.json");
|
|
18
|
+
try {
|
|
19
|
+
if (existsSync(cfgPath)) return JSON.parse(readFileSync(cfgPath, "utf-8"));
|
|
20
|
+
} catch {
|
|
21
|
+
}
|
|
22
|
+
return null;
|
|
23
|
+
}
|
|
24
|
+
function getAgentId(harnessHint) {
|
|
25
|
+
if (process.env.CRYSTAL_AGENT_ID) return process.env.CRYSTAL_AGENT_ID;
|
|
26
|
+
const agentsDir = join(LDM_ROOT, "agents");
|
|
27
|
+
if (existsSync(agentsDir)) {
|
|
28
|
+
try {
|
|
29
|
+
for (const d of readdirSync(agentsDir)) {
|
|
30
|
+
const cfg = loadAgentConfig(d);
|
|
31
|
+
if (!cfg || !cfg.agentId) continue;
|
|
32
|
+
if (!harnessHint) return cfg.agentId;
|
|
33
|
+
if (harnessHint === "claude-code" && cfg.harness === "claude-code-cli") return cfg.agentId;
|
|
34
|
+
if (harnessHint === "openclaw" && cfg.harness === "openclaw") return cfg.agentId;
|
|
35
|
+
}
|
|
36
|
+
} catch {
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
return harnessHint === "openclaw" ? "oc-lesa-mini" : "cc-mini";
|
|
40
|
+
}
|
|
41
|
+
function ldmPaths(agentId) {
|
|
42
|
+
const id = agentId || getAgentId();
|
|
43
|
+
const agentRoot = join(LDM_ROOT, "agents", id);
|
|
44
|
+
return {
|
|
45
|
+
root: LDM_ROOT,
|
|
46
|
+
bin: join(LDM_ROOT, "bin"),
|
|
47
|
+
secrets: join(LDM_ROOT, "secrets"),
|
|
48
|
+
state: join(LDM_ROOT, "state"),
|
|
49
|
+
config: join(LDM_ROOT, "config.json"),
|
|
50
|
+
crystalDb: join(LDM_ROOT, "memory", "crystal.db"),
|
|
51
|
+
crystalLance: join(LDM_ROOT, "memory", "lance"),
|
|
52
|
+
agentRoot,
|
|
53
|
+
transcripts: join(agentRoot, "memory", "transcripts"),
|
|
54
|
+
sessions: join(agentRoot, "memory", "sessions"),
|
|
55
|
+
daily: join(agentRoot, "memory", "daily"),
|
|
56
|
+
journals: join(agentRoot, "memory", "journals"),
|
|
57
|
+
workspace: join(agentRoot, "memory", "workspace")
|
|
58
|
+
};
|
|
59
|
+
}
|
|
60
|
+
var LEGACY_OC_DIR = join(HOME, ".openclaw");
|
|
61
|
+
function resolveSecretPath(filename) {
|
|
62
|
+
const paths = ldmPaths();
|
|
63
|
+
const ldmPath = join(paths.secrets, filename);
|
|
64
|
+
if (existsSync(ldmPath)) return ldmPath;
|
|
65
|
+
const legacyPath = join(LEGACY_OC_DIR, "secrets", filename);
|
|
66
|
+
if (existsSync(legacyPath)) return legacyPath;
|
|
67
|
+
return ldmPath;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
// src/crypto.ts
|
|
71
|
+
import { createHash } from "crypto";
|
|
72
|
+
var KEY_PATH = process.env.CRYSTAL_RELAY_KEY_PATH || resolveSecretPath("crystal-relay-key");
|
|
73
|
+
function loadRelayKey() {
|
|
74
|
+
if (!existsSync2(KEY_PATH)) {
|
|
75
|
+
throw new Error(
|
|
76
|
+
`Relay key not found at ${KEY_PATH}
|
|
77
|
+
Generate one: mkdir -p ~/.ldm/secrets && openssl rand -base64 32 > ~/.ldm/secrets/crystal-relay-key && chmod 600 ~/.ldm/secrets/crystal-relay-key
|
|
78
|
+
Or run: crystal pair`
|
|
79
|
+
);
|
|
80
|
+
}
|
|
81
|
+
const raw = readFileSync2(KEY_PATH, "utf-8").trim();
|
|
82
|
+
const key = Buffer.from(raw, "base64");
|
|
83
|
+
if (key.length !== 32) {
|
|
84
|
+
throw new Error(`Relay key must be 32 bytes (256 bits). Got ${key.length} bytes. Regenerate with: openssl rand -base64 32`);
|
|
85
|
+
}
|
|
86
|
+
return key;
|
|
87
|
+
}
|
|
88
|
+
var RELAY_KEY_PATH = KEY_PATH;
|
|
89
|
+
function generateRelayKey() {
|
|
90
|
+
return randomBytes(32);
|
|
91
|
+
}
|
|
92
|
+
function encodePairingString(key) {
|
|
93
|
+
if (key.length !== 32) throw new Error("Key must be 32 bytes");
|
|
94
|
+
return `mc1:${key.toString("base64")}`;
|
|
95
|
+
}
|
|
96
|
+
function decodePairingString(str) {
|
|
97
|
+
const trimmed = str.trim();
|
|
98
|
+
if (!trimmed.startsWith("mc1:")) {
|
|
99
|
+
throw new Error("Invalid pairing string (expected mc1: prefix)");
|
|
100
|
+
}
|
|
101
|
+
const key = Buffer.from(trimmed.slice(4), "base64");
|
|
102
|
+
if (key.length !== 32) {
|
|
103
|
+
throw new Error(`Invalid key length: expected 32 bytes, got ${key.length}`);
|
|
104
|
+
}
|
|
105
|
+
return key;
|
|
106
|
+
}
|
|
9
107
|
|
|
10
108
|
// src/pair.ts
|
|
11
|
-
import { existsSync, writeFileSync, mkdirSync, chmodSync } from "fs";
|
|
12
|
-
import { dirname } from "path";
|
|
13
109
|
import qrcode from "qrcode-terminal";
|
|
14
110
|
function generateQR(text) {
|
|
15
111
|
return new Promise((resolve) => {
|
|
@@ -19,13 +115,13 @@ function generateQR(text) {
|
|
|
19
115
|
});
|
|
20
116
|
}
|
|
21
117
|
function saveKey(key) {
|
|
22
|
-
const dir =
|
|
23
|
-
if (!
|
|
24
|
-
|
|
118
|
+
const dir = dirname2(RELAY_KEY_PATH);
|
|
119
|
+
if (!existsSync3(dir)) {
|
|
120
|
+
mkdirSync2(dir, { recursive: true });
|
|
25
121
|
}
|
|
26
|
-
|
|
122
|
+
writeFileSync2(RELAY_KEY_PATH, key.toString("base64") + "\n", { mode: 384 });
|
|
27
123
|
try {
|
|
28
|
-
|
|
124
|
+
chmodSync2(RELAY_KEY_PATH, 384);
|
|
29
125
|
} catch {
|
|
30
126
|
}
|
|
31
127
|
}
|
|
@@ -54,7 +150,7 @@ async function pairShow() {
|
|
|
54
150
|
}
|
|
55
151
|
function pairReceive(code) {
|
|
56
152
|
const key = decodePairingString(code);
|
|
57
|
-
if (
|
|
153
|
+
if (existsSync3(RELAY_KEY_PATH)) {
|
|
58
154
|
try {
|
|
59
155
|
const existing = loadRelayKey();
|
|
60
156
|
if (existing.equals(key)) {
|