openclaw-workspace-sync 2.2.0 → 2.3.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 +189 -9
- package/README.src.md +206 -9
- package/dist/backup-manager.d.ts +67 -0
- package/dist/backup-manager.d.ts.map +1 -0
- package/dist/backup-manager.js +520 -0
- package/dist/backup-manager.js.map +1 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +159 -2
- package/dist/index.js.map +1 -1
- package/dist/types.d.ts +82 -0
- package/dist/types.d.ts.map +1 -1
- package/docs/diagrams/mode-3.svg +42 -0
- package/openclaw.plugin.json +122 -1
- package/package.json +1 -1
- package/skills/workspace-sync/SKILL.md +57 -10
|
@@ -0,0 +1,520 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Backup manager — encrypted snapshot backups to cloud storage.
|
|
3
|
+
*
|
|
4
|
+
* Streams tar | [openssl enc] | rclone rcat directly to the remote —
|
|
5
|
+
* zero local disk usage. Optionally encrypts with AES-256 via openssl.
|
|
6
|
+
* Prunes old snapshots based on retention policy.
|
|
7
|
+
*/
|
|
8
|
+
import { execFile, spawn } from "node:child_process";
|
|
9
|
+
import { existsSync, mkdirSync } from "node:fs";
|
|
10
|
+
import { dirname, join, basename } from "node:path";
|
|
11
|
+
import { getRcloneBinary, isRcloneInstalled, isRcloneConfigured, ensureRcloneConfigFromConfig, getDefaultRcloneConfigPath, } from "./rclone.js";
|
|
12
|
+
function isErr(r) {
|
|
13
|
+
return !r.ok;
|
|
14
|
+
}
|
|
15
|
+
// ── Paths for each include item ──────────────────────────────────────
|
|
16
|
+
function resolveIncludePaths(include, workspaceDir, stateDir) {
|
|
17
|
+
const extensionsDir = join(stateDir, "extensions");
|
|
18
|
+
const map = {
|
|
19
|
+
workspace: workspaceDir,
|
|
20
|
+
config: join(stateDir, "openclaw.json"),
|
|
21
|
+
cron: join(stateDir, "cron"),
|
|
22
|
+
memory: join(workspaceDir, "memory"),
|
|
23
|
+
sessions: join(stateDir, "sessions"),
|
|
24
|
+
credentials: join(stateDir, "credentials"),
|
|
25
|
+
skills: join(workspaceDir, "skills"),
|
|
26
|
+
hooks: join(stateDir, "hooks"),
|
|
27
|
+
extensions: existsSync(extensionsDir) ? extensionsDir : join(stateDir, "extensions"),
|
|
28
|
+
env: join(workspaceDir, ".env"),
|
|
29
|
+
agents: join(stateDir, "agents"),
|
|
30
|
+
pages: join(workspaceDir, "pages"),
|
|
31
|
+
transcripts: join(stateDir, "sessions"),
|
|
32
|
+
};
|
|
33
|
+
return include.map((item) => {
|
|
34
|
+
const p = map[item];
|
|
35
|
+
return { item, path: p, exists: existsSync(p) };
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
// ── Archive ──────────────────────────────────────────────────────────
|
|
39
|
+
function buildTarArgs(sources, exclude) {
|
|
40
|
+
const args = ["cz"];
|
|
41
|
+
for (const pattern of exclude) {
|
|
42
|
+
args.push(`--exclude=${pattern}`);
|
|
43
|
+
}
|
|
44
|
+
for (const src of sources) {
|
|
45
|
+
args.push("-C", dirname(src.path), basename(src.path));
|
|
46
|
+
}
|
|
47
|
+
return args;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Stream tar output directly to rclone rcat — no local temp file needed.
|
|
51
|
+
* For encryption, pipes through openssl enc in between.
|
|
52
|
+
*/
|
|
53
|
+
async function streamBackup(params) {
|
|
54
|
+
const { sources, exclude, remoteDest, configPath, encrypt, passphrase, logger } = params;
|
|
55
|
+
const rcloneBinary = await getRcloneBinary();
|
|
56
|
+
const tarArgs = buildTarArgs(sources, exclude);
|
|
57
|
+
return new Promise((resolve) => {
|
|
58
|
+
const tar = spawn("tar", tarArgs, { stdio: ["ignore", "pipe", "pipe"] });
|
|
59
|
+
let lastStdout = tar.stdout;
|
|
60
|
+
let encExitCode = null;
|
|
61
|
+
if (encrypt && passphrase) {
|
|
62
|
+
const enc = spawn("openssl", [
|
|
63
|
+
"enc", "-aes-256-cbc", "-salt", "-pbkdf2", "-iter", "100000",
|
|
64
|
+
"-pass", "env:OPENCLAW_BACKUP_PASS",
|
|
65
|
+
], {
|
|
66
|
+
stdio: ["pipe", "pipe", "pipe"],
|
|
67
|
+
env: { ...process.env, OPENCLAW_BACKUP_PASS: passphrase },
|
|
68
|
+
});
|
|
69
|
+
tar.stdout.pipe(enc.stdin);
|
|
70
|
+
enc.stderr.on("data", (chunk) => {
|
|
71
|
+
const msg = chunk.toString().trim();
|
|
72
|
+
if (msg)
|
|
73
|
+
logger.warn(`[backup] openssl: ${msg}`);
|
|
74
|
+
});
|
|
75
|
+
enc.on("error", (err) => {
|
|
76
|
+
resolve({ ok: false, error: `openssl spawn failed: ${err.message}` });
|
|
77
|
+
});
|
|
78
|
+
enc.on("close", (code) => { encExitCode = code; });
|
|
79
|
+
lastStdout = enc.stdout;
|
|
80
|
+
}
|
|
81
|
+
const rcat = spawn(rcloneBinary, [
|
|
82
|
+
"rcat", remoteDest, "--config", configPath,
|
|
83
|
+
], { stdio: ["pipe", "pipe", "pipe"] });
|
|
84
|
+
lastStdout.pipe(rcat.stdin);
|
|
85
|
+
let sizeBytes = 0;
|
|
86
|
+
lastStdout.on("data", (chunk) => {
|
|
87
|
+
sizeBytes += chunk.length;
|
|
88
|
+
});
|
|
89
|
+
let tarErr = "";
|
|
90
|
+
let tarExitCode = null;
|
|
91
|
+
tar.stderr.on("data", (chunk) => { tarErr += chunk.toString(); });
|
|
92
|
+
tar.on("close", (code) => { tarExitCode = code; });
|
|
93
|
+
let rcatErr = "";
|
|
94
|
+
rcat.stderr.on("data", (chunk) => { rcatErr += chunk.toString(); });
|
|
95
|
+
rcat.on("close", (code) => {
|
|
96
|
+
if (tarExitCode !== 0 && tarExitCode !== null) {
|
|
97
|
+
resolve({ ok: false, error: `tar failed (exit ${tarExitCode}): ${tarErr}` });
|
|
98
|
+
}
|
|
99
|
+
else if (encExitCode !== 0 && encExitCode !== null) {
|
|
100
|
+
resolve({ ok: false, error: `openssl enc failed (exit ${encExitCode})` });
|
|
101
|
+
}
|
|
102
|
+
else if (code !== 0) {
|
|
103
|
+
resolve({ ok: false, error: `rclone rcat failed (exit ${code}): ${rcatErr}` });
|
|
104
|
+
}
|
|
105
|
+
else {
|
|
106
|
+
resolve({ ok: true, sizeBytes });
|
|
107
|
+
}
|
|
108
|
+
});
|
|
109
|
+
tar.on("error", (err) => {
|
|
110
|
+
resolve({ ok: false, error: `tar spawn failed: ${err.message}` });
|
|
111
|
+
});
|
|
112
|
+
rcat.on("error", (err) => {
|
|
113
|
+
resolve({ ok: false, error: `rclone rcat spawn failed: ${err.message}` });
|
|
114
|
+
});
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
// ── rclone list / delete ─────────────────────────────────────────────
|
|
118
|
+
function resolveBackupRcloneConfig(syncConfig, backupConfig, stateDir) {
|
|
119
|
+
const rawProvider = backupConfig.provider ?? syncConfig.provider;
|
|
120
|
+
const provider = (rawProvider === "off" || !rawProvider) ? "s3" : rawProvider;
|
|
121
|
+
const remoteName = backupConfig.remoteName ?? "backup";
|
|
122
|
+
const configPath = backupConfig.configPath ?? syncConfig.configPath ?? getDefaultRcloneConfigPath(stateDir);
|
|
123
|
+
let remotePath = (backupConfig.remotePath ?? "").replace(/^\/+|\/+$/g, "");
|
|
124
|
+
if (backupConfig.bucket) {
|
|
125
|
+
remotePath = backupConfig.bucket.replace(/^\/+|\/+$/g, "");
|
|
126
|
+
}
|
|
127
|
+
else if (backupConfig.s3?.bucket) {
|
|
128
|
+
remotePath = backupConfig.s3.bucket.replace(/^\/+|\/+$/g, "");
|
|
129
|
+
}
|
|
130
|
+
const prefix = (backupConfig.prefix ?? "").replace(/^\/+|\/+$/g, "");
|
|
131
|
+
if (prefix) {
|
|
132
|
+
remotePath = remotePath ? `${remotePath}/${prefix}` : prefix;
|
|
133
|
+
}
|
|
134
|
+
return { configPath, remoteName, remotePath, provider };
|
|
135
|
+
}
|
|
136
|
+
function ensureBackupRcloneConfig(syncConfig, backupConfig, configPath, remoteName, provider) {
|
|
137
|
+
if (isRcloneConfigured(configPath, remoteName))
|
|
138
|
+
return;
|
|
139
|
+
const providerConfig = backupConfig.s3 ?? backupConfig.dropbox ?? backupConfig.gdrive ??
|
|
140
|
+
backupConfig.onedrive ?? backupConfig.custom ??
|
|
141
|
+
syncConfig.s3 ?? syncConfig.dropbox ?? syncConfig.gdrive ??
|
|
142
|
+
syncConfig.onedrive ?? syncConfig.custom;
|
|
143
|
+
if (!providerConfig)
|
|
144
|
+
return;
|
|
145
|
+
const effectiveSyncConfig = {
|
|
146
|
+
provider,
|
|
147
|
+
s3: backupConfig.s3 ?? syncConfig.s3,
|
|
148
|
+
dropbox: backupConfig.dropbox ?? syncConfig.dropbox,
|
|
149
|
+
gdrive: backupConfig.gdrive ?? syncConfig.gdrive,
|
|
150
|
+
onedrive: backupConfig.onedrive ?? syncConfig.onedrive,
|
|
151
|
+
custom: backupConfig.custom ?? syncConfig.custom,
|
|
152
|
+
};
|
|
153
|
+
ensureRcloneConfigFromConfig(effectiveSyncConfig, configPath, remoteName);
|
|
154
|
+
}
|
|
155
|
+
async function listSnapshots(configPath, remoteName, remotePath) {
|
|
156
|
+
try {
|
|
157
|
+
const binary = await getRcloneBinary();
|
|
158
|
+
const remote = remotePath ? `${remoteName}:${remotePath}` : `${remoteName}:`;
|
|
159
|
+
const args = ["lsf", remote, "--config", configPath, "--files-only"];
|
|
160
|
+
return new Promise((resolve) => {
|
|
161
|
+
execFile(binary, args, { timeout: 30_000, maxBuffer: 1_000_000 }, (err, stdout, stderr) => {
|
|
162
|
+
if (err) {
|
|
163
|
+
resolve({ ok: false, error: stderr || err.message });
|
|
164
|
+
return;
|
|
165
|
+
}
|
|
166
|
+
const files = stdout
|
|
167
|
+
.split("\n")
|
|
168
|
+
.map((f) => f.trim())
|
|
169
|
+
.filter((f) => f.startsWith("backup-"))
|
|
170
|
+
.sort();
|
|
171
|
+
resolve({ ok: true, files });
|
|
172
|
+
});
|
|
173
|
+
});
|
|
174
|
+
}
|
|
175
|
+
catch (err) {
|
|
176
|
+
return { ok: false, error: err instanceof Error ? err.message : String(err) };
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
async function deleteRemoteFile(configPath, remoteName, remotePath, fileName, logger) {
|
|
180
|
+
const binary = await getRcloneBinary();
|
|
181
|
+
const remote = remotePath ? `${remoteName}:${remotePath}/${fileName}` : `${remoteName}:${fileName}`;
|
|
182
|
+
await new Promise((resolve) => {
|
|
183
|
+
execFile(binary, ["deletefile", remote, "--config", configPath], { timeout: 30_000 }, (err, _stdout, stderr) => {
|
|
184
|
+
if (err) {
|
|
185
|
+
logger.warn(`[backup] Failed to delete ${fileName}: ${stderr || err.message}`);
|
|
186
|
+
}
|
|
187
|
+
else {
|
|
188
|
+
logger.info(`[backup] Deleted: ${fileName}`);
|
|
189
|
+
}
|
|
190
|
+
resolve();
|
|
191
|
+
});
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
// ── Retention / pruning ──────────────────────────────────────────────
|
|
195
|
+
function resolveRetainCount(retain) {
|
|
196
|
+
if (retain === undefined)
|
|
197
|
+
return 7;
|
|
198
|
+
if (typeof retain === "number")
|
|
199
|
+
return Math.max(retain, 1);
|
|
200
|
+
return Math.max((retain.daily ?? 7) + (retain.weekly ?? 0) + (retain.monthly ?? 0), 1);
|
|
201
|
+
}
|
|
202
|
+
function selectSnapshotsToKeep(files, retain) {
|
|
203
|
+
if (retain === undefined || typeof retain === "number") {
|
|
204
|
+
const count = resolveRetainCount(retain);
|
|
205
|
+
const sorted = [...files].sort().reverse();
|
|
206
|
+
return new Set(sorted.slice(0, count));
|
|
207
|
+
}
|
|
208
|
+
const keep = new Set();
|
|
209
|
+
const sorted = [...files].sort().reverse();
|
|
210
|
+
const dailyCount = retain.daily ?? 7;
|
|
211
|
+
const weeklyCount = retain.weekly ?? 0;
|
|
212
|
+
const monthlyCount = retain.monthly ?? 0;
|
|
213
|
+
function extractDate(name) {
|
|
214
|
+
const m = name.match(/backup-(\d{4})(\d{2})(\d{2})T/);
|
|
215
|
+
return m ? `${m[1]}-${m[2]}-${m[3]}` : null;
|
|
216
|
+
}
|
|
217
|
+
const seenDays = new Set();
|
|
218
|
+
const seenWeeks = new Set();
|
|
219
|
+
const seenMonths = new Set();
|
|
220
|
+
for (const file of sorted) {
|
|
221
|
+
const date = extractDate(file);
|
|
222
|
+
if (!date)
|
|
223
|
+
continue;
|
|
224
|
+
const d = new Date(date);
|
|
225
|
+
const dayKey = date;
|
|
226
|
+
const weekNum = Math.floor(d.getTime() / (7 * 86400000));
|
|
227
|
+
const weekKey = `${weekNum}`;
|
|
228
|
+
const monthKey = `${d.getFullYear()}-${d.getMonth()}`;
|
|
229
|
+
if (seenDays.size < dailyCount && !seenDays.has(dayKey)) {
|
|
230
|
+
keep.add(file);
|
|
231
|
+
seenDays.add(dayKey);
|
|
232
|
+
}
|
|
233
|
+
else if (seenWeeks.size < weeklyCount && !seenWeeks.has(weekKey)) {
|
|
234
|
+
keep.add(file);
|
|
235
|
+
seenWeeks.add(weekKey);
|
|
236
|
+
}
|
|
237
|
+
else if (seenMonths.size < monthlyCount && !seenMonths.has(monthKey)) {
|
|
238
|
+
keep.add(file);
|
|
239
|
+
seenMonths.add(monthKey);
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
if (keep.size === 0 && sorted.length > 0)
|
|
243
|
+
keep.add(sorted[0]);
|
|
244
|
+
return keep;
|
|
245
|
+
}
|
|
246
|
+
export async function runBackup(params) {
|
|
247
|
+
const { syncConfig, workspaceDir, stateDir, logger } = params;
|
|
248
|
+
const backup = syncConfig.backup;
|
|
249
|
+
if (!backup?.enabled) {
|
|
250
|
+
return { ok: false, error: "Backup not enabled" };
|
|
251
|
+
}
|
|
252
|
+
const installed = await isRcloneInstalled();
|
|
253
|
+
if (!installed) {
|
|
254
|
+
return { ok: false, error: "rclone not installed" };
|
|
255
|
+
}
|
|
256
|
+
const include = backup.include ?? ["workspace", "config", "cron", "memory"];
|
|
257
|
+
const excludePatterns = (backup.exclude ?? syncConfig.exclude ?? [".git", "node_modules", "__pycache__", ".venv", "venv"])
|
|
258
|
+
.map((p) => p.replace(/^\*\*\//, "").replace(/\/\*\*$/, ""));
|
|
259
|
+
const doEncrypt = backup.encrypt ?? false;
|
|
260
|
+
const passphrase = backup.passphrase;
|
|
261
|
+
if (doEncrypt && !passphrase) {
|
|
262
|
+
return { ok: false, error: "Backup encryption enabled but no passphrase configured" };
|
|
263
|
+
}
|
|
264
|
+
const { configPath, remoteName, remotePath, provider } = resolveBackupRcloneConfig(syncConfig, backup, stateDir);
|
|
265
|
+
ensureBackupRcloneConfig(syncConfig, backup, configPath, remoteName, provider);
|
|
266
|
+
if (!isRcloneConfigured(configPath, remoteName)) {
|
|
267
|
+
return { ok: false, error: `rclone not configured for backup remote "${remoteName}"` };
|
|
268
|
+
}
|
|
269
|
+
const paths = resolveIncludePaths(include, workspaceDir, stateDir);
|
|
270
|
+
const existing = paths.filter((p) => p.exists);
|
|
271
|
+
if (existing.length === 0) {
|
|
272
|
+
return { ok: false, error: "No backup sources found" };
|
|
273
|
+
}
|
|
274
|
+
logger.info(`[backup] Starting backup: ${existing.map((p) => p.item).join(", ")}`);
|
|
275
|
+
const now = new Date();
|
|
276
|
+
const timestamp = [
|
|
277
|
+
now.getUTCFullYear(),
|
|
278
|
+
String(now.getUTCMonth() + 1).padStart(2, "0"),
|
|
279
|
+
String(now.getUTCDate()).padStart(2, "0"),
|
|
280
|
+
"T",
|
|
281
|
+
String(now.getUTCHours()).padStart(2, "0"),
|
|
282
|
+
String(now.getUTCMinutes()).padStart(2, "0"),
|
|
283
|
+
String(now.getUTCSeconds()).padStart(2, "0"),
|
|
284
|
+
"Z",
|
|
285
|
+
].join("");
|
|
286
|
+
const ext = doEncrypt ? ".tar.gz.enc" : ".tar.gz";
|
|
287
|
+
const snapshotName = `backup-${timestamp}${ext}`;
|
|
288
|
+
const dest = remotePath ? `${remoteName}:${remotePath}/${snapshotName}` : `${remoteName}:${snapshotName}`;
|
|
289
|
+
logger.info(`[backup] Streaming to ${dest} (no local temp file needed)`);
|
|
290
|
+
try {
|
|
291
|
+
const result = await streamBackup({
|
|
292
|
+
sources: existing,
|
|
293
|
+
exclude: excludePatterns,
|
|
294
|
+
remoteDest: dest,
|
|
295
|
+
configPath,
|
|
296
|
+
encrypt: doEncrypt,
|
|
297
|
+
passphrase,
|
|
298
|
+
logger,
|
|
299
|
+
});
|
|
300
|
+
if (isErr(result)) {
|
|
301
|
+
return { ok: false, error: result.error };
|
|
302
|
+
}
|
|
303
|
+
logger.info(`[backup] Snapshot uploaded: ${snapshotName} (~${(result.sizeBytes / 1024 / 1024).toFixed(1)} MB)`);
|
|
304
|
+
await pruneSnapshots(configPath, remoteName, remotePath, backup.retain, logger);
|
|
305
|
+
return { ok: true, snapshotName, encrypted: doEncrypt, sizeBytes: result.sizeBytes };
|
|
306
|
+
}
|
|
307
|
+
catch (err) {
|
|
308
|
+
return { ok: false, error: err instanceof Error ? err.message : String(err) };
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
async function pruneSnapshots(configPath, remoteName, remotePath, retain, logger) {
|
|
312
|
+
const listResult = await listSnapshots(configPath, remoteName, remotePath);
|
|
313
|
+
if (isErr(listResult)) {
|
|
314
|
+
logger.warn(`[backup] Could not list snapshots for pruning: ${listResult.error}`);
|
|
315
|
+
return;
|
|
316
|
+
}
|
|
317
|
+
const keep = selectSnapshotsToKeep(listResult.files, retain);
|
|
318
|
+
const toDelete = listResult.files.filter((f) => !keep.has(f));
|
|
319
|
+
if (toDelete.length === 0)
|
|
320
|
+
return;
|
|
321
|
+
logger.info(`[backup] Pruning ${toDelete.length} old snapshot(s), keeping ${keep.size}`);
|
|
322
|
+
for (const file of toDelete) {
|
|
323
|
+
await deleteRemoteFile(configPath, remoteName, remotePath, file, logger);
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
export async function listBackupSnapshots(params) {
|
|
327
|
+
const { syncConfig, stateDir, logger } = params;
|
|
328
|
+
const backup = syncConfig.backup;
|
|
329
|
+
if (!backup) {
|
|
330
|
+
return { ok: false, error: "Backup not configured" };
|
|
331
|
+
}
|
|
332
|
+
const { configPath, remoteName, remotePath, provider } = resolveBackupRcloneConfig(syncConfig, backup, stateDir);
|
|
333
|
+
ensureBackupRcloneConfig(syncConfig, backup, configPath, remoteName, provider);
|
|
334
|
+
const result = await listSnapshots(configPath, remoteName, remotePath);
|
|
335
|
+
if (isErr(result))
|
|
336
|
+
return { ok: false, error: result.error };
|
|
337
|
+
return { ok: true, snapshots: result.files.sort().reverse() };
|
|
338
|
+
}
|
|
339
|
+
export async function runRestore(params) {
|
|
340
|
+
const { syncConfig, workspaceDir, stateDir, snapshotName, restoreTo, logger } = params;
|
|
341
|
+
const backup = syncConfig.backup;
|
|
342
|
+
if (!backup) {
|
|
343
|
+
return { ok: false, error: "Backup not configured" };
|
|
344
|
+
}
|
|
345
|
+
const installed = await isRcloneInstalled();
|
|
346
|
+
if (!installed) {
|
|
347
|
+
return { ok: false, error: "rclone not installed" };
|
|
348
|
+
}
|
|
349
|
+
const doEncrypt = backup.encrypt ?? false;
|
|
350
|
+
const passphrase = backup.passphrase;
|
|
351
|
+
if (doEncrypt && !passphrase) {
|
|
352
|
+
return { ok: false, error: "Backup is encrypted but no passphrase configured" };
|
|
353
|
+
}
|
|
354
|
+
const { configPath, remoteName, remotePath, provider } = resolveBackupRcloneConfig(syncConfig, backup, stateDir);
|
|
355
|
+
ensureBackupRcloneConfig(syncConfig, backup, configPath, remoteName, provider);
|
|
356
|
+
let targetSnapshot = snapshotName;
|
|
357
|
+
if (targetSnapshot === "latest") {
|
|
358
|
+
const listResult = await listSnapshots(configPath, remoteName, remotePath);
|
|
359
|
+
if (isErr(listResult))
|
|
360
|
+
return { ok: false, error: `Cannot list snapshots: ${listResult.error}` };
|
|
361
|
+
if (listResult.files.length === 0)
|
|
362
|
+
return { ok: false, error: "No snapshots found" };
|
|
363
|
+
targetSnapshot = listResult.files.sort().reverse()[0];
|
|
364
|
+
}
|
|
365
|
+
logger.info(`[backup] Restoring snapshot: ${targetSnapshot}`);
|
|
366
|
+
const restoreDir = restoreTo;
|
|
367
|
+
if (!existsSync(restoreDir))
|
|
368
|
+
mkdirSync(restoreDir, { recursive: true });
|
|
369
|
+
const rcloneBinary = await getRcloneBinary();
|
|
370
|
+
const remoteSrc = remotePath
|
|
371
|
+
? `${remoteName}:${remotePath}/${targetSnapshot}`
|
|
372
|
+
: `${remoteName}:${targetSnapshot}`;
|
|
373
|
+
try {
|
|
374
|
+
logger.info(`[backup] Streaming restore to ${restoreDir} (no local temp file needed)`);
|
|
375
|
+
await new Promise((resolve, reject) => {
|
|
376
|
+
const rcat = spawn(rcloneBinary, [
|
|
377
|
+
"cat", remoteSrc, "--config", configPath,
|
|
378
|
+
], { stdio: ["ignore", "pipe", "pipe"] });
|
|
379
|
+
let lastStdout = rcat.stdout;
|
|
380
|
+
if (targetSnapshot.endsWith(".enc")) {
|
|
381
|
+
if (!passphrase) {
|
|
382
|
+
reject(new Error("Snapshot is encrypted but no passphrase"));
|
|
383
|
+
return;
|
|
384
|
+
}
|
|
385
|
+
const dec = spawn("openssl", [
|
|
386
|
+
"enc", "-d", "-aes-256-cbc", "-salt", "-pbkdf2", "-iter", "100000",
|
|
387
|
+
"-pass", "env:OPENCLAW_BACKUP_PASS",
|
|
388
|
+
], {
|
|
389
|
+
stdio: ["pipe", "pipe", "pipe"],
|
|
390
|
+
env: { ...process.env, OPENCLAW_BACKUP_PASS: passphrase },
|
|
391
|
+
});
|
|
392
|
+
rcat.stdout.pipe(dec.stdin);
|
|
393
|
+
dec.stderr.on("data", (chunk) => {
|
|
394
|
+
const msg = chunk.toString().trim();
|
|
395
|
+
if (msg)
|
|
396
|
+
logger.warn(`[backup] openssl: ${msg}`);
|
|
397
|
+
});
|
|
398
|
+
lastStdout = dec.stdout;
|
|
399
|
+
dec.on("error", (err) => reject(new Error(`openssl spawn failed: ${err.message}`)));
|
|
400
|
+
}
|
|
401
|
+
const tar = spawn("tar", ["xz", "--no-same-owner", "-C", restoreDir], { stdio: ["pipe", "ignore", "pipe"] });
|
|
402
|
+
lastStdout.pipe(tar.stdin);
|
|
403
|
+
let tarErr = "";
|
|
404
|
+
tar.stderr.on("data", (chunk) => { tarErr += chunk.toString(); });
|
|
405
|
+
let rcatErr = "";
|
|
406
|
+
rcat.stderr.on("data", (chunk) => { rcatErr += chunk.toString(); });
|
|
407
|
+
tar.on("close", (code) => {
|
|
408
|
+
if (code !== 0)
|
|
409
|
+
reject(new Error(`tar extract failed (exit ${code}): ${tarErr}`));
|
|
410
|
+
else
|
|
411
|
+
resolve();
|
|
412
|
+
});
|
|
413
|
+
rcat.on("close", (code) => {
|
|
414
|
+
if (code !== 0 && code !== null) {
|
|
415
|
+
reject(new Error(`rclone cat failed (exit ${code}): ${rcatErr}`));
|
|
416
|
+
}
|
|
417
|
+
});
|
|
418
|
+
rcat.on("error", (err) => reject(err));
|
|
419
|
+
tar.on("error", (err) => reject(err));
|
|
420
|
+
});
|
|
421
|
+
logger.info(`[backup] Restore complete: ${targetSnapshot} → ${restoreDir}`);
|
|
422
|
+
return { ok: true, snapshotName: targetSnapshot, restoredTo: restoreDir };
|
|
423
|
+
}
|
|
424
|
+
catch (err) {
|
|
425
|
+
return { ok: false, error: err instanceof Error ? err.message : String(err) };
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
const backupState = {
|
|
429
|
+
timeoutId: null,
|
|
430
|
+
running: false,
|
|
431
|
+
lastBackupAt: null,
|
|
432
|
+
lastBackupOk: null,
|
|
433
|
+
backupCount: 0,
|
|
434
|
+
errorCount: 0,
|
|
435
|
+
};
|
|
436
|
+
let bkSyncConfig = null;
|
|
437
|
+
let bkWorkspaceDir = null;
|
|
438
|
+
let bkStateDir = null;
|
|
439
|
+
let bkLogger = null;
|
|
440
|
+
function scheduleNextBackup() {
|
|
441
|
+
if (!backupState.running || !bkSyncConfig?.backup)
|
|
442
|
+
return;
|
|
443
|
+
const intervalMs = Math.max(bkSyncConfig.backup.interval ?? 86400, 300) * 1000;
|
|
444
|
+
if (intervalMs <= 0)
|
|
445
|
+
return;
|
|
446
|
+
backupState.timeoutId = setTimeout(() => {
|
|
447
|
+
void doBackupLoop();
|
|
448
|
+
}, intervalMs);
|
|
449
|
+
}
|
|
450
|
+
async function doBackupLoop() {
|
|
451
|
+
if (!bkSyncConfig || !bkWorkspaceDir || !bkStateDir || !bkLogger)
|
|
452
|
+
return;
|
|
453
|
+
try {
|
|
454
|
+
const result = await runBackup({
|
|
455
|
+
syncConfig: bkSyncConfig,
|
|
456
|
+
workspaceDir: bkWorkspaceDir,
|
|
457
|
+
stateDir: bkStateDir,
|
|
458
|
+
logger: bkLogger,
|
|
459
|
+
});
|
|
460
|
+
backupState.lastBackupAt = new Date();
|
|
461
|
+
backupState.backupCount++;
|
|
462
|
+
if (result.ok) {
|
|
463
|
+
backupState.lastBackupOk = true;
|
|
464
|
+
bkLogger.info(`[backup] Scheduled backup complete: ${result.snapshotName} (${(result.sizeBytes / 1024 / 1024).toFixed(1)} MB)`);
|
|
465
|
+
}
|
|
466
|
+
else if (isErr(result)) {
|
|
467
|
+
backupState.lastBackupOk = false;
|
|
468
|
+
backupState.errorCount++;
|
|
469
|
+
bkLogger.error(`[backup] Scheduled backup failed: ${result.error}`);
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
catch (err) {
|
|
473
|
+
backupState.lastBackupOk = false;
|
|
474
|
+
backupState.errorCount++;
|
|
475
|
+
bkLogger?.error(`[backup] Unexpected error: ${err instanceof Error ? err.message : String(err)}`);
|
|
476
|
+
}
|
|
477
|
+
scheduleNextBackup();
|
|
478
|
+
}
|
|
479
|
+
export function startBackupManager(syncConfig, workspaceDir, stateDir, logger) {
|
|
480
|
+
stopBackupManager();
|
|
481
|
+
const backup = syncConfig.backup;
|
|
482
|
+
if (!backup?.enabled)
|
|
483
|
+
return;
|
|
484
|
+
bkSyncConfig = syncConfig;
|
|
485
|
+
bkWorkspaceDir = workspaceDir;
|
|
486
|
+
bkStateDir = stateDir;
|
|
487
|
+
bkLogger = logger;
|
|
488
|
+
const intervalSeconds = backup.interval ?? 86400;
|
|
489
|
+
if (intervalSeconds <= 0) {
|
|
490
|
+
logger.info("[backup] Scheduled backups disabled (interval=0)");
|
|
491
|
+
return;
|
|
492
|
+
}
|
|
493
|
+
const effectiveInterval = Math.max(intervalSeconds, 300);
|
|
494
|
+
backupState.running = true;
|
|
495
|
+
logger.info(`[backup] Backup service started — every ${effectiveInterval}s`);
|
|
496
|
+
backupState.timeoutId = setTimeout(() => {
|
|
497
|
+
void doBackupLoop();
|
|
498
|
+
}, 30_000);
|
|
499
|
+
}
|
|
500
|
+
export function stopBackupManager() {
|
|
501
|
+
backupState.running = false;
|
|
502
|
+
if (backupState.timeoutId) {
|
|
503
|
+
clearTimeout(backupState.timeoutId);
|
|
504
|
+
backupState.timeoutId = null;
|
|
505
|
+
}
|
|
506
|
+
bkSyncConfig = null;
|
|
507
|
+
bkWorkspaceDir = null;
|
|
508
|
+
bkStateDir = null;
|
|
509
|
+
bkLogger = null;
|
|
510
|
+
}
|
|
511
|
+
export function getBackupManagerStatus() {
|
|
512
|
+
return {
|
|
513
|
+
running: backupState.running,
|
|
514
|
+
lastBackupAt: backupState.lastBackupAt,
|
|
515
|
+
lastBackupOk: backupState.lastBackupOk,
|
|
516
|
+
backupCount: backupState.backupCount,
|
|
517
|
+
errorCount: backupState.errorCount,
|
|
518
|
+
};
|
|
519
|
+
}
|
|
520
|
+
//# sourceMappingURL=backup-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"backup-manager.js","sourceRoot":"","sources":["../src/backup-manager.ts"],"names":[],"mappings":"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"}
|
package/dist/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAGH,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAGH,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAwD7D,QAAA,MAAM,mBAAmB;;;;kBAMT,iBAAiB;CAmyBhC,CAAC;AAEF,eAAe,mBAAmB,CAAC"}
|