shennian 0.2.89 → 0.2.90
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/dist/assets/wechat-channel/macos/manifest.json +13 -4
- package/dist/assets/wechat-channel/macos/shennian-wechat-channel-helper +0 -0
- package/dist/bin/shennian.js +1 -1
- package/dist/publish-build-manifest.json +548 -0
- package/dist/scripts/wechat-rpa-confirmation.mjs +5 -97
- package/dist/src/agent-env.js +4 -105
- package/dist/src/agents/adapter.js +1 -19
- package/dist/src/agents/claude.js +8 -305
- package/dist/src/agents/codex-control.js +2 -188
- package/dist/src/agents/codex-utils.js +7 -200
- package/dist/src/agents/codex.js +15 -916
- package/dist/src/agents/command-spec.js +2 -413
- package/dist/src/agents/config-status.js +1 -226
- package/dist/src/agents/cursor.js +1 -249
- package/dist/src/agents/custom.js +4 -271
- package/dist/src/agents/detect.js +1 -56
- package/dist/src/agents/external-channel-instructions.js +10 -94
- package/dist/src/agents/gemini.js +1 -173
- package/dist/src/agents/manager.js +13 -157
- package/dist/src/agents/model-registry/cache.js +1 -37
- package/dist/src/agents/model-registry/discovery.js +2 -187
- package/dist/src/agents/model-registry/parsers.js +4 -447
- package/dist/src/agents/model-registry/runner.js +1 -30
- package/dist/src/agents/model-registry/service.js +1 -78
- package/dist/src/agents/model-registry/types.js +1 -8
- package/dist/src/agents/model-registry.js +1 -18
- package/dist/src/agents/openclaw.js +2 -275
- package/dist/src/agents/opencode.js +1 -231
- package/dist/src/agents/pi-context.js +12 -217
- package/dist/src/agents/pi.js +14 -723
- package/dist/src/agents/platform-instructions.js +9 -54
- package/dist/src/channels/base.js +1 -3
- package/dist/src/channels/registry.js +1 -30
- package/dist/src/channels/reply-split.js +10 -89
- package/dist/src/channels/runtime.js +5 -564
- package/dist/src/channels/secret-registry.js +1 -46
- package/dist/src/channels/websocket.js +8 -378
- package/dist/src/channels/wechat-channel/anchor.js +1 -65
- package/dist/src/channels/wechat-channel/client.js +1 -96
- package/dist/src/channels/wechat-channel/cooldown.js +1 -38
- package/dist/src/channels/wechat-channel/fingerprint.js +1 -71
- package/dist/src/channels/wechat-channel/helper-assets.d.ts +10 -1
- package/dist/src/channels/wechat-channel/helper-assets.js +1 -68
- package/dist/src/channels/wechat-channel/helper-client.js +3 -149
- package/dist/src/channels/wechat-channel/helper-protocol.d.ts +1 -1
- package/dist/src/channels/wechat-channel/helper-protocol.js +1 -115
- package/dist/src/channels/wechat-channel/index.d.ts +1 -0
- package/dist/src/channels/wechat-channel/index.js +1 -19
- package/dist/src/channels/wechat-channel/ledger.js +1 -54
- package/dist/src/channels/wechat-channel/media-resolver.js +1 -181
- package/dist/src/channels/wechat-channel/message-key.js +1 -105
- package/dist/src/channels/wechat-channel/observer.js +1 -118
- package/dist/src/channels/wechat-channel/outbound-ledger.d.ts +3 -0
- package/dist/src/channels/wechat-channel/outbound-ledger.js +2 -112
- package/dist/src/channels/wechat-channel/outbound-sender.d.ts +26 -0
- package/dist/src/channels/wechat-channel/outbound-sender.js +1 -0
- package/dist/src/channels/wechat-channel/preflight.js +1 -48
- package/dist/src/channels/wechat-channel/runner.js +1 -84
- package/dist/src/channels/wechat-channel/runtime.js +1 -66
- package/dist/src/channels/wechat-channel/scheduler.d.ts +5 -0
- package/dist/src/channels/wechat-channel/scheduler.js +1 -152
- package/dist/src/channels/wechat-rpa/macos-flow.js +1 -96
- package/dist/src/channels/wechat-rpa/macos.js +6 -48
- package/dist/src/channels/wechat-rpa/normalizer.js +7 -127
- package/dist/src/channels/wechat-rpa.js +6 -1028
- package/dist/src/channels/wecom.js +4 -357
- package/dist/src/commands/agent.js +6 -131
- package/dist/src/commands/daemon-windows.js +8 -48
- package/dist/src/commands/daemon.js +19 -1013
- package/dist/src/commands/external-attachments.js +1 -51
- package/dist/src/commands/external.js +1 -137
- package/dist/src/commands/manager.js +2 -391
- package/dist/src/commands/pair-qr.js +1 -6
- package/dist/src/commands/pair.js +9 -287
- package/dist/src/commands/tools.js +1 -34
- package/dist/src/commands/upgrade.js +1 -198
- package/dist/src/config/index.js +1 -35
- package/dist/src/daemon-log.js +6 -58
- package/dist/src/env-path.js +1 -64
- package/dist/src/fs/boundary.js +1 -126
- package/dist/src/fs/handler.js +1 -130
- package/dist/src/fs/security.js +1 -32
- package/dist/src/fs/text-decoder.js +1 -110
- package/dist/src/index.js +2 -404
- package/dist/src/log-reporter.js +1 -16
- package/dist/src/manager/prompt.js +29 -34
- package/dist/src/manager/registry.js +2 -269
- package/dist/src/manager/runtime.js +19 -1007
- package/dist/src/native-fusion/config.js +1 -5
- package/dist/src/native-fusion/opencode-parser.js +3 -123
- package/dist/src/native-fusion/parser-common.js +8 -264
- package/dist/src/native-fusion/parsers.js +8 -729
- package/dist/src/native-fusion/service.js +2 -225
- package/dist/src/native-fusion/state.js +1 -22
- package/dist/src/native-fusion/types.js +1 -1
- package/dist/src/region.js +1 -88
- package/dist/src/relay/client.js +1 -343
- package/dist/src/session/archive-zip.js +1 -220
- package/dist/src/session/handlers/agent-config.js +1 -150
- package/dist/src/session/handlers/agents.js +1 -55
- package/dist/src/session/handlers/chat.js +2 -751
- package/dist/src/session/handlers/control.js +1 -55
- package/dist/src/session/handlers/fs.js +1 -783
- package/dist/src/session/handlers/session-refresh.js +1 -47
- package/dist/src/session/handlers/skills.js +1 -121
- package/dist/src/session/handlers/title.js +1 -60
- package/dist/src/session/handlers/tool-detail.js +1 -218
- package/dist/src/session/manager.js +1 -319
- package/dist/src/session/projection.js +1 -54
- package/dist/src/session/queue.js +4 -317
- package/dist/src/session/remote-attachments.js +1 -72
- package/dist/src/session/store.js +3 -109
- package/dist/src/session/types.js +1 -4
- package/dist/src/skills/registry.js +15 -148
- package/dist/src/skills/setup.js +1 -101
- package/dist/src/tools/markdown-to-pdf.js +10 -346
- package/dist/src/upgrade/engine.js +3 -347
- package/package.json +3 -2
|
@@ -1,347 +1,3 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
import { execSync, exec as execCb } from 'node:child_process';
|
|
5
|
-
import { promisify } from 'node:util';
|
|
6
|
-
import { fileURLToPath } from 'node:url';
|
|
7
|
-
import { getShennianDir, resolveShennianPath } from '../config/index.js';
|
|
8
|
-
const exec = promisify(execCb);
|
|
9
|
-
const BACKUP_DIR = resolveShennianPath('backup');
|
|
10
|
-
const UPGRADE_ATTEMPT_FILE = resolveShennianPath('upgrade-attempt.json');
|
|
11
|
-
const MAX_CRASH_COUNT = 3;
|
|
12
|
-
const BACKUP_TTL_DAYS = 7;
|
|
13
|
-
const NPM_REGISTRY_FALLBACK = 'https://registry.npmjs.org';
|
|
14
|
-
const RETRY_DELAYS_MS = [
|
|
15
|
-
5 * 60_000,
|
|
16
|
-
30 * 60_000,
|
|
17
|
-
2 * 60 * 60_000,
|
|
18
|
-
6 * 60 * 60_000,
|
|
19
|
-
];
|
|
20
|
-
// ─── Version helpers ─────────────────────────────────────────────────────────
|
|
21
|
-
export function getCurrentVersion() {
|
|
22
|
-
try {
|
|
23
|
-
const pkgPath = findPackageJson();
|
|
24
|
-
if (pkgPath) {
|
|
25
|
-
const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'));
|
|
26
|
-
return pkg.version;
|
|
27
|
-
}
|
|
28
|
-
}
|
|
29
|
-
catch { /* noop */ }
|
|
30
|
-
return '0.0.0';
|
|
31
|
-
}
|
|
32
|
-
export function findPackageJson(startFilePath = fileURLToPath(import.meta.url)) {
|
|
33
|
-
// Walk up from this file's location to find package.json
|
|
34
|
-
let dir = path.dirname(startFilePath);
|
|
35
|
-
for (let i = 0; i < 6; i++) {
|
|
36
|
-
const candidate = path.join(dir, 'package.json');
|
|
37
|
-
if (fs.existsSync(candidate)) {
|
|
38
|
-
const pkg = JSON.parse(fs.readFileSync(candidate, 'utf-8'));
|
|
39
|
-
if (pkg.name === 'shennian')
|
|
40
|
-
return candidate;
|
|
41
|
-
}
|
|
42
|
-
dir = path.dirname(dir);
|
|
43
|
-
}
|
|
44
|
-
return null;
|
|
45
|
-
}
|
|
46
|
-
export async function fetchLatestVersion() {
|
|
47
|
-
const res = await fetch('https://registry.npmjs.org/shennian/latest', {
|
|
48
|
-
headers: { Accept: 'application/json' },
|
|
49
|
-
signal: AbortSignal.timeout(10_000),
|
|
50
|
-
});
|
|
51
|
-
if (!res.ok)
|
|
52
|
-
throw new Error(`npm registry returned ${res.status}`);
|
|
53
|
-
const data = (await res.json());
|
|
54
|
-
return data.version;
|
|
55
|
-
}
|
|
56
|
-
/**
|
|
57
|
-
* Compare semver strings. Returns:
|
|
58
|
-
* 'none' — same or older
|
|
59
|
-
* 'patch' — only patch differs
|
|
60
|
-
* 'minor' — minor or patch differs
|
|
61
|
-
* 'major' — major differs
|
|
62
|
-
*/
|
|
63
|
-
export function compareVersions(current, latest) {
|
|
64
|
-
const parse = (v) => v.replace(/^v/, '').split('.').map(Number);
|
|
65
|
-
const [cMaj, cMin, cPat] = parse(current);
|
|
66
|
-
const [lMaj, lMin, lPat] = parse(latest);
|
|
67
|
-
if (lMaj > cMaj)
|
|
68
|
-
return 'major';
|
|
69
|
-
if (lMaj === cMaj && lMin > cMin)
|
|
70
|
-
return 'minor';
|
|
71
|
-
if (lMaj === cMaj && lMin === cMin && lPat > cPat)
|
|
72
|
-
return 'patch';
|
|
73
|
-
return 'none';
|
|
74
|
-
}
|
|
75
|
-
// ─── npm global path helpers ──────────────────────────────────────────────────
|
|
76
|
-
function getNpmGlobalRoot() {
|
|
77
|
-
return execSync('npm root -g', { encoding: 'utf-8', stdio: 'pipe', windowsHide: true }).trim();
|
|
78
|
-
}
|
|
79
|
-
function getGlobalPkgDir() {
|
|
80
|
-
return path.join(getNpmGlobalRoot(), 'shennian');
|
|
81
|
-
}
|
|
82
|
-
function getGlobalBinScript() {
|
|
83
|
-
const root = getNpmGlobalRoot();
|
|
84
|
-
return path.join(root, 'shennian', 'dist', 'bin', 'shennian.js');
|
|
85
|
-
}
|
|
86
|
-
// ─── Backup / Restore ─────────────────────────────────────────────────────────
|
|
87
|
-
function backupVersion(version) {
|
|
88
|
-
const pkgDir = getGlobalPkgDir();
|
|
89
|
-
const dest = path.join(BACKUP_DIR, version);
|
|
90
|
-
fs.rmSync(dest, { recursive: true, force: true });
|
|
91
|
-
fs.mkdirSync(dest, { recursive: true });
|
|
92
|
-
copyPackageRuntimeFiles(pkgDir, dest);
|
|
93
|
-
}
|
|
94
|
-
function restoreVersion(version) {
|
|
95
|
-
const src = path.join(BACKUP_DIR, version);
|
|
96
|
-
if (!fs.existsSync(src))
|
|
97
|
-
throw new Error(`Backup for ${version} not found`);
|
|
98
|
-
const pkgDir = getGlobalPkgDir();
|
|
99
|
-
fs.cpSync(src, pkgDir, { recursive: true, force: true });
|
|
100
|
-
}
|
|
101
|
-
export function copyPackageRuntimeFiles(src, dest) {
|
|
102
|
-
for (const entry of ['package.json', 'README.md', 'dist']) {
|
|
103
|
-
const from = path.join(src, entry);
|
|
104
|
-
if (!fs.existsSync(from))
|
|
105
|
-
continue;
|
|
106
|
-
fs.cpSync(from, path.join(dest, entry), { recursive: true, force: true });
|
|
107
|
-
}
|
|
108
|
-
}
|
|
109
|
-
function cleanOldBackups() {
|
|
110
|
-
try {
|
|
111
|
-
if (!fs.existsSync(BACKUP_DIR))
|
|
112
|
-
return;
|
|
113
|
-
const cutoff = Date.now() - BACKUP_TTL_DAYS * 86400_000;
|
|
114
|
-
for (const entry of fs.readdirSync(BACKUP_DIR)) {
|
|
115
|
-
const full = path.join(BACKUP_DIR, entry);
|
|
116
|
-
const stat = fs.statSync(full);
|
|
117
|
-
if (stat.mtimeMs < cutoff) {
|
|
118
|
-
fs.rmSync(full, { recursive: true, force: true });
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
catch { /* noop */ }
|
|
123
|
-
}
|
|
124
|
-
// ─── Upgrade attempt tracking (crash detection) ───────────────────────────────
|
|
125
|
-
export function readUpgradeAttempt() {
|
|
126
|
-
try {
|
|
127
|
-
return JSON.parse(fs.readFileSync(UPGRADE_ATTEMPT_FILE, 'utf-8'));
|
|
128
|
-
}
|
|
129
|
-
catch {
|
|
130
|
-
return null;
|
|
131
|
-
}
|
|
132
|
-
}
|
|
133
|
-
export function writeUpgradeAttempt(attempt) {
|
|
134
|
-
fs.mkdirSync(getShennianDir(), { recursive: true });
|
|
135
|
-
fs.writeFileSync(UPGRADE_ATTEMPT_FILE, JSON.stringify(attempt, null, 2));
|
|
136
|
-
}
|
|
137
|
-
export function clearUpgradeAttempt() {
|
|
138
|
-
try {
|
|
139
|
-
fs.unlinkSync(UPGRADE_ATTEMPT_FILE);
|
|
140
|
-
}
|
|
141
|
-
catch { /* noop */ }
|
|
142
|
-
}
|
|
143
|
-
function getUpgradeFailuresFile() {
|
|
144
|
-
return resolveShennianPath('upgrade-failures.json');
|
|
145
|
-
}
|
|
146
|
-
export function readUpgradeFailures(filePath = getUpgradeFailuresFile()) {
|
|
147
|
-
try {
|
|
148
|
-
const parsed = JSON.parse(fs.readFileSync(filePath, 'utf-8'));
|
|
149
|
-
if (!parsed || typeof parsed !== 'object' || !parsed.versions)
|
|
150
|
-
return { versions: {} };
|
|
151
|
-
return { versions: parsed.versions };
|
|
152
|
-
}
|
|
153
|
-
catch {
|
|
154
|
-
return { versions: {} };
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
function writeUpgradeFailures(failures) {
|
|
158
|
-
fs.mkdirSync(getShennianDir(), { recursive: true });
|
|
159
|
-
fs.writeFileSync(getUpgradeFailuresFile(), JSON.stringify(failures, null, 2));
|
|
160
|
-
}
|
|
161
|
-
export function getUpgradeRetryDelayMs(failureCount) {
|
|
162
|
-
return RETRY_DELAYS_MS[Math.min(Math.max(failureCount - 1, 0), RETRY_DELAYS_MS.length - 1)];
|
|
163
|
-
}
|
|
164
|
-
export function isUpgradeVersionInCooldown(version, now = Date.now(), failures = readUpgradeFailures()) {
|
|
165
|
-
const record = failures.versions[version];
|
|
166
|
-
return Boolean(record && record.nextRetryAt > now);
|
|
167
|
-
}
|
|
168
|
-
export function recordUpgradeFailure(version, error, now = Date.now()) {
|
|
169
|
-
const failures = readUpgradeFailures();
|
|
170
|
-
const previous = failures.versions[version];
|
|
171
|
-
const failureCount = (previous?.failureCount ?? 0) + 1;
|
|
172
|
-
const record = {
|
|
173
|
-
version,
|
|
174
|
-
failureCount,
|
|
175
|
-
firstFailedAt: previous?.firstFailedAt ?? now,
|
|
176
|
-
lastFailedAt: now,
|
|
177
|
-
nextRetryAt: now + getUpgradeRetryDelayMs(failureCount),
|
|
178
|
-
error,
|
|
179
|
-
};
|
|
180
|
-
failures.versions[version] = record;
|
|
181
|
-
writeUpgradeFailures(failures);
|
|
182
|
-
return record;
|
|
183
|
-
}
|
|
184
|
-
export function clearUpgradeFailure(version) {
|
|
185
|
-
const failures = readUpgradeFailures();
|
|
186
|
-
if (!failures.versions[version])
|
|
187
|
-
return;
|
|
188
|
-
delete failures.versions[version];
|
|
189
|
-
writeUpgradeFailures(failures);
|
|
190
|
-
}
|
|
191
|
-
/**
|
|
192
|
-
* Called on daemon startup. Detects crash loops from a failed upgrade and
|
|
193
|
-
* initiates rollback if crash count exceeds threshold.
|
|
194
|
-
* Returns true if rollback was performed (caller should exit and let service manager restart).
|
|
195
|
-
*/
|
|
196
|
-
export async function handleStartupCrashCheck() {
|
|
197
|
-
const attempt = readUpgradeAttempt();
|
|
198
|
-
if (!attempt)
|
|
199
|
-
return false;
|
|
200
|
-
const elapsed = Date.now() - attempt.attemptAt;
|
|
201
|
-
// If more than 10 minutes have passed, this is a stale record — clear and continue
|
|
202
|
-
if (elapsed > 10 * 60 * 1000) {
|
|
203
|
-
clearUpgradeAttempt();
|
|
204
|
-
return false;
|
|
205
|
-
}
|
|
206
|
-
attempt.attemptCount++;
|
|
207
|
-
if (attempt.attemptCount >= MAX_CRASH_COUNT) {
|
|
208
|
-
console.error(`[upgrade] New version crashed ${attempt.attemptCount} times, rolling back to ${attempt.from}...`);
|
|
209
|
-
try {
|
|
210
|
-
restoreVersion(attempt.from);
|
|
211
|
-
clearUpgradeAttempt();
|
|
212
|
-
console.error(`[upgrade] Rolled back to ${attempt.from}. Restarting...`);
|
|
213
|
-
return true;
|
|
214
|
-
}
|
|
215
|
-
catch (err) {
|
|
216
|
-
console.error(`[upgrade] Rollback failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
217
|
-
clearUpgradeAttempt();
|
|
218
|
-
return false;
|
|
219
|
-
}
|
|
220
|
-
}
|
|
221
|
-
writeUpgradeAttempt(attempt);
|
|
222
|
-
return false;
|
|
223
|
-
}
|
|
224
|
-
// ─── Core upgrade ─────────────────────────────────────────────────────────────
|
|
225
|
-
export async function performUpgrade(targetVersion, onProgress, opts = {}) {
|
|
226
|
-
const currentVersion = opts.currentVersion ?? getCurrentVersion();
|
|
227
|
-
const installedVersion = getCurrentVersion();
|
|
228
|
-
if (currentVersion === targetVersion) {
|
|
229
|
-
clearUpgradeFailure(targetVersion);
|
|
230
|
-
return { ok: false, error: `Already on version ${targetVersion}` };
|
|
231
|
-
}
|
|
232
|
-
// Step 1: Verify npm is accessible
|
|
233
|
-
onProgress({ step: 'checking' });
|
|
234
|
-
try {
|
|
235
|
-
execSync('npm --version', { stdio: 'pipe', windowsHide: true });
|
|
236
|
-
}
|
|
237
|
-
catch {
|
|
238
|
-
return { ok: false, error: 'npm is not available in PATH' };
|
|
239
|
-
}
|
|
240
|
-
if (installedVersion === targetVersion) {
|
|
241
|
-
onProgress({ step: 'verifying', version: targetVersion });
|
|
242
|
-
try {
|
|
243
|
-
const binScript = getGlobalBinScript();
|
|
244
|
-
const { stdout } = await exec(`node "${binScript}" --version`, { timeout: 10_000, windowsHide: true });
|
|
245
|
-
if (!stdout.trim())
|
|
246
|
-
throw new Error('Empty output from --version check');
|
|
247
|
-
}
|
|
248
|
-
catch (err) {
|
|
249
|
-
return {
|
|
250
|
-
ok: false,
|
|
251
|
-
error: `Smoke test failed: ${err instanceof Error ? err.message : String(err)}`,
|
|
252
|
-
};
|
|
253
|
-
}
|
|
254
|
-
writeUpgradeAttempt({
|
|
255
|
-
from: currentVersion,
|
|
256
|
-
to: targetVersion,
|
|
257
|
-
attemptCount: 0,
|
|
258
|
-
attemptAt: Date.now(),
|
|
259
|
-
});
|
|
260
|
-
clearUpgradeFailure(targetVersion);
|
|
261
|
-
onProgress({ step: 'restarting', from: currentVersion, to: targetVersion });
|
|
262
|
-
return { ok: true, from: currentVersion, to: targetVersion };
|
|
263
|
-
}
|
|
264
|
-
// Step 2: Backup current version
|
|
265
|
-
onProgress({ step: 'backing-up' });
|
|
266
|
-
try {
|
|
267
|
-
fs.mkdirSync(BACKUP_DIR, { recursive: true });
|
|
268
|
-
backupVersion(currentVersion);
|
|
269
|
-
}
|
|
270
|
-
catch (err) {
|
|
271
|
-
return { ok: false, error: `Backup failed: ${err instanceof Error ? err.message : String(err)}` };
|
|
272
|
-
}
|
|
273
|
-
// Step 3: npm install new version
|
|
274
|
-
onProgress({ step: 'installing', version: targetVersion });
|
|
275
|
-
try {
|
|
276
|
-
await installShennianVersion(targetVersion);
|
|
277
|
-
}
|
|
278
|
-
catch (err) {
|
|
279
|
-
// Restore backup and abort
|
|
280
|
-
try {
|
|
281
|
-
restoreVersion(currentVersion);
|
|
282
|
-
}
|
|
283
|
-
catch { /* best effort */ }
|
|
284
|
-
return {
|
|
285
|
-
ok: false,
|
|
286
|
-
error: `npm install failed: ${err instanceof Error ? err.message : String(err)}`,
|
|
287
|
-
rolledBack: true,
|
|
288
|
-
};
|
|
289
|
-
}
|
|
290
|
-
// Step 4: Smoke test — verify the new binary works
|
|
291
|
-
onProgress({ step: 'verifying', version: targetVersion });
|
|
292
|
-
try {
|
|
293
|
-
const binScript = getGlobalBinScript();
|
|
294
|
-
const { stdout } = await exec(`node "${binScript}" --version`, { timeout: 10_000, windowsHide: true });
|
|
295
|
-
if (!stdout.trim())
|
|
296
|
-
throw new Error('Empty output from --version check');
|
|
297
|
-
}
|
|
298
|
-
catch (err) {
|
|
299
|
-
try {
|
|
300
|
-
restoreVersion(currentVersion);
|
|
301
|
-
}
|
|
302
|
-
catch { /* best effort */ }
|
|
303
|
-
return {
|
|
304
|
-
ok: false,
|
|
305
|
-
error: `Smoke test failed: ${err instanceof Error ? err.message : String(err)}`,
|
|
306
|
-
rolledBack: true,
|
|
307
|
-
};
|
|
308
|
-
}
|
|
309
|
-
// Step 5: Write upgrade attempt file (crash detection on restart)
|
|
310
|
-
writeUpgradeAttempt({
|
|
311
|
-
from: currentVersion,
|
|
312
|
-
to: targetVersion,
|
|
313
|
-
attemptCount: 0,
|
|
314
|
-
attemptAt: Date.now(),
|
|
315
|
-
});
|
|
316
|
-
cleanOldBackups();
|
|
317
|
-
clearUpgradeFailure(targetVersion);
|
|
318
|
-
// Step 6: Signal restart (caller sends upgrade.status: restarting before killing)
|
|
319
|
-
onProgress({ step: 'restarting', from: currentVersion, to: targetVersion });
|
|
320
|
-
return { ok: true, from: currentVersion, to: targetVersion };
|
|
321
|
-
}
|
|
322
|
-
async function installShennianVersion(targetVersion) {
|
|
323
|
-
try {
|
|
324
|
-
await exec(`npm install -g shennian@${targetVersion}`, { timeout: 120_000, windowsHide: true });
|
|
325
|
-
}
|
|
326
|
-
catch (error) {
|
|
327
|
-
const firstMessage = error instanceof Error ? error.message : String(error);
|
|
328
|
-
try {
|
|
329
|
-
await exec(`npm install -g shennian@${targetVersion} --registry ${NPM_REGISTRY_FALLBACK}`, {
|
|
330
|
-
timeout: 120_000,
|
|
331
|
-
windowsHide: true,
|
|
332
|
-
});
|
|
333
|
-
}
|
|
334
|
-
catch (fallbackError) {
|
|
335
|
-
const secondMessage = fallbackError instanceof Error ? fallbackError.message : String(fallbackError);
|
|
336
|
-
throw new Error(`${firstMessage}\n--- npmjs fallback ---\n${secondMessage}`);
|
|
337
|
-
}
|
|
338
|
-
}
|
|
339
|
-
}
|
|
340
|
-
export async function checkForUpdate(currentVersion) {
|
|
341
|
-
const current = currentVersion ?? getCurrentVersion();
|
|
342
|
-
const latest = await fetchLatestVersion();
|
|
343
|
-
const changeType = compareVersions(current, latest);
|
|
344
|
-
if (changeType === 'none')
|
|
345
|
-
return { hasUpdate: false };
|
|
346
|
-
return { hasUpdate: true, current, latest, changeType };
|
|
347
|
-
}
|
|
1
|
+
import i from"node:fs";import c from"node:path";import{execSync as w,exec as $}from"node:child_process";import{promisify as C}from"node:util";import{fileURLToPath as R}from"node:url";import{getShennianDir as v,resolveShennianPath as p}from"../config/index.js";const f=C($),u=p("backup"),l=p("upgrade-attempt.json"),D=3,M=7,N="https://registry.npmjs.org",x=[5*6e4,30*6e4,120*6e4,360*6e4];function m(){try{const t=T();if(t)return JSON.parse(i.readFileSync(t,"utf-8")).version}catch{}return"0.0.0"}function T(t=R(import.meta.url)){let r=c.dirname(t);for(let n=0;n<6;n++){const e=c.join(r,"package.json");if(i.existsSync(e)&&JSON.parse(i.readFileSync(e,"utf-8")).name==="shennian")return e;r=c.dirname(r)}return null}async function B(){const t=await fetch("https://registry.npmjs.org/shennian/latest",{headers:{Accept:"application/json"},signal:AbortSignal.timeout(1e4)});if(!t.ok)throw new Error(`npm registry returned ${t.status}`);return(await t.json()).version}function H(t,r){const n=U=>U.replace(/^v/,"").split(".").map(Number),[e,s,o]=n(t),[a,k,F]=n(r);return a>e?"major":a===e&&k>s?"minor":a===e&&k===s&&F>o?"patch":"none"}function A(){return w("npm root -g",{encoding:"utf-8",stdio:"pipe",windowsHide:!0}).trim()}function E(){return c.join(A(),"shennian")}function j(){const t=A();return c.join(t,"shennian","dist","bin","shennian.js")}function O(t){const r=E(),n=c.join(u,t);i.rmSync(n,{recursive:!0,force:!0}),i.mkdirSync(n,{recursive:!0}),P(r,n)}function d(t){const r=c.join(u,t);if(!i.existsSync(r))throw new Error(`Backup for ${t} not found`);const n=E();i.cpSync(r,n,{recursive:!0,force:!0})}function P(t,r){for(const n of["package.json","README.md","dist"]){const e=c.join(t,n);i.existsSync(e)&&i.cpSync(e,c.join(r,n),{recursive:!0,force:!0})}}function J(){try{if(!i.existsSync(u))return;const t=Date.now()-M*864e5;for(const r of i.readdirSync(u)){const n=c.join(u,r);i.statSync(n).mtimeMs<t&&i.rmSync(n,{recursive:!0,force:!0})}}catch{}}function L(){try{return JSON.parse(i.readFileSync(l,"utf-8"))}catch{return null}}function y(t){i.mkdirSync(v(),{recursive:!0}),i.writeFileSync(l,JSON.stringify(t,null,2))}function g(){try{i.unlinkSync(l)}catch{}}function _(){return p("upgrade-failures.json")}function h(t=_()){try{const r=JSON.parse(i.readFileSync(t,"utf-8"));return!r||typeof r!="object"||!r.versions?{versions:{}}:{versions:r.versions}}catch{return{versions:{}}}}function b(t){i.mkdirSync(v(),{recursive:!0}),i.writeFileSync(_(),JSON.stringify(t,null,2))}function G(t){return x[Math.min(Math.max(t-1,0),x.length-1)]}function W(t,r=Date.now(),n=h()){const e=n.versions[t];return!!(e&&e.nextRetryAt>r)}function Z(t,r,n=Date.now()){const e=h(),s=e.versions[t],o=(s?.failureCount??0)+1,a={version:t,failureCount:o,firstFailedAt:s?.firstFailedAt??n,lastFailedAt:n,nextRetryAt:n+G(o),error:r};return e.versions[t]=a,b(e),a}function S(t){const r=h();r.versions[t]&&(delete r.versions[t],b(r))}async function V(){const t=L();if(!t)return!1;if(Date.now()-t.attemptAt>600*1e3)return g(),!1;if(t.attemptCount++,t.attemptCount>=D){console.error(`[upgrade] New version crashed ${t.attemptCount} times, rolling back to ${t.from}...`);try{return d(t.from),g(),console.error(`[upgrade] Rolled back to ${t.from}. Restarting...`),!0}catch(n){return console.error(`[upgrade] Rollback failed: ${n instanceof Error?n.message:String(n)}`),g(),!1}}return y(t),!1}async function tt(t,r,n={}){const e=n.currentVersion??m(),s=m();if(e===t)return S(t),{ok:!1,error:`Already on version ${t}`};r({step:"checking"});try{w("npm --version",{stdio:"pipe",windowsHide:!0})}catch{return{ok:!1,error:"npm is not available in PATH"}}if(s===t){r({step:"verifying",version:t});try{const o=j(),{stdout:a}=await f(`node "${o}" --version`,{timeout:1e4,windowsHide:!0});if(!a.trim())throw new Error("Empty output from --version check")}catch(o){return{ok:!1,error:`Smoke test failed: ${o instanceof Error?o.message:String(o)}`}}return y({from:e,to:t,attemptCount:0,attemptAt:Date.now()}),S(t),r({step:"restarting",from:e,to:t}),{ok:!0,from:e,to:t}}r({step:"backing-up"});try{i.mkdirSync(u,{recursive:!0}),O(e)}catch(o){return{ok:!1,error:`Backup failed: ${o instanceof Error?o.message:String(o)}`}}r({step:"installing",version:t});try{await I(t)}catch(o){try{d(e)}catch{}return{ok:!1,error:`npm install failed: ${o instanceof Error?o.message:String(o)}`,rolledBack:!0}}r({step:"verifying",version:t});try{const o=j(),{stdout:a}=await f(`node "${o}" --version`,{timeout:1e4,windowsHide:!0});if(!a.trim())throw new Error("Empty output from --version check")}catch(o){try{d(e)}catch{}return{ok:!1,error:`Smoke test failed: ${o instanceof Error?o.message:String(o)}`,rolledBack:!0}}return y({from:e,to:t,attemptCount:0,attemptAt:Date.now()}),J(),S(t),r({step:"restarting",from:e,to:t}),{ok:!0,from:e,to:t}}async function I(t){try{await f(`npm install -g shennian@${t}`,{timeout:12e4,windowsHide:!0})}catch(r){const n=r instanceof Error?r.message:String(r);try{await f(`npm install -g shennian@${t} --registry ${N}`,{timeout:12e4,windowsHide:!0})}catch(e){const s=e instanceof Error?e.message:String(e);throw new Error(`${n}
|
|
2
|
+
--- npmjs fallback ---
|
|
3
|
+
${s}`)}}}async function rt(t){const r=t??m(),n=await B(),e=H(r,n);return e==="none"?{hasUpdate:!1}:{hasUpdate:!0,current:r,latest:n,changeType:e}}export{rt as checkForUpdate,g as clearUpgradeAttempt,S as clearUpgradeFailure,H as compareVersions,P as copyPackageRuntimeFiles,B as fetchLatestVersion,T as findPackageJson,m as getCurrentVersion,G as getUpgradeRetryDelayMs,V as handleStartupCrashCheck,W as isUpgradeVersionInCooldown,tt as performUpgrade,L as readUpgradeAttempt,h as readUpgradeFailures,Z as recordUpgradeFailure,y as writeUpgradeAttempt};
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "shennian",
|
|
3
|
-
"version": "0.2.
|
|
3
|
+
"version": "0.2.90",
|
|
4
4
|
"description": "Shennian — AI Agent Control Plane CLI",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"bin": {
|
|
@@ -46,12 +46,13 @@
|
|
|
46
46
|
"@types/node": "^20",
|
|
47
47
|
"@types/qrcode-terminal": "^0.12.2",
|
|
48
48
|
"@types/ws": "^8.18.1",
|
|
49
|
+
"esbuild": "^0.27.4",
|
|
49
50
|
"tsx": "^4.19.4",
|
|
50
51
|
"typescript": "^5.9.3"
|
|
51
52
|
},
|
|
52
53
|
"scripts": {
|
|
53
54
|
"build": "tsc && node scripts/copy-wechat-rpa-assets.mjs && node -e \"require('node:fs').chmodSync('dist/bin/shennian.js', 0o755)\"",
|
|
54
|
-
"build:publish": "node -e \"const fs=require('node:fs'); fs.rmSync('dist', { recursive: true, force: true }); fs.rmSync('.tsbuildinfo.publish', { force: true })\" && tsc -p tsconfig.publish.json && node scripts/copy-wechat-rpa-assets.mjs && node -e \"require('node:fs').chmodSync('dist/bin/shennian.js', 0o755)\" && node scripts/check-publish-artifact.mjs",
|
|
55
|
+
"build:publish": "node -e \"const fs=require('node:fs'); fs.rmSync('dist', { recursive: true, force: true }); fs.rmSync('.tsbuildinfo.publish', { force: true })\" && tsc -p tsconfig.publish.json && node scripts/copy-wechat-rpa-assets.mjs && node scripts/minify-publish-js.mjs && node -e \"require('node:fs').chmodSync('dist/bin/shennian.js', 0o755)\" && node scripts/check-publish-artifact.mjs",
|
|
55
56
|
"dev": "tsc --watch"
|
|
56
57
|
}
|
|
57
58
|
}
|