mirror of
https://github.com/nexu-io/open-design.git
synced 2026-06-01 03:14:35 +07:00
* Prebundle standalone web packaged runtime * Harden mac standalone prebundle policy * Prebundle mac daemon packaged runtime * Prune mac Electron locales * Maximize mac release artifact compression * Publish beta mac artifacts to R2 * Use remote R2 uploads for beta releases * Fail fast on beta R2 access issues * Use S3-compatible uploads for beta R2 releases * Decouple beta versioning from GitHub releases * Remove legacy beta metadata source * Address release beta review notes
852 lines
30 KiB
JavaScript
852 lines
30 KiB
JavaScript
const { access, cp, lstat, mkdir, readFile, readlink, readdir, realpath, rm, stat, symlink, writeFile } = require("node:fs/promises");
|
|
const { createRequire } = require("node:module");
|
|
const path = require("node:path");
|
|
|
|
const CONFIG_ENV = "OD_TOOLS_PACK_WEB_STANDALONE_HOOK_CONFIG";
|
|
const STANDALONE_RESOURCE_NAME = "open-design-web-standalone";
|
|
const REQUIRED_MODULES = ["next/package.json", "react/package.json", "react-dom/package.json", "styled-jsx/package.json"];
|
|
|
|
function isRecord(value) {
|
|
return typeof value === "object" && value !== null && !Array.isArray(value);
|
|
}
|
|
|
|
function requireString(record, key) {
|
|
const value = record[key];
|
|
if (typeof value !== "string" || value.length === 0) {
|
|
throw new Error(`[tools-pack web-standalone] config.${key} must be a non-empty string`);
|
|
}
|
|
return value;
|
|
}
|
|
|
|
function requireBoolean(record, key) {
|
|
const value = record[key];
|
|
if (typeof value !== "boolean") {
|
|
throw new Error(`[tools-pack web-standalone] config.${key} must be a boolean`);
|
|
}
|
|
return value;
|
|
}
|
|
|
|
function requireAbsolutePath(record, key) {
|
|
const value = requireString(record, key);
|
|
if (!path.isAbsolute(value)) {
|
|
throw new Error(`[tools-pack web-standalone] config.${key} must be absolute: ${value}`);
|
|
}
|
|
return path.resolve(value);
|
|
}
|
|
|
|
function isWithin(parent, child) {
|
|
const relative = path.relative(parent, child);
|
|
return relative.length === 0 || (!relative.startsWith("..") && !path.isAbsolute(relative));
|
|
}
|
|
|
|
async function isWithinPhysicalPath(parent, child) {
|
|
const [realParent, realChild] = await Promise.all([realpath(parent), realpath(child)]);
|
|
return isWithin(realParent, realChild);
|
|
}
|
|
|
|
async function pathExists(filePath) {
|
|
try {
|
|
await access(filePath);
|
|
return true;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
async function pathLstatExists(filePath) {
|
|
try {
|
|
await lstat(filePath);
|
|
return true;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
async function readHookConfig() {
|
|
const configPath = process.env[CONFIG_ENV];
|
|
if (configPath == null || configPath.length === 0) {
|
|
throw new Error(`[tools-pack web-standalone] missing ${CONFIG_ENV}`);
|
|
}
|
|
if (!path.isAbsolute(configPath)) {
|
|
throw new Error(`[tools-pack web-standalone] ${CONFIG_ENV} must be absolute: ${configPath}`);
|
|
}
|
|
|
|
const raw = JSON.parse(await readFile(configPath, "utf8"));
|
|
if (!isRecord(raw) || raw.version !== 1) {
|
|
throw new Error("[tools-pack web-standalone] hook config must be an object with version=1");
|
|
}
|
|
|
|
const workspaceRoot = requireAbsolutePath(raw, "workspaceRoot");
|
|
const standaloneSourceRoot = requireAbsolutePath(raw, "standaloneSourceRoot");
|
|
const webStaticSourceRoot = requireAbsolutePath(raw, "webStaticSourceRoot");
|
|
const webPublicSourceRoot = requireAbsolutePath(raw, "webPublicSourceRoot");
|
|
const auditReportPath = requireAbsolutePath(raw, "auditReportPath");
|
|
const resourceName = requireString(raw, "resourceName");
|
|
if (resourceName !== STANDALONE_RESOURCE_NAME) {
|
|
throw new Error(`[tools-pack web-standalone] unsupported resourceName: ${resourceName}`);
|
|
}
|
|
|
|
for (const [key, value] of Object.entries({ standaloneSourceRoot, webStaticSourceRoot, webPublicSourceRoot })) {
|
|
if (!isWithin(workspaceRoot, value)) {
|
|
throw new Error(`[tools-pack web-standalone] config.${key} must stay under workspaceRoot: ${value}`);
|
|
}
|
|
}
|
|
|
|
return {
|
|
auditReportPath,
|
|
pruneCopiedSharp: requireBoolean(raw, "pruneCopiedSharp"),
|
|
pruneRootNext: requireBoolean(raw, "pruneRootNext"),
|
|
pruneRootSharp: requireBoolean(raw, "pruneRootSharp"),
|
|
resourceName,
|
|
standaloneSourceRoot,
|
|
webPublicSourceRoot,
|
|
webStaticSourceRoot,
|
|
workspaceRoot,
|
|
};
|
|
}
|
|
|
|
function resolveAppPath(context) {
|
|
if (context == null || typeof context.appOutDir !== "string" || context.appOutDir.length === 0) {
|
|
throw new Error("[tools-pack web-standalone] electron-builder context.appOutDir is missing");
|
|
}
|
|
const productFilename = context.packager?.appInfo?.productFilename;
|
|
if (typeof productFilename !== "string" || productFilename.length === 0) {
|
|
throw new Error("[tools-pack web-standalone] electron-builder productFilename is missing");
|
|
}
|
|
if (context.electronPlatformName === "win32") return context.appOutDir;
|
|
return path.join(context.appOutDir, `${productFilename}.app`);
|
|
}
|
|
|
|
function resolveResourcesRoot(context, appPath) {
|
|
switch (context?.electronPlatformName) {
|
|
case "darwin":
|
|
return path.join(appPath, "Contents", "Resources");
|
|
case "win32":
|
|
return path.join(context.appOutDir, "resources");
|
|
default:
|
|
throw new Error(`[tools-pack web-standalone] unsupported platform: ${context?.electronPlatformName ?? "unknown"}`);
|
|
}
|
|
}
|
|
|
|
function resolveRootAppNodeModulesRoot(resourcesRoot) {
|
|
return path.join(resourcesRoot, "app", "node_modules");
|
|
}
|
|
|
|
async function sizePathBytes(filePath) {
|
|
let metadata;
|
|
try {
|
|
metadata = await lstat(filePath);
|
|
} catch {
|
|
return 0;
|
|
}
|
|
|
|
if (!metadata.isDirectory()) return metadata.size;
|
|
|
|
const entries = await readdir(filePath, { withFileTypes: true }).catch(() => []);
|
|
let total = 0;
|
|
for (const entry of entries) {
|
|
total += await sizePathBytes(path.join(filePath, entry.name));
|
|
}
|
|
return total;
|
|
}
|
|
|
|
async function copyRequired(sourcePath, destinationPath, options = {}) {
|
|
if (!(await pathExists(sourcePath))) {
|
|
throw new Error(`[tools-pack web-standalone] required source missing: ${sourcePath}`);
|
|
}
|
|
await rm(destinationPath, { force: true, recursive: true });
|
|
await mkdir(path.dirname(destinationPath), { recursive: true });
|
|
await cp(sourcePath, destinationPath, {
|
|
dereference: options.dereference === true,
|
|
recursive: true,
|
|
verbatimSymlinks: options.dereference === true ? false : true,
|
|
});
|
|
}
|
|
|
|
async function copyOptional(sourcePath, destinationPath, options = {}) {
|
|
if (!(await pathExists(sourcePath))) return false;
|
|
await copyRequired(sourcePath, destinationPath, options);
|
|
return true;
|
|
}
|
|
|
|
async function linkRelative(sourcePath, destinationPath) {
|
|
if (!(await pathExists(sourcePath))) return false;
|
|
if (await pathLstatExists(destinationPath)) return false;
|
|
await mkdir(path.dirname(destinationPath), { recursive: true });
|
|
const relativeTarget = path.relative(path.dirname(destinationPath), sourcePath);
|
|
await symlink(relativeTarget.length === 0 ? "." : relativeTarget, destinationPath);
|
|
return true;
|
|
}
|
|
|
|
async function linkPnpmPublicHoist(destinationRoot) {
|
|
const nodeModulesRoot = path.join(destinationRoot, "node_modules");
|
|
const hoistRoot = path.join(nodeModulesRoot, ".pnpm", "node_modules");
|
|
const entries = await readdir(hoistRoot, { withFileTypes: true }).catch(() => []);
|
|
const linked = [];
|
|
|
|
for (const entry of entries) {
|
|
const sourcePath = path.join(hoistRoot, entry.name);
|
|
if (entry.name.startsWith("@") && entry.isDirectory()) {
|
|
const scopedEntries = await readdir(sourcePath).catch(() => []);
|
|
for (const scopedEntry of scopedEntries) {
|
|
const scopedSource = path.join(sourcePath, scopedEntry);
|
|
const scopedDestination = path.join(nodeModulesRoot, entry.name, scopedEntry);
|
|
if (await linkRelative(scopedSource, scopedDestination)) linked.push(scopedDestination);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
const destinationPath = path.join(nodeModulesRoot, entry.name);
|
|
if (await linkRelative(sourcePath, destinationPath)) linked.push(destinationPath);
|
|
}
|
|
|
|
return linked;
|
|
}
|
|
|
|
async function resolveStandaloneSourceWebRoot(standaloneSourceRoot) {
|
|
const candidates = [
|
|
path.join(standaloneSourceRoot, "apps", "web"),
|
|
standaloneSourceRoot,
|
|
];
|
|
|
|
for (const candidate of candidates) {
|
|
if (await pathExists(path.join(candidate, "server.js"))) return candidate;
|
|
}
|
|
|
|
throw new Error(`[tools-pack web-standalone] standalone server.js not found under ${standaloneSourceRoot}`);
|
|
}
|
|
|
|
async function installStandaloneResource(config, resourcesRoot, platformName) {
|
|
const sourceWebRoot = await resolveStandaloneSourceWebRoot(config.standaloneSourceRoot);
|
|
const destinationRoot = path.join(resourcesRoot, config.resourceName);
|
|
const destinationWebRoot = path.join(destinationRoot, "apps", "web");
|
|
const copyOptions = { dereference: platformName === "win32" };
|
|
|
|
await rm(destinationRoot, { force: true, recursive: true });
|
|
await mkdir(destinationWebRoot, { recursive: true });
|
|
|
|
await copyRequired(path.join(config.standaloneSourceRoot, "node_modules"), path.join(destinationRoot, "node_modules"), copyOptions);
|
|
await copyRequired(path.join(sourceWebRoot, "server.js"), path.join(destinationWebRoot, "server.js"));
|
|
await copyOptional(path.join(sourceWebRoot, "package.json"), path.join(destinationWebRoot, "package.json"));
|
|
const copiedNestedNodeModules = await copyOptional(path.join(sourceWebRoot, "node_modules"), path.join(destinationWebRoot, "node_modules"), copyOptions);
|
|
const linkedHoistEntries = await linkPnpmPublicHoist(destinationRoot);
|
|
await copyRequired(path.join(sourceWebRoot, ".next"), path.join(destinationWebRoot, ".next"));
|
|
const copiedStatic = await copyOptional(config.webStaticSourceRoot, path.join(destinationWebRoot, ".next", "static"));
|
|
const copiedPublic = await copyOptional(config.webPublicSourceRoot, path.join(destinationWebRoot, "public"));
|
|
const rewrittenSymlinks = platformName === "win32"
|
|
? []
|
|
: await rewriteCopiedStandaloneSymlinks({
|
|
destinationRoot,
|
|
destinationWebRoot,
|
|
sourceWebRoot,
|
|
standaloneSourceRoot: config.standaloneSourceRoot,
|
|
});
|
|
|
|
return {
|
|
copiedNestedNodeModules,
|
|
copiedPublic,
|
|
copiedStatic,
|
|
destinationRoot,
|
|
destinationWebRoot,
|
|
linkedHoistEntries,
|
|
rewrittenSymlinks,
|
|
sourceWebRoot,
|
|
};
|
|
}
|
|
|
|
async function rewriteCopiedStandaloneSymlinks(options) {
|
|
const mappings = [
|
|
{
|
|
destinationRoot: path.join(options.destinationWebRoot, "node_modules"),
|
|
sourceRoot: path.join(options.sourceWebRoot, "node_modules"),
|
|
},
|
|
{
|
|
destinationRoot: path.join(options.destinationRoot, "node_modules"),
|
|
sourceRoot: path.join(options.standaloneSourceRoot, "node_modules"),
|
|
},
|
|
{
|
|
destinationRoot: options.destinationWebRoot,
|
|
sourceRoot: options.sourceWebRoot,
|
|
},
|
|
{
|
|
destinationRoot: options.destinationRoot,
|
|
sourceRoot: options.standaloneSourceRoot,
|
|
},
|
|
];
|
|
const rewrittenSymlinks = [];
|
|
|
|
function mapPath(pathToMap, fromKey, toKey) {
|
|
for (const mapping of mappings) {
|
|
if (!isWithin(mapping[fromKey], pathToMap)) continue;
|
|
return path.join(mapping[toKey], path.relative(mapping[fromKey], pathToMap));
|
|
}
|
|
return null;
|
|
}
|
|
|
|
async function visit(current) {
|
|
let metadata;
|
|
try {
|
|
metadata = await lstat(current);
|
|
} catch {
|
|
return;
|
|
}
|
|
|
|
if (metadata.isSymbolicLink()) {
|
|
const copiedSourcePath = mapPath(current, "destinationRoot", "sourceRoot");
|
|
if (copiedSourcePath == null) return;
|
|
|
|
const currentTarget = await readlink(current);
|
|
const sourceTarget = path.resolve(path.dirname(copiedSourcePath), currentTarget);
|
|
const destinationTarget = mapPath(sourceTarget, "sourceRoot", "destinationRoot");
|
|
// External source-tree symlinks are not rewritten; the closure audit below
|
|
// reports them as externalSymlink and fails the package instead.
|
|
if (destinationTarget == null) return;
|
|
|
|
const nextTarget = path.relative(path.dirname(current), destinationTarget) || ".";
|
|
if (nextTarget === currentTarget) return;
|
|
|
|
await rm(current, { force: true, recursive: true });
|
|
await symlink(nextTarget, current);
|
|
rewrittenSymlinks.push({
|
|
path: current,
|
|
target: nextTarget,
|
|
});
|
|
return;
|
|
}
|
|
|
|
if (!metadata.isDirectory()) return;
|
|
|
|
const entries = await readdir(current, { withFileTypes: true }).catch(() => []);
|
|
for (const entry of entries) {
|
|
await visit(path.join(current, entry.name));
|
|
}
|
|
}
|
|
|
|
await visit(options.destinationRoot);
|
|
return rewrittenSymlinks;
|
|
}
|
|
|
|
async function removePathAndRecord(targetPath, reason, removedPaths) {
|
|
const existed = await pathExists(targetPath);
|
|
const bytes = await sizePathBytes(targetPath);
|
|
await rm(targetPath, { force: true, recursive: true });
|
|
if (existed || bytes > 0) {
|
|
removedPaths.push({ bytes, path: targetPath, reason });
|
|
}
|
|
}
|
|
|
|
function isPrunablePnpmSharpEntry(name) {
|
|
return name.startsWith("sharp@") || name.startsWith("@img+colour@") || name.startsWith("@img+sharp-");
|
|
}
|
|
|
|
function isPrunableImgEntry(name) {
|
|
return name === "colour" || name.startsWith("sharp-");
|
|
}
|
|
|
|
async function pruneImgScope(scopePath, reason, removedPaths) {
|
|
const entries = await readdir(scopePath).catch(() => []);
|
|
for (const entry of entries) {
|
|
if (isPrunableImgEntry(entry)) {
|
|
await removePathAndRecord(path.join(scopePath, entry), reason, removedPaths);
|
|
}
|
|
}
|
|
}
|
|
|
|
async function pruneCopiedSharp(destinationRoot) {
|
|
const nodeModulesRoot = path.join(destinationRoot, "node_modules");
|
|
const pnpmRoot = path.join(nodeModulesRoot, ".pnpm");
|
|
const removedPaths = [];
|
|
|
|
await removePathAndRecord(path.join(nodeModulesRoot, "sharp"), "copied top-level sharp symlink", removedPaths);
|
|
await pruneImgScope(path.join(nodeModulesRoot, "@img"), "copied top-level @img sharp symlink", removedPaths);
|
|
await removePathAndRecord(path.join(pnpmRoot, "node_modules", "sharp"), "copied pnpm sharp symlink", removedPaths);
|
|
await pruneImgScope(path.join(pnpmRoot, "node_modules", "@img"), "copied pnpm @img sharp symlink", removedPaths);
|
|
|
|
const pnpmEntries = await readdir(pnpmRoot).catch(() => []);
|
|
for (const entry of pnpmEntries) {
|
|
if (isPrunablePnpmSharpEntry(entry)) {
|
|
await removePathAndRecord(path.join(pnpmRoot, entry), "copied pnpm sharp package", removedPaths);
|
|
continue;
|
|
}
|
|
|
|
if (entry.startsWith("next@")) {
|
|
await removePathAndRecord(path.join(pnpmRoot, entry, "node_modules", "sharp"), "copied next sharp symlink", removedPaths);
|
|
}
|
|
}
|
|
|
|
return removedPaths;
|
|
}
|
|
|
|
async function dedupeCopiedStandaloneNext(destinationRoot, destinationWebRoot, platformName) {
|
|
if (platformName !== "win32") return null;
|
|
|
|
const nodeModulesRoot = path.join(destinationRoot, "node_modules");
|
|
const rootNextRoot = path.join(nodeModulesRoot, "next");
|
|
const pnpmHoistedNextRoot = path.join(nodeModulesRoot, ".pnpm", "node_modules", "next");
|
|
const webNextRoot = path.join(destinationWebRoot, "node_modules", "next");
|
|
const removedPaths = [];
|
|
|
|
if (!(await pathExists(path.join(webNextRoot, "package.json")))) {
|
|
throw new Error(`[tools-pack web-standalone] copied standalone app-local Next package missing: ${webNextRoot}`);
|
|
}
|
|
|
|
await removePathAndRecord(
|
|
rootNextRoot,
|
|
"copied standalone root next public-hoist duplicate",
|
|
removedPaths,
|
|
);
|
|
await removePathAndRecord(
|
|
pnpmHoistedNextRoot,
|
|
"copied standalone pnpm-hoisted next duplicate superseded by app-local next",
|
|
removedPaths,
|
|
);
|
|
|
|
return {
|
|
removedPaths,
|
|
retainedPath: webNextRoot,
|
|
};
|
|
}
|
|
|
|
async function pruneBrokenSymlinks(root, current = root, removedPaths = [], reason = "broken symlink") {
|
|
let metadata;
|
|
try {
|
|
metadata = await lstat(current);
|
|
} catch {
|
|
return removedPaths;
|
|
}
|
|
|
|
if (metadata.isSymbolicLink()) {
|
|
try {
|
|
await stat(current);
|
|
} catch {
|
|
await removePathAndRecord(current, reason, removedPaths);
|
|
}
|
|
return removedPaths;
|
|
}
|
|
|
|
if (!metadata.isDirectory()) return removedPaths;
|
|
|
|
const entries = await readdir(current, { withFileTypes: true }).catch(() => []);
|
|
for (const entry of entries) {
|
|
await pruneBrokenSymlinks(root, path.join(current, entry.name), removedPaths, reason);
|
|
}
|
|
return removedPaths;
|
|
}
|
|
|
|
function isSourceBuildResidue(relativePath) {
|
|
const normalized = relativePath.split(path.sep).join("/");
|
|
return normalized.endsWith(".map") || normalized.endsWith(".tsbuildinfo");
|
|
}
|
|
|
|
async function pruneMatchingFilesSummary(root, includeRelativePath, reason) {
|
|
const summary = {
|
|
bytes: 0,
|
|
count: 0,
|
|
reason,
|
|
root,
|
|
};
|
|
|
|
async function visit(current) {
|
|
let metadata;
|
|
try {
|
|
metadata = await lstat(current);
|
|
} catch {
|
|
return;
|
|
}
|
|
|
|
if (metadata.isDirectory()) {
|
|
const entries = await readdir(current, { withFileTypes: true }).catch(() => []);
|
|
for (const entry of entries) {
|
|
await visit(path.join(current, entry.name));
|
|
}
|
|
return;
|
|
}
|
|
|
|
const relativePath = path.relative(root, current);
|
|
if (relativePath.length === 0 || !includeRelativePath(relativePath)) return;
|
|
summary.bytes += metadata.size;
|
|
summary.count += 1;
|
|
await rm(current, { force: true });
|
|
}
|
|
|
|
await visit(root);
|
|
return summary.count > 0 ? [summary] : [];
|
|
}
|
|
|
|
async function pruneSourceBuildResidue(root, reason) {
|
|
return await pruneMatchingFilesSummary(root, isSourceBuildResidue, reason);
|
|
}
|
|
|
|
function isForbiddenCopiedEntry(relativePath, platformName) {
|
|
const normalized = relativePath.split(path.sep).join("/");
|
|
const withRootSlash = `/${normalized}`;
|
|
const forbiddenSwc = platformName === "win32"
|
|
? withRootSlash.includes("swc-darwin") || withRootSlash.includes("swc-linux")
|
|
: withRootSlash.includes("swc-darwin");
|
|
return (
|
|
withRootSlash.includes("/node_modules/.pnpm/sharp@") ||
|
|
withRootSlash.includes("/node_modules/.pnpm/@img+colour@") ||
|
|
withRootSlash.includes("/node_modules/.pnpm/@img+sharp-") ||
|
|
withRootSlash.includes("/node_modules/sharp") ||
|
|
withRootSlash.includes("/node_modules/@img/colour") ||
|
|
withRootSlash.includes("/node_modules/@img/sharp-") ||
|
|
withRootSlash.includes("sharp-libvips") ||
|
|
forbiddenSwc
|
|
);
|
|
}
|
|
|
|
async function collectClosureStats(
|
|
root,
|
|
current = root,
|
|
stats = { brokenSymlinks: [], externalSymlinks: [], forbiddenEntries: [], symlinks: 0 },
|
|
platformName = process.platform,
|
|
) {
|
|
let metadata;
|
|
try {
|
|
metadata = await lstat(current);
|
|
} catch {
|
|
return stats;
|
|
}
|
|
|
|
const relativePath = path.relative(root, current);
|
|
if (relativePath.length > 0 && isForbiddenCopiedEntry(relativePath, platformName)) {
|
|
stats.forbiddenEntries.push(relativePath.split(path.sep).join("/"));
|
|
}
|
|
|
|
if (metadata.isSymbolicLink()) {
|
|
stats.symlinks += 1;
|
|
try {
|
|
await stat(current);
|
|
if (!(await isWithinPhysicalPath(root, current))) {
|
|
stats.externalSymlinks.push(relativePath.split(path.sep).join("/"));
|
|
}
|
|
} catch {
|
|
stats.brokenSymlinks.push(relativePath.split(path.sep).join("/"));
|
|
}
|
|
return stats;
|
|
}
|
|
|
|
if (!metadata.isDirectory()) return stats;
|
|
|
|
const entries = await readdir(current, { withFileTypes: true }).catch(() => []);
|
|
for (const entry of entries) {
|
|
await collectClosureStats(root, path.join(current, entry.name), stats, platformName);
|
|
}
|
|
return stats;
|
|
}
|
|
|
|
async function assertResolvedInside(root, moduleName, resolvedPath) {
|
|
if (!(await isWithinPhysicalPath(root, resolvedPath))) {
|
|
throw new Error(`[tools-pack web-standalone] ${moduleName} resolved outside copied standalone: ${resolvedPath}`);
|
|
}
|
|
}
|
|
|
|
async function auditCopiedStandalone(config, installResult, platformName) {
|
|
const serverPath = path.join(installResult.destinationWebRoot, "server.js");
|
|
const staticRoot = path.join(installResult.destinationWebRoot, ".next", "static");
|
|
const publicRoot = path.join(installResult.destinationWebRoot, "public");
|
|
const nodeModulesRoot = path.join(installResult.destinationRoot, "node_modules");
|
|
const webNodeModulesRoot = path.join(installResult.destinationWebRoot, "node_modules");
|
|
const requiredPaths = [serverPath, staticRoot, nodeModulesRoot];
|
|
if (await pathExists(config.webPublicSourceRoot)) requiredPaths.push(publicRoot);
|
|
|
|
for (const requiredPath of requiredPaths) {
|
|
if (!(await pathExists(requiredPath))) {
|
|
throw new Error(`[tools-pack web-standalone] copied standalone audit missing: ${requiredPath}`);
|
|
}
|
|
}
|
|
|
|
const localRequire = createRequire(serverPath);
|
|
const resolvedModules = {};
|
|
for (const moduleName of REQUIRED_MODULES) {
|
|
const resolvedPath = localRequire.resolve(moduleName);
|
|
await assertResolvedInside(installResult.destinationRoot, moduleName, resolvedPath);
|
|
resolvedModules[moduleName] = resolvedPath;
|
|
}
|
|
|
|
const closureStats = await collectClosureStats(installResult.destinationRoot, installResult.destinationRoot, undefined, platformName);
|
|
if (closureStats.brokenSymlinks.length > 0) {
|
|
throw new Error(`[tools-pack web-standalone] copied standalone has broken symlinks: ${closureStats.brokenSymlinks.join(", ")}`);
|
|
}
|
|
if (closureStats.externalSymlinks.length > 0) {
|
|
throw new Error(`[tools-pack web-standalone] copied standalone has external symlinks: ${closureStats.externalSymlinks.join(", ")}`);
|
|
}
|
|
if (closureStats.forbiddenEntries.length > 0) {
|
|
throw new Error(`[tools-pack web-standalone] copied standalone has forbidden entries: ${closureStats.forbiddenEntries.join(", ")}`);
|
|
}
|
|
|
|
return {
|
|
brokenSymlinks: closureStats.brokenSymlinks,
|
|
bytes: await sizePathBytes(installResult.destinationRoot),
|
|
destinationRoot: installResult.destinationRoot,
|
|
destinationWebRoot: installResult.destinationWebRoot,
|
|
externalSymlinks: closureStats.externalSymlinks,
|
|
forbiddenEntries: closureStats.forbiddenEntries,
|
|
nodeModulesBytes: await sizePathBytes(nodeModulesRoot),
|
|
resolvedModules,
|
|
serverPath,
|
|
symlinks: closureStats.symlinks,
|
|
webNextBytes: await sizePathBytes(path.join(webNodeModulesRoot, "next")),
|
|
webNodeModulesBytes: await sizePathBytes(webNodeModulesRoot),
|
|
};
|
|
}
|
|
|
|
async function auditCopiedStandaloneNextDedupe(installResult, platformName) {
|
|
if (platformName !== "win32") return null;
|
|
|
|
const serverPath = path.join(installResult.destinationWebRoot, "server.js");
|
|
const retainedNextRoot = path.join(installResult.destinationWebRoot, "node_modules", "next");
|
|
const retainedNextPackagePath = path.join(retainedNextRoot, "package.json");
|
|
const checkedPaths = [
|
|
path.join(installResult.destinationRoot, "node_modules", "next"),
|
|
path.join(installResult.destinationRoot, "node_modules", ".pnpm", "node_modules", "next"),
|
|
];
|
|
const remainingPaths = [];
|
|
|
|
for (const checkedPath of checkedPaths) {
|
|
if (await pathExists(checkedPath)) remainingPaths.push(checkedPath);
|
|
}
|
|
if (remainingPaths.length > 0) {
|
|
throw new Error(
|
|
`[tools-pack web-standalone] copied standalone next dedupe audit found remaining duplicate paths: ${remainingPaths.join(", ")}`,
|
|
);
|
|
}
|
|
|
|
if (!(await pathExists(retainedNextPackagePath))) {
|
|
throw new Error(`[tools-pack web-standalone] copied standalone retained app-local next missing: ${retainedNextPackagePath}`);
|
|
}
|
|
|
|
const resolvedNextPackagePath = createRequire(serverPath).resolve("next/package.json");
|
|
if (!(await isWithinPhysicalPath(retainedNextRoot, resolvedNextPackagePath))) {
|
|
throw new Error(
|
|
`[tools-pack web-standalone] copied standalone next resolved outside retained app-local next: ${resolvedNextPackagePath}`,
|
|
);
|
|
}
|
|
|
|
return {
|
|
checkedPaths,
|
|
remainingPaths,
|
|
resolvedNextPackagePath,
|
|
retainedNextRoot,
|
|
};
|
|
}
|
|
|
|
async function pruneRootNext(appNodeModulesRoot, platformName) {
|
|
const removedPaths = [];
|
|
|
|
if (platformName === "win32") {
|
|
await removePathAndRecord(
|
|
path.join(appNodeModulesRoot, "next"),
|
|
"root next package superseded by copied standalone resource",
|
|
removedPaths,
|
|
);
|
|
await removePathAndRecord(
|
|
path.join(appNodeModulesRoot, "@next"),
|
|
"root @next package scope superseded by copied standalone resource",
|
|
removedPaths,
|
|
);
|
|
} else {
|
|
const nextScopeRoot = path.join(appNodeModulesRoot, "@next");
|
|
const nextScopeEntries = await readdir(nextScopeRoot).catch(() => []);
|
|
for (const entry of nextScopeEntries) {
|
|
if (platformName === "darwin" && entry.startsWith("swc-darwin-")) {
|
|
await removePathAndRecord(path.join(nextScopeRoot, entry), "root next darwin swc package", removedPaths);
|
|
}
|
|
}
|
|
}
|
|
|
|
await removePathAndRecord(
|
|
path.join(appNodeModulesRoot, "@open-design", "web", ".next", "standalone"),
|
|
"root @open-design/web standalone output",
|
|
removedPaths,
|
|
);
|
|
|
|
return removedPaths;
|
|
}
|
|
|
|
async function auditRootNextPruned(appNodeModulesRoot, platformName, enabled) {
|
|
if (platformName !== "win32" || !enabled) return null;
|
|
|
|
const checkedPaths = [
|
|
path.join(appNodeModulesRoot, "next"),
|
|
path.join(appNodeModulesRoot, "@next"),
|
|
];
|
|
const remainingPaths = [];
|
|
for (const checkedPath of checkedPaths) {
|
|
if (await pathExists(checkedPath)) remainingPaths.push(checkedPath);
|
|
}
|
|
if (remainingPaths.length > 0) {
|
|
throw new Error(`[tools-pack web-standalone] root next pruning audit found remaining paths: ${remainingPaths.join(", ")}`);
|
|
}
|
|
|
|
return {
|
|
checkedPaths,
|
|
remainingPaths,
|
|
};
|
|
}
|
|
|
|
async function pruneRootSharp(appNodeModulesRoot) {
|
|
const pnpmRoot = path.join(appNodeModulesRoot, ".pnpm");
|
|
const removedPaths = [];
|
|
|
|
await removePathAndRecord(path.join(appNodeModulesRoot, "sharp"), "root sharp package", removedPaths);
|
|
await pruneImgScope(path.join(appNodeModulesRoot, "@img"), "root @img sharp package", removedPaths);
|
|
await removePathAndRecord(path.join(pnpmRoot, "node_modules", "sharp"), "root pnpm sharp symlink", removedPaths);
|
|
await pruneImgScope(path.join(pnpmRoot, "node_modules", "@img"), "root pnpm @img sharp symlink", removedPaths);
|
|
|
|
const pnpmEntries = await readdir(pnpmRoot).catch(() => []);
|
|
for (const entry of pnpmEntries) {
|
|
if (isPrunablePnpmSharpEntry(entry)) {
|
|
await removePathAndRecord(path.join(pnpmRoot, entry), "root pnpm sharp package", removedPaths);
|
|
}
|
|
}
|
|
|
|
return removedPaths;
|
|
}
|
|
|
|
async function pruneRootWebPackage(appNodeModulesRoot, platformName) {
|
|
if (platformName !== "win32") return [];
|
|
|
|
const webPackageRoot = path.join(appNodeModulesRoot, "@open-design", "web");
|
|
const removedPaths = [];
|
|
for (const entry of [".next", "app", "next.config.ts", "public", "src"]) {
|
|
await removePathAndRecord(
|
|
path.join(webPackageRoot, entry),
|
|
"root @open-design/web standalone-safe package residue",
|
|
removedPaths,
|
|
);
|
|
}
|
|
return removedPaths;
|
|
}
|
|
|
|
async function auditRootWebPackage(appNodeModulesRoot) {
|
|
const webPackageRoot = path.join(appNodeModulesRoot, "@open-design", "web");
|
|
const packageJsonPath = path.join(webPackageRoot, "package.json");
|
|
const sidecarEntryPath = path.join(webPackageRoot, "dist", "sidecar", "index.js");
|
|
for (const requiredPath of [packageJsonPath, sidecarEntryPath]) {
|
|
if (!(await pathExists(requiredPath))) {
|
|
throw new Error(`[tools-pack web-standalone] root @open-design/web audit missing: ${requiredPath}`);
|
|
}
|
|
}
|
|
return {
|
|
bytes: await sizePathBytes(webPackageRoot),
|
|
packageJsonPath,
|
|
retainedDistBytes: await sizePathBytes(path.join(webPackageRoot, "dist")),
|
|
sidecarEntryPath,
|
|
webPackageRoot,
|
|
};
|
|
}
|
|
|
|
async function auditNoBrokenSymlinks(root, label) {
|
|
const stats = await collectClosureStats(root);
|
|
if (stats.brokenSymlinks.length > 0) {
|
|
throw new Error(`[tools-pack web-standalone] ${label} has broken symlinks: ${stats.brokenSymlinks.join(", ")}`);
|
|
}
|
|
return {
|
|
brokenSymlinks: stats.brokenSymlinks,
|
|
symlinks: stats.symlinks,
|
|
};
|
|
}
|
|
|
|
async function runWebStandaloneAfterPack(context) {
|
|
if (context?.electronPlatformName != null && context.electronPlatformName !== "darwin" && context.electronPlatformName !== "win32") return;
|
|
|
|
const config = await readHookConfig();
|
|
const appPath = resolveAppPath(context);
|
|
const resourcesRoot = resolveResourcesRoot(context, appPath);
|
|
if (context.electronPlatformName === "darwin" && !(await pathExists(appPath))) {
|
|
throw new Error(`[tools-pack web-standalone] app bundle not found: ${appPath}`);
|
|
}
|
|
if (!(await pathExists(resourcesRoot))) {
|
|
throw new Error(`[tools-pack web-standalone] resources root not found: ${resourcesRoot}`);
|
|
}
|
|
|
|
const appNodeModulesRoot = resolveRootAppNodeModulesRoot(resourcesRoot);
|
|
const installResult = await installStandaloneResource(config, resourcesRoot, context.electronPlatformName);
|
|
const copiedPrune = config.pruneCopiedSharp ? await pruneCopiedSharp(installResult.destinationRoot) : [];
|
|
const copiedNextDedupe = await dedupeCopiedStandaloneNext(
|
|
installResult.destinationRoot,
|
|
installResult.destinationWebRoot,
|
|
context.electronPlatformName,
|
|
);
|
|
const copiedBuildResiduePrune = context.electronPlatformName === "win32"
|
|
? await pruneSourceBuildResidue(installResult.destinationRoot, "copied standalone source/build residue")
|
|
: [];
|
|
const brokenSymlinkPrune = await pruneBrokenSymlinks(
|
|
installResult.destinationRoot,
|
|
installResult.destinationRoot,
|
|
[],
|
|
"copied broken symlink",
|
|
);
|
|
const copiedNextDedupeAudit = await auditCopiedStandaloneNextDedupe(
|
|
installResult,
|
|
context.electronPlatformName,
|
|
);
|
|
const copiedAudit = await auditCopiedStandalone(config, installResult, context.electronPlatformName);
|
|
const rootPrune = config.pruneRootNext ? await pruneRootNext(appNodeModulesRoot, context.electronPlatformName) : [];
|
|
const rootSharpPrune = config.pruneRootSharp ? await pruneRootSharp(appNodeModulesRoot) : [];
|
|
const rootWebPackagePrune = await pruneRootWebPackage(appNodeModulesRoot, context.electronPlatformName);
|
|
const rootBuildResiduePrune = context.electronPlatformName === "win32"
|
|
? await pruneSourceBuildResidue(appNodeModulesRoot, "root app source/build residue")
|
|
: [];
|
|
const rootWebPackageAudit = context.electronPlatformName === "win32"
|
|
? await auditRootWebPackage(appNodeModulesRoot)
|
|
: null;
|
|
const rootNextPruneAudit = await auditRootNextPruned(
|
|
appNodeModulesRoot,
|
|
context.electronPlatformName,
|
|
config.pruneRootNext,
|
|
);
|
|
const rootBrokenSymlinkPrune = await pruneBrokenSymlinks(
|
|
appNodeModulesRoot,
|
|
appNodeModulesRoot,
|
|
[],
|
|
"root broken symlink",
|
|
);
|
|
const rootSymlinkAudit = await auditNoBrokenSymlinks(
|
|
appNodeModulesRoot,
|
|
"root app node_modules",
|
|
);
|
|
const report = {
|
|
appPath,
|
|
brokenSymlinkPrune,
|
|
copiedAudit,
|
|
copiedBuildResiduePrune,
|
|
copiedNextDedupe,
|
|
copiedNextDedupeAudit,
|
|
copiedPrune,
|
|
generatedAt: new Date().toISOString(),
|
|
platformName: context.electronPlatformName,
|
|
resourcesRoot,
|
|
rootBuildResiduePrune,
|
|
rootBrokenSymlinkPrune,
|
|
rootNextPruneAudit,
|
|
rootPrune,
|
|
rootSharpPrune,
|
|
rootSymlinkAudit,
|
|
rootWebPackageAudit,
|
|
rootWebPackagePrune,
|
|
sourceWebRoot: installResult.sourceWebRoot,
|
|
version: 1,
|
|
};
|
|
|
|
await mkdir(path.dirname(config.auditReportPath), { recursive: true });
|
|
await writeFile(config.auditReportPath, `${JSON.stringify(report, null, 2)}\n`, "utf8");
|
|
}
|
|
|
|
module.exports = async function webStandaloneAfterPack(context) {
|
|
try {
|
|
await runWebStandaloneAfterPack(context);
|
|
} catch (error) {
|
|
console.error(
|
|
"[tools-pack web-standalone] after-pack hook failed:",
|
|
error instanceof Error ? error.message : error,
|
|
);
|
|
console.error("[tools-pack web-standalone] electron-builder context:", {
|
|
appOutDir: context?.appOutDir,
|
|
electronPlatformName: context?.electronPlatformName,
|
|
productFilename: context?.packager?.appInfo?.productFilename,
|
|
});
|
|
throw error;
|
|
}
|
|
};
|