electron/spec/api-ipc-main-spec.ts

119 lines
3.3 KiB
TypeScript
Raw Normal View History

import { ipcMain, BrowserWindow } from 'electron/main';
2020-03-20 20:28:31 +00:00
import { expect } from 'chai';
import * as cp from 'node:child_process';
import { once } from 'node:events';
import * as path from 'node:path';
import { defer } from './lib/spec-helpers';
import { closeAllWindows } from './lib/window-helpers';
2017-12-01 20:57:41 +00:00
2017-12-01 21:01:03 +00:00
describe('ipc main module', () => {
2020-03-20 20:28:31 +00:00
const fixtures = path.join(__dirname, 'fixtures');
2017-12-01 20:57:41 +00:00
2020-03-20 20:28:31 +00:00
afterEach(closeAllWindows);
2017-12-01 20:57:41 +00:00
describe('ipc.sendSync', () => {
2020-03-20 20:28:31 +00:00
afterEach(() => { ipcMain.removeAllListeners('send-sync-message'); });
2017-12-01 20:57:41 +00:00
it('does not crash when reply is not sent and browser is destroyed', (done) => {
2019-07-15 08:23:59 +00:00
const w = new BrowserWindow({
show: false,
webPreferences: {
nodeIntegration: true,
contextIsolation: false
}
2020-03-20 20:28:31 +00:00
});
2017-12-01 20:57:41 +00:00
ipcMain.once('send-sync-message', (event) => {
2020-03-20 20:28:31 +00:00
event.returnValue = null;
done();
});
w.loadFile(path.join(fixtures, 'api', 'send-sync-message.html'));
});
2017-12-01 20:57:41 +00:00
it('does not crash when reply is sent by multiple listeners', (done) => {
2019-07-15 08:23:59 +00:00
const w = new BrowserWindow({
show: false,
webPreferences: {
nodeIntegration: true,
contextIsolation: false
}
2020-03-20 20:28:31 +00:00
});
2017-12-01 20:57:41 +00:00
ipcMain.on('send-sync-message', (event) => {
2020-03-20 20:28:31 +00:00
event.returnValue = null;
});
2017-12-01 20:57:41 +00:00
ipcMain.on('send-sync-message', (event) => {
2020-03-20 20:28:31 +00:00
event.returnValue = null;
done();
});
w.loadFile(path.join(fixtures, 'api', 'send-sync-message.html'));
});
});
2017-12-01 20:57:41 +00:00
describe('ipcMain.on', () => {
it('is not used for internals', async () => {
2020-03-20 20:28:31 +00:00
const appPath = path.join(fixtures, 'api', 'ipc-main-listeners');
const electronPath = process.execPath;
const appProcess = cp.spawn(electronPath, [appPath]);
2020-03-20 20:28:31 +00:00
let output = '';
appProcess.stdout.on('data', (data) => { output += data; });
await once(appProcess.stdout, 'end');
2020-03-20 20:28:31 +00:00
output = JSON.parse(output);
expect(output).to.deep.equal(['error']);
});
it('can be replied to', async () => {
ipcMain.on('test-echo', (e, arg) => {
e.reply('test-echo', arg);
});
defer(() => {
ipcMain.removeAllListeners('test-echo');
});
const w = new BrowserWindow({
show: false,
webPreferences: {
nodeIntegration: true,
contextIsolation: false
}
});
w.loadURL('about:blank');
const v = await w.webContents.executeJavaScript(`new Promise((resolve, reject) => {
const { ipcRenderer } = require('electron')
ipcRenderer.send('test-echo', 'hello')
ipcRenderer.on('test-echo', (e, v) => {
resolve(v)
})
})`);
expect(v).to.equal('hello');
});
2020-03-20 20:28:31 +00:00
});
describe('ipcMain.removeAllListeners', () => {
beforeEach(() => { ipcMain.removeAllListeners(); });
beforeEach(() => { ipcMain.removeAllListeners(); });
it('removes only the given channel', () => {
ipcMain.on('channel1', () => {});
ipcMain.on('channel2', () => {});
ipcMain.removeAllListeners('channel1');
expect(ipcMain.eventNames()).to.deep.equal(['channel2']);
});
it('removes all channels if no channel is specified', () => {
ipcMain.on('channel1', () => {});
ipcMain.on('channel2', () => {});
ipcMain.removeAllListeners();
expect(ipcMain.eventNames()).to.deep.equal([]);
});
});
2020-03-20 20:28:31 +00:00
});