// Copyright 2019 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { pathToFileURL } from 'url'; import { autoUpdater } from 'electron'; import { writeFile } from 'fs/promises'; import { join } from 'path'; import { Updater, createTempDir, deleteTempDir } from './common'; import { explodePromise } from '../util/explodePromise'; import * as Errors from '../types/errors'; import { DialogType } from '../types/Dialogs'; export class MacOSUpdater extends Updater { protected async deletePreviousInstallers(): Promise { // No installers are cache on macOS } protected async installUpdate( updateFilePath: string ): Promise<() => Promise> { const { logger } = this; logger.info('downloadAndInstall: handing download to electron...'); try { await this.handToAutoUpdate(updateFilePath); } catch (error) { const readOnly = 'Cannot update while running on a read-only volume'; const message: string = error.message || ''; this.markCannotUpdate( error, message.includes(readOnly) ? DialogType.MacOS_Read_Only : DialogType.Cannot_Update ); throw error; } // At this point, closing the app will cause the update to be installed automatically // because Squirrel has cached the update file and will do the right thing. return async () => { logger.info('downloadAndInstall: restarting...'); this.markRestarting(); autoUpdater.quitAndInstall(); }; } private async handToAutoUpdate(filePath: string): Promise { const { logger } = this; const { promise, resolve, reject } = explodePromise(); const onError = (...args: Array) => { logger.error('autoUpdater: error', ...args.map(Errors.toLogFormat)); const [error] = args; cleanup(); reject(error); }; const onDownloaded = () => { logger.info('autoUpdater: update-downloaded event fired'); cleanup(); resolve(); }; function cleanup() { autoUpdater.removeListener('error', onError); autoUpdater.removeListener('update-downloaded', onDownloaded); } autoUpdater.on('error', onError); autoUpdater.on('update-downloaded', onDownloaded); // See: https://github.com/electron/electron/issues/5020#issuecomment-477636990 const updateUrl = pathToFileURL(filePath).href; const tempDir = await createTempDir(); try { const feedPath = join(tempDir, 'feed.json'); await writeFile( feedPath, JSON.stringify({ url: updateUrl, }) ); autoUpdater.setFeedURL({ url: pathToFileURL(feedPath).href, serverType: 'json', }); autoUpdater.checkForUpdates(); await promise; } finally { await deleteTempDir(this.logger, tempDir); } } }