mirror of
https://github.com/Syngnat/GoNavi.git
synced 2026-05-30 15:19:59 +08:00
- 修正 Wails 前端安装脚本在 Windows 下启动 npm 失败的问题 - 统一从脚本路径解析 frontend 目录,避免 cwd 变化导致 package.json 定位错误 - 增加 CI 安装诊断日志与 npm 失败状态输出
126 lines
3.9 KiB
JavaScript
126 lines
3.9 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
import { spawnSync } from 'node:child_process';
|
|
import { createHash } from 'node:crypto';
|
|
import { existsSync, readFileSync, statSync, writeFileSync } from 'node:fs';
|
|
import path from 'node:path';
|
|
import { fileURLToPath } from 'node:url';
|
|
|
|
const scriptDir = path.dirname(fileURLToPath(import.meta.url));
|
|
const frontendDir = path.resolve(scriptDir, '..');
|
|
const packageJsonPath = path.join(frontendDir, 'package.json');
|
|
const packageLockPath = path.join(frontendDir, 'package-lock.json');
|
|
const nodeModulesPath = path.join(frontendDir, 'node_modules');
|
|
const npmHiddenLockPath = path.join(nodeModulesPath, '.package-lock.json');
|
|
const installStatePath = path.join(nodeModulesPath, '.gonavi-install-state.json');
|
|
const npmCommand = 'npm';
|
|
const commonArgs = [
|
|
'--prefer-offline',
|
|
'--no-audit',
|
|
'--fund=false',
|
|
'--fetch-retries=5',
|
|
'--fetch-retry-mintimeout=20000',
|
|
'--fetch-retry-maxtimeout=120000',
|
|
];
|
|
const isCI = process.env.CI === 'true' || process.env.GITHUB_ACTIONS === 'true';
|
|
|
|
const fail = (message) => {
|
|
console.error(`[gonavi-frontend-install] ${message}`);
|
|
process.exit(1);
|
|
};
|
|
|
|
const exitWithStatus = (status) => {
|
|
process.exit(typeof status === 'number' && status > 0 && status <= 255 ? status : 1);
|
|
};
|
|
|
|
const hashFile = (filePath) => {
|
|
const hash = createHash('sha256');
|
|
hash.update(readFileSync(filePath));
|
|
return hash.digest('hex');
|
|
};
|
|
|
|
const currentState = () => ({
|
|
packageJson: hashFile(packageJsonPath),
|
|
packageLock: existsSync(packageLockPath) ? hashFile(packageLockPath) : '',
|
|
});
|
|
|
|
const readInstalledState = () => {
|
|
if (!existsSync(installStatePath)) return null;
|
|
try {
|
|
return JSON.parse(readFileSync(installStatePath, 'utf8'));
|
|
} catch {
|
|
return null;
|
|
}
|
|
};
|
|
|
|
const writeInstalledState = (state) => {
|
|
writeFileSync(installStatePath, `${JSON.stringify(state, null, 2)}\n`, 'utf8');
|
|
};
|
|
|
|
const packageInputsAreOlderThanNpmLock = () => {
|
|
if (!existsSync(npmHiddenLockPath)) return false;
|
|
const markerTime = statSync(npmHiddenLockPath).mtimeMs;
|
|
return [packageJsonPath, packageLockPath]
|
|
.filter(existsSync)
|
|
.every((filePath) => statSync(filePath).mtimeMs <= markerTime);
|
|
};
|
|
|
|
const runNpm = (subcommand) => {
|
|
const args = [subcommand, ...commonArgs];
|
|
if (isCI) {
|
|
console.log(
|
|
`[gonavi-frontend-install] cwd=${process.cwd()} frontend=${frontendDir} node=${process.version} platform=${process.platform}/${process.arch} command=${npmCommand} ${args.join(' ')}`,
|
|
);
|
|
}
|
|
|
|
const result = spawnSync(npmCommand, args, {
|
|
cwd: frontendDir,
|
|
env: process.env,
|
|
stdio: 'inherit',
|
|
shell: process.platform === 'win32',
|
|
});
|
|
if (result.error) {
|
|
fail(`failed to start npm: ${result.error.message}`);
|
|
}
|
|
if (result.signal) {
|
|
fail(`npm was terminated by signal ${result.signal}`);
|
|
}
|
|
if (result.status !== 0) {
|
|
console.error(`[gonavi-frontend-install] npm exited with status ${result.status ?? 'unknown'}`);
|
|
exitWithStatus(result.status);
|
|
}
|
|
};
|
|
|
|
if (!existsSync(packageJsonPath)) {
|
|
fail(`package.json not found at ${packageJsonPath}; cwd=${process.cwd()}`);
|
|
}
|
|
|
|
const state = currentState();
|
|
const installedState = readInstalledState();
|
|
const forceInstall = process.env.GONAVI_FORCE_FRONTEND_INSTALL === '1';
|
|
|
|
if (!forceInstall && existsSync(nodeModulesPath)) {
|
|
if (
|
|
installedState?.packageJson === state.packageJson &&
|
|
installedState?.packageLock === state.packageLock
|
|
) {
|
|
console.log('Frontend dependencies are up to date; skipping npm install.');
|
|
process.exit(0);
|
|
}
|
|
|
|
if (!installedState && isCI && existsSync(npmHiddenLockPath)) {
|
|
writeInstalledState(state);
|
|
console.log('Frontend dependencies are up to date from CI cache; recorded install state.');
|
|
process.exit(0);
|
|
}
|
|
|
|
if (!installedState && packageInputsAreOlderThanNpmLock()) {
|
|
writeInstalledState(state);
|
|
console.log('Frontend dependencies are up to date; recorded install state.');
|
|
process.exit(0);
|
|
}
|
|
}
|
|
|
|
runNpm(isCI ? 'ci' : 'install');
|
|
writeInstalledState(state);
|