mirror of
https://gitee.com/myxzgzs/boyue_jnpf.git
synced 2025-08-08 15:22:43 +08:00
690 lines
33 KiB
JavaScript
690 lines
33 KiB
JavaScript
"use strict";
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.PlatformPackager = void 0;
|
|
exports.isSafeGithubName = isSafeGithubName;
|
|
exports.computeSafeArtifactNameIfNeeded = computeSafeArtifactNameIfNeeded;
|
|
exports.normalizeExt = normalizeExt;
|
|
exports.chooseNotNull = chooseNotNull;
|
|
const fuses_1 = require("@electron/fuses");
|
|
const builder_util_1 = require("builder-util");
|
|
const promises_1 = require("fs/promises");
|
|
const lazy_val_1 = require("lazy-val");
|
|
const path = require("path");
|
|
const appInfo_1 = require("./appInfo");
|
|
const asarFileChecker_1 = require("./asar/asarFileChecker");
|
|
const asarUtil_1 = require("./asar/asarUtil");
|
|
const integrity_1 = require("./asar/integrity");
|
|
const fileMatcher_1 = require("./fileMatcher");
|
|
const fileTransformer_1 = require("./fileTransformer");
|
|
const Framework_1 = require("./Framework");
|
|
const index_1 = require("./index");
|
|
const appBuilder_1 = require("./util/appBuilder");
|
|
const appFileCopier_1 = require("./util/appFileCopier");
|
|
const macroExpander_1 = require("./util/macroExpander");
|
|
class PlatformPackager {
|
|
get packagerOptions() {
|
|
return this.info.options;
|
|
}
|
|
get buildResourcesDir() {
|
|
return this.info.buildResourcesDir;
|
|
}
|
|
get projectDir() {
|
|
return this.info.projectDir;
|
|
}
|
|
get config() {
|
|
return this.info.config;
|
|
}
|
|
get resourceList() {
|
|
return this._resourceList.value;
|
|
}
|
|
constructor(info, platform) {
|
|
this.info = info;
|
|
this.platform = platform;
|
|
this._resourceList = new lazy_val_1.Lazy(() => (0, builder_util_1.orIfFileNotExist)((0, promises_1.readdir)(this.info.buildResourcesDir), []));
|
|
this.platformSpecificBuildOptions = PlatformPackager.normalizePlatformSpecificBuildOptions(this.config[platform.buildConfigurationKey]);
|
|
this.appInfo = this.prepareAppInfo(info.appInfo);
|
|
}
|
|
get compression() {
|
|
const compression = this.platformSpecificBuildOptions.compression;
|
|
// explicitly set to null - request to use default value instead of parent (in the config)
|
|
if (compression === null) {
|
|
return "normal";
|
|
}
|
|
return compression || this.config.compression || "normal";
|
|
}
|
|
get debugLogger() {
|
|
return this.info.debugLogger;
|
|
}
|
|
// eslint-disable-next-line
|
|
prepareAppInfo(appInfo) {
|
|
return new appInfo_1.AppInfo(this.info, null, this.platformSpecificBuildOptions);
|
|
}
|
|
static normalizePlatformSpecificBuildOptions(options) {
|
|
return options == null ? Object.create(null) : options;
|
|
}
|
|
getCscPassword() {
|
|
const password = this.doGetCscPassword();
|
|
if ((0, builder_util_1.isEmptyOrSpaces)(password)) {
|
|
builder_util_1.log.info({ reason: "CSC_KEY_PASSWORD is not defined" }, "empty password will be used for code signing");
|
|
return "";
|
|
}
|
|
else {
|
|
return password.trim();
|
|
}
|
|
}
|
|
getCscLink(extraEnvName) {
|
|
// allow to specify as empty string
|
|
const envValue = chooseNotNull(extraEnvName == null ? null : process.env[extraEnvName], process.env.CSC_LINK);
|
|
return chooseNotNull(chooseNotNull(this.info.config.cscLink, this.platformSpecificBuildOptions.cscLink), envValue);
|
|
}
|
|
doGetCscPassword() {
|
|
// allow to specify as empty string
|
|
return chooseNotNull(chooseNotNull(this.info.config.cscKeyPassword, this.platformSpecificBuildOptions.cscKeyPassword), process.env.CSC_KEY_PASSWORD);
|
|
}
|
|
computeAppOutDir(outDir, arch) {
|
|
return (this.packagerOptions.prepackaged ||
|
|
path.join(outDir, `${this.platform.buildConfigurationKey}${(0, builder_util_1.getArchSuffix)(arch, this.platformSpecificBuildOptions.defaultArch)}${this.platform === index_1.Platform.MAC ? "" : "-unpacked"}`));
|
|
}
|
|
dispatchArtifactCreated(file, target, arch, safeArtifactName) {
|
|
return this.info.emitArtifactBuildCompleted({
|
|
file,
|
|
safeArtifactName,
|
|
target,
|
|
arch,
|
|
packager: this,
|
|
});
|
|
}
|
|
async pack(outDir, arch, targets, taskManager) {
|
|
const appOutDir = this.computeAppOutDir(outDir, arch);
|
|
await this.doPack({
|
|
outDir,
|
|
appOutDir,
|
|
platformName: this.platform.nodeName,
|
|
arch,
|
|
platformSpecificBuildOptions: this.platformSpecificBuildOptions,
|
|
targets,
|
|
});
|
|
this.packageInDistributableFormat(appOutDir, arch, targets, taskManager);
|
|
}
|
|
packageInDistributableFormat(appOutDir, arch, targets, taskManager) {
|
|
if (targets.find(it => !it.isAsyncSupported) == null) {
|
|
PlatformPackager.buildAsyncTargets(targets, taskManager, appOutDir, arch);
|
|
return;
|
|
}
|
|
taskManager.add(async () => {
|
|
// BluebirdPromise.map doesn't invoke target.build immediately, but for RemoteTarget it is very critical to call build() before finishBuild()
|
|
const subTaskManager = new builder_util_1.AsyncTaskManager(this.info.cancellationToken);
|
|
PlatformPackager.buildAsyncTargets(targets, subTaskManager, appOutDir, arch);
|
|
await subTaskManager.awaitTasks();
|
|
for (const target of targets) {
|
|
if (!target.isAsyncSupported && !this.info.cancellationToken.cancelled) {
|
|
await target.build(appOutDir, arch);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
static buildAsyncTargets(targets, taskManager, appOutDir, arch) {
|
|
for (const target of targets) {
|
|
if (target.isAsyncSupported) {
|
|
taskManager.addTask(target.build(appOutDir, arch));
|
|
}
|
|
}
|
|
}
|
|
getExtraFileMatchers(isResources, appOutDir, options) {
|
|
const base = isResources
|
|
? this.getResourcesDir(appOutDir)
|
|
: this.platform === index_1.Platform.MAC
|
|
? path.join(appOutDir, `${this.appInfo.productFilename}.app`, "Contents")
|
|
: appOutDir;
|
|
return (0, fileMatcher_1.getFileMatchers)(this.config, isResources ? "extraResources" : "extraFiles", base, options);
|
|
}
|
|
createGetFileMatchersOptions(outDir, arch, customBuildOptions) {
|
|
return {
|
|
macroExpander: it => this.expandMacro(it, arch == null ? null : builder_util_1.Arch[arch], { "/*": "{,/**/*}" }),
|
|
customBuildOptions,
|
|
globalOutDir: outDir,
|
|
defaultSrc: this.projectDir,
|
|
};
|
|
}
|
|
async doPack(packOptions) {
|
|
var _a;
|
|
if (this.packagerOptions.prepackaged != null) {
|
|
return;
|
|
}
|
|
if (this.info.cancellationToken.cancelled) {
|
|
return;
|
|
}
|
|
// Due to node-gyp rewriting GYP_MSVS_VERSION when reused across the same session, we must reset the env var: https://github.com/electron-userland/electron-builder/issues/7256
|
|
delete process.env.GYP_MSVS_VERSION;
|
|
const { outDir, appOutDir, platformName, arch, platformSpecificBuildOptions, targets, options } = packOptions;
|
|
await this.info.emitBeforePack({
|
|
appOutDir,
|
|
outDir,
|
|
arch,
|
|
targets,
|
|
packager: this,
|
|
electronPlatformName: platformName,
|
|
});
|
|
await this.info.installAppDependencies(this.platform, arch);
|
|
if (this.info.cancellationToken.cancelled) {
|
|
return;
|
|
}
|
|
const framework = this.info.framework;
|
|
builder_util_1.log.info({
|
|
platform: platformName,
|
|
arch: builder_util_1.Arch[arch],
|
|
[`${framework.name}`]: framework.version,
|
|
appOutDir: builder_util_1.log.filePath(appOutDir),
|
|
}, `packaging`);
|
|
await framework.prepareApplicationStageDirectory({
|
|
packager: this,
|
|
appOutDir,
|
|
platformName,
|
|
arch: builder_util_1.Arch[arch],
|
|
version: framework.version,
|
|
});
|
|
await this.info.emitAfterExtract({
|
|
appOutDir,
|
|
outDir,
|
|
arch,
|
|
targets,
|
|
packager: this,
|
|
electronPlatformName: platformName,
|
|
});
|
|
const excludePatterns = [];
|
|
const computeParsedPatterns = (patterns) => {
|
|
if (patterns != null) {
|
|
for (const pattern of patterns) {
|
|
pattern.computeParsedPatterns(excludePatterns, this.info.projectDir);
|
|
}
|
|
}
|
|
};
|
|
const getFileMatchersOptions = this.createGetFileMatchersOptions(outDir, arch, platformSpecificBuildOptions);
|
|
const macroExpander = getFileMatchersOptions.macroExpander;
|
|
const extraResourceMatchers = this.getExtraFileMatchers(true, appOutDir, getFileMatchersOptions);
|
|
computeParsedPatterns(extraResourceMatchers);
|
|
const extraFileMatchers = this.getExtraFileMatchers(false, appOutDir, getFileMatchersOptions);
|
|
computeParsedPatterns(extraFileMatchers);
|
|
const packContext = {
|
|
appOutDir,
|
|
outDir,
|
|
arch,
|
|
targets,
|
|
packager: this,
|
|
electronPlatformName: platformName,
|
|
};
|
|
const asarOptions = await this.computeAsarOptions(platformSpecificBuildOptions);
|
|
const resourcesPath = this.platform === index_1.Platform.MAC
|
|
? path.join(appOutDir, framework.distMacOsAppName, "Contents", "Resources")
|
|
: (0, Framework_1.isElectronBased)(framework)
|
|
? path.join(appOutDir, "resources")
|
|
: appOutDir;
|
|
const taskManager = new builder_util_1.AsyncTaskManager(this.info.cancellationToken);
|
|
this.copyAppFiles(taskManager, asarOptions, resourcesPath, path.join(resourcesPath, "app"), packContext, platformSpecificBuildOptions, excludePatterns, macroExpander);
|
|
await taskManager.awaitTasks();
|
|
if (this.info.cancellationToken.cancelled) {
|
|
return;
|
|
}
|
|
if (framework.beforeCopyExtraFiles != null) {
|
|
const resourcesRelativePath = this.platform === index_1.Platform.MAC ? "Resources" : (0, Framework_1.isElectronBased)(framework) ? "resources" : "";
|
|
let asarIntegrity = null;
|
|
if (!(asarOptions == null || (options === null || options === void 0 ? void 0 : options.disableAsarIntegrity))) {
|
|
asarIntegrity = await (0, integrity_1.computeData)({ resourcesPath, resourcesRelativePath, resourcesDestinationPath: this.getResourcesDir(appOutDir), extraResourceMatchers });
|
|
}
|
|
await framework.beforeCopyExtraFiles({
|
|
packager: this,
|
|
appOutDir,
|
|
asarIntegrity,
|
|
platformName,
|
|
});
|
|
}
|
|
if (this.info.cancellationToken.cancelled) {
|
|
return;
|
|
}
|
|
const transformerForExtraFiles = this.createTransformerForExtraFiles(packContext);
|
|
await (0, fileMatcher_1.copyFiles)(extraResourceMatchers, transformerForExtraFiles);
|
|
await (0, fileMatcher_1.copyFiles)(extraFileMatchers, transformerForExtraFiles);
|
|
if (this.info.cancellationToken.cancelled) {
|
|
return;
|
|
}
|
|
await this.info.emitAfterPack(packContext);
|
|
if (framework.afterPack != null) {
|
|
await framework.afterPack(packContext);
|
|
}
|
|
const isAsar = asarOptions != null;
|
|
await this.sanityCheckPackage(appOutDir, isAsar, framework, !!this.config.disableSanityCheckAsar);
|
|
if (!(options === null || options === void 0 ? void 0 : options.disableFuses)) {
|
|
await this.doAddElectronFuses(packContext);
|
|
}
|
|
if ((_a = options === null || options === void 0 ? void 0 : options.sign) !== null && _a !== void 0 ? _a : true) {
|
|
await this.doSignAfterPack(outDir, appOutDir, platformName, arch, platformSpecificBuildOptions, targets);
|
|
}
|
|
}
|
|
// the fuses MUST be flipped right before signing
|
|
async doAddElectronFuses(packContext) {
|
|
if (this.config.electronFuses == null) {
|
|
return;
|
|
}
|
|
const fuseConfig = this.generateFuseConfig(this.config.electronFuses);
|
|
await this.addElectronFuses(packContext, fuseConfig);
|
|
}
|
|
generateFuseConfig(fuses) {
|
|
const config = {
|
|
version: fuses_1.FuseVersion.V1,
|
|
resetAdHocDarwinSignature: fuses.resetAdHocDarwinSignature,
|
|
};
|
|
// this is annoying, but we must filter out undefined entries because some older electron versions will receive `the fuse wire in this version of Electron is not long enough` even if entry is set undefined
|
|
if (fuses.runAsNode != null) {
|
|
config[fuses_1.FuseV1Options.RunAsNode] = fuses.runAsNode;
|
|
}
|
|
if (fuses.enableCookieEncryption != null) {
|
|
config[fuses_1.FuseV1Options.EnableCookieEncryption] = fuses.enableCookieEncryption;
|
|
}
|
|
if (fuses.enableNodeOptionsEnvironmentVariable != null) {
|
|
config[fuses_1.FuseV1Options.EnableNodeOptionsEnvironmentVariable] = fuses.enableNodeOptionsEnvironmentVariable;
|
|
}
|
|
if (fuses.enableNodeCliInspectArguments != null) {
|
|
config[fuses_1.FuseV1Options.EnableNodeCliInspectArguments] = fuses.enableNodeCliInspectArguments;
|
|
}
|
|
if (fuses.enableEmbeddedAsarIntegrityValidation != null) {
|
|
config[fuses_1.FuseV1Options.EnableEmbeddedAsarIntegrityValidation] = fuses.enableEmbeddedAsarIntegrityValidation;
|
|
}
|
|
if (fuses.onlyLoadAppFromAsar != null) {
|
|
config[fuses_1.FuseV1Options.OnlyLoadAppFromAsar] = fuses.onlyLoadAppFromAsar;
|
|
}
|
|
if (fuses.loadBrowserProcessSpecificV8Snapshot != null) {
|
|
config[fuses_1.FuseV1Options.LoadBrowserProcessSpecificV8Snapshot] = fuses.loadBrowserProcessSpecificV8Snapshot;
|
|
}
|
|
if (fuses.grantFileProtocolExtraPrivileges != null) {
|
|
config[fuses_1.FuseV1Options.GrantFileProtocolExtraPrivileges] = fuses.grantFileProtocolExtraPrivileges;
|
|
}
|
|
return config;
|
|
}
|
|
/**
|
|
* Use `AfterPackContext` here to keep available for public API
|
|
* @param {AfterPackContext} context
|
|
* @param {FuseConfig} fuses
|
|
*
|
|
* Can be used in `afterPack` hook for custom fuse logic like below. It's an alternative approach if one wants to override electron-builder's @electron/fuses version
|
|
* ```
|
|
* await context.packager.addElectronFuses(context, { ... })
|
|
* ```
|
|
*/
|
|
addElectronFuses(context, fuses) {
|
|
const { appOutDir, electronPlatformName } = context;
|
|
const ext = {
|
|
darwin: ".app",
|
|
mas: ".app",
|
|
win32: ".exe",
|
|
linux: "",
|
|
}[electronPlatformName];
|
|
const executableName = this instanceof index_1.LinuxPackager ? this.executableName : this.appInfo.productFilename;
|
|
const electronBinaryPath = path.join(appOutDir, `${executableName}${ext}`);
|
|
builder_util_1.log.info({ electronPath: builder_util_1.log.filePath(electronBinaryPath) }, "executing @electron/fuses");
|
|
return (0, fuses_1.flipFuses)(electronBinaryPath, fuses);
|
|
}
|
|
async doSignAfterPack(outDir, appOutDir, platformName, arch, platformSpecificBuildOptions, targets) {
|
|
const asarOptions = await this.computeAsarOptions(platformSpecificBuildOptions);
|
|
const isAsar = asarOptions != null;
|
|
const packContext = {
|
|
appOutDir,
|
|
outDir,
|
|
arch,
|
|
targets,
|
|
packager: this,
|
|
electronPlatformName: platformName,
|
|
};
|
|
const didSign = await this.signApp(packContext, isAsar);
|
|
if (didSign) {
|
|
await this.info.emitAfterSign(packContext);
|
|
}
|
|
else if (this.info.filterPackagerEventListeners("afterSign", "user").length) {
|
|
builder_util_1.log.warn(null, `skipping "afterSign" hook as no signing occurred, perhaps you intended "afterPack"?`);
|
|
}
|
|
}
|
|
// eslint-disable-next-line
|
|
createTransformerForExtraFiles(packContext) {
|
|
return null;
|
|
}
|
|
copyAppFiles(taskManager, asarOptions, resourcePath, defaultDestination, packContext, platformSpecificBuildOptions, excludePatterns, macroExpander) {
|
|
const appDir = this.info.appDir;
|
|
const config = this.config;
|
|
const isElectronCompile = asarOptions != null && (0, fileTransformer_1.isElectronCompileUsed)(this.info);
|
|
const mainMatchers = (0, fileMatcher_1.getMainFileMatchers)(appDir, defaultDestination, macroExpander, platformSpecificBuildOptions, this, packContext.outDir, isElectronCompile);
|
|
if (excludePatterns.length > 0) {
|
|
for (const matcher of mainMatchers) {
|
|
matcher.excludePatterns = excludePatterns;
|
|
}
|
|
}
|
|
const framework = this.info.framework;
|
|
const transformer = (0, fileTransformer_1.createTransformer)(appDir, config, isElectronCompile
|
|
? {
|
|
originalMain: this.info.metadata.main,
|
|
main: appFileCopier_1.ELECTRON_COMPILE_SHIM_FILENAME,
|
|
...config.extraMetadata,
|
|
}
|
|
: config.extraMetadata, framework.createTransformer == null ? null : framework.createTransformer());
|
|
const _computeFileSets = (matchers) => {
|
|
return (0, appFileCopier_1.computeFileSets)(matchers, this.info.isPrepackedAppAsar ? null : transformer, this, isElectronCompile).then(async (result) => {
|
|
if (!this.info.isPrepackedAppAsar && !this.info.areNodeModulesHandledExternally) {
|
|
const moduleFileMatcher = (0, fileMatcher_1.getNodeModuleFileMatcher)(appDir, defaultDestination, macroExpander, platformSpecificBuildOptions, this.info);
|
|
result = result.concat(await (0, appFileCopier_1.computeNodeModuleFileSets)(this, moduleFileMatcher));
|
|
}
|
|
return result.filter(it => it.files.length > 0);
|
|
});
|
|
};
|
|
if (this.info.isPrepackedAppAsar) {
|
|
taskManager.add(async () => {
|
|
const fileSets = await _computeFileSets([new fileMatcher_1.FileMatcher(appDir, resourcePath, macroExpander)]);
|
|
fileSets.forEach(it => taskManager.addTask((0, appFileCopier_1.copyAppFiles)(it, this.info, transformer)));
|
|
await taskManager.awaitTasks();
|
|
});
|
|
}
|
|
else if (asarOptions == null) {
|
|
// for ASAR all asar unpacked files will be extra transformed (e.g. sign of EXE and DLL) later,
|
|
// for prepackaged asar extra transformation not supported yet,
|
|
// so, extra transform if asar is disabled
|
|
const transformerForExtraFiles = this.createTransformerForExtraFiles(packContext);
|
|
const combinedTransformer = file => {
|
|
if (transformerForExtraFiles != null) {
|
|
const result = transformerForExtraFiles(file);
|
|
if (result != null) {
|
|
return result;
|
|
}
|
|
}
|
|
return transformer(file);
|
|
};
|
|
taskManager.add(async () => {
|
|
const fileSets = await _computeFileSets(mainMatchers);
|
|
fileSets.forEach(it => taskManager.addTask((0, appFileCopier_1.copyAppFiles)(it, this.info, combinedTransformer)));
|
|
await taskManager.awaitTasks();
|
|
});
|
|
}
|
|
else {
|
|
const unpackPattern = (0, fileMatcher_1.getFileMatchers)(config, "asarUnpack", defaultDestination, {
|
|
macroExpander,
|
|
customBuildOptions: platformSpecificBuildOptions,
|
|
globalOutDir: packContext.outDir,
|
|
defaultSrc: appDir,
|
|
});
|
|
const fileMatcher = unpackPattern == null ? null : unpackPattern[0];
|
|
taskManager.addTask(_computeFileSets(mainMatchers).then(async (fileSets) => {
|
|
for (const fileSet of fileSets) {
|
|
await (0, appFileCopier_1.transformFiles)(transformer, fileSet);
|
|
}
|
|
await new asarUtil_1.AsarPackager(this, {
|
|
defaultDestination,
|
|
resourcePath,
|
|
options: asarOptions,
|
|
unpackPattern: fileMatcher === null || fileMatcher === void 0 ? void 0 : fileMatcher.createFilter(),
|
|
}).pack(fileSets);
|
|
}));
|
|
}
|
|
}
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
signApp(packContext, isAsar) {
|
|
return Promise.resolve(false);
|
|
}
|
|
getIconPath() {
|
|
return Promise.resolve(null);
|
|
}
|
|
async computeAsarOptions(customBuildOptions) {
|
|
if (!(0, Framework_1.isElectronBased)(this.info.framework)) {
|
|
return null;
|
|
}
|
|
function errorMessage(name) {
|
|
return `${name} is deprecated is deprecated and not supported — please use asarUnpack`;
|
|
}
|
|
const buildMetadata = this.config;
|
|
if (buildMetadata["asar-unpack"] != null) {
|
|
throw new Error(errorMessage("asar-unpack"));
|
|
}
|
|
if (buildMetadata["asar-unpack-dir"] != null) {
|
|
throw new Error(errorMessage("asar-unpack-dir"));
|
|
}
|
|
const platformSpecific = customBuildOptions.asar;
|
|
const result = platformSpecific == null ? this.config.asar : platformSpecific;
|
|
if (result === false) {
|
|
const appAsarStat = await (0, builder_util_1.statOrNull)(path.join(this.info.appDir, "app.asar"));
|
|
//noinspection ES6MissingAwait
|
|
if (appAsarStat == null || !appAsarStat.isFile()) {
|
|
builder_util_1.log.warn({
|
|
solution: "enable asar and use asarUnpack to unpack files that must be externally available",
|
|
}, "asar usage is disabled — this is strongly not recommended");
|
|
}
|
|
return null;
|
|
}
|
|
if (result == null || result === true) {
|
|
return {};
|
|
}
|
|
for (const name of ["unpackDir", "unpack"]) {
|
|
if (result[name] != null) {
|
|
throw new Error(errorMessage(`asar.${name}`));
|
|
}
|
|
}
|
|
return (0, builder_util_1.deepAssign)({}, result);
|
|
}
|
|
getElectronSrcDir(dist) {
|
|
return path.resolve(this.projectDir, dist);
|
|
}
|
|
getElectronDestinationDir(appOutDir) {
|
|
return appOutDir;
|
|
}
|
|
getResourcesDir(appOutDir) {
|
|
if (this.platform === index_1.Platform.MAC) {
|
|
return this.getMacOsResourcesDir(appOutDir);
|
|
}
|
|
else if ((0, Framework_1.isElectronBased)(this.info.framework)) {
|
|
return path.join(appOutDir, "resources");
|
|
}
|
|
else {
|
|
return appOutDir;
|
|
}
|
|
}
|
|
getMacOsResourcesDir(appOutDir) {
|
|
return path.join(appOutDir, `${this.appInfo.productFilename}.app`, "Contents", "Resources");
|
|
}
|
|
async checkFileInPackage(resourcesDir, file, messagePrefix, isAsar, disableSanityCheckAsar) {
|
|
if (isAsar && disableSanityCheckAsar) {
|
|
return;
|
|
}
|
|
const relativeFile = path.relative(this.info.appDir, path.resolve(this.info.appDir, file));
|
|
if (isAsar) {
|
|
(0, asarFileChecker_1.checkFileInArchive)(path.join(resourcesDir, "app.asar"), relativeFile, messagePrefix);
|
|
return;
|
|
}
|
|
const pathParsed = path.parse(file);
|
|
// Even when packaging to asar is disabled, it does not imply that the main file can not be inside an .asar archive.
|
|
// This may occur when the packaging is done manually before processing with electron-builder.
|
|
if (pathParsed.dir.includes(".asar")) {
|
|
// The path needs to be split to the part with an asar archive which acts like a directory and the part with
|
|
// the path to main file itself. (e.g. path/arch.asar/dir/index.js -> path/arch.asar, dir/index.js)
|
|
// noinspection TypeScriptValidateJSTypes
|
|
const pathSplit = pathParsed.dir.split(path.sep);
|
|
let partWithAsarIndex = 0;
|
|
pathSplit.some((pathPart, index) => {
|
|
partWithAsarIndex = index;
|
|
return pathPart.endsWith(".asar");
|
|
});
|
|
const asarPath = path.join(...pathSplit.slice(0, partWithAsarIndex + 1));
|
|
let mainPath = pathSplit.length > partWithAsarIndex + 1 ? path.join.apply(pathSplit.slice(partWithAsarIndex + 1)) : "";
|
|
mainPath += path.join(mainPath, pathParsed.base);
|
|
(0, asarFileChecker_1.checkFileInArchive)(path.join(resourcesDir, "app", asarPath), mainPath, messagePrefix);
|
|
}
|
|
else {
|
|
const fullPath = path.join(resourcesDir, "app", relativeFile);
|
|
const outStat = await (0, builder_util_1.statOrNull)(fullPath);
|
|
if (outStat == null) {
|
|
throw new Error(`${messagePrefix} "${fullPath}" does not exist. Seems like a wrong configuration.`);
|
|
}
|
|
else {
|
|
//noinspection ES6MissingAwait
|
|
if (!outStat.isFile()) {
|
|
throw new Error(`${messagePrefix} "${fullPath}" is not a file. Seems like a wrong configuration.`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
async sanityCheckPackage(appOutDir, isAsar, framework, disableSanityCheckAsar) {
|
|
const outStat = await (0, builder_util_1.statOrNull)(appOutDir);
|
|
if (outStat == null) {
|
|
throw new Error(`Output directory "${appOutDir}" does not exist. Seems like a wrong configuration.`);
|
|
}
|
|
else {
|
|
//noinspection ES6MissingAwait
|
|
if (!outStat.isDirectory()) {
|
|
throw new Error(`Output directory "${appOutDir}" is not a directory. Seems like a wrong configuration.`);
|
|
}
|
|
}
|
|
const resourcesDir = this.getResourcesDir(appOutDir);
|
|
const mainFile = (framework.getMainFile == null ? null : framework.getMainFile(this.platform)) || this.info.metadata.main || "index.js";
|
|
await this.checkFileInPackage(resourcesDir, mainFile, "Application entry file", isAsar, disableSanityCheckAsar);
|
|
await this.checkFileInPackage(resourcesDir, "package.json", "Application", isAsar, disableSanityCheckAsar);
|
|
}
|
|
// tslint:disable-next-line:no-invalid-template-strings
|
|
computeSafeArtifactName(suggestedName, ext, arch, skipDefaultArch = true, defaultArch, safePattern = "${name}-${version}-${arch}.${ext}") {
|
|
return computeSafeArtifactNameIfNeeded(suggestedName, () => this.computeArtifactName(safePattern, ext, skipDefaultArch && arch === (0, builder_util_1.defaultArchFromString)(defaultArch) ? null : arch));
|
|
}
|
|
expandArtifactNamePattern(targetSpecificOptions, ext, arch, defaultPattern, skipDefaultArch = true, defaultArch) {
|
|
const { pattern, isUserForced } = this.artifactPatternConfig(targetSpecificOptions, defaultPattern);
|
|
return this.computeArtifactName(pattern, ext, !isUserForced && skipDefaultArch && arch === (0, builder_util_1.defaultArchFromString)(defaultArch) ? null : arch);
|
|
}
|
|
artifactPatternConfig(targetSpecificOptions, defaultPattern) {
|
|
const userSpecifiedPattern = (targetSpecificOptions === null || targetSpecificOptions === void 0 ? void 0 : targetSpecificOptions.artifactName) || this.platformSpecificBuildOptions.artifactName || this.config.artifactName;
|
|
return {
|
|
isUserForced: !!userSpecifiedPattern,
|
|
pattern: userSpecifiedPattern || defaultPattern || "${productName}-${version}-${arch}.${ext}",
|
|
};
|
|
}
|
|
expandArtifactBeautyNamePattern(targetSpecificOptions, ext, arch) {
|
|
// tslint:disable-next-line:no-invalid-template-strings
|
|
return this.expandArtifactNamePattern(targetSpecificOptions, ext, arch, "${productName} ${version} ${arch}.${ext}", true);
|
|
}
|
|
computeArtifactName(pattern, ext, arch) {
|
|
const archName = arch == null ? null : (0, builder_util_1.getArtifactArchName)(arch, ext);
|
|
return this.expandMacro(pattern, archName, {
|
|
ext,
|
|
});
|
|
}
|
|
expandMacro(pattern, arch, extra = {}, isProductNameSanitized = true) {
|
|
return (0, macroExpander_1.expandMacro)(pattern, arch, this.appInfo, { os: this.platform.buildConfigurationKey, ...extra }, isProductNameSanitized);
|
|
}
|
|
generateName2(ext, classifier, deployment) {
|
|
const dotExt = ext == null ? "" : `.${ext}`;
|
|
const separator = ext === "deb" ? "_" : "-";
|
|
return `${deployment ? this.appInfo.name : this.appInfo.productFilename}${separator}${this.appInfo.version}${classifier == null ? "" : `${separator}${classifier}`}${dotExt}`;
|
|
}
|
|
getTempFile(suffix) {
|
|
return this.info.tempDirManager.getTempFile({ suffix });
|
|
}
|
|
get fileAssociations() {
|
|
return (0, builder_util_1.asArray)(this.config.fileAssociations).concat((0, builder_util_1.asArray)(this.platformSpecificBuildOptions.fileAssociations));
|
|
}
|
|
async getResource(custom, ...names) {
|
|
const resourcesDir = this.info.buildResourcesDir;
|
|
if (custom === undefined) {
|
|
const resourceList = await this.resourceList;
|
|
for (const name of names) {
|
|
if (resourceList.includes(name)) {
|
|
return path.join(resourcesDir, name);
|
|
}
|
|
}
|
|
}
|
|
else if (custom != null && !(0, builder_util_1.isEmptyOrSpaces)(custom)) {
|
|
const resourceList = await this.resourceList;
|
|
if (resourceList.includes(custom)) {
|
|
return path.join(resourcesDir, custom);
|
|
}
|
|
let p = path.resolve(resourcesDir, custom);
|
|
if ((await (0, builder_util_1.statOrNull)(p)) == null) {
|
|
p = path.resolve(this.projectDir, custom);
|
|
if ((await (0, builder_util_1.statOrNull)(p)) == null) {
|
|
throw new builder_util_1.InvalidConfigurationError(`cannot find specified resource "${custom}", nor relative to "${resourcesDir}", neither relative to project dir ("${this.projectDir}")`);
|
|
}
|
|
}
|
|
return p;
|
|
}
|
|
return null;
|
|
}
|
|
get forceCodeSigning() {
|
|
const forceCodeSigningPlatform = this.platformSpecificBuildOptions.forceCodeSigning;
|
|
return (forceCodeSigningPlatform == null ? this.config.forceCodeSigning : forceCodeSigningPlatform) || false;
|
|
}
|
|
async getOrConvertIcon(format) {
|
|
const result = await this.resolveIcon((0, builder_util_1.asArray)(this.platformSpecificBuildOptions.icon || this.config.icon), [], format);
|
|
if (result.length === 0) {
|
|
const framework = this.info.framework;
|
|
if (framework.getDefaultIcon != null) {
|
|
return framework.getDefaultIcon(this.platform);
|
|
}
|
|
builder_util_1.log.warn({ reason: "application icon is not set" }, `default ${capitalizeFirstLetter(framework.name)} icon is used`);
|
|
return this.getDefaultFrameworkIcon();
|
|
}
|
|
else {
|
|
return result[0].file;
|
|
}
|
|
}
|
|
getDefaultFrameworkIcon() {
|
|
const framework = this.info.framework;
|
|
return framework.getDefaultIcon == null ? null : framework.getDefaultIcon(this.platform);
|
|
}
|
|
// convert if need, validate size (it is a reason why tool is called even if file has target extension (already specified as foo.icns for example))
|
|
async resolveIcon(sources, fallbackSources, outputFormat) {
|
|
const output = this.expandMacro(this.config.directories.output);
|
|
const args = [
|
|
"icon",
|
|
"--format",
|
|
outputFormat,
|
|
"--root",
|
|
this.buildResourcesDir,
|
|
"--root",
|
|
this.projectDir,
|
|
"--out",
|
|
path.resolve(this.projectDir, output, `.icon-${outputFormat}`),
|
|
];
|
|
for (const source of sources) {
|
|
args.push("--input", source);
|
|
}
|
|
for (const source of fallbackSources) {
|
|
args.push("--fallback-input", source);
|
|
}
|
|
const result = await (0, appBuilder_1.executeAppBuilderAsJson)(args);
|
|
const errorMessage = result.error;
|
|
if (errorMessage != null) {
|
|
throw new builder_util_1.InvalidConfigurationError(errorMessage, result.errorCode);
|
|
}
|
|
if (result.isFallback) {
|
|
builder_util_1.log.warn({ reason: "application icon is not set" }, `default ${capitalizeFirstLetter(this.info.framework.name)} icon is used`);
|
|
}
|
|
return result.icons || [];
|
|
}
|
|
}
|
|
exports.PlatformPackager = PlatformPackager;
|
|
function isSafeGithubName(name) {
|
|
return /^[0-9A-Za-z._-]+$/.test(name);
|
|
}
|
|
function computeSafeArtifactNameIfNeeded(suggestedName, safeNameProducer) {
|
|
// GitHub only allows the listed characters in file names.
|
|
if (suggestedName != null) {
|
|
if (isSafeGithubName(suggestedName)) {
|
|
return null;
|
|
}
|
|
// prefer to use suggested name - so, if space is the only problem, just replace only space to dash
|
|
suggestedName = suggestedName.replace(/ /g, "-");
|
|
if (isSafeGithubName(suggestedName)) {
|
|
return suggestedName;
|
|
}
|
|
}
|
|
return safeNameProducer();
|
|
}
|
|
// remove leading dot
|
|
function normalizeExt(ext) {
|
|
return ext.startsWith(".") ? ext.substring(1) : ext;
|
|
}
|
|
function chooseNotNull(v1, v2) {
|
|
return v1 == null ? v2 : v1;
|
|
}
|
|
function capitalizeFirstLetter(text) {
|
|
return text.charAt(0).toUpperCase() + text.slice(1);
|
|
}
|
|
//# sourceMappingURL=platformPackager.js.map
|