api-ipc-spec.ts 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562
  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 as Error).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 sent a message without a transfer', 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. require('electron').ipcRenderer.postMessage('port', 'hi');
  206. }})()`);
  207. const [ev, msg] = await p;
  208. expect(msg).to.equal('hi');
  209. expect(ev.ports).to.deep.equal([]);
  210. });
  211. it('can communicate between main and renderer', async () => {
  212. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  213. w.loadURL('about:blank');
  214. const p = emittedOnce(ipcMain, 'port');
  215. await w.webContents.executeJavaScript(`(${function () {
  216. const channel = new MessageChannel();
  217. (channel.port2 as any).onmessage = (ev: any) => {
  218. channel.port2.postMessage(ev.data * 2);
  219. };
  220. require('electron').ipcRenderer.postMessage('port', '', [channel.port1]);
  221. }})()`);
  222. const [ev] = await p;
  223. expect(ev.ports).to.have.length(1);
  224. const [port] = ev.ports;
  225. port.start();
  226. port.postMessage(42);
  227. const [ev2] = await emittedOnce(port, 'message');
  228. expect(ev2.data).to.equal(84);
  229. });
  230. it('can receive a port from a renderer over a MessagePort connection', async () => {
  231. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  232. w.loadURL('about:blank');
  233. function fn () {
  234. const channel1 = new MessageChannel();
  235. const channel2 = new MessageChannel();
  236. channel1.port2.postMessage('', [channel2.port1]);
  237. channel2.port2.postMessage('matryoshka');
  238. require('electron').ipcRenderer.postMessage('port', '', [channel1.port1]);
  239. }
  240. w.webContents.executeJavaScript(`(${fn})()`);
  241. const [{ ports: [port1] }] = await emittedOnce(ipcMain, 'port');
  242. port1.start();
  243. const [{ ports: [port2] }] = await emittedOnce(port1, 'message');
  244. port2.start();
  245. const [{ data }] = await emittedOnce(port2, 'message');
  246. expect(data).to.equal('matryoshka');
  247. });
  248. it('can forward a port from one renderer to another renderer', async () => {
  249. const w1 = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  250. const w2 = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  251. w1.loadURL('about:blank');
  252. w2.loadURL('about:blank');
  253. w1.webContents.executeJavaScript(`(${function () {
  254. const channel = new MessageChannel();
  255. (channel.port2 as any).onmessage = (ev: any) => {
  256. require('electron').ipcRenderer.send('message received', ev.data);
  257. };
  258. require('electron').ipcRenderer.postMessage('port', '', [channel.port1]);
  259. }})()`);
  260. const [{ ports: [port] }] = await emittedOnce(ipcMain, 'port');
  261. await w2.webContents.executeJavaScript(`(${function () {
  262. require('electron').ipcRenderer.on('port', ({ ports: [port] }: any) => {
  263. port.postMessage('a message');
  264. });
  265. }})()`);
  266. w2.webContents.postMessage('port', '', [port]);
  267. const [, data] = await emittedOnce(ipcMain, 'message received');
  268. expect(data).to.equal('a message');
  269. });
  270. describe('close event', () => {
  271. describe('in renderer', () => {
  272. it('is emitted when the main process closes its end of the port', async () => {
  273. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  274. w.loadURL('about:blank');
  275. await w.webContents.executeJavaScript(`(${function () {
  276. const { ipcRenderer } = require('electron');
  277. ipcRenderer.on('port', e => {
  278. const [port] = e.ports;
  279. port.start();
  280. (port as any).onclose = () => {
  281. ipcRenderer.send('closed');
  282. };
  283. });
  284. }})()`);
  285. const { port1, port2 } = new MessageChannelMain();
  286. w.webContents.postMessage('port', null, [port2]);
  287. port1.close();
  288. await emittedOnce(ipcMain, 'closed');
  289. });
  290. it('is emitted when the other end of a port is garbage-collected', async () => {
  291. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  292. w.loadURL('about:blank');
  293. await w.webContents.executeJavaScript(`(${async function () {
  294. const { port2 } = new MessageChannel();
  295. await new Promise(resolve => {
  296. port2.start();
  297. (port2 as any).onclose = resolve;
  298. process._linkedBinding('electron_common_v8_util').requestGarbageCollectionForTesting();
  299. });
  300. }})()`);
  301. });
  302. it('is emitted when the other end of a port is sent to nowhere', async () => {
  303. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  304. w.loadURL('about:blank');
  305. ipcMain.once('do-a-gc', () => v8Util.requestGarbageCollectionForTesting());
  306. await w.webContents.executeJavaScript(`(${async function () {
  307. const { port1, port2 } = new MessageChannel();
  308. await new Promise(resolve => {
  309. port2.start();
  310. (port2 as any).onclose = resolve;
  311. require('electron').ipcRenderer.postMessage('nobody-listening', null, [port1]);
  312. require('electron').ipcRenderer.send('do-a-gc');
  313. });
  314. }})()`);
  315. });
  316. });
  317. });
  318. describe('MessageChannelMain', () => {
  319. it('can be created', () => {
  320. const { port1, port2 } = new MessageChannelMain();
  321. expect(port1).not.to.be.null();
  322. expect(port2).not.to.be.null();
  323. });
  324. it('can send messages within the process', async () => {
  325. const { port1, port2 } = new MessageChannelMain();
  326. port2.postMessage('hello');
  327. port1.start();
  328. const [ev] = await emittedOnce(port1, 'message');
  329. expect(ev.data).to.equal('hello');
  330. });
  331. it('can pass one end to a WebContents', async () => {
  332. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  333. w.loadURL('about:blank');
  334. await w.webContents.executeJavaScript(`(${function () {
  335. const { ipcRenderer } = require('electron');
  336. ipcRenderer.on('port', ev => {
  337. const [port] = ev.ports;
  338. port.onmessage = () => {
  339. ipcRenderer.send('done');
  340. };
  341. });
  342. }})()`);
  343. const { port1, port2 } = new MessageChannelMain();
  344. port1.postMessage('hello');
  345. w.webContents.postMessage('port', null, [port2]);
  346. await emittedOnce(ipcMain, 'done');
  347. });
  348. it('can be passed over another channel', async () => {
  349. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  350. w.loadURL('about:blank');
  351. await w.webContents.executeJavaScript(`(${function () {
  352. const { ipcRenderer } = require('electron');
  353. ipcRenderer.on('port', e1 => {
  354. e1.ports[0].onmessage = e2 => {
  355. e2.ports[0].onmessage = e3 => {
  356. ipcRenderer.send('done', e3.data);
  357. };
  358. };
  359. });
  360. }})()`);
  361. const { port1, port2 } = new MessageChannelMain();
  362. const { port1: port3, port2: port4 } = new MessageChannelMain();
  363. port1.postMessage(null, [port4]);
  364. port3.postMessage('hello');
  365. w.webContents.postMessage('port', null, [port2]);
  366. const [, message] = await emittedOnce(ipcMain, 'done');
  367. expect(message).to.equal('hello');
  368. });
  369. it('can send messages to a closed port', () => {
  370. const { port1, port2 } = new MessageChannelMain();
  371. port2.start();
  372. port2.on('message', () => { throw new Error('unexpected message received'); });
  373. port1.close();
  374. port1.postMessage('hello');
  375. });
  376. it('can send messages to a port whose remote end is closed', () => {
  377. const { port1, port2 } = new MessageChannelMain();
  378. port2.start();
  379. port2.on('message', () => { throw new Error('unexpected message received'); });
  380. port2.close();
  381. port1.postMessage('hello');
  382. });
  383. it('throws when passing null ports', () => {
  384. const { port1 } = new MessageChannelMain();
  385. expect(() => {
  386. port1.postMessage(null, [null] as any);
  387. }).to.throw(/conversion failure/);
  388. });
  389. it('throws when passing duplicate ports', () => {
  390. const { port1 } = new MessageChannelMain();
  391. const { port1: port3 } = new MessageChannelMain();
  392. expect(() => {
  393. port1.postMessage(null, [port3, port3]);
  394. }).to.throw(/duplicate/);
  395. });
  396. it('throws when passing ports that have already been neutered', () => {
  397. const { port1 } = new MessageChannelMain();
  398. const { port1: port3 } = new MessageChannelMain();
  399. port1.postMessage(null, [port3]);
  400. expect(() => {
  401. port1.postMessage(null, [port3]);
  402. }).to.throw(/already neutered/);
  403. });
  404. it('throws when passing itself', () => {
  405. const { port1 } = new MessageChannelMain();
  406. expect(() => {
  407. port1.postMessage(null, [port1]);
  408. }).to.throw(/contains the source port/);
  409. });
  410. describe('GC behavior', () => {
  411. it('is not collected while it could still receive messages', async () => {
  412. let trigger: Function;
  413. const promise = new Promise(resolve => { trigger = resolve; });
  414. const port1 = (() => {
  415. const { port1, port2 } = new MessageChannelMain();
  416. port2.on('message', (e) => { trigger(e.data); });
  417. port2.start();
  418. return port1;
  419. })();
  420. v8Util.requestGarbageCollectionForTesting();
  421. port1.postMessage('hello');
  422. expect(await promise).to.equal('hello');
  423. });
  424. });
  425. });
  426. const generateTests = (title: string, postMessage: (contents: WebContents) => WebContents['postMessage']) => {
  427. describe(title, () => {
  428. it('sends a message', async () => {
  429. const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false } });
  430. w.loadURL('about:blank');
  431. await w.webContents.executeJavaScript(`(${function () {
  432. const { ipcRenderer } = require('electron');
  433. ipcRenderer.on('foo', (_e, msg) => {
  434. ipcRenderer.send('bar', msg);
  435. });
  436. }})()`);
  437. postMessage(w.webContents)('foo', { some: 'message' });
  438. const [, msg] = await emittedOnce(ipcMain, 'bar');
  439. expect(msg).to.deep.equal({ some: 'message' });
  440. });
  441. describe('error handling', () => {
  442. it('throws on missing channel', async () => {
  443. const w = new BrowserWindow({ show: false });
  444. await w.loadURL('about:blank');
  445. expect(() => {
  446. (postMessage(w.webContents) as any)();
  447. }).to.throw(/Insufficient number of arguments/);
  448. });
  449. it('throws on invalid channel', async () => {
  450. const w = new BrowserWindow({ show: false });
  451. await w.loadURL('about:blank');
  452. expect(() => {
  453. postMessage(w.webContents)(null as any, '', []);
  454. }).to.throw(/Error processing argument at index 0/);
  455. });
  456. it('throws on missing message', async () => {
  457. const w = new BrowserWindow({ show: false });
  458. await w.loadURL('about:blank');
  459. expect(() => {
  460. (postMessage(w.webContents) as any)('channel');
  461. }).to.throw(/Insufficient number of arguments/);
  462. });
  463. it('throws on non-serializable message', async () => {
  464. const w = new BrowserWindow({ show: false });
  465. await w.loadURL('about:blank');
  466. expect(() => {
  467. postMessage(w.webContents)('channel', w);
  468. }).to.throw(/An object could not be cloned/);
  469. });
  470. it('throws on invalid transferable list', async () => {
  471. const w = new BrowserWindow({ show: false });
  472. await w.loadURL('about:blank');
  473. expect(() => {
  474. postMessage(w.webContents)('', '', null as any);
  475. }).to.throw(/Invalid value for transfer/);
  476. });
  477. it('throws on transferring non-transferable', async () => {
  478. const w = new BrowserWindow({ show: false });
  479. await w.loadURL('about:blank');
  480. expect(() => {
  481. (postMessage(w.webContents) as any)('channel', '', [123]);
  482. }).to.throw(/Invalid value for transfer/);
  483. });
  484. it('throws when passing null ports', async () => {
  485. const w = new BrowserWindow({ show: false });
  486. await w.loadURL('about:blank');
  487. expect(() => {
  488. postMessage(w.webContents)('foo', null, [null] as any);
  489. }).to.throw(/Invalid value for transfer/);
  490. });
  491. it('throws when passing duplicate ports', async () => {
  492. const w = new BrowserWindow({ show: false });
  493. await w.loadURL('about:blank');
  494. const { port1 } = new MessageChannelMain();
  495. expect(() => {
  496. postMessage(w.webContents)('foo', null, [port1, port1]);
  497. }).to.throw(/duplicate/);
  498. });
  499. it('throws when passing ports that have already been neutered', async () => {
  500. const w = new BrowserWindow({ show: false });
  501. await w.loadURL('about:blank');
  502. const { port1 } = new MessageChannelMain();
  503. postMessage(w.webContents)('foo', null, [port1]);
  504. expect(() => {
  505. postMessage(w.webContents)('foo', null, [port1]);
  506. }).to.throw(/already neutered/);
  507. });
  508. });
  509. });
  510. };
  511. generateTests('WebContents.postMessage', contents => contents.postMessage.bind(contents));
  512. generateTests('WebFrameMain.postMessage', contents => contents.mainFrame.postMessage.bind(contents.mainFrame));
  513. });
  514. });