UNPKG

electron-updater

Version:
169 lines 6.91 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.DownloadedUpdateHelper = void 0; exports.createTempUpdateFile = createTempUpdateFile; const crypto_1 = require("crypto"); const fs_1 = require("fs"); // @ts-ignore const isEqual = require("lodash.isequal"); const fs_extra_1 = require("fs-extra"); const path = require("path"); /** @private **/ class DownloadedUpdateHelper { constructor(cacheDir) { this.cacheDir = cacheDir; this._file = null; this._packageFile = null; this.versionInfo = null; this.fileInfo = null; this._downloadedFileInfo = null; } get downloadedFileInfo() { return this._downloadedFileInfo; } get file() { return this._file; } get packageFile() { return this._packageFile; } get cacheDirForPendingUpdate() { return path.join(this.cacheDir, "pending"); } async validateDownloadedPath(updateFile, updateInfo, fileInfo, logger) { if (this.versionInfo != null && this.file === updateFile && this.fileInfo != null) { // update has already been downloaded from this running instance // check here only existence, not checksum if (isEqual(this.versionInfo, updateInfo) && isEqual(this.fileInfo.info, fileInfo.info) && (await (0, fs_extra_1.pathExists)(updateFile))) { return updateFile; } else { return null; } } // update has already been downloaded from some previous app launch const cachedUpdateFile = await this.getValidCachedUpdateFile(fileInfo, logger); if (cachedUpdateFile === null) { return null; } logger.info(`Update has already been downloaded to ${updateFile}).`); this._file = cachedUpdateFile; return cachedUpdateFile; } async setDownloadedFile(downloadedFile, packageFile, versionInfo, fileInfo, updateFileName, isSaveCache) { this._file = downloadedFile; this._packageFile = packageFile; this.versionInfo = versionInfo; this.fileInfo = fileInfo; this._downloadedFileInfo = { fileName: updateFileName, sha512: fileInfo.info.sha512, isAdminRightsRequired: fileInfo.info.isAdminRightsRequired === true, }; if (isSaveCache) { await (0, fs_extra_1.outputJson)(this.getUpdateInfoFile(), this._downloadedFileInfo); } } async clear() { this._file = null; this._packageFile = null; this.versionInfo = null; this.fileInfo = null; await this.cleanCacheDirForPendingUpdate(); } async cleanCacheDirForPendingUpdate() { try { // remove stale data await (0, fs_extra_1.emptyDir)(this.cacheDirForPendingUpdate); } catch (_ignore) { // ignore } } /** * Returns "update-info.json" which is created in the update cache directory's "pending" subfolder after the first update is downloaded. If the update file does not exist then the cache is cleared and recreated. If the update file exists then its properties are validated. * @param fileInfo * @param logger */ async getValidCachedUpdateFile(fileInfo, logger) { const updateInfoFilePath = this.getUpdateInfoFile(); const doesUpdateInfoFileExist = await (0, fs_extra_1.pathExists)(updateInfoFilePath); if (!doesUpdateInfoFileExist) { return null; } let cachedInfo; try { cachedInfo = await (0, fs_extra_1.readJson)(updateInfoFilePath); } catch (error) { let message = `No cached update info available`; if (error.code !== "ENOENT") { await this.cleanCacheDirForPendingUpdate(); message += ` (error on read: ${error.message})`; } logger.info(message); return null; } const isCachedInfoFileNameValid = (cachedInfo === null || cachedInfo === void 0 ? void 0 : cachedInfo.fileName) !== null; if (!isCachedInfoFileNameValid) { logger.warn(`Cached update info is corrupted: no fileName, directory for cached update will be cleaned`); await this.cleanCacheDirForPendingUpdate(); return null; } if (fileInfo.info.sha512 !== cachedInfo.sha512) { logger.info(`Cached update sha512 checksum doesn't match the latest available update. New update must be downloaded. Cached: ${cachedInfo.sha512}, expected: ${fileInfo.info.sha512}. Directory for cached update will be cleaned`); await this.cleanCacheDirForPendingUpdate(); return null; } const updateFile = path.join(this.cacheDirForPendingUpdate, cachedInfo.fileName); if (!(await (0, fs_extra_1.pathExists)(updateFile))) { logger.info("Cached update file doesn't exist"); return null; } const sha512 = await hashFile(updateFile); if (fileInfo.info.sha512 !== sha512) { logger.warn(`Sha512 checksum doesn't match the latest available update. New update must be downloaded. Cached: ${sha512}, expected: ${fileInfo.info.sha512}`); await this.cleanCacheDirForPendingUpdate(); return null; } this._downloadedFileInfo = cachedInfo; return updateFile; } getUpdateInfoFile() { return path.join(this.cacheDirForPendingUpdate, "update-info.json"); } } exports.DownloadedUpdateHelper = DownloadedUpdateHelper; function hashFile(file, algorithm = "sha512", encoding = "base64", options) { return new Promise((resolve, reject) => { const hash = (0, crypto_1.createHash)(algorithm); hash.on("error", reject).setEncoding(encoding); (0, fs_1.createReadStream)(file, { ...options, highWaterMark: 1024 * 1024 /* better to use more memory but hash faster */ }) .on("error", reject) .on("end", () => { hash.end(); resolve(hash.read()); }) .pipe(hash, { end: false }); }); } async function createTempUpdateFile(name, cacheDir, log) { // https://github.com/electron-userland/electron-builder/pull/2474#issuecomment-366481912 let nameCounter = 0; let result = path.join(cacheDir, name); for (let i = 0; i < 3; i++) { try { await (0, fs_extra_1.unlink)(result); return result; } catch (e) { if (e.code === "ENOENT") { return result; } log.warn(`Error on remove temp update file: ${e}`); result = path.join(cacheDir, `${nameCounter++}-${name}`); } } return result; } //# sourceMappingURL=DownloadedUpdateHelper.js.map