/* eslint-disable no-console */ import { app, BrowserWindow, dialog, ipcMain as ipc, Menu, protocol as electronProtocol, screen, shell, systemPreferences, } from 'electron'; import path, { join } from 'path'; import { platform as osPlatform } from 'process'; import url from 'url'; import os from 'os'; import fs from 'fs'; import crypto from 'crypto'; import _ from 'lodash'; import pify from 'pify'; import Logger from 'bunyan'; import { setup as setupSpellChecker } from '../node/spell_check'; // checked - only node import { setupGlobalErrorHandler } from '../node/global_errors'; // checked - only node import packageJson from '../../package.json'; // checked - only node setupGlobalErrorHandler(); import electronLocalshortcut from 'electron-localshortcut'; // tslint:disable: no-console const getRealPath = pify(fs.realpath); // Hardcoding appId to prevent build failures on release. // const appUserModelId = packageJson.build.appId; const appUserModelId = 'com.loki-project.messenger-desktop'; console.log('Set Windows Application User Model ID (AUMID)', { appUserModelId, }); app.setAppUserModelId(appUserModelId); // Keep a global reference of the window object, if you don't, the window will // be closed automatically when the JavaScript object is garbage collected. let mainWindow: BrowserWindow | null = null; function getMainWindow() { return mainWindow; } let readyForShutdown: boolean = false; // Tray icon and related objects let tray: any = null; import { config } from '../node/config'; // checked - only node // Very important to put before the single instance check, since it is based on the // userData directory. import { userConfig } from '../node/config/user_config'; // checked - only node import * as PasswordUtil from '../util/passwordUtils'; // checked - only node const development = (config as any).environment === 'development'; const appInstance = config.util.getEnv('NODE_APP_INSTANCE') || 0; // We generally want to pull in our own modules after this point, after the user // data directory has been set. import { initAttachmentsChannel } from '../node/attachment_channel'; import * as updater from '../updater/index'; // checked - only node import { createTrayIcon } from '../node/tray_icon'; // checked - only node import { ephemeralConfig } from '../node/config/ephemeral_config'; // checked - only node import { getLogger, initializeLogger } from '../node/logging'; // checked - only node import { sqlNode } from '../node/sql'; // checked - only node import * as sqlChannels from '../node/sql_channel'; // checked - only node import { windowMarkShouldQuit, windowShouldQuit } from '../node/window_state'; // checked - only node import { createTemplate } from '../node/menu'; // checked - only node import { installFileHandler, installWebHandler } from '../node/protocol_filter'; // checked - only node import { installPermissionsHandler } from '../node/permissions'; // checked - only node let appStartInitialSpellcheckSetting = true; const enableTestIntegrationWiderWindow = true; const isTestIntegration = enableTestIntegrationWiderWindow && Boolean( process.env.NODE_APP_INSTANCE && process.env.NODE_APP_INSTANCE.includes('test-integration') ); async function getSpellCheckSetting() { const json = sqlNode.getItemById('spell-check'); // Default to `true` if setting doesn't exist yet if (!json) { return true; } return json.value; } function showWindow() { if (!mainWindow) { return; } // Using focus() instead of show() seems to be important on Windows when our window // has been docked using Aero Snap/Snap Assist. A full .show() call here will cause // the window to reposition: // https://github.com/signalapp/Signal-Desktop/issues/1429 if (mainWindow.isVisible()) { mainWindow.focus(); } else { mainWindow.show(); } // toggle the visibility of the show/hide tray icon menu entries if (tray) { tray.updateContextMenu(); } } if (!process.mas) { console.log('making app single instance'); const gotLock = app.requestSingleInstanceLock(); if (!gotLock) { // Don't allow second instance if we are in prod if (appInstance === 0) { console.log('quitting; we are the second instance'); app.exit(); } } else { app.on('second-instance', () => { // Someone tried to run a second instance, we should focus our window if (mainWindow) { if (mainWindow.isMinimized()) { mainWindow.restore(); } showWindow(); } return true; }); } } const windowFromUserConfig = userConfig.get('window'); const windowFromEphemeral = ephemeralConfig.get('window'); let windowConfig = windowFromEphemeral || windowFromUserConfig; if (windowFromUserConfig) { userConfig.set('window', null); ephemeralConfig.set('window', windowConfig); } // import {load as loadLocale} from '../..' import { load as loadLocale, LocaleMessagesWithNameType } from '../node/locale'; import { setLastestRelease } from '../node/latest_desktop_release'; import { getAppRootPath } from '../node/getRootPath'; import { classicDark } from '../themes'; // Both of these will be set after app fires the 'ready' event let logger: Logger | null = null; let locale: LocaleMessagesWithNameType; function assertLogger(): Logger { if (!logger) { throw new Error('assertLogger: logger is not set'); } return logger; } function prepareURL(pathSegments: Array, moreKeys?: { theme: any }) { const urlObject: url.UrlObject = { pathname: join(...pathSegments), protocol: 'file:', slashes: true, query: { name: packageJson.productName, locale: locale.name, version: app.getVersion(), commitHash: config.get('commitHash'), environment: (config as any).environment, node_version: process.versions.node, hostname: os.hostname(), appInstance: process.env.NODE_APP_INSTANCE, proxyUrl: process.env.HTTPS_PROXY || process.env.https_proxy, appStartInitialSpellcheckSetting, ...moreKeys, }, }; return url.format(urlObject); } function handleUrl(event: any, target: string) { event.preventDefault(); const { protocol } = url.parse(target); // tslint:disable-next-line: no-http-string if (protocol === 'http:' || protocol === 'https:') { void shell.openExternal(target); } } function captureClicks(window: BrowserWindow) { window.webContents.on('will-navigate', handleUrl); window.webContents.on('new-window', handleUrl); } function getDefaultWindowSize() { return { defaultWidth: isTestIntegration ? 1500 : 880, defaultHeight: 820, minWidth: 880, minHeight: 600, }; } function getWindowSize() { const screenSize = screen.getPrimaryDisplay().workAreaSize; const { minWidth, minHeight, defaultWidth, defaultHeight } = getDefaultWindowSize(); // Ensure that the screen can fit within the default size const width = Math.min(defaultWidth, Math.max(minWidth, screenSize.width)); const height = Math.min(defaultHeight, Math.max(minHeight, screenSize.height)); return { width, height, minWidth, minHeight }; } function isVisible(window: { x: number; y: number; width: number }, bounds: any) { const boundsX = _.get(bounds, 'x') || 0; const boundsY = _.get(bounds, 'y') || 0; const boundsWidth = _.get(bounds, 'width') || getDefaultWindowSize().defaultWidth; const boundsHeight = _.get(bounds, 'height') || getDefaultWindowSize().defaultHeight; const BOUNDS_BUFFER = 100; // requiring BOUNDS_BUFFER pixels on the left or right side // tslint:disable: restrict-plus-operands const rightSideClearOfLeftBound = window.x + window.width >= boundsX + BOUNDS_BUFFER; const leftSideClearOfRightBound = window.x <= boundsX + boundsWidth - BOUNDS_BUFFER; // top can't be offscreen, and must show at least BOUNDS_BUFFER pixels at bottom const topClearOfUpperBound = window.y >= boundsY; const topClearOfLowerBound = window.y <= boundsY + boundsHeight - BOUNDS_BUFFER; return ( rightSideClearOfLeftBound && leftSideClearOfRightBound && topClearOfUpperBound && topClearOfLowerBound ); } function getStartInTray() { const startInTray = process.argv.some(arg => arg === '--start-in-tray') || userConfig.get('startInTray'); const usingTrayIcon = startInTray || process.argv.some(arg => arg === '--use-tray-icon'); return { usingTrayIcon, startInTray }; } // tslint:disable-next-line: max-func-body-length async function createWindow() { const { minWidth, minHeight, width, height } = getWindowSize(); windowConfig = windowConfig || {}; const picked = { maximized: (windowConfig as any).maximized || false, autoHideMenuBar: (windowConfig as any).autoHideMenuBar || false, width: (windowConfig as any).width || width, height: (windowConfig as any).height || height, x: (windowConfig as any).x, y: (windowConfig as any).y, }; if (isTestIntegration) { const screenWidth = screen.getPrimaryDisplay().workAreaSize.width - getDefaultWindowSize().defaultWidth; const screenHeight = screen.getPrimaryDisplay().workAreaSize.height - getDefaultWindowSize().defaultHeight; // tslint:disable: insecure-random picked.x = Math.floor(Math.random() * screenWidth); picked.y = Math.floor(Math.random() * screenHeight); } const windowOptions = { show: true, minWidth, minHeight, fullscreen: false as boolean | undefined, // Default theme is Classic Dark backgroundColor: classicDark['--background-primary-color'], webPreferences: { nodeIntegration: true, enableRemoteModule: true, nodeIntegrationInWorker: true, contextIsolation: false, preload: path.join(getAppRootPath(), 'preload.js'), nativeWindowOpen: true, spellcheck: await getSpellCheckSetting(), }, // only set icon for Linux, the executable one will be used by default for other platforms icon: (osPlatform === 'linux' && path.join(getAppRootPath(), 'images/session/session_icon.png')) || undefined, ...picked, }; if (!_.isNumber(windowOptions.width) || windowOptions.width < minWidth) { windowOptions.width = Math.max(minWidth, width); } if (!_.isNumber(windowOptions.height) || windowOptions.height < minHeight) { windowOptions.height = Math.max(minHeight, height); } if (!_.isBoolean(windowOptions.maximized)) { delete windowOptions.maximized; } if (!_.isBoolean(windowOptions.autoHideMenuBar)) { delete windowOptions.autoHideMenuBar; } const visibleOnAnyScreen = _.some(screen.getAllDisplays(), display => { if (!_.isNumber(windowOptions.x) || !_.isNumber(windowOptions.y)) { return false; } return isVisible(windowOptions, _.get(display, 'bounds')); }); if (!visibleOnAnyScreen) { console.log('Location reset needed'); delete windowOptions.x; delete windowOptions.y; } if (windowOptions.fullscreen === false) { delete windowOptions.fullscreen; } assertLogger().info('Initializing BrowserWindow config: %s', JSON.stringify(windowOptions)); // Create the browser window. mainWindow = new BrowserWindow(windowOptions); setupSpellChecker(mainWindow, locale.messages); const setWindowFocus = () => { if (!mainWindow) { return; } mainWindow.webContents.send('set-window-focus', mainWindow.isFocused()); }; mainWindow.on('focus', setWindowFocus); mainWindow.on('blur', setWindowFocus); mainWindow.once('ready-to-show', setWindowFocus); // This is a fallback in case we drop an event for some reason. global.setInterval(setWindowFocus, 5000); electronLocalshortcut.register(mainWindow, 'F5', () => { if (!mainWindow) { return; } mainWindow.reload(); }); electronLocalshortcut.register(mainWindow, 'CommandOrControl+R', () => { if (!mainWindow) { return; } mainWindow.reload(); }); function captureAndSaveWindowStats() { if (!mainWindow) { return; } const size = mainWindow.getSize(); const position = mainWindow.getPosition(); // so if we need to recreate the window, we have the most recent settings windowConfig = { maximized: mainWindow.isMaximized(), autoHideMenuBar: mainWindow.isMenuBarAutoHide(), width: size[0], height: size[1], x: position[0], y: position[1], fullscreen: false as boolean | undefined, }; if (mainWindow.isFullScreen()) { // Only include this property if true, because when explicitly set to // false the fullscreen button will be disabled on osx (windowConfig as any).fullscreen = true; } assertLogger().info('Updating BrowserWindow config: %s', JSON.stringify(windowConfig)); ephemeralConfig.set('window', windowConfig); } const debouncedCaptureStats = _.debounce(captureAndSaveWindowStats, 500); mainWindow.on('resize', debouncedCaptureStats); mainWindow.on('move', debouncedCaptureStats); mainWindow.on('focus', () => { if (!mainWindow) { return; } mainWindow.flashFrame(false); if (passwordWindow) { passwordWindow.close(); passwordWindow = null; } }); const urlToLoad = prepareURL([getAppRootPath(), 'background.html']); await mainWindow.loadURL(urlToLoad); if (isTestIntegration) { setTimeout(() => { if (mainWindow && mainWindow.webContents) { mainWindow.webContents.openDevTools({ mode: 'right', activate: false, }); } }, 5000); } if ((process.env.NODE_APP_INSTANCE || '').startsWith('devprod')) { // Open the DevTools. mainWindow.webContents.openDevTools({ mode: 'bottom', activate: false, }); } captureClicks(mainWindow); // Emitted when the window is about to be closed. // Note: We do most of our shutdown logic here because all windows are closed by // Electron before the app quits. mainWindow.on('close', async e => { console.log('close event', { readyForShutdown: mainWindow ? readyForShutdown : null, shouldQuit: windowShouldQuit(), }); // If the application is terminating, just do the default if (mainWindow && readyForShutdown && windowShouldQuit()) { return; } // Prevent the shutdown e.preventDefault(); mainWindow?.hide(); // On Mac, or on other platforms when the tray icon is in use, the window // should be only hidden, not closed, when the user clicks the close button if (!windowShouldQuit() && (getStartInTray().usingTrayIcon || process.platform === 'darwin')) { // toggle the visibility of the show/hide tray icon menu entries if (tray) { tray.updateContextMenu(); } return; } await requestShutdown(); if (mainWindow) { readyForShutdown = true; } app.quit(); }); // Emitted when the window is closed. mainWindow.on('closed', () => { // Dereference the window object, usually you would store windows // in an array if your app supports multi windows, this is the time // when you should delete the corresponding element. mainWindow = null; }); } ipc.on('show-window', () => { showWindow(); }); ipc.on('set-release-from-file-server', (_event, releaseGotFromFileServer) => { setLastestRelease(releaseGotFromFileServer); }); let isReadyForUpdates = false; async function readyForUpdates() { console.log('[updater] isReadyForUpdates', isReadyForUpdates); if (isReadyForUpdates) { return; } isReadyForUpdates = true; // Second, start checking for app updates try { // if the user disabled auto updates, this will actually not start the updater await updater.start(getMainWindow, userConfig, locale.messages, logger); } catch (error) { const log = logger || console; log.error('Error starting update checks:', error && error.stack ? error.stack : error); } } ipc.once('ready-for-updates', readyForUpdates); // Forcefully call readyForUpdates after 10 minutes. // This ensures we start the updater. const TEN_MINUTES = 10 * 60 * 1000; setTimeout(readyForUpdates, TEN_MINUTES); function openReleaseNotes() { void shell.openExternal( `https://github.com/oxen-io/session-desktop/releases/tag/v${app.getVersion()}` ); } function openSupportPage() { void shell.openExternal('https://docs.oxen.io/products-built-on-oxen/session'); } let passwordWindow: BrowserWindow | null = null; async function showPasswordWindow() { if (passwordWindow) { passwordWindow.show(); return; } const { minWidth, minHeight, width, height } = getWindowSize(); const windowOptions = { show: true, // allow to start minimised in tray width, height, minWidth, minHeight, autoHideMenuBar: false, // Default theme is Classic Dark backgroundColor: classicDark['--background-primary-color'], webPreferences: { nodeIntegration: true, enableRemoteModule: true, nodeIntegrationInWorker: false, contextIsolation: false, // sandbox: true, preload: path.join(getAppRootPath(), 'password_preload.js'), nativeWindowOpen: true, }, // don't setup icon, the executable one will be used by default }; passwordWindow = new BrowserWindow(windowOptions); await passwordWindow.loadURL(prepareURL([getAppRootPath(), 'password.html'])); captureClicks(passwordWindow); passwordWindow.on('close', e => { // If the application is terminating, just do the default if (windowShouldQuit()) { return; } // Prevent the shutdown e.preventDefault(); passwordWindow?.hide(); // On Mac, or on other platforms when the tray icon is in use, the window // should be only hidden, not closed, when the user clicks the close button if (!windowShouldQuit() && (getStartInTray().usingTrayIcon || process.platform === 'darwin')) { // toggle the visibility of the show/hide tray icon menu entries if (tray) { tray.updateContextMenu(); } return; } if (passwordWindow) { (passwordWindow as any).readyForShutdown = true; } // Quit the app if we don't have a main window if (!mainWindow) { app.quit(); } }); passwordWindow.on('closed', () => { passwordWindow = null; }); } let aboutWindow: BrowserWindow | null; // tslint:disable-next-line: max-func-body-length async function showAbout() { if (aboutWindow) { aboutWindow.show(); return; } if (!mainWindow) { console.info('about window needs mainwindow as parent'); return; } const theme = await getThemeFromMainWindow(); const options = { width: 500, height: 500, resizable: true, title: locale.messages.about, autoHideMenuBar: true, backgroundColor: classicDark['--background-primary-color'], show: false, webPreferences: { nodeIntegration: true, nodeIntegrationInWorker: false, contextIsolation: false, preload: path.join(getAppRootPath(), 'about_preload.js'), nativeWindowOpen: true, }, parent: mainWindow, }; aboutWindow = new BrowserWindow(options); captureClicks(aboutWindow); await aboutWindow.loadURL(prepareURL([getAppRootPath(), 'about.html'], { theme })); aboutWindow.on('closed', () => { aboutWindow = null; }); aboutWindow.once('ready-to-show', () => { aboutWindow?.setBackgroundColor(classicDark['--background-primary-color']); }); // looks like sometimes ready-to-show is not fired by electron. // the fix mentioned here does not work neither: https://github.com/electron/electron/issues/7779. // But, just showing the aboutWindow right away works correctly, so just force it to be shown when just created. // It might take half a second to render it's content though. aboutWindow?.show(); } let debugLogWindow: BrowserWindow | null = null; async function showDebugLogWindow() { if (debugLogWindow) { debugLogWindow.show(); return; } if (!mainWindow) { console.info('debug log neeeds mainwindow size to open'); return; } const theme = await getThemeFromMainWindow(); const size = mainWindow.getSize(); const options = { width: Math.max(size[0] - 100, getDefaultWindowSize().minWidth), height: Math.max(size[1] - 100, getDefaultWindowSize().minHeight), resizable: true, title: locale.messages.debugLog, autoHideMenuBar: true, backgroundColor: classicDark['--background-primary-color'], shadow: true, show: false, modal: true, webPreferences: { nodeIntegration: true, nodeIntegrationInWorker: false, contextIsolation: false, preload: path.join(getAppRootPath(), 'debug_log_preload.js'), nativeWindowOpen: true, }, parent: mainWindow, }; debugLogWindow = new BrowserWindow(options); captureClicks(debugLogWindow); await debugLogWindow.loadURL(prepareURL([getAppRootPath(), 'debug_log.html'], { theme })); debugLogWindow.on('closed', () => { debugLogWindow = null; }); debugLogWindow.once('ready-to-show', () => { debugLogWindow?.setBackgroundColor(classicDark['--background-primary-color']); }); // see above: looks like sometimes ready-to-show is not fired by electron debugLogWindow?.show(); } async function saveDebugLog(_event: any, logText: any) { const options: Electron.SaveDialogOptions = { title: 'Save debug log', defaultPath: path.join(app.getPath('desktop'), `session_debug_${Date.now()}.txt`), properties: ['createDirectory'], }; try { const result = await dialog.showSaveDialog(options); const outputPath = result.filePath; console.info(`Trying to save logs to ${outputPath}`); if (result === undefined || outputPath === undefined || outputPath === '') { throw Error("User clicked Save button but didn't create a file"); } // tslint:disable: non-literal-fs-path fs.writeFile(outputPath, logText, err => { if (err) { throw Error(`${err}`); } console.info(`Saved log - ${outputPath}`); }); } catch (err) { console.error('Error saving debug log', err); } } // This method will be called when Electron has finished // initialization and is ready to create browser windows. // Some APIs can only be used after this event occurs. let ready = false; app.on('ready', async () => { const userDataPath = await getRealPath(app.getPath('userData')); const installPath = await getRealPath(join(app.getAppPath(), '..', '..')); installFileHandler({ protocol: electronProtocol, userDataPath, installPath, isWindows: process.platform === 'win32', }); installWebHandler({ protocol: electronProtocol, }); installPermissionsHandler({ userConfig }); await initializeLogger(); logger = getLogger(); assertLogger().info('app ready'); assertLogger().info(`starting version ${packageJson.version}`); if (!locale) { const appLocale = app.getLocale() || 'en'; locale = loadLocale({ appLocale, logger }); } const key = getDefaultSQLKey(); // Try to show the main window with the default key // If that fails then show the password window const dbHasPassword = userConfig.get('dbHasPassword'); if (dbHasPassword) { await showPasswordWindow(); } else { await showMainWindow(key); } }); function getDefaultSQLKey() { let key = userConfig.get('key'); if (!key) { console.log('key/initialize: Generating new encryption key, since we did not find it on disk'); // https://www.zetetic.net/sqlcipher/sqlcipher-api/#key key = crypto.randomBytes(32).toString('hex'); userConfig.set('key', key); } return key as string; } async function removeDB() { // this don't remove attachments and stuff like that... const userDir = await getRealPath(app.getPath('userData')); sqlNode.removeDB(userDir); try { console.error('Remove DB: removing.', userDir); userConfig.remove(); ephemeralConfig.remove(); } catch (e) { console.error('Remove DB: Failed to remove configs.', e); } } async function showMainWindow(sqlKey: string, passwordAttempt = false) { const userDataPath = await getRealPath(app.getPath('userData')); await sqlNode.initializeSql({ configDir: userDataPath, key: sqlKey, messages: locale.messages, passwordAttempt, }); appStartInitialSpellcheckSetting = await getSpellCheckSetting(); sqlChannels.initializeSqlChannel(); await initAttachmentsChannel({ userDataPath, }); ready = true; await createWindow(); if (getStartInTray().usingTrayIcon) { tray = createTrayIcon(getMainWindow, locale.messages); } setupMenu(); } function setupMenu() { const { platform } = process; const menuOptions = { development, showDebugLog: showDebugLogWindow, showWindow, showAbout, openReleaseNotes, openSupportPage, platform, }; const template = createTemplate(menuOptions, locale.messages); const menu = Menu.buildFromTemplate(template); Menu.setApplicationMenu(menu); } async function requestShutdown() { if (!mainWindow || !mainWindow.webContents) { return; } console.log('requestShutdown: Requesting close of mainWindow...'); const request = new Promise((resolve, reject) => { ipc.once('now-ready-for-shutdown', (_event, error) => { console.log('requestShutdown: Response received'); if (error) { reject(error); return; } resolve(undefined); return; }); mainWindow?.webContents.send('get-ready-for-shutdown'); // We'll wait two minutes, then force the app to go down. This can happen if someone // exits the app before we've set everything up in preload() (so the browser isn't // yet listening for these events), or if there are a whole lot of stacked-up tasks. // Note: two minutes is also our timeout for SQL tasks in data.ts in the browser. setTimeout(() => { console.log('requestShutdown: Response never received; forcing shutdown.'); resolve(undefined); }, 2 * 60 * 1000); }); try { await request; } catch (error) { console.log('requestShutdown error:', error && error.stack ? error.stack : error); } } app.on('before-quit', () => { console.log('before-quit event', { readyForShutdown: mainWindow ? readyForShutdown : null, shouldQuit: windowShouldQuit(), }); if (tray) { tray.destroy(); } windowMarkShouldQuit(); }); // Quit when all windows are closed. app.on('window-all-closed', () => { // On OS X it is common for applications and their menu bar // to stay active until the user quits explicitly with Cmd + Q if (process.platform !== 'darwin') { app.quit(); } }); app.on('activate', async () => { if (!ready) { return; } // On OS X it's common to re-create a window in the app when the // dock icon is clicked and there are no other windows open. if (mainWindow) { mainWindow.show(); } else { await createWindow(); } }); // Defense in depth. We never intend to open webviews or windows. Prevent it completely. app.on('web-contents-created', (_createEvent, contents) => { contents.on('will-attach-webview', attachEvent => { attachEvent.preventDefault(); }); contents.on('new-window', newEvent => { newEvent.preventDefault(); }); }); // Ingested in preload.js via a sendSync call ipc.on('locale-data', event => { // eslint-disable-next-line no-param-reassign event.returnValue = locale.messages; }); ipc.on('draw-attention', () => { if (!mainWindow) { return; } if (process.platform === 'win32') { mainWindow.flashFrame(true); } }); ipc.on('restart', () => { app.relaunch(); app.quit(); }); ipc.on('resetDatabase', async () => { await removeDB(); app.relaunch(); app.quit(); }); ipc.on('set-auto-hide-menu-bar', (_event, autoHide) => { if (mainWindow) { mainWindow.setAutoHideMenuBar(autoHide); } }); ipc.on('set-menu-bar-visibility', (_event, visibility) => { if (mainWindow) { mainWindow.setMenuBarVisibility(visibility); } }); ipc.on('close-about', () => { if (aboutWindow) { aboutWindow.close(); } }); // Password screen related IPC calls ipc.on('password-window-login', async (event, passPhrase) => { const sendResponse = (e: string | undefined) => { event.sender.send('password-window-login-response', e); }; try { const passwordAttempt = true; await showMainWindow(passPhrase, passwordAttempt); sendResponse(undefined); } catch (e) { const localisedError = locale.messages.removePasswordInvalid; sendResponse(localisedError); } }); ipc.on('start-in-tray-on-start', (event, newValue) => { try { userConfig.set('startInTray', newValue); if (newValue) { if (!tray) { tray = createTrayIcon(getMainWindow, locale.messages); } } else { // destroy is not working for a lot of desktop env. So for simplicity, we don't destroy it here but just // show a toast to explain to the user that he needs to restart // tray.destroy(); // tray = null; } event.sender.send('start-in-tray-on-start-response', null); } catch (e) { event.sender.send('start-in-tray-on-start-response', e); } }); ipc.on('get-start-in-tray', event => { try { const val = userConfig.get('startInTray'); event.sender.send('get-start-in-tray-response', val); } catch (e) { event.sender.send('get-start-in-tray-response', false); } }); ipc.on('get-opengroup-pruning', event => { try { const val = userConfig.get('opengroupPruning'); event.sender.send('get-opengroup-pruning-response', val); } catch (e) { event.sender.send('get-opengroup-pruning-response', false); } }); ipc.on('set-opengroup-pruning', (event, newValue) => { try { userConfig.set('opengroupPruning', newValue); event.sender.send('set-opengroup-pruning-response', null); } catch (e) { event.sender.send('set-opengroup-pruning-response', e); } }); ipc.on('set-password', async (event, passPhrase, oldPhrase) => { const sendResponse = (response: string | undefined) => { event.sender.send('set-password-response', response); }; try { // Check if the hash we have stored matches the hash of the old passphrase. const hash = sqlNode.getPasswordHash(); const hashMatches = oldPhrase && PasswordUtil.matchesHash(oldPhrase, hash); if (hash && !hashMatches) { const incorrectOldPassword = locale.messages.invalidOldPassword; sendResponse( incorrectOldPassword || 'Failed to set password: Old password provided is invalid' ); return; } if (_.isEmpty(passPhrase)) { const defaultKey = getDefaultSQLKey(); sqlNode.setSQLPassword(defaultKey); sqlNode.removePasswordHash(); userConfig.set('dbHasPassword', false); } else { sqlNode.setSQLPassword(passPhrase); const newHash = PasswordUtil.generateHash(passPhrase); sqlNode.savePasswordHash(newHash); userConfig.set('dbHasPassword', true); } sendResponse(undefined); } catch (e) { const localisedError = locale.messages.setPasswordFail; sendResponse(localisedError || 'Failed to set password'); } }); // Debug Log-related IPC calls ipc.on('show-debug-log', showDebugLogWindow); ipc.on('close-debug-log', () => { if (debugLogWindow) { debugLogWindow.close(); } }); ipc.on('save-debug-log', saveDebugLog); // This should be called with an ipc sendSync ipc.on('get-media-permissions', event => { // eslint-disable-next-line no-param-reassign event.returnValue = userConfig.get('mediaPermissions') || false; }); ipc.on('set-media-permissions', (event, value) => { userConfig.set('mediaPermissions', value); // We reinstall permissions handler to ensure that a revoked permission takes effect installPermissionsHandler({ userConfig }); event.sender.send('set-success-media-permissions', null); }); // This should be called with an ipc sendSync ipc.on('get-call-media-permissions', event => { // eslint-disable-next-line no-param-reassign event.returnValue = userConfig.get('callMediaPermissions') || false; }); ipc.on('set-call-media-permissions', (event, value) => { userConfig.set('callMediaPermissions', value); // We reinstall permissions handler to ensure that a revoked permission takes effect installPermissionsHandler({ userConfig }); event.sender.send('set-success-call-media-permissions', null); }); // Session - Auto updating ipc.on('get-auto-update-setting', event => { const configValue = userConfig.get('autoUpdate'); // eslint-disable-next-line no-param-reassign event.returnValue = typeof configValue !== 'boolean' ? true : configValue; }); ipc.on('set-auto-update-setting', async (_event, enabled) => { userConfig.set('autoUpdate', !!enabled); if (enabled) { await readyForUpdates(); } else { updater.stop(); isReadyForUpdates = false; } }); async function getThemeFromMainWindow() { return new Promise(resolve => { ipc.once('get-success-theme-setting', (_event, value) => { resolve(value); }); mainWindow?.webContents.send('get-theme-setting'); }); } async function askForMediaAccess() { // Microphone part let status = systemPreferences.getMediaAccessStatus('microphone'); if (status !== 'granted') { await systemPreferences.askForMediaAccess('microphone'); } // Camera part status = systemPreferences.getMediaAccessStatus('camera'); if (status !== 'granted') { await systemPreferences.askForMediaAccess('camera'); } } ipc.on('media-access', async () => { await askForMediaAccess(); });