UNPKG

electron-builder-squirrel-windows

Version:

Plugin for [electron-builder](https://github.com/electron-userland/electron-builder) to build Squirrel.Windows installer.

235 lines 12.7 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const builder_util_1 = require("builder-util"); const wine_1 = require("app-builder-lib/out/wine"); const binDownload_1 = require("app-builder-lib/out/binDownload"); const filename_1 = require("builder-util/out/filename"); const app_builder_lib_1 = require("app-builder-lib"); const path = require("path"); const fs = require("fs"); const os = require("os"); const electron_winstaller_1 = require("electron-winstaller"); class SquirrelWindowsTarget extends app_builder_lib_1.Target { constructor(packager, outDir) { super("squirrel"); this.packager = packager; this.outDir = outDir; //tslint:disable-next-line:no-object-literal-type-assertion this.options = { ...this.packager.platformSpecificBuildOptions, ...this.packager.config.squirrelWindows }; this.isAsyncSupported = false; } async prepareSignedVendorDirectory() { const customSquirrelVendorDirectory = this.options.customSquirrelVendorDir; const tmpVendorDirectory = await this.packager.info.tempDirManager.createTempDir({ prefix: "squirrel-windows-vendor" }); if ((0, builder_util_1.isEmptyOrSpaces)(customSquirrelVendorDirectory) || !fs.existsSync(customSquirrelVendorDirectory)) { builder_util_1.log.warn({ customSquirrelVendorDirectory: customSquirrelVendorDirectory }, "unable to access custom Squirrel.Windows vendor directory, falling back to default vendor "); const windowInstallerPackage = require.resolve("electron-winstaller/package.json"); const vendorDirectory = path.join(path.dirname(windowInstallerPackage), "vendor"); const squirrelBin = await (0, binDownload_1.getBinFromUrl)("squirrel.windows@1.0.0", "squirrel.windows-2.0.1-patched.7z", "DWijIRRElidu/Rq0yegAKqo2g6aVJUPvcRyvkzUoBPbRasIk61P6xY2fBMdXw6wT17md7NzrTI9/zA1wT9vEqg=="); await fs.promises.cp(vendorDirectory, tmpVendorDirectory, { recursive: true }); // copy the patched squirrel to tmp vendor directory await fs.promises.cp(path.join(squirrelBin, "electron-winstaller", "vendor"), tmpVendorDirectory, { recursive: true }); } else { // copy the custom squirrel vendor directory to tmp vendor directory await fs.promises.cp(customSquirrelVendorDirectory, tmpVendorDirectory, { recursive: true }); } const files = await fs.promises.readdir(tmpVendorDirectory); const squirrelExe = files.find(f => f === "Squirrel.exe"); if (squirrelExe) { const filePath = path.join(tmpVendorDirectory, squirrelExe); builder_util_1.log.debug({ file: filePath }, "signing vendor executable"); await this.packager.signIf(filePath); } else { builder_util_1.log.warn("Squirrel.exe not found in vendor directory, skipping signing"); } return tmpVendorDirectory; } async generateStubExecutableExe(appOutDir, vendorDir) { const files = await fs.promises.readdir(appOutDir, { withFileTypes: true }); const appExe = files.find(f => f.name === `${this.exeName}.exe`); if (!appExe) { throw new Error(`App executable not found in app directory: ${appOutDir}`); } const filePath = path.join(appOutDir, appExe.name); const stubExePath = path.join(appOutDir, `${this.exeName}_ExecutionStub.exe`); await fs.promises.copyFile(path.join(vendorDir, "StubExecutable.exe"), stubExePath); await (0, wine_1.execWine)(path.join(vendorDir, "WriteZipToSetup.exe"), null, ["--copy-stub-resources", filePath, stubExePath]); await this.packager.signIf(stubExePath); builder_util_1.log.debug({ file: filePath }, "signing app executable"); await this.packager.signIf(filePath); } async build(appOutDir, arch) { const packager = this.packager; const version = packager.appInfo.version; const sanitizedName = (0, filename_1.sanitizeFileName)(this.appName); const setupFile = packager.expandArtifactNamePattern(this.options, "exe", arch, "${productName} Setup ${version}.${ext}"); const installerOutDir = path.join(this.outDir, `squirrel-windows${(0, app_builder_lib_1.getArchSuffix)(arch)}`); const artifactPath = path.join(installerOutDir, setupFile); const msiArtifactPath = path.join(installerOutDir, packager.expandArtifactNamePattern(this.options, "msi", arch, "${productName} Setup ${version}.${ext}")); this.buildQueueManager.add(async () => { await packager.info.emitArtifactBuildStarted({ targetPresentableName: "Squirrel.Windows", file: artifactPath, arch, }); const distOptions = await this.computeEffectiveDistOptions(appOutDir, installerOutDir, setupFile); await this.generateStubExecutableExe(appOutDir, distOptions.vendorDirectory); await (0, electron_winstaller_1.createWindowsInstaller)(distOptions); await packager.signAndEditResources(artifactPath, arch, installerOutDir); if (this.options.msi) { await packager.signIf(msiArtifactPath); } const safeArtifactName = (ext) => `${sanitizedName}-Setup-${version}${(0, app_builder_lib_1.getArchSuffix)(arch)}.${ext}`; await packager.info.emitArtifactBuildCompleted({ file: artifactPath, target: this, arch, safeArtifactName: safeArtifactName("exe"), packager: this.packager, }); if (this.options.msi) { await packager.info.emitArtifactCreated({ file: msiArtifactPath, target: this, arch, safeArtifactName: safeArtifactName("msi"), packager: this.packager, }); } const packagePrefix = `${this.appName}-${(0, electron_winstaller_1.convertVersion)(version)}-`; await packager.info.emitArtifactCreated({ file: path.join(installerOutDir, `${packagePrefix}full.nupkg`), target: this, arch, packager, }); if (distOptions.remoteReleases != null) { await packager.info.emitArtifactCreated({ file: path.join(installerOutDir, `${packagePrefix}delta.nupkg`), target: this, arch, packager, }); } await packager.info.emitArtifactCreated({ file: path.join(installerOutDir, "RELEASES"), target: this, arch, packager, }); }); return Promise.resolve(); } get appName() { return this.options.name || this.packager.appInfo.name; } get exeName() { return this.packager.appInfo.productFilename || this.options.name || this.packager.appInfo.productName; } select7zipArch(vendorDirectory) { // https://github.com/electron/windows-installer/blob/main/script/select-7z-arch.js // Even if we're cross-compiling for a different arch like arm64, // we still need to use the 7-Zip executable for the host arch const resolvedArch = os.arch; fs.copyFileSync(path.join(vendorDirectory, `7z-${resolvedArch}.exe`), path.join(vendorDirectory, "7z.exe")); fs.copyFileSync(path.join(vendorDirectory, `7z-${resolvedArch}.dll`), path.join(vendorDirectory, "7z.dll")); } async createNuspecTemplateWithProjectUrl() { const templatePath = path.resolve(__dirname, "..", "template.nuspectemplate"); const projectUrl = await this.packager.appInfo.computePackageUrl(); if (projectUrl != null) { const nuspecTemplate = await this.packager.info.tempDirManager.getTempFile({ prefix: "template", suffix: ".nuspectemplate" }); let templateContent = await fs.promises.readFile(templatePath, "utf8"); const searchString = "<copyright><%- copyright %></copyright>"; templateContent = templateContent.replace(searchString, `${searchString}\n <projectUrl>${projectUrl}</projectUrl>`); await fs.promises.writeFile(nuspecTemplate, templateContent); return nuspecTemplate; } return templatePath; } async computeEffectiveDistOptions(appDirectory, outputDirectory, setupFile) { const packager = this.packager; let iconUrl = this.options.iconUrl; if (iconUrl == null) { const info = await packager.info.repositoryInfo; if (info != null) { iconUrl = `https://github.com/${info.user}/${info.project}/blob/master/${packager.info.relativeBuildResourcesDirname}/icon.ico?raw=true`; } if (iconUrl == null) { throw new builder_util_1.InvalidConfigurationError("squirrelWindows.iconUrl is not specified, please see https://www.electron.build/squirrel-windows#SquirrelWindowsOptions-iconUrl"); } } checkConflictingOptions(this.options); const appInfo = packager.appInfo; const options = { appDirectory: appDirectory, outputDirectory: outputDirectory, name: this.options.useAppIdAsId ? appInfo.id : this.appName, title: appInfo.productName || appInfo.name, version: appInfo.version, description: appInfo.description, exe: `${appInfo.productFilename || this.options.name || appInfo.productName}.exe`, authors: appInfo.companyName || "", nuspecTemplate: await this.createNuspecTemplateWithProjectUrl(), iconUrl, copyright: appInfo.copyright, noMsi: !this.options.msi, usePackageJson: false, }; options.vendorDirectory = await this.prepareSignedVendorDirectory(); this.select7zipArch(options.vendorDirectory); options.fixUpPaths = true; options.setupExe = setupFile; if (this.options.msi) { options.setupMsi = setupFile.replace(".exe", ".msi"); } if ((0, builder_util_1.isEmptyOrSpaces)(options.description)) { options.description = this.options.name || appInfo.productName; } if (options.remoteToken == null) { options.remoteToken = process.env.GH_TOKEN || process.env.GITHUB_TOKEN; } if (this.options.remoteReleases === true) { const info = await packager.info.repositoryInfo; if (info == null) { builder_util_1.log.warn("remoteReleases set to true, but cannot get repository info"); } else { options.remoteReleases = `https://github.com/${info.user}/${info.project}`; builder_util_1.log.info({ remoteReleases: options.remoteReleases }, `remoteReleases is set`); } } else if (typeof this.options.remoteReleases === "string" && !(0, builder_util_1.isEmptyOrSpaces)(this.options.remoteReleases)) { options.remoteReleases = this.options.remoteReleases; } if (this.options.loadingGif) { options.loadingGif = path.resolve(packager.projectDir, this.options.loadingGif); } else { const resourceList = await packager.resourceList; if (resourceList.includes("install-spinner.gif")) { options.loadingGif = path.join(packager.buildResourcesDir, "install-spinner.gif"); } } return options; } } exports.default = SquirrelWindowsTarget; function checkConflictingOptions(options) { for (const name of ["outputDirectory", "appDirectory", "exe", "fixUpPaths", "usePackageJson", "extraFileSpecs", "extraMetadataSpecs", "skipUpdateIcon", "setupExe"]) { if (name in options) { throw new builder_util_1.InvalidConfigurationError(`Option ${name} is ignored, do not specify it.`); } } if ("noMsi" in options) { builder_util_1.log.warn(`noMsi is deprecated, please specify as "msi": true if you want to create an MSI installer`); options.msi = !options.noMsi; } const msi = options.msi; if (msi != null && typeof msi !== "boolean") { throw new builder_util_1.InvalidConfigurationError(`msi expected to be boolean value, but string '"${msi}"' was specified`); } } //# sourceMappingURL=SquirrelWindowsTarget.js.map