workflow_launcher/node_modules/app-builder-lib/out/packager.js
2024-03-12 12:57:14 +01:00

481 lines
21 KiB
JavaScript

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.Packager = void 0;
const builder_util_1 = require("builder-util");
const builder_util_runtime_1 = require("builder-util-runtime");
const promise_1 = require("builder-util/out/promise");
const events_1 = require("events");
const fs_extra_1 = require("fs-extra");
const isCI = require("is-ci");
const lazy_val_1 = require("lazy-val");
const path = require("path");
const arch_1 = require("builder-util/out/arch");
const appInfo_1 = require("./appInfo");
const asar_1 = require("./asar/asar");
const core_1 = require("./core");
const ElectronFramework_1 = require("./electron/ElectronFramework");
const LibUiFramework_1 = require("./frameworks/LibUiFramework");
const platformPackager_1 = require("./platformPackager");
const ProtonFramework_1 = require("./ProtonFramework");
const targetFactory_1 = require("./targets/targetFactory");
const config_1 = require("./util/config");
const macroExpander_1 = require("./util/macroExpander");
const packageDependencies_1 = require("./util/packageDependencies");
const packageMetadata_1 = require("./util/packageMetadata");
const repositoryInfo_1 = require("./util/repositoryInfo");
const yarn_1 = require("./util/yarn");
const version_1 = require("./version");
const os_1 = require("os");
function addHandler(emitter, event, handler) {
emitter.on(event, handler);
}
async function createFrameworkInfo(configuration, packager) {
let framework = configuration.framework;
if (framework != null) {
framework = framework.toLowerCase();
}
let nodeVersion = configuration.nodeVersion;
if (framework === "electron" || framework == null) {
return await (0, ElectronFramework_1.createElectronFrameworkSupport)(configuration, packager);
}
if (nodeVersion == null || nodeVersion === "current") {
nodeVersion = process.versions.node;
}
const distMacOsName = `${packager.appInfo.productFilename}.app`;
const isUseLaunchUi = configuration.launchUiVersion !== false;
if (framework === "proton" || framework === "proton-native") {
return new ProtonFramework_1.ProtonFramework(nodeVersion, distMacOsName, isUseLaunchUi);
}
else if (framework === "libui") {
return new LibUiFramework_1.LibUiFramework(nodeVersion, distMacOsName, isUseLaunchUi);
}
else {
throw new builder_util_1.InvalidConfigurationError(`Unknown framework: ${framework}`);
}
}
class Packager {
get appDir() {
return this._appDir;
}
get metadata() {
return this._metadata;
}
get areNodeModulesHandledExternally() {
return this._nodeModulesHandledExternally;
}
get isPrepackedAppAsar() {
return this._isPrepackedAppAsar;
}
get devMetadata() {
return this._devMetadata;
}
get config() {
return this._configuration;
}
get appInfo() {
return this._appInfo;
}
get repositoryInfo() {
return this._repositoryInfo.value;
}
getNodeDependencyInfo(platform) {
let key = "";
let excludedDependencies = null;
if (platform != null && this.framework.getExcludedDependencies != null) {
excludedDependencies = this.framework.getExcludedDependencies(platform);
if (excludedDependencies != null) {
key += `-${platform.name}`;
}
}
let result = this.nodeDependencyInfo.get(key);
if (result == null) {
result = (0, packageDependencies_1.createLazyProductionDeps)(this.appDir, excludedDependencies);
this.nodeDependencyInfo.set(key, result);
}
return result;
}
get buildResourcesDir() {
let result = this._buildResourcesDir;
if (result == null) {
result = path.resolve(this.projectDir, this.relativeBuildResourcesDirname);
this._buildResourcesDir = result;
}
return result;
}
get relativeBuildResourcesDirname() {
return this.config.directories.buildResources;
}
get framework() {
return this._framework;
}
disposeOnBuildFinish(disposer) {
this.toDispose.push(disposer);
}
//noinspection JSUnusedGlobalSymbols
constructor(options, cancellationToken = new builder_util_runtime_1.CancellationToken()) {
this.cancellationToken = cancellationToken;
this._metadata = null;
this._nodeModulesHandledExternally = false;
this._isPrepackedAppAsar = false;
this._devMetadata = null;
this._configuration = null;
this.isTwoPackageJsonProjectLayoutUsed = false;
this.eventEmitter = new events_1.EventEmitter();
this._appInfo = null;
this.tempDirManager = new builder_util_1.TmpDir("packager");
this._repositoryInfo = new lazy_val_1.Lazy(() => (0, repositoryInfo_1.getRepositoryInfo)(this.projectDir, this.metadata, this.devMetadata));
this.afterPackHandlers = [];
this.debugLogger = new builder_util_1.DebugLogger(builder_util_1.log.isDebugEnabled);
this.nodeDependencyInfo = new Map();
this.stageDirPathCustomizer = (target, packager, arch) => {
return path.join(target.outDir, `__${target.name}-${(0, arch_1.getArtifactArchName)(arch, target.name)}`);
};
this._buildResourcesDir = null;
this._framework = null;
this.toDispose = [];
if ("devMetadata" in options) {
throw new builder_util_1.InvalidConfigurationError("devMetadata in the options is deprecated, please use config instead");
}
if ("extraMetadata" in options) {
throw new builder_util_1.InvalidConfigurationError("extraMetadata in the options is deprecated, please use config.extraMetadata instead");
}
const targets = options.targets || new Map();
if (options.targets == null) {
options.targets = targets;
}
function processTargets(platform, types) {
function commonArch(currentIfNotSpecified) {
const result = Array();
return result.length === 0 && currentIfNotSpecified ? [(0, builder_util_1.archFromString)(process.arch)] : result;
}
let archToType = targets.get(platform);
if (archToType == null) {
archToType = new Map();
targets.set(platform, archToType);
}
if (types.length === 0) {
for (const arch of commonArch(false)) {
archToType.set(arch, []);
}
return;
}
for (const type of types) {
const suffixPos = type.lastIndexOf(":");
if (suffixPos > 0) {
(0, builder_util_1.addValue)(archToType, (0, builder_util_1.archFromString)(type.substring(suffixPos + 1)), type.substring(0, suffixPos));
}
else {
for (const arch of commonArch(true)) {
(0, builder_util_1.addValue)(archToType, arch, type);
}
}
}
}
if (options.mac != null) {
processTargets(core_1.Platform.MAC, options.mac);
}
if (options.linux != null) {
processTargets(core_1.Platform.LINUX, options.linux);
}
if (options.win != null) {
processTargets(core_1.Platform.WINDOWS, options.win);
}
this.projectDir = options.projectDir == null ? process.cwd() : path.resolve(options.projectDir);
this._appDir = this.projectDir;
this.options = {
...options,
prepackaged: options.prepackaged == null ? null : path.resolve(this.projectDir, options.prepackaged),
};
try {
builder_util_1.log.info({ version: version_1.PACKAGE_VERSION, os: (0, os_1.release)() }, "electron-builder");
}
catch (e) {
// error in dev mode without babel
if (!(e instanceof ReferenceError)) {
throw e;
}
}
}
addAfterPackHandler(handler) {
this.afterPackHandlers.push(handler);
}
artifactCreated(handler) {
addHandler(this.eventEmitter, "artifactCreated", handler);
return this;
}
async callArtifactBuildStarted(event, logFields) {
builder_util_1.log.info(logFields || {
target: event.targetPresentableName,
arch: event.arch == null ? null : builder_util_1.Arch[event.arch],
file: builder_util_1.log.filePath(event.file),
}, "building");
const handler = await (0, platformPackager_1.resolveFunction)(this.appInfo.type, this.config.artifactBuildStarted, "artifactBuildStarted");
if (handler != null) {
await Promise.resolve(handler(event));
}
}
/**
* Only for sub artifacts (update info), for main artifacts use `callArtifactBuildCompleted`.
*/
dispatchArtifactCreated(event) {
this.eventEmitter.emit("artifactCreated", event);
}
async callArtifactBuildCompleted(event) {
const handler = await (0, platformPackager_1.resolveFunction)(this.appInfo.type, this.config.artifactBuildCompleted, "artifactBuildCompleted");
if (handler != null) {
await Promise.resolve(handler(event));
}
this.dispatchArtifactCreated(event);
}
async callAppxManifestCreated(path) {
const handler = await (0, platformPackager_1.resolveFunction)(this.appInfo.type, this.config.appxManifestCreated, "appxManifestCreated");
if (handler != null) {
await Promise.resolve(handler(path));
}
}
async callMsiProjectCreated(path) {
const handler = await (0, platformPackager_1.resolveFunction)(this.appInfo.type, this.config.msiProjectCreated, "msiProjectCreated");
if (handler != null) {
await Promise.resolve(handler(path));
}
}
async build() {
let configPath = null;
let configFromOptions = this.options.config;
if (typeof configFromOptions === "string") {
// it is a path to config file
configPath = configFromOptions;
configFromOptions = null;
}
else if (configFromOptions != null && typeof configFromOptions.extends === "string" && configFromOptions.extends.includes(".")) {
configPath = configFromOptions.extends;
delete configFromOptions.extends;
}
const projectDir = this.projectDir;
const devPackageFile = path.join(projectDir, "package.json");
this._devMetadata = await (0, promise_1.orNullIfFileNotExist)((0, packageMetadata_1.readPackageJson)(devPackageFile));
const devMetadata = this.devMetadata;
const configuration = await (0, config_1.getConfig)(projectDir, configPath, configFromOptions, new lazy_val_1.Lazy(() => Promise.resolve(devMetadata)));
if (builder_util_1.log.isDebugEnabled) {
builder_util_1.log.debug({ config: getSafeEffectiveConfig(configuration) }, "effective config");
}
this._appDir = await (0, config_1.computeDefaultAppDirectory)(projectDir, configuration.directories.app);
this.isTwoPackageJsonProjectLayoutUsed = this._appDir !== projectDir;
const appPackageFile = this.isTwoPackageJsonProjectLayoutUsed ? path.join(this.appDir, "package.json") : devPackageFile;
// tslint:disable:prefer-conditional-expression
if (this.devMetadata != null && !this.isTwoPackageJsonProjectLayoutUsed) {
this._metadata = this.devMetadata;
}
else {
this._metadata = await this.readProjectMetadataIfTwoPackageStructureOrPrepacked(appPackageFile);
}
(0, builder_util_1.deepAssign)(this.metadata, configuration.extraMetadata);
if (this.isTwoPackageJsonProjectLayoutUsed) {
builder_util_1.log.debug({ devPackageFile, appPackageFile }, "two package.json structure is used");
}
(0, packageMetadata_1.checkMetadata)(this.metadata, this.devMetadata, appPackageFile, devPackageFile);
return await this._build(configuration, this._metadata, this._devMetadata);
}
// external caller of this method always uses isTwoPackageJsonProjectLayoutUsed=false and appDir=projectDir, no way (and need) to use another values
async _build(configuration, metadata, devMetadata, repositoryInfo) {
await (0, config_1.validateConfig)(configuration, this.debugLogger);
this._configuration = configuration;
this._metadata = metadata;
this._devMetadata = devMetadata;
if (repositoryInfo != null) {
this._repositoryInfo.value = Promise.resolve(repositoryInfo);
}
this._appInfo = new appInfo_1.AppInfo(this, null);
this._framework = await createFrameworkInfo(this.config, this);
const commonOutDirWithoutPossibleOsMacro = path.resolve(this.projectDir, (0, macroExpander_1.expandMacro)(configuration.directories.output, null, this._appInfo, {
os: "",
}));
if (!isCI && process.stdout.isTTY) {
const effectiveConfigFile = path.join(commonOutDirWithoutPossibleOsMacro, "builder-effective-config.yaml");
builder_util_1.log.info({ file: builder_util_1.log.filePath(effectiveConfigFile) }, "writing effective config");
await (0, fs_extra_1.outputFile)(effectiveConfigFile, getSafeEffectiveConfig(configuration));
}
// because artifact event maybe dispatched several times for different publish providers
const artifactPaths = new Set();
this.artifactCreated(event => {
if (event.file != null) {
artifactPaths.add(event.file);
}
});
this.disposeOnBuildFinish(() => this.tempDirManager.cleanup());
const platformToTargets = await (0, promise_1.executeFinally)(this.doBuild(), async () => {
if (this.debugLogger.isEnabled) {
await this.debugLogger.save(path.join(commonOutDirWithoutPossibleOsMacro, "builder-debug.yml"));
}
const toDispose = this.toDispose.slice();
this.toDispose.length = 0;
for (const disposer of toDispose) {
await disposer().catch((e) => {
builder_util_1.log.warn({ error: e }, "cannot dispose");
});
}
});
return {
outDir: commonOutDirWithoutPossibleOsMacro,
artifactPaths: Array.from(artifactPaths),
platformToTargets,
configuration,
};
}
async readProjectMetadataIfTwoPackageStructureOrPrepacked(appPackageFile) {
let data = await (0, promise_1.orNullIfFileNotExist)((0, packageMetadata_1.readPackageJson)(appPackageFile));
if (data != null) {
return data;
}
data = await (0, promise_1.orNullIfFileNotExist)((0, asar_1.readAsarJson)(path.join(this.projectDir, "app.asar"), "package.json"));
if (data != null) {
this._isPrepackedAppAsar = true;
return data;
}
throw new Error(`Cannot find package.json in the ${path.dirname(appPackageFile)}`);
}
async doBuild() {
const taskManager = new builder_util_1.AsyncTaskManager(this.cancellationToken);
const syncTargetsIfAny = [];
const platformToTarget = new Map();
const createdOutDirs = new Set();
for (const [platform, archToType] of this.options.targets) {
if (this.cancellationToken.cancelled) {
break;
}
if (platform === core_1.Platform.MAC && process.platform === core_1.Platform.WINDOWS.nodeName) {
throw new builder_util_1.InvalidConfigurationError("Build for macOS is supported only on macOS, please see https://electron.build/multi-platform-build");
}
const packager = await this.createHelper(platform);
const nameToTarget = new Map();
platformToTarget.set(platform, nameToTarget);
for (const [arch, targetNames] of (0, targetFactory_1.computeArchToTargetNamesMap)(archToType, packager, platform)) {
if (this.cancellationToken.cancelled) {
break;
}
// support os and arch macro in output value
const outDir = path.resolve(this.projectDir, packager.expandMacro(this._configuration.directories.output, builder_util_1.Arch[arch]));
const targetList = (0, targetFactory_1.createTargets)(nameToTarget, targetNames.length === 0 ? packager.defaultTarget : targetNames, outDir, packager);
await createOutDirIfNeed(targetList, createdOutDirs);
await packager.pack(outDir, arch, targetList, taskManager);
}
if (this.cancellationToken.cancelled) {
break;
}
for (const target of nameToTarget.values()) {
if (target.isAsyncSupported) {
taskManager.addTask(target.finishBuild());
}
else {
syncTargetsIfAny.push(target);
}
}
}
await taskManager.awaitTasks();
for (const target of syncTargetsIfAny) {
await target.finishBuild();
}
return platformToTarget;
}
async createHelper(platform) {
if (this.options.platformPackagerFactory != null) {
return this.options.platformPackagerFactory(this, platform);
}
switch (platform) {
case core_1.Platform.MAC: {
const helperClass = (await Promise.resolve().then(() => require("./macPackager"))).default;
return new helperClass(this);
}
case core_1.Platform.WINDOWS: {
const helperClass = (await Promise.resolve().then(() => require("./winPackager"))).WinPackager;
return new helperClass(this);
}
case core_1.Platform.LINUX:
return new (await Promise.resolve().then(() => require("./linuxPackager"))).LinuxPackager(this);
default:
throw new Error(`Unknown platform: ${platform}`);
}
}
async installAppDependencies(platform, arch) {
if (this.options.prepackaged != null || !this.framework.isNpmRebuildRequired) {
return;
}
const frameworkInfo = { version: this.framework.version, useCustomDist: true };
const config = this.config;
if (config.nodeGypRebuild === true) {
await (0, yarn_1.nodeGypRebuild)(platform.nodeName, builder_util_1.Arch[arch], frameworkInfo);
}
if (config.npmRebuild === false) {
builder_util_1.log.info({ reason: "npmRebuild is set to false" }, "skipped dependencies rebuild");
return;
}
const beforeBuild = await (0, platformPackager_1.resolveFunction)(this.appInfo.type, config.beforeBuild, "beforeBuild");
if (beforeBuild != null) {
const performDependenciesInstallOrRebuild = await beforeBuild({
appDir: this.appDir,
electronVersion: this.config.electronVersion,
platform,
arch: builder_util_1.Arch[arch],
});
// If beforeBuild resolves to false, it means that handling node_modules is done outside of electron-builder.
this._nodeModulesHandledExternally = !performDependenciesInstallOrRebuild;
if (!performDependenciesInstallOrRebuild) {
return;
}
}
if (config.buildDependenciesFromSource === true && platform.nodeName !== process.platform) {
builder_util_1.log.info({ reason: "platform is different and buildDependenciesFromSource is set to true" }, "skipped dependencies rebuild");
}
else {
await (0, yarn_1.installOrRebuild)(config, this.appDir, {
frameworkInfo,
platform: platform.nodeName,
arch: builder_util_1.Arch[arch],
productionDeps: this.getNodeDependencyInfo(null),
});
}
}
async afterPack(context) {
const afterPack = await (0, platformPackager_1.resolveFunction)(this.appInfo.type, this.config.afterPack, "afterPack");
const handlers = this.afterPackHandlers.slice();
if (afterPack != null) {
// user handler should be last
handlers.push(afterPack);
}
for (const handler of handlers) {
await Promise.resolve(handler(context));
}
}
}
exports.Packager = Packager;
function createOutDirIfNeed(targetList, createdOutDirs) {
const ourDirs = new Set();
for (const target of targetList) {
// noinspection SuspiciousInstanceOfGuard
if (target instanceof targetFactory_1.NoOpTarget) {
continue;
}
const outDir = target.outDir;
if (!createdOutDirs.has(outDir)) {
ourDirs.add(outDir);
}
}
if (ourDirs.size === 0) {
return Promise.resolve();
}
return Promise.all(Array.from(ourDirs)
.sort()
.map(dir => {
return (0, fs_extra_1.mkdirs)(dir)
.then(() => (0, fs_extra_1.chmod)(dir, 0o755) /* set explicitly */)
.then(() => createdOutDirs.add(dir));
}));
}
function getSafeEffectiveConfig(configuration) {
const o = JSON.parse((0, builder_util_1.safeStringifyJson)(configuration));
if (o.cscLink != null) {
o.cscLink = "<hidden by builder>";
}
return (0, builder_util_1.serializeToYaml)(o, true);
}
//# sourceMappingURL=packager.js.map