api-ipc-spec.ts 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550
  1. import { EventEmitter } from 'events';
  2. import { expect } from 'chai';
  3. import { BrowserWindow, ipcMain, IpcMainInvokeEvent, MessageChannelMain, WebContents } from 'electron/main';
  4. import { closeAllWindows } from './window-helpers';
  5. import { emittedOnce } from './events-helpers';
  6. const v8Util = process._linkedBinding('electron_common_v8_util');
  7. describe('ipc module', () => {
  8. describe('invoke', () => {
  9. let w = (null as unknown as BrowserWindow);
  10. before(async () => {
  11. w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  12. await w.loadURL('about:blank');
  13. });
  14. after(async () => {
  15. w.destroy();
  16. });
  17. async function rendererInvoke (...args: any[]) {
  18. const { ipcRenderer } = require('electron');
  19. try {
  20. const result = await ipcRenderer.invoke('test', ...args);
  21. ipcRenderer.send('result', { result });
  22. } catch (e) {
  23. ipcRenderer.send('result', { error: e.message });
  24. }
  25. }
  26. it('receives a response from a synchronous handler', async () => {
  27. ipcMain.handleOnce('test', (e: IpcMainInvokeEvent, arg: number) => {
  28. expect(arg).to.equal(123);
  29. return 3;
  30. });
  31. const done = new Promise<void>(resolve => ipcMain.once('result', (e, arg) => {
  32. expect(arg).to.deep.equal({ result: 3 });
  33. resolve();
  34. }));
  35. await w.webContents.executeJavaScript(`(${rendererInvoke})(123)`);
  36. await done;
  37. });
  38. it('receives a response from an asynchronous handler', async () => {
  39. ipcMain.handleOnce('test', async (e: IpcMainInvokeEvent, arg: number) => {
  40. expect(arg).to.equal(123);
  41. await new Promise(setImmediate);
  42. return 3;
  43. });
  44. const done = new Promise<void>(resolve => ipcMain.once('result', (e, arg) => {
  45. expect(arg).to.deep.equal({ result: 3 });
  46. resolve();
  47. }));
  48. await w.webContents.executeJavaScript(`(${rendererInvoke})(123)`);
  49. await done;
  50. });
  51. it('receives an error from a synchronous handler', async () => {
  52. ipcMain.handleOnce('test', () => {
  53. throw new Error('some error');
  54. });
  55. const done = new Promise<void>(resolve => ipcMain.once('result', (e, arg) => {
  56. expect(arg.error).to.match(/some error/);
  57. resolve();
  58. }));
  59. await w.webContents.executeJavaScript(`(${rendererInvoke})()`);
  60. await done;
  61. });
  62. it('receives an error from an asynchronous handler', async () => {
  63. ipcMain.handleOnce('test', async () => {
  64. await new Promise(setImmediate);
  65. throw new Error('some error');
  66. });
  67. const done = new Promise<void>(resolve => ipcMain.once('result', (e, arg) => {
  68. expect(arg.error).to.match(/some error/);
  69. resolve();
  70. }));
  71. await w.webContents.executeJavaScript(`(${rendererInvoke})()`);
  72. await done;
  73. });
  74. it('throws an error if no handler is registered', async () => {
  75. const done = new Promise<void>(resolve => ipcMain.once('result', (e, arg) => {
  76. expect(arg.error).to.match(/No handler registered/);
  77. resolve();
  78. }));
  79. await w.webContents.executeJavaScript(`(${rendererInvoke})()`);
  80. await done;
  81. });
  82. it('throws an error when invoking a handler that was removed', async () => {
  83. ipcMain.handle('test', () => {});
  84. ipcMain.removeHandler('test');
  85. const done = new Promise<void>(resolve => ipcMain.once('result', (e, arg) => {
  86. expect(arg.error).to.match(/No handler registered/);
  87. resolve();
  88. }));
  89. await w.webContents.executeJavaScript(`(${rendererInvoke})()`);
  90. await done;
  91. });
  92. it('forbids multiple handlers', async () => {
  93. ipcMain.handle('test', () => {});
  94. try {
  95. expect(() => { ipcMain.handle('test', () => {}); }).to.throw(/second handler/);
  96. } finally {
  97. ipcMain.removeHandler('test');
  98. }
  99. });
  100. it('throws an error in the renderer if the reply callback is dropped', async () => {
  101. // eslint-disable-next-line @typescript-eslint/no-unused-vars
  102. ipcMain.handleOnce('test', () => new Promise(resolve => {
  103. setTimeout(() => v8Util.requestGarbageCollectionForTesting());
  104. /* never resolve */
  105. }));
  106. w.webContents.executeJavaScript(`(${rendererInvoke})()`);
  107. const [, { error }] = await emittedOnce(ipcMain, 'result');
  108. expect(error).to.match(/reply was never sent/);
  109. });
  110. });
  111. describe('ordering', () => {
  112. let w = (null as unknown as BrowserWindow);
  113. before(async () => {
  114. w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  115. await w.loadURL('about:blank');
  116. });
  117. after(async () => {
  118. w.destroy();
  119. });
  120. it('between send and sendSync is consistent', async () => {
  121. const received: number[] = [];
  122. ipcMain.on('test-async', (e, i) => { received.push(i); });
  123. ipcMain.on('test-sync', (e, i) => { received.push(i); e.returnValue = null; });
  124. const done = new Promise<void>(resolve => ipcMain.once('done', () => { resolve(); }));
  125. function rendererStressTest () {
  126. const { ipcRenderer } = require('electron');
  127. for (let i = 0; i < 1000; i++) {
  128. switch ((Math.random() * 2) | 0) {
  129. case 0:
  130. ipcRenderer.send('test-async', i);
  131. break;
  132. case 1:
  133. ipcRenderer.sendSync('test-sync', i);
  134. break;
  135. }
  136. }
  137. ipcRenderer.send('done');
  138. }
  139. try {
  140. w.webContents.executeJavaScript(`(${rendererStressTest})()`);
  141. await done;
  142. } finally {
  143. ipcMain.removeAllListeners('test-async');
  144. ipcMain.removeAllListeners('test-sync');
  145. }
  146. expect(received).to.have.lengthOf(1000);
  147. expect(received).to.deep.equal([...received].sort((a, b) => a - b));
  148. });
  149. it('between send, sendSync, and invoke is consistent', async () => {
  150. const received: number[] = [];
  151. ipcMain.handle('test-invoke', (e, i) => { received.push(i); });
  152. ipcMain.on('test-async', (e, i) => { received.push(i); });
  153. ipcMain.on('test-sync', (e, i) => { received.push(i); e.returnValue = null; });
  154. const done = new Promise<void>(resolve => ipcMain.once('done', () => { resolve(); }));
  155. function rendererStressTest () {
  156. const { ipcRenderer } = require('electron');
  157. for (let i = 0; i < 1000; i++) {
  158. switch ((Math.random() * 3) | 0) {
  159. case 0:
  160. ipcRenderer.send('test-async', i);
  161. break;
  162. case 1:
  163. ipcRenderer.sendSync('test-sync', i);
  164. break;
  165. case 2:
  166. ipcRenderer.invoke('test-invoke', i);
  167. break;
  168. }
  169. }
  170. ipcRenderer.send('done');
  171. }
  172. try {
  173. w.webContents.executeJavaScript(`(${rendererStressTest})()`);
  174. await done;
  175. } finally {
  176. ipcMain.removeHandler('test-invoke');
  177. ipcMain.removeAllListeners('test-async');
  178. ipcMain.removeAllListeners('test-sync');
  179. }
  180. expect(received).to.have.lengthOf(1000);
  181. expect(received).to.deep.equal([...received].sort((a, b) => a - b));
  182. });
  183. });
  184. describe('MessagePort', () => {
  185. afterEach(closeAllWindows);
  186. it('can send a port to the main process', async () => {
  187. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  188. w.loadURL('about:blank');
  189. const p = emittedOnce(ipcMain, 'port');
  190. await w.webContents.executeJavaScript(`(${function () {
  191. const channel = new MessageChannel();
  192. require('electron').ipcRenderer.postMessage('port', 'hi', [channel.port1]);
  193. }})()`);
  194. const [ev, msg] = await p;
  195. expect(msg).to.equal('hi');
  196. expect(ev.ports).to.have.length(1);
  197. const [port] = ev.ports;
  198. expect(port).to.be.an.instanceOf(EventEmitter);
  199. });
  200. it('can communicate between main and renderer', async () => {
  201. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  202. w.loadURL('about:blank');
  203. const p = emittedOnce(ipcMain, 'port');
  204. await w.webContents.executeJavaScript(`(${function () {
  205. const channel = new MessageChannel();
  206. (channel.port2 as any).onmessage = (ev: any) => {
  207. channel.port2.postMessage(ev.data * 2);
  208. };
  209. require('electron').ipcRenderer.postMessage('port', '', [channel.port1]);
  210. }})()`);
  211. const [ev] = await p;
  212. expect(ev.ports).to.have.length(1);
  213. const [port] = ev.ports;
  214. port.start();
  215. port.postMessage(42);
  216. const [ev2] = await emittedOnce(port, 'message');
  217. expect(ev2.data).to.equal(84);
  218. });
  219. it('can receive a port from a renderer over a MessagePort connection', async () => {
  220. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  221. w.loadURL('about:blank');
  222. function fn () {
  223. const channel1 = new MessageChannel();
  224. const channel2 = new MessageChannel();
  225. channel1.port2.postMessage('', [channel2.port1]);
  226. channel2.port2.postMessage('matryoshka');
  227. require('electron').ipcRenderer.postMessage('port', '', [channel1.port1]);
  228. }
  229. w.webContents.executeJavaScript(`(${fn})()`);
  230. const [{ ports: [port1] }] = await emittedOnce(ipcMain, 'port');
  231. port1.start();
  232. const [{ ports: [port2] }] = await emittedOnce(port1, 'message');
  233. port2.start();
  234. const [{ data }] = await emittedOnce(port2, 'message');
  235. expect(data).to.equal('matryoshka');
  236. });
  237. it('can forward a port from one renderer to another renderer', async () => {
  238. const w1 = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  239. const w2 = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  240. w1.loadURL('about:blank');
  241. w2.loadURL('about:blank');
  242. w1.webContents.executeJavaScript(`(${function () {
  243. const channel = new MessageChannel();
  244. (channel.port2 as any).onmessage = (ev: any) => {
  245. require('electron').ipcRenderer.send('message received', ev.data);
  246. };
  247. require('electron').ipcRenderer.postMessage('port', '', [channel.port1]);
  248. }})()`);
  249. const [{ ports: [port] }] = await emittedOnce(ipcMain, 'port');
  250. await w2.webContents.executeJavaScript(`(${function () {
  251. require('electron').ipcRenderer.on('port', ({ ports: [port] }: any) => {
  252. port.postMessage('a message');
  253. });
  254. }})()`);
  255. w2.webContents.postMessage('port', '', [port]);
  256. const [, data] = await emittedOnce(ipcMain, 'message received');
  257. expect(data).to.equal('a message');
  258. });
  259. describe('close event', () => {
  260. describe('in renderer', () => {
  261. it('is emitted when the main process closes its end of the port', async () => {
  262. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  263. w.loadURL('about:blank');
  264. await w.webContents.executeJavaScript(`(${function () {
  265. const { ipcRenderer } = require('electron');
  266. ipcRenderer.on('port', e => {
  267. const [port] = e.ports;
  268. port.start();
  269. (port as any).onclose = () => {
  270. ipcRenderer.send('closed');
  271. };
  272. });
  273. }})()`);
  274. const { port1, port2 } = new MessageChannelMain();
  275. w.webContents.postMessage('port', null, [port2]);
  276. port1.close();
  277. await emittedOnce(ipcMain, 'closed');
  278. });
  279. it('is emitted when the other end of a port is garbage-collected', async () => {
  280. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  281. w.loadURL('about:blank');
  282. await w.webContents.executeJavaScript(`(${async function () {
  283. const { port2 } = new MessageChannel();
  284. await new Promise(resolve => {
  285. port2.start();
  286. (port2 as any).onclose = resolve;
  287. process._linkedBinding('electron_common_v8_util').requestGarbageCollectionForTesting();
  288. });
  289. }})()`);
  290. });
  291. it('is emitted when the other end of a port is sent to nowhere', async () => {
  292. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  293. w.loadURL('about:blank');
  294. ipcMain.once('do-a-gc', () => v8Util.requestGarbageCollectionForTesting());
  295. await w.webContents.executeJavaScript(`(${async function () {
  296. const { port1, port2 } = new MessageChannel();
  297. await new Promise(resolve => {
  298. port2.start();
  299. (port2 as any).onclose = resolve;
  300. require('electron').ipcRenderer.postMessage('nobody-listening', null, [port1]);
  301. require('electron').ipcRenderer.send('do-a-gc');
  302. });
  303. }})()`);
  304. });
  305. });
  306. });
  307. describe('MessageChannelMain', () => {
  308. it('can be created', () => {
  309. const { port1, port2 } = new MessageChannelMain();
  310. expect(port1).not.to.be.null();
  311. expect(port2).not.to.be.null();
  312. });
  313. it('can send messages within the process', async () => {
  314. const { port1, port2 } = new MessageChannelMain();
  315. port2.postMessage('hello');
  316. port1.start();
  317. const [ev] = await emittedOnce(port1, 'message');
  318. expect(ev.data).to.equal('hello');
  319. });
  320. it('can pass one end to a WebContents', async () => {
  321. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  322. w.loadURL('about:blank');
  323. await w.webContents.executeJavaScript(`(${function () {
  324. const { ipcRenderer } = require('electron');
  325. ipcRenderer.on('port', ev => {
  326. const [port] = ev.ports;
  327. port.onmessage = () => {
  328. ipcRenderer.send('done');
  329. };
  330. });
  331. }})()`);
  332. const { port1, port2 } = new MessageChannelMain();
  333. port1.postMessage('hello');
  334. w.webContents.postMessage('port', null, [port2]);
  335. await emittedOnce(ipcMain, 'done');
  336. });
  337. it('can be passed over another channel', async () => {
  338. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  339. w.loadURL('about:blank');
  340. await w.webContents.executeJavaScript(`(${function () {
  341. const { ipcRenderer } = require('electron');
  342. ipcRenderer.on('port', e1 => {
  343. e1.ports[0].onmessage = e2 => {
  344. e2.ports[0].onmessage = e3 => {
  345. ipcRenderer.send('done', e3.data);
  346. };
  347. };
  348. });
  349. }})()`);
  350. const { port1, port2 } = new MessageChannelMain();
  351. const { port1: port3, port2: port4 } = new MessageChannelMain();
  352. port1.postMessage(null, [port4]);
  353. port3.postMessage('hello');
  354. w.webContents.postMessage('port', null, [port2]);
  355. const [, message] = await emittedOnce(ipcMain, 'done');
  356. expect(message).to.equal('hello');
  357. });
  358. it('can send messages to a closed port', () => {
  359. const { port1, port2 } = new MessageChannelMain();
  360. port2.start();
  361. port2.on('message', () => { throw new Error('unexpected message received'); });
  362. port1.close();
  363. port1.postMessage('hello');
  364. });
  365. it('can send messages to a port whose remote end is closed', () => {
  366. const { port1, port2 } = new MessageChannelMain();
  367. port2.start();
  368. port2.on('message', () => { throw new Error('unexpected message received'); });
  369. port2.close();
  370. port1.postMessage('hello');
  371. });
  372. it('throws when passing null ports', () => {
  373. const { port1 } = new MessageChannelMain();
  374. expect(() => {
  375. port1.postMessage(null, [null] as any);
  376. }).to.throw(/conversion failure/);
  377. });
  378. it('throws when passing duplicate ports', () => {
  379. const { port1 } = new MessageChannelMain();
  380. const { port1: port3 } = new MessageChannelMain();
  381. expect(() => {
  382. port1.postMessage(null, [port3, port3]);
  383. }).to.throw(/duplicate/);
  384. });
  385. it('throws when passing ports that have already been neutered', () => {
  386. const { port1 } = new MessageChannelMain();
  387. const { port1: port3 } = new MessageChannelMain();
  388. port1.postMessage(null, [port3]);
  389. expect(() => {
  390. port1.postMessage(null, [port3]);
  391. }).to.throw(/already neutered/);
  392. });
  393. it('throws when passing itself', () => {
  394. const { port1 } = new MessageChannelMain();
  395. expect(() => {
  396. port1.postMessage(null, [port1]);
  397. }).to.throw(/contains the source port/);
  398. });
  399. describe('GC behavior', () => {
  400. it('is not collected while it could still receive messages', async () => {
  401. let trigger: Function;
  402. const promise = new Promise(resolve => { trigger = resolve; });
  403. const port1 = (() => {
  404. const { port1, port2 } = new MessageChannelMain();
  405. port2.on('message', (e) => { trigger(e.data); });
  406. port2.start();
  407. return port1;
  408. })();
  409. v8Util.requestGarbageCollectionForTesting();
  410. port1.postMessage('hello');
  411. expect(await promise).to.equal('hello');
  412. });
  413. });
  414. });
  415. const generateTests = (title: string, postMessage: (contents: WebContents) => typeof WebContents.prototype.postMessage) => {
  416. describe(title, () => {
  417. it('sends a message', async () => {
  418. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  419. w.loadURL('about:blank');
  420. await w.webContents.executeJavaScript(`(${function () {
  421. const { ipcRenderer } = require('electron');
  422. ipcRenderer.on('foo', (_e, msg) => {
  423. ipcRenderer.send('bar', msg);
  424. });
  425. }})()`);
  426. postMessage(w.webContents)('foo', { some: 'message' });
  427. const [, msg] = await emittedOnce(ipcMain, 'bar');
  428. expect(msg).to.deep.equal({ some: 'message' });
  429. });
  430. describe('error handling', () => {
  431. it('throws on missing channel', async () => {
  432. const w = new BrowserWindow({ show: false });
  433. await w.loadURL('about:blank');
  434. expect(() => {
  435. (postMessage(w.webContents) as any)();
  436. }).to.throw(/Insufficient number of arguments/);
  437. });
  438. it('throws on invalid channel', async () => {
  439. const w = new BrowserWindow({ show: false });
  440. await w.loadURL('about:blank');
  441. expect(() => {
  442. postMessage(w.webContents)(null as any, '', []);
  443. }).to.throw(/Error processing argument at index 0/);
  444. });
  445. it('throws on missing message', async () => {
  446. const w = new BrowserWindow({ show: false });
  447. await w.loadURL('about:blank');
  448. expect(() => {
  449. (postMessage(w.webContents) as any)('channel');
  450. }).to.throw(/Insufficient number of arguments/);
  451. });
  452. it('throws on non-serializable message', async () => {
  453. const w = new BrowserWindow({ show: false });
  454. await w.loadURL('about:blank');
  455. expect(() => {
  456. postMessage(w.webContents)('channel', w);
  457. }).to.throw(/An object could not be cloned/);
  458. });
  459. it('throws on invalid transferable list', async () => {
  460. const w = new BrowserWindow({ show: false });
  461. await w.loadURL('about:blank');
  462. expect(() => {
  463. postMessage(w.webContents)('', '', null as any);
  464. }).to.throw(/Invalid value for transfer/);
  465. });
  466. it('throws on transferring non-transferable', async () => {
  467. const w = new BrowserWindow({ show: false });
  468. await w.loadURL('about:blank');
  469. expect(() => {
  470. (postMessage(w.webContents) as any)('channel', '', [123]);
  471. }).to.throw(/Invalid value for transfer/);
  472. });
  473. it('throws when passing null ports', async () => {
  474. const w = new BrowserWindow({ show: false });
  475. await w.loadURL('about:blank');
  476. expect(() => {
  477. postMessage(w.webContents)('foo', null, [null] as any);
  478. }).to.throw(/Invalid value for transfer/);
  479. });
  480. it('throws when passing duplicate ports', async () => {
  481. const w = new BrowserWindow({ show: false });
  482. await w.loadURL('about:blank');
  483. const { port1 } = new MessageChannelMain();
  484. expect(() => {
  485. postMessage(w.webContents)('foo', null, [port1, port1]);
  486. }).to.throw(/duplicate/);
  487. });
  488. it('throws when passing ports that have already been neutered', async () => {
  489. const w = new BrowserWindow({ show: false });
  490. await w.loadURL('about:blank');
  491. const { port1 } = new MessageChannelMain();
  492. postMessage(w.webContents)('foo', null, [port1]);
  493. expect(() => {
  494. postMessage(w.webContents)('foo', null, [port1]);
  495. }).to.throw(/already neutered/);
  496. });
  497. });
  498. });
  499. };
  500. generateTests('WebContents.postMessage', contents => contents.postMessage.bind(contents));
  501. generateTests('WebFrameMain.postMessage', contents => contents.mainFrame.postMessage.bind(contents.mainFrame));
  502. });
  503. });