session-desktop/ts/mains/main_node.ts

1129 lines
32 KiB
TypeScript

/* 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);
// FIXME 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';
// 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<string>, 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,
backgroundColor: '#000',
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,
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;
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: '#000',
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('#000');
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: '#000',
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('#000');
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.invalidPassword;
sendResponse(localisedError || 'Invalid password');
}
});
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();
});