8900 commited on
Update hf-sync-manager.mjs
Browse files- hf-sync-manager.mjs +274 -272
hf-sync-manager.mjs
CHANGED
|
@@ -13,338 +13,340 @@
|
|
| 13 |
// Excluded: openclaw.json, sessions, qmd, canvas, *.bak
|
| 14 |
// ============================================================
|
| 15 |
|
| 16 |
-
import fs from
|
| 17 |
-
import path from
|
| 18 |
-
import { execSync } from
|
| 19 |
-
|
| 20 |
-
//
|
| 21 |
-
|
| 22 |
-
var HOME = process.env.OPENCLAW_HOME || process.env.HOME ||
|
| 23 |
-
var OPENCLAW_DIR = path.join(HOME,
|
| 24 |
-
var WORKSPACE = path.join(OPENCLAW_DIR,
|
| 25 |
-
var HF_TOKEN = (process.env.HF_TOKEN ||
|
| 26 |
-
var DATASET_ID = (process.env.HF_DATASET_ID ||
|
| 27 |
-
var REPO_DIR =
|
| 28 |
var INTERVAL = 30 * 60 * 1000;
|
| 29 |
-
var RUNNING_FLAG =
|
| 30 |
|
| 31 |
var EXCLUDE_NAMES = [
|
| 32 |
-
|
| 33 |
-
|
| 34 |
-
|
| 35 |
-
|
| 36 |
-
|
| 37 |
-
|
| 38 |
];
|
| 39 |
-
var EXCLUDE_EXT = [
|
| 40 |
|
| 41 |
-
//
|
| 42 |
|
| 43 |
-
function log(msg) { console.log(
|
| 44 |
|
| 45 |
function sleep(ms) {
|
| 46 |
-
|
| 47 |
}
|
| 48 |
|
| 49 |
function repoUrl() {
|
| 50 |
-
|
| 51 |
-
|
| 52 |
}
|
| 53 |
|
| 54 |
function git(args) {
|
| 55 |
-
|
| 56 |
-
|
| 57 |
-
|
| 58 |
-
|
| 59 |
-
|
| 60 |
-
|
| 61 |
}
|
| 62 |
|
| 63 |
function shouldExclude(name) {
|
| 64 |
-
|
| 65 |
-
|
| 66 |
-
|
| 67 |
-
|
| 68 |
-
|
| 69 |
-
|
| 70 |
-
|
| 71 |
-
|
| 72 |
}
|
| 73 |
|
| 74 |
-
//
|
| 75 |
|
| 76 |
function ensureRepo() {
|
| 77 |
-
|
| 78 |
-
|
| 79 |
-
|
| 80 |
-
|
| 81 |
-
|
| 82 |
-
|
| 83 |
-
|
| 84 |
-
|
| 85 |
-
|
| 86 |
-
|
| 87 |
-
|
| 88 |
-
|
| 89 |
-
|
| 90 |
-
|
| 91 |
-
|
| 92 |
-
|
| 93 |
-
|
| 94 |
-
|
| 95 |
-
|
| 96 |
}
|
| 97 |
|
| 98 |
function pull() {
|
| 99 |
-
|
| 100 |
-
|
| 101 |
-
|
| 102 |
-
|
| 103 |
-
|
| 104 |
-
|
| 105 |
}
|
| 106 |
|
| 107 |
function push() {
|
| 108 |
-
|
| 109 |
-
|
| 110 |
-
|
| 111 |
-
|
| 112 |
-
|
| 113 |
-
|
| 114 |
-
|
| 115 |
-
|
| 116 |
-
|
| 117 |
-
|
| 118 |
-
|
| 119 |
-
|
| 120 |
-
|
| 121 |
-
|
| 122 |
}
|
| 123 |
|
| 124 |
-
//
|
| 125 |
|
| 126 |
function mirrorToRepo(srcDir, dstDir) {
|
| 127 |
-
|
| 128 |
-
|
| 129 |
-
|
| 130 |
-
|
| 131 |
-
|
| 132 |
-
|
| 133 |
-
|
| 134 |
-
|
| 135 |
-
|
| 136 |
-
|
| 137 |
-
|
| 138 |
-
|
| 139 |
-
|
| 140 |
-
|
| 141 |
-
|
| 142 |
-
|
| 143 |
-
|
| 144 |
-
|
| 145 |
-
|
| 146 |
}
|
| 147 |
|
| 148 |
function mirrorFromRepo(srcDir, dstDir) {
|
| 149 |
-
|
| 150 |
-
|
| 151 |
-
|
| 152 |
-
|
| 153 |
-
|
| 154 |
-
|
| 155 |
-
|
| 156 |
-
|
| 157 |
-
|
| 158 |
-
|
| 159 |
-
|
| 160 |
-
|
| 161 |
-
|
| 162 |
-
|
| 163 |
-
|
| 164 |
-
|
| 165 |
-
|
| 166 |
-
|
| 167 |
-
|
| 168 |
-
}
|
| 169 |
-
|
| 170 |
-
//
|
| 171 |
|
| 172 |
function seedWorkspaceIfNeeded() {
|
| 173 |
-
|
| 174 |
-
|
| 175 |
-
|
| 176 |
-
|
| 177 |
-
|
| 178 |
-
|
| 179 |
-
|
| 180 |
-
|
| 181 |
-
|
| 182 |
-
|
| 183 |
-
|
| 184 |
-
|
| 185 |
-
|
| 186 |
-
|
| 187 |
-
|
| 188 |
-
|
| 189 |
-
|
| 190 |
-
|
| 191 |
-
|
| 192 |
-
|
| 193 |
-
|
| 194 |
-
|
| 195 |
-
|
| 196 |
|
| 197 |
-
|
| 198 |
-
|
| 199 |
-
|
| 200 |
-
|
| 201 |
-
|
| 202 |
-
|
| 203 |
-
|
| 204 |
-
|
| 205 |
-
|
| 206 |
-
|
| 207 |
-
|
| 208 |
-
|
| 209 |
-
|
| 210 |
-
|
| 211 |
-
|
| 212 |
-
|
| 213 |
-
|
| 214 |
-
|
| 215 |
-
|
| 216 |
-
|
| 217 |
|
| 218 |
-
|
| 219 |
-
|
| 220 |
-
|
| 221 |
-
|
| 222 |
-
|
| 223 |
-
|
| 224 |
-
|
| 225 |
-
|
| 226 |
-
|
| 227 |
-
|
| 228 |
}
|
| 229 |
|
| 230 |
-
//
|
| 231 |
|
| 232 |
function bootRestore() {
|
| 233 |
-
|
| 234 |
-
|
| 235 |
-
|
| 236 |
-
|
| 237 |
-
|
| 238 |
|
| 239 |
-
|
| 240 |
-
|
| 241 |
-
|
| 242 |
-
|
| 243 |
-
|
| 244 |
-
|
| 245 |
-
|
| 246 |
-
|
| 247 |
-
|
| 248 |
|
| 249 |
-
|
| 250 |
-
|
| 251 |
-
|
| 252 |
-
|
| 253 |
-
|
| 254 |
-
|
| 255 |
-
|
| 256 |
|
| 257 |
-
|
| 258 |
-
|
| 259 |
}
|
| 260 |
|
| 261 |
-
//
|
| 262 |
|
| 263 |
function runSync() {
|
| 264 |
-
|
| 265 |
|
| 266 |
-
|
| 267 |
-
|
| 268 |
|
| 269 |
-
|
| 270 |
-
|
| 271 |
-
|
| 272 |
-
|
| 273 |
-
|
| 274 |
-
|
| 275 |
-
|
| 276 |
-
|
| 277 |
-
|
| 278 |
-
|
| 279 |
-
|
| 280 |
-
|
| 281 |
-
// Upload openclaw.json with JSON validation (excluded from restore, not from upload)
|
| 282 |
-
var localCfg = path.join(OPENCLAW_DIR, "openclaw.json");
|
| 283 |
-
if (fs.existsSync(localCfg)) {
|
| 284 |
-
try {
|
| 285 |
-
JSON.parse(fs.readFileSync(localCfg, "utf-8").trim());
|
| 286 |
-
fs.copyFileSync(localCfg, path.join(repoOC, "openclaw.json"));
|
| 287 |
-
n++;
|
| 288 |
-
log("openclaw.json uploaded (JSON valid)");
|
| 289 |
-
} catch (e) {
|
| 290 |
-
log("openclaw.json skipped (invalid JSON)");
|
| 291 |
-
}
|
| 292 |
-
}
|
| 293 |
-
|
| 294 |
-
var pushed = push();
|
| 295 |
-
if (pushed > 0) {
|
| 296 |
-
log("Sync done: " + pushed + " file(s) pushed");
|
| 297 |
-
} else {
|
| 298 |
-
log("Sync done: no changes");
|
| 299 |
-
}
|
| 300 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 301 |
} catch (e) {
|
| 302 |
-
log("
|
| 303 |
}
|
| 304 |
}
|
| 305 |
|
| 306 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 307 |
|
| 308 |
-
|
| 309 |
-
|
| 310 |
-
|
| 311 |
-
|
| 312 |
-
log("RUNNING_FLAG : " + RUNNING_FLAG);
|
| 313 |
|
| 314 |
-
|
| 315 |
-
fs.mkdirSync(WORKSPACE, { recursive: true });
|
| 316 |
|
| 317 |
-
|
| 318 |
-
|
| 319 |
-
|
| 320 |
-
|
| 321 |
-
|
| 322 |
-
|
| 323 |
-
|
| 324 |
-
|
| 325 |
-
|
| 326 |
-
|
| 327 |
-
|
| 328 |
-
|
| 329 |
-
|
| 330 |
-
|
| 331 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 332 |
|
| 333 |
-
|
| 334 |
-
|
| 335 |
-
|
| 336 |
-
|
| 337 |
|
| 338 |
-
|
| 339 |
-
|
| 340 |
-
|
| 341 |
-
|
| 342 |
|
| 343 |
-
|
| 344 |
-
|
| 345 |
|
| 346 |
-
|
| 347 |
-
|
| 348 |
-
|
| 349 |
-
|
| 350 |
-
})().catch(function(e) { console.error(
|
|
|
|
| 13 |
// Excluded: openclaw.json, sessions, qmd, canvas, *.bak
|
| 14 |
// ============================================================
|
| 15 |
|
| 16 |
+
import fs from βnode:fsβ;
|
| 17 |
+
import path from βnode:pathβ;
|
| 18 |
+
import { execSync } from βnode:child_processβ;
|
| 19 |
+
|
| 20 |
+
// ββ config ββββββββββββββββ
|
| 21 |
+
|
| 22 |
+
var HOME = process.env.OPENCLAW_HOME || process.env.HOME || β/home/userβ;
|
| 23 |
+
var OPENCLAW_DIR = path.join(HOME, β.openclawβ);
|
| 24 |
+
var WORKSPACE = path.join(OPENCLAW_DIR, βworkspaceβ);
|
| 25 |
+
var HF_TOKEN = (process.env.HF_TOKEN || ββ).trim();
|
| 26 |
+
var DATASET_ID = (process.env.HF_DATASET_ID || ββ).trim();
|
| 27 |
+
var REPO_DIR = β/tmp/oc-datasetβ;
|
| 28 |
var INTERVAL = 30 * 60 * 1000;
|
| 29 |
+
var RUNNING_FLAG = β/tmp/.hf-sync-runningβ;
|
| 30 |
|
| 31 |
var EXCLUDE_NAMES = [
|
| 32 |
+
βopenclaw.jsonβ,
|
| 33 |
+
βopenclaw.json.bakβ,
|
| 34 |
+
βsessionsβ,
|
| 35 |
+
βqmdβ,
|
| 36 |
+
βcanvasβ,
|
| 37 |
+
β.gitβ
|
| 38 |
];
|
| 39 |
+
var EXCLUDE_EXT = [β.bakβ, β.tmpβ, β.logβ];
|
| 40 |
|
| 41 |
+
// ββ helpers ββββββββββββββββ
|
| 42 |
|
| 43 |
+
function log(msg) { console.log(β[hf-sync] β + msg); }
|
| 44 |
|
| 45 |
function sleep(ms) {
|
| 46 |
+
return new Promise(function(r) { setTimeout(r, ms); });
|
| 47 |
}
|
| 48 |
|
| 49 |
function repoUrl() {
|
| 50 |
+
return βhttps://user:β + HF_TOKEN +
|
| 51 |
+
β@huggingface.co/datasets/β + DATASET_ID;
|
| 52 |
}
|
| 53 |
|
| 54 |
function git(args) {
|
| 55 |
+
return execSync(βgit β + args, {
|
| 56 |
+
cwd: REPO_DIR,
|
| 57 |
+
stdio: βpipeβ,
|
| 58 |
+
timeout: 120000,
|
| 59 |
+
env: Object.assign({}, process.env, { GIT_TERMINAL_PROMPT: β0β })
|
| 60 |
+
}).toString().trim();
|
| 61 |
}
|
| 62 |
|
| 63 |
function shouldExclude(name) {
|
| 64 |
+
var i;
|
| 65 |
+
for (i = 0; i < EXCLUDE_NAMES.length; i++) {
|
| 66 |
+
if (name === EXCLUDE_NAMES[i]) return true;
|
| 67 |
+
}
|
| 68 |
+
for (i = 0; i < EXCLUDE_EXT.length; i++) {
|
| 69 |
+
if (name.indexOf(EXCLUDE_EXT[i]) >= 0) return true;
|
| 70 |
+
}
|
| 71 |
+
return false;
|
| 72 |
}
|
| 73 |
|
| 74 |
+
// ββ git ββββββββββββββββββββββββββ
|
| 75 |
|
| 76 |
function ensureRepo() {
|
| 77 |
+
if (fs.existsSync(path.join(REPO_DIR, β.gitβ))) return;
|
| 78 |
+
execSync(βrm -rf β + REPO_DIR, { stdio: βpipeβ });
|
| 79 |
+
try {
|
| 80 |
+
execSync(βgit clone βdepth 1 β + repoUrl() + β β + REPO_DIR, {
|
| 81 |
+
stdio: βpipeβ, timeout: 120000,
|
| 82 |
+
env: Object.assign({}, process.env, { GIT_TERMINAL_PROMPT: β0β })
|
| 83 |
+
});
|
| 84 |
+
log(βRepo clonedβ);
|
| 85 |
+
} catch (e) {
|
| 86 |
+
log(βClone failed, init empty: β + e.message);
|
| 87 |
+
fs.mkdirSync(REPO_DIR, { recursive: true });
|
| 88 |
+
execSync(βgit init β + REPO_DIR, { stdio: βpipeβ });
|
| 89 |
+
git(βremote add origin β + repoUrl());
|
| 90 |
+
}
|
| 91 |
+
try {
|
| 92 |
+
git(βconfig user.email openclaw@hf.spaceβ);
|
| 93 |
+
git(βconfig user.name OpenClaw-Syncβ);
|
| 94 |
+
git(βconfig pull.rebase falseβ);
|
| 95 |
+
} catch (e) { /* non-fatal */ }
|
| 96 |
}
|
| 97 |
|
| 98 |
function pull() {
|
| 99 |
+
try { git(βfetch βquiet originβ); } catch (e) { /* ignore */ }
|
| 100 |
+
try { git(βpull βquiet βno-rebase origin mainβ); }
|
| 101 |
+
catch (e) {
|
| 102 |
+
try { git(βpull βquiet βno-rebase origin masterβ); }
|
| 103 |
+
catch (e2) { /* empty repo */ }
|
| 104 |
+
}
|
| 105 |
}
|
| 106 |
|
| 107 |
function push() {
|
| 108 |
+
try {
|
| 109 |
+
git(βadd -Aβ);
|
| 110 |
+
var changed = git(βdiff βcached βname-onlyβ);
|
| 111 |
+
if (!changed) return 0;
|
| 112 |
+
var n = changed.split(β\nβ).filter(Boolean).length;
|
| 113 |
+
var ts = new Date().toISOString().replace(βTβ, β β).substring(0, 19);
|
| 114 |
+
git(βcommit -m "sync β + ts + β"β);
|
| 115 |
+
try { git(βpush βquiet origin HEAD:mainβ); }
|
| 116 |
+
catch (e) { git(βpush βquiet origin HEAD:masterβ); }
|
| 117 |
+
return n;
|
| 118 |
+
} catch (e) {
|
| 119 |
+
log(βPush failed: β + e.message);
|
| 120 |
+
return 0;
|
| 121 |
+
}
|
| 122 |
}
|
| 123 |
|
| 124 |
+
// ββ mirror ββββββββββββββββ
|
| 125 |
|
| 126 |
function mirrorToRepo(srcDir, dstDir) {
|
| 127 |
+
if (!fs.existsSync(srcDir)) return 0;
|
| 128 |
+
fs.mkdirSync(dstDir, { recursive: true });
|
| 129 |
+
var count = 0;
|
| 130 |
+
fs.readdirSync(srcDir).forEach(function(name) {
|
| 131 |
+
if (shouldExclude(name)) return;
|
| 132 |
+
var src = path.join(srcDir, name);
|
| 133 |
+
var dst = path.join(dstDir, name);
|
| 134 |
+
try {
|
| 135 |
+
if (fs.statSync(src).isDirectory()) {
|
| 136 |
+
count += mirrorToRepo(src, dst);
|
| 137 |
+
} else {
|
| 138 |
+
fs.copyFileSync(src, dst);
|
| 139 |
+
count++;
|
| 140 |
+
}
|
| 141 |
+
} catch (e) {
|
| 142 |
+
log(βCopy error [β + name + β]: β + e.message);
|
| 143 |
+
}
|
| 144 |
+
});
|
| 145 |
+
return count;
|
| 146 |
}
|
| 147 |
|
| 148 |
function mirrorFromRepo(srcDir, dstDir) {
|
| 149 |
+
if (!fs.existsSync(srcDir)) return 0;
|
| 150 |
+
fs.mkdirSync(dstDir, { recursive: true });
|
| 151 |
+
var count = 0;
|
| 152 |
+
fs.readdirSync(srcDir).forEach(function(name) {
|
| 153 |
+
if (shouldExclude(name)) return;
|
| 154 |
+
var src = path.join(srcDir, name);
|
| 155 |
+
var dst = path.join(dstDir, name);
|
| 156 |
+
try {
|
| 157 |
+
if (fs.statSync(src).isDirectory()) {
|
| 158 |
+
count += mirrorFromRepo(src, dst);
|
| 159 |
+
} else {
|
| 160 |
+
fs.copyFileSync(src, dst);
|
| 161 |
+
count++;
|
| 162 |
+
}
|
| 163 |
+
} catch (e) {
|
| 164 |
+
log(βRestore error [β + name + β]: β + e.message);
|
| 165 |
+
}
|
| 166 |
+
});
|
| 167 |
+
return count;
|
| 168 |
+
}
|
| 169 |
+
|
| 170 |
+
// ββ seed workspace ββββββββββββββββββββ
|
| 171 |
|
| 172 |
function seedWorkspaceIfNeeded() {
|
| 173 |
+
fs.mkdirSync(WORKSPACE, { recursive: true });
|
| 174 |
+
fs.mkdirSync(path.join(WORKSPACE, βmemoryβ), { recursive: true });
|
| 175 |
+
|
| 176 |
+
var soul = path.join(WORKSPACE, βSOUL.mdβ);
|
| 177 |
+
if (!fs.existsSync(soul)) {
|
| 178 |
+
fs.writeFileSync(soul, [
|
| 179 |
+
β# Soulβ,
|
| 180 |
+
ββ,
|
| 181 |
+
βYou are a helpful, warm, concise AI assistant.β,
|
| 182 |
+
ββ,
|
| 183 |
+
β## Languageβ,
|
| 184 |
+
ββ,
|
| 185 |
+
βDefault language: Simplified Chinese.β,
|
| 186 |
+
βAlways reply in Chinese unless the user writes in another language first.β,
|
| 187 |
+
ββ,
|
| 188 |
+
β## Toneβ,
|
| 189 |
+
ββ,
|
| 190 |
+
β- Natural and friendly, not overly formalβ,
|
| 191 |
+
β- Concise and to the pointβ,
|
| 192 |
+
β- Ask one clarifying question at a time when neededβ
|
| 193 |
+
].join(β\nβ) + β\nβ, βutf-8β);
|
| 194 |
+
log(βSeeded workspace/SOUL.mdβ);
|
| 195 |
+
}
|
| 196 |
|
| 197 |
+
var agents = path.join(WORKSPACE, βAGENTS.mdβ);
|
| 198 |
+
if (!fs.existsSync(agents)) {
|
| 199 |
+
fs.writeFileSync(agents, [
|
| 200 |
+
β# Agent Instructionsβ,
|
| 201 |
+
ββ,
|
| 202 |
+
β## Boot sequenceβ,
|
| 203 |
+
ββ,
|
| 204 |
+
β1. Read SOUL.md - language and personaβ,
|
| 205 |
+
β2. Read USER.md if present - user profileβ,
|
| 206 |
+
β3. Read MEMORY.md - long-term facts and rulesβ,
|
| 207 |
+
β4. Read today and yesterday memory/YYYY-MM-DD.md if presentβ,
|
| 208 |
+
ββ,
|
| 209 |
+
β## Memory rulesβ,
|
| 210 |
+
ββ,
|
| 211 |
+
β- Write important facts to MEMORY.md when askedβ,
|
| 212 |
+
β- Log daily context to memory/YYYY-MM-DD.mdβ,
|
| 213 |
+
β- Do not ask for information already providedβ
|
| 214 |
+
].join(β\nβ) + β\nβ, βutf-8β);
|
| 215 |
+
log(βSeeded workspace/AGENTS.mdβ);
|
| 216 |
+
}
|
| 217 |
|
| 218 |
+
var mem = path.join(WORKSPACE, βMEMORY.mdβ);
|
| 219 |
+
if (!fs.existsSync(mem)) {
|
| 220 |
+
fs.writeFileSync(mem, [
|
| 221 |
+
β# Long-term Memoryβ,
|
| 222 |
+
ββ,
|
| 223 |
+
β<!-- OpenClaw writes important facts here. -->β,
|
| 224 |
+
β<!-- Loaded at the start of every session. -->β
|
| 225 |
+
].join(β\nβ) + β\nβ, βutf-8β);
|
| 226 |
+
log(βSeeded workspace/MEMORY.mdβ);
|
| 227 |
+
}
|
| 228 |
}
|
| 229 |
|
| 230 |
+
// ββ boot restore ββββββββββββββ
|
| 231 |
|
| 232 |
function bootRestore() {
|
| 233 |
+
if (!HF_TOKEN || !DATASET_ID) {
|
| 234 |
+
log(βNo HF_TOKEN/HF_DATASET_ID - skipping Dataset restoreβ);
|
| 235 |
+
seedWorkspaceIfNeeded();
|
| 236 |
+
return;
|
| 237 |
+
}
|
| 238 |
|
| 239 |
+
log(βBoot restore from β + DATASET_ID + ββ¦β);
|
| 240 |
+
try {
|
| 241 |
+
ensureRepo();
|
| 242 |
+
pull();
|
| 243 |
+
} catch (e) {
|
| 244 |
+
log(βRepo init failed (β + e.message + β) - starting freshβ);
|
| 245 |
+
seedWorkspaceIfNeeded();
|
| 246 |
+
return;
|
| 247 |
+
}
|
| 248 |
|
| 249 |
+
var repoOC = path.join(REPO_DIR, βopenclawβ);
|
| 250 |
+
if (fs.existsSync(repoOC)) {
|
| 251 |
+
var n = mirrorFromRepo(repoOC, OPENCLAW_DIR);
|
| 252 |
+
log(βBoot restore done: β + n + β file(s) restored (openclaw.json excluded)β);
|
| 253 |
+
} else {
|
| 254 |
+
log(βNo data in Dataset yet - starting freshβ);
|
| 255 |
+
}
|
| 256 |
|
| 257 |
+
seedWorkspaceIfNeeded();
|
| 258 |
+
log(βBoot restore completeβ);
|
| 259 |
}
|
| 260 |
|
| 261 |
+
// ββ sync cycle ββββββββββββββββββββββ
|
| 262 |
|
| 263 |
function runSync() {
|
| 264 |
+
if (!HF_TOKEN || !DATASET_ID) return;
|
| 265 |
|
| 266 |
+
var ts = new Date().toISOString().replace(βTβ, β β).substring(0, 19);
|
| 267 |
+
log(βSync at β + ts + ββ¦β);
|
| 268 |
|
| 269 |
+
try {
|
| 270 |
+
ensureRepo();
|
| 271 |
+
pull();
|
| 272 |
+
|
| 273 |
+
```
|
| 274 |
+
var repoOC = path.join(REPO_DIR, "openclaw");
|
| 275 |
+
if (fs.existsSync(repoOC)) {
|
| 276 |
+
execSync("rm -rf " + repoOC, { stdio: "pipe" });
|
| 277 |
+
}
|
| 278 |
+
fs.mkdirSync(repoOC, { recursive: true });
|
| 279 |
+
|
| 280 |
+
var n = mirrorToRepo(OPENCLAW_DIR, repoOC);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 281 |
|
| 282 |
+
// Upload openclaw.json with JSON validation (excluded from restore, not from upload)
|
| 283 |
+
var localCfg = path.join(OPENCLAW_DIR, "openclaw.json");
|
| 284 |
+
if (fs.existsSync(localCfg)) {
|
| 285 |
+
try {
|
| 286 |
+
JSON.parse(fs.readFileSync(localCfg, "utf-8").trim());
|
| 287 |
+
fs.copyFileSync(localCfg, path.join(repoOC, "openclaw.json"));
|
| 288 |
+
n++;
|
| 289 |
+
log("openclaw.json uploaded (JSON valid)");
|
| 290 |
} catch (e) {
|
| 291 |
+
log("openclaw.json skipped (invalid JSON)");
|
| 292 |
}
|
| 293 |
}
|
| 294 |
|
| 295 |
+
var pushed = push();
|
| 296 |
+
if (pushed > 0) {
|
| 297 |
+
log("Sync done: " + pushed + " file(s) pushed");
|
| 298 |
+
} else {
|
| 299 |
+
log("Sync done: no changes");
|
| 300 |
+
}
|
| 301 |
+
```
|
| 302 |
|
| 303 |
+
} catch (e) {
|
| 304 |
+
log(βSync failed (retry in 30 min): β + e.message);
|
| 305 |
+
}
|
| 306 |
+
}
|
|
|
|
| 307 |
|
| 308 |
+
// ββ main βββββββββββββββββββββββββ
|
|
|
|
| 309 |
|
| 310 |
+
(async function() {
|
| 311 |
+
log(βStartingβ¦β);
|
| 312 |
+
log(βOPENCLAW_DIR : β + OPENCLAW_DIR);
|
| 313 |
+
log(βDATASET : β + (DATASET_ID || βNOT SETβ));
|
| 314 |
+
log(βRUNNING_FLAG : β + RUNNING_FLAG);
|
| 315 |
+
|
| 316 |
+
fs.mkdirSync(OPENCLAW_DIR, { recursive: true });
|
| 317 |
+
fs.mkdirSync(WORKSPACE, { recursive: true });
|
| 318 |
+
|
| 319 |
+
try {
|
| 320 |
+
execSync(βgit config βglobal user.email openclaw@hf.spaceβ, { stdio: βpipeβ });
|
| 321 |
+
execSync(βgit config βglobal user.name OpenClaw-Syncβ, { stdio: βpipeβ });
|
| 322 |
+
execSync(βgit config βglobal http.postBuffer 52428800β, { stdio: βpipeβ });
|
| 323 |
+
execSync(βgit config βglobal pull.rebase falseβ, { stdio: βpipeβ });
|
| 324 |
+
} catch (e) { /* non-fatal */ }
|
| 325 |
+
|
| 326 |
+
var isGatewayRestart = fs.existsSync(RUNNING_FLAG);
|
| 327 |
+
|
| 328 |
+
if (isGatewayRestart) {
|
| 329 |
+
log(βGateway restart - skipping boot restore (data already on disk)β);
|
| 330 |
+
} else {
|
| 331 |
+
log(βContainer start - running full boot restoreβ);
|
| 332 |
+
bootRestore();
|
| 333 |
+
}
|
| 334 |
|
| 335 |
+
// Write flag. NEVER delete it on exit.
|
| 336 |
+
// Only container restart clears /tmp.
|
| 337 |
+
fs.writeFileSync(RUNNING_FLAG, String(process.pid), βutf-8β);
|
| 338 |
+
log(βRunning flag writtenβ);
|
| 339 |
|
| 340 |
+
if (!HF_TOKEN || !DATASET_ID) {
|
| 341 |
+
log(βSync disabled: set HF_TOKEN and HF_DATASET_ID in Secretsβ);
|
| 342 |
+
return;
|
| 343 |
+
}
|
| 344 |
|
| 345 |
+
log(βSync loop starting (first run in 10 min)β¦β);
|
| 346 |
+
await sleep(10 * 60 * 1000);
|
| 347 |
|
| 348 |
+
while (true) {
|
| 349 |
+
runSync();
|
| 350 |
+
await sleep(INTERVAL);
|
| 351 |
+
}
|
| 352 |
+
})().catch(function(e) { console.error(β[hf-sync] Fatal: β + e.message); });
|