test: migrate webview attribute specs to spec-main (#35076)
This commit is contained in:
parent
3baf713648
commit
d15348ecc2
2 changed files with 645 additions and 591 deletions
|
@ -3,13 +3,18 @@ import * as url from 'url';
|
||||||
import { BrowserWindow, session, ipcMain, app, WebContents } from 'electron/main';
|
import { BrowserWindow, session, ipcMain, app, WebContents } from 'electron/main';
|
||||||
import { closeAllWindows } from './window-helpers';
|
import { closeAllWindows } from './window-helpers';
|
||||||
import { emittedOnce, emittedUntil } from './events-helpers';
|
import { emittedOnce, emittedUntil } from './events-helpers';
|
||||||
import { ifit, delay } from './spec-helpers';
|
import { ifit, delay, defer } from './spec-helpers';
|
||||||
import { expect } from 'chai';
|
import { AssertionError, expect } from 'chai';
|
||||||
|
import * as http from 'http';
|
||||||
|
import { AddressInfo } from 'net';
|
||||||
|
|
||||||
|
declare let WebView: any;
|
||||||
|
|
||||||
async function loadWebView (w: WebContents, attributes: Record<string, string>, openDevTools: boolean = false): Promise<void> {
|
async function loadWebView (w: WebContents, attributes: Record<string, string>, openDevTools: boolean = false): Promise<void> {
|
||||||
await w.executeJavaScript(`
|
await w.executeJavaScript(`
|
||||||
new Promise((resolve, reject) => {
|
new Promise((resolve, reject) => {
|
||||||
const webview = new WebView()
|
const webview = new WebView()
|
||||||
|
webview.id = 'webview'
|
||||||
for (const [k, v] of Object.entries(${JSON.stringify(attributes)})) {
|
for (const [k, v] of Object.entries(${JSON.stringify(attributes)})) {
|
||||||
webview.setAttribute(k, v)
|
webview.setAttribute(k, v)
|
||||||
}
|
}
|
||||||
|
@ -25,11 +30,40 @@ async function loadWebView (w: WebContents, attributes: Record<string, string>,
|
||||||
})
|
})
|
||||||
`);
|
`);
|
||||||
}
|
}
|
||||||
|
async function loadWebViewAndWaitForMessage (w: WebContents, attributes: Record<string, string>): Promise<string> {
|
||||||
|
return await w.executeJavaScript(`new Promise((resolve, reject) => {
|
||||||
|
const webview = new WebView()
|
||||||
|
for (const [k, v] of Object.entries(${JSON.stringify(attributes)})) {
|
||||||
|
webview.setAttribute(k, v)
|
||||||
|
}
|
||||||
|
webview.addEventListener('console-message', (e) => {
|
||||||
|
resolve(e.message)
|
||||||
|
})
|
||||||
|
document.body.appendChild(webview)
|
||||||
|
})`);
|
||||||
|
};
|
||||||
|
|
||||||
|
async function itremote (name: string, fn: Function, args?: any[]) {
|
||||||
|
it(name, async () => {
|
||||||
|
const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false, webviewTag: true } });
|
||||||
|
defer(() => w.close());
|
||||||
|
w.loadURL('about:blank');
|
||||||
|
const { ok, message } = await w.webContents.executeJavaScript(`(async () => {
|
||||||
|
try {
|
||||||
|
const chai_1 = require('chai')
|
||||||
|
await (${fn})(...${JSON.stringify(args ?? [])})
|
||||||
|
return {ok: true};
|
||||||
|
} catch (e) {
|
||||||
|
return {ok: false, message: e.message}
|
||||||
|
}
|
||||||
|
})()`);
|
||||||
|
if (!ok) { throw new AssertionError(message); }
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
describe('<webview> tag', function () {
|
describe('<webview> tag', function () {
|
||||||
const fixtures = path.join(__dirname, '..', 'spec', 'fixtures');
|
const fixtures = path.join(__dirname, '..', 'spec', 'fixtures');
|
||||||
|
const blankPageUrl = url.pathToFileURL(path.join(fixtures, 'pages', 'blank.html')).toString();
|
||||||
afterEach(closeAllWindows);
|
|
||||||
|
|
||||||
function hideChildWindows (e: any, wc: WebContents) {
|
function hideChildWindows (e: any, wc: WebContents) {
|
||||||
wc.setWindowOpenHandler(() => ({
|
wc.setWindowOpenHandler(() => ({
|
||||||
|
@ -48,81 +82,85 @@ describe('<webview> tag', function () {
|
||||||
app.off('web-contents-created', hideChildWindows);
|
app.off('web-contents-created', hideChildWindows);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('works without script tag in page', async () => {
|
describe('behavior', () => {
|
||||||
const w = new BrowserWindow({
|
afterEach(closeAllWindows);
|
||||||
show: false,
|
|
||||||
webPreferences: {
|
|
||||||
webviewTag: true,
|
|
||||||
nodeIntegration: true,
|
|
||||||
contextIsolation: false
|
|
||||||
}
|
|
||||||
});
|
|
||||||
w.loadFile(path.join(fixtures, 'pages', 'webview-no-script.html'));
|
|
||||||
await emittedOnce(ipcMain, 'pong');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('works with sandbox', async () => {
|
it('works without script tag in page', async () => {
|
||||||
const w = new BrowserWindow({
|
const w = new BrowserWindow({
|
||||||
show: false,
|
show: false,
|
||||||
webPreferences: {
|
webPreferences: {
|
||||||
webviewTag: true,
|
webviewTag: true,
|
||||||
sandbox: true
|
nodeIntegration: true,
|
||||||
}
|
contextIsolation: false
|
||||||
});
|
}
|
||||||
w.loadFile(path.join(fixtures, 'pages', 'webview-isolated.html'));
|
});
|
||||||
await emittedOnce(ipcMain, 'pong');
|
w.loadFile(path.join(fixtures, 'pages', 'webview-no-script.html'));
|
||||||
});
|
await emittedOnce(ipcMain, 'pong');
|
||||||
|
|
||||||
it('works with contextIsolation', async () => {
|
|
||||||
const w = new BrowserWindow({
|
|
||||||
show: false,
|
|
||||||
webPreferences: {
|
|
||||||
webviewTag: true,
|
|
||||||
contextIsolation: true
|
|
||||||
}
|
|
||||||
});
|
|
||||||
w.loadFile(path.join(fixtures, 'pages', 'webview-isolated.html'));
|
|
||||||
await emittedOnce(ipcMain, 'pong');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('works with contextIsolation + sandbox', async () => {
|
|
||||||
const w = new BrowserWindow({
|
|
||||||
show: false,
|
|
||||||
webPreferences: {
|
|
||||||
webviewTag: true,
|
|
||||||
contextIsolation: true,
|
|
||||||
sandbox: true
|
|
||||||
}
|
|
||||||
});
|
|
||||||
w.loadFile(path.join(fixtures, 'pages', 'webview-isolated.html'));
|
|
||||||
await emittedOnce(ipcMain, 'pong');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('works with Trusted Types', async () => {
|
|
||||||
const w = new BrowserWindow({
|
|
||||||
show: false,
|
|
||||||
webPreferences: {
|
|
||||||
webviewTag: true
|
|
||||||
}
|
|
||||||
});
|
|
||||||
w.loadFile(path.join(fixtures, 'pages', 'webview-trusted-types.html'));
|
|
||||||
await emittedOnce(ipcMain, 'pong');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('is disabled by default', async () => {
|
|
||||||
const w = new BrowserWindow({
|
|
||||||
show: false,
|
|
||||||
webPreferences: {
|
|
||||||
preload: path.join(fixtures, 'module', 'preload-webview.js'),
|
|
||||||
nodeIntegration: true
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const webview = emittedOnce(ipcMain, 'webview');
|
it('works with sandbox', async () => {
|
||||||
w.loadFile(path.join(fixtures, 'pages', 'webview-no-script.html'));
|
const w = new BrowserWindow({
|
||||||
const [, type] = await webview;
|
show: false,
|
||||||
|
webPreferences: {
|
||||||
|
webviewTag: true,
|
||||||
|
sandbox: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
w.loadFile(path.join(fixtures, 'pages', 'webview-isolated.html'));
|
||||||
|
await emittedOnce(ipcMain, 'pong');
|
||||||
|
});
|
||||||
|
|
||||||
expect(type).to.equal('undefined', 'WebView still exists');
|
it('works with contextIsolation', async () => {
|
||||||
|
const w = new BrowserWindow({
|
||||||
|
show: false,
|
||||||
|
webPreferences: {
|
||||||
|
webviewTag: true,
|
||||||
|
contextIsolation: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
w.loadFile(path.join(fixtures, 'pages', 'webview-isolated.html'));
|
||||||
|
await emittedOnce(ipcMain, 'pong');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('works with contextIsolation + sandbox', async () => {
|
||||||
|
const w = new BrowserWindow({
|
||||||
|
show: false,
|
||||||
|
webPreferences: {
|
||||||
|
webviewTag: true,
|
||||||
|
contextIsolation: true,
|
||||||
|
sandbox: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
w.loadFile(path.join(fixtures, 'pages', 'webview-isolated.html'));
|
||||||
|
await emittedOnce(ipcMain, 'pong');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('works with Trusted Types', async () => {
|
||||||
|
const w = new BrowserWindow({
|
||||||
|
show: false,
|
||||||
|
webPreferences: {
|
||||||
|
webviewTag: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
w.loadFile(path.join(fixtures, 'pages', 'webview-trusted-types.html'));
|
||||||
|
await emittedOnce(ipcMain, 'pong');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('is disabled by default', async () => {
|
||||||
|
const w = new BrowserWindow({
|
||||||
|
show: false,
|
||||||
|
webPreferences: {
|
||||||
|
preload: path.join(fixtures, 'module', 'preload-webview.js'),
|
||||||
|
nodeIntegration: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const webview = emittedOnce(ipcMain, 'webview');
|
||||||
|
w.loadFile(path.join(fixtures, 'pages', 'webview-no-script.html'));
|
||||||
|
const [, type] = await webview;
|
||||||
|
|
||||||
|
expect(type).to.equal('undefined', 'WebView still exists');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// FIXME(deepak1556): Ch69 follow up.
|
// FIXME(deepak1556): Ch69 follow up.
|
||||||
|
@ -131,6 +169,8 @@ describe('<webview> tag', function () {
|
||||||
ipcMain.removeAllListeners('pong');
|
ipcMain.removeAllListeners('pong');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
afterEach(closeAllWindows);
|
||||||
|
|
||||||
it('updates when the window is shown after the ready-to-show event', async () => {
|
it('updates when the window is shown after the ready-to-show event', async () => {
|
||||||
const w = new BrowserWindow({ show: false });
|
const w = new BrowserWindow({ show: false });
|
||||||
const readyToShowSignal = emittedOnce(w, 'ready-to-show');
|
const readyToShowSignal = emittedOnce(w, 'ready-to-show');
|
||||||
|
@ -166,6 +206,7 @@ describe('<webview> tag', function () {
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('did-attach-webview event', () => {
|
describe('did-attach-webview event', () => {
|
||||||
|
afterEach(closeAllWindows);
|
||||||
it('is emitted when a webview has been attached', async () => {
|
it('is emitted when a webview has been attached', async () => {
|
||||||
const w = new BrowserWindow({
|
const w = new BrowserWindow({
|
||||||
show: false,
|
show: false,
|
||||||
|
@ -186,6 +227,7 @@ describe('<webview> tag', function () {
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('did-attach event', () => {
|
describe('did-attach event', () => {
|
||||||
|
afterEach(closeAllWindows);
|
||||||
it('is emitted when a webview has been attached', async () => {
|
it('is emitted when a webview has been attached', async () => {
|
||||||
const w = new BrowserWindow({
|
const w = new BrowserWindow({
|
||||||
webPreferences: {
|
webPreferences: {
|
||||||
|
@ -206,6 +248,7 @@ describe('<webview> tag', function () {
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('did-change-theme-color event', () => {
|
describe('did-change-theme-color event', () => {
|
||||||
|
afterEach(closeAllWindows);
|
||||||
it('emits when theme color changes', async () => {
|
it('emits when theme color changes', async () => {
|
||||||
const w = new BrowserWindow({
|
const w = new BrowserWindow({
|
||||||
webPreferences: {
|
webPreferences: {
|
||||||
|
@ -230,55 +273,60 @@ describe('<webview> tag', function () {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// This test is flaky on WOA, so skip it there.
|
describe('devtools', () => {
|
||||||
ifit(process.platform !== 'win32' || process.arch !== 'arm64')('loads devtools extensions registered on the parent window', async () => {
|
afterEach(closeAllWindows);
|
||||||
const w = new BrowserWindow({
|
// This test is flaky on WOA, so skip it there.
|
||||||
show: false,
|
ifit(process.platform !== 'win32' || process.arch !== 'arm64')('loads devtools extensions registered on the parent window', async () => {
|
||||||
webPreferences: {
|
const w = new BrowserWindow({
|
||||||
webviewTag: true,
|
show: false,
|
||||||
nodeIntegration: true,
|
webPreferences: {
|
||||||
contextIsolation: false
|
webviewTag: true,
|
||||||
}
|
nodeIntegration: true,
|
||||||
});
|
contextIsolation: false
|
||||||
w.webContents.session.removeExtension('foo');
|
}
|
||||||
|
|
||||||
const extensionPath = path.join(__dirname, 'fixtures', 'devtools-extensions', 'foo');
|
|
||||||
await w.webContents.session.loadExtension(extensionPath);
|
|
||||||
|
|
||||||
w.loadFile(path.join(__dirname, 'fixtures', 'pages', 'webview-devtools.html'));
|
|
||||||
loadWebView(w.webContents, {
|
|
||||||
nodeintegration: 'on',
|
|
||||||
webpreferences: 'contextIsolation=no',
|
|
||||||
src: `file://${path.join(__dirname, 'fixtures', 'blank.html')}`
|
|
||||||
}, true);
|
|
||||||
let childWebContentsId = 0;
|
|
||||||
app.once('web-contents-created', (e, webContents) => {
|
|
||||||
childWebContentsId = webContents.id;
|
|
||||||
webContents.on('devtools-opened', function () {
|
|
||||||
const showPanelIntervalId = setInterval(function () {
|
|
||||||
if (!webContents.isDestroyed() && webContents.devToolsWebContents) {
|
|
||||||
webContents.devToolsWebContents.executeJavaScript('(' + function () {
|
|
||||||
const { UI } = (window as any);
|
|
||||||
const tabs = UI.inspectorView.tabbedPane.tabs;
|
|
||||||
const lastPanelId: any = tabs[tabs.length - 1].id;
|
|
||||||
UI.inspectorView.showPanel(lastPanelId);
|
|
||||||
}.toString() + ')()');
|
|
||||||
} else {
|
|
||||||
clearInterval(showPanelIntervalId);
|
|
||||||
}
|
|
||||||
}, 100);
|
|
||||||
});
|
});
|
||||||
});
|
w.webContents.session.removeExtension('foo');
|
||||||
|
|
||||||
const [, { runtimeId, tabId }] = await emittedOnce(ipcMain, 'answer');
|
const extensionPath = path.join(__dirname, 'fixtures', 'devtools-extensions', 'foo');
|
||||||
expect(runtimeId).to.match(/^[a-z]{32}$/);
|
await w.webContents.session.loadExtension(extensionPath);
|
||||||
expect(tabId).to.equal(childWebContentsId);
|
|
||||||
|
w.loadFile(path.join(__dirname, 'fixtures', 'pages', 'webview-devtools.html'));
|
||||||
|
loadWebView(w.webContents, {
|
||||||
|
nodeintegration: 'on',
|
||||||
|
webpreferences: 'contextIsolation=no',
|
||||||
|
src: `file://${path.join(__dirname, 'fixtures', 'blank.html')}`
|
||||||
|
}, true);
|
||||||
|
let childWebContentsId = 0;
|
||||||
|
app.once('web-contents-created', (e, webContents) => {
|
||||||
|
childWebContentsId = webContents.id;
|
||||||
|
webContents.on('devtools-opened', function () {
|
||||||
|
const showPanelIntervalId = setInterval(function () {
|
||||||
|
if (!webContents.isDestroyed() && webContents.devToolsWebContents) {
|
||||||
|
webContents.devToolsWebContents.executeJavaScript('(' + function () {
|
||||||
|
const { UI } = (window as any);
|
||||||
|
const tabs = UI.inspectorView.tabbedPane.tabs;
|
||||||
|
const lastPanelId: any = tabs[tabs.length - 1].id;
|
||||||
|
UI.inspectorView.showPanel(lastPanelId);
|
||||||
|
}.toString() + ')()');
|
||||||
|
} else {
|
||||||
|
clearInterval(showPanelIntervalId);
|
||||||
|
}
|
||||||
|
}, 100);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const [, { runtimeId, tabId }] = await emittedOnce(ipcMain, 'answer');
|
||||||
|
expect(runtimeId).to.match(/^[a-z]{32}$/);
|
||||||
|
expect(tabId).to.equal(childWebContentsId);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('zoom behavior', () => {
|
describe('zoom behavior', () => {
|
||||||
const zoomScheme = standardScheme;
|
const zoomScheme = standardScheme;
|
||||||
const webviewSession = session.fromPartition('webview-temp');
|
const webviewSession = session.fromPartition('webview-temp');
|
||||||
|
|
||||||
|
afterEach(closeAllWindows);
|
||||||
|
|
||||||
before(() => {
|
before(() => {
|
||||||
const protocol = webviewSession.protocol;
|
const protocol = webviewSession.protocol;
|
||||||
protocol.registerStringProtocol(zoomScheme, (request, callback) => {
|
protocol.registerStringProtocol(zoomScheme, (request, callback) => {
|
||||||
|
@ -421,6 +469,7 @@ describe('<webview> tag', function () {
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('requestFullscreen from webview', () => {
|
describe('requestFullscreen from webview', () => {
|
||||||
|
afterEach(closeAllWindows);
|
||||||
const loadWebViewWindow = async () => {
|
const loadWebViewWindow = async () => {
|
||||||
const w = new BrowserWindow({
|
const w = new BrowserWindow({
|
||||||
webPreferences: {
|
webPreferences: {
|
||||||
|
@ -893,4 +942,484 @@ describe('<webview> tag', function () {
|
||||||
})`);
|
})`);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('attributes', () => {
|
||||||
|
let w: WebContents;
|
||||||
|
before(async () => {
|
||||||
|
const window = new BrowserWindow({
|
||||||
|
show: false,
|
||||||
|
webPreferences: {
|
||||||
|
webviewTag: true,
|
||||||
|
nodeIntegration: true,
|
||||||
|
contextIsolation: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
await window.loadURL(`file://${fixtures}/pages/blank.html`);
|
||||||
|
w = window.webContents;
|
||||||
|
});
|
||||||
|
afterEach(async () => {
|
||||||
|
await w.executeJavaScript(`{
|
||||||
|
document.querySelectorAll('webview').forEach(el => el.remove())
|
||||||
|
}`);
|
||||||
|
});
|
||||||
|
after(closeAllWindows);
|
||||||
|
|
||||||
|
describe('src attribute', () => {
|
||||||
|
it('specifies the page to load', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
src: `file://${fixtures}/pages/a.html`
|
||||||
|
});
|
||||||
|
expect(message).to.equal('a');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('navigates to new page when changed', async () => {
|
||||||
|
await loadWebView(w, {
|
||||||
|
src: `file://${fixtures}/pages/a.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const { message } = await w.executeJavaScript(`new Promise(resolve => {
|
||||||
|
webview.addEventListener('console-message', e => resolve({message: e.message}))
|
||||||
|
webview.src = ${JSON.stringify(`file://${fixtures}/pages/b.html`)}
|
||||||
|
})`);
|
||||||
|
|
||||||
|
expect(message).to.equal('b');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resolves relative URLs', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
src: './e.html'
|
||||||
|
});
|
||||||
|
expect(message).to.equal('Window script is loaded before preload script');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ignores empty values', async () => {
|
||||||
|
loadWebView(w, {});
|
||||||
|
|
||||||
|
for (const emptyValue of ['""', 'null', 'undefined']) {
|
||||||
|
const src = await w.executeJavaScript(`webview.src = ${emptyValue}, webview.src`);
|
||||||
|
expect(src).to.equal('');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not wait until loadURL is resolved', async () => {
|
||||||
|
await loadWebView(w, { src: 'about:blank' });
|
||||||
|
|
||||||
|
const delay = await w.executeJavaScript(`new Promise(resolve => {
|
||||||
|
const before = Date.now();
|
||||||
|
webview.src = 'file://${fixtures}/pages/blank.html';
|
||||||
|
const now = Date.now();
|
||||||
|
resolve(now - before);
|
||||||
|
})`);
|
||||||
|
|
||||||
|
// Setting src is essentially sending a sync IPC message, which should
|
||||||
|
// not exceed more than a few ms.
|
||||||
|
//
|
||||||
|
// This is for testing #18638.
|
||||||
|
expect(delay).to.be.below(100);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('nodeintegration attribute', () => {
|
||||||
|
it('inserts no node symbols when not set', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
src: `file://${fixtures}/pages/c.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'undefined',
|
||||||
|
module: 'undefined',
|
||||||
|
process: 'undefined',
|
||||||
|
global: 'undefined'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('inserts node symbols when set', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
nodeintegration: 'on',
|
||||||
|
webpreferences: 'contextIsolation=no',
|
||||||
|
src: `file://${fixtures}/pages/d.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function',
|
||||||
|
module: 'object',
|
||||||
|
process: 'object'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('loads node symbols after POST navigation when set', async function () {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
nodeintegration: 'on',
|
||||||
|
webpreferences: 'contextIsolation=no',
|
||||||
|
src: `file://${fixtures}/pages/post.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function',
|
||||||
|
module: 'object',
|
||||||
|
process: 'object'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('disables node integration on child windows when it is disabled on the webview', async () => {
|
||||||
|
const src = url.format({
|
||||||
|
pathname: `${fixtures}/pages/webview-opener-no-node-integration.html`,
|
||||||
|
protocol: 'file',
|
||||||
|
query: {
|
||||||
|
p: `${fixtures}/pages/window-opener-node.html`
|
||||||
|
},
|
||||||
|
slashes: true
|
||||||
|
});
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
allowpopups: 'on',
|
||||||
|
webpreferences: 'contextIsolation=no',
|
||||||
|
src
|
||||||
|
});
|
||||||
|
expect(JSON.parse(message).isProcessGlobalUndefined).to.be.true();
|
||||||
|
});
|
||||||
|
|
||||||
|
ifit(!process.env.ELECTRON_SKIP_NATIVE_MODULE_TESTS)('loads native modules when navigation happens', async function () {
|
||||||
|
await loadWebView(w, {
|
||||||
|
nodeintegration: 'on',
|
||||||
|
webpreferences: 'contextIsolation=no',
|
||||||
|
src: `file://${fixtures}/pages/native-module.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const message = await w.executeJavaScript(`new Promise(resolve => {
|
||||||
|
webview.addEventListener('console-message', e => resolve(e.message))
|
||||||
|
webview.reload();
|
||||||
|
})`);
|
||||||
|
|
||||||
|
expect(message).to.equal('function');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('preload attribute', () => {
|
||||||
|
it('loads the script before other scripts in window', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
preload: `${fixtures}/module/preload.js`,
|
||||||
|
src: `file://${fixtures}/pages/e.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(message).to.be.a('string');
|
||||||
|
expect(message).to.be.not.equal('Window script is loaded before preload script');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('preload script can still use "process" and "Buffer" when nodeintegration is off', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
preload: `${fixtures}/module/preload-node-off.js`,
|
||||||
|
src: `file://${fixtures}/api/blank.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
process: 'object',
|
||||||
|
Buffer: 'function'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('runs in the correct scope when sandboxed', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
preload: `${fixtures}/module/preload-context.js`,
|
||||||
|
src: `file://${fixtures}/api/blank.html`,
|
||||||
|
webpreferences: 'sandbox=yes'
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function', // arguments passed to it should be available
|
||||||
|
electron: 'undefined', // objects from the scope it is called from should not be available
|
||||||
|
window: 'object', // the window object should be available
|
||||||
|
localVar: 'undefined' // but local variables should not be exposed to the window
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('preload script can require modules that still use "process" and "Buffer" when nodeintegration is off', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
preload: `${fixtures}/module/preload-node-off-wrapper.js`,
|
||||||
|
webpreferences: 'sandbox=no',
|
||||||
|
src: `file://${fixtures}/api/blank.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
process: 'object',
|
||||||
|
Buffer: 'function'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('receives ipc message in preload script', async () => {
|
||||||
|
await loadWebView(w, {
|
||||||
|
preload: `${fixtures}/module/preload-ipc.js`,
|
||||||
|
src: `file://${fixtures}/pages/e.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const message = 'boom!';
|
||||||
|
const { channel, args } = await w.executeJavaScript(`new Promise(resolve => {
|
||||||
|
webview.send('ping', ${JSON.stringify(message)})
|
||||||
|
webview.addEventListener('ipc-message', ({channel, args}) => resolve({channel, args}))
|
||||||
|
})`);
|
||||||
|
|
||||||
|
expect(channel).to.equal('pong');
|
||||||
|
expect(args).to.deep.equal([message]);
|
||||||
|
});
|
||||||
|
|
||||||
|
itremote('<webview>.sendToFrame()', async (fixtures: string) => {
|
||||||
|
const w = new WebView();
|
||||||
|
w.setAttribute('nodeintegration', 'on');
|
||||||
|
w.setAttribute('webpreferences', 'contextIsolation=no');
|
||||||
|
w.setAttribute('preload', `file://${fixtures}/module/preload-ipc.js`);
|
||||||
|
w.setAttribute('src', `file://${fixtures}/pages/ipc-message.html`);
|
||||||
|
document.body.appendChild(w);
|
||||||
|
const { frameId } = await new Promise(resolve => w.addEventListener('ipc-message', resolve, { once: true }));
|
||||||
|
|
||||||
|
const message = 'boom!';
|
||||||
|
|
||||||
|
w.sendToFrame(frameId, 'ping', message);
|
||||||
|
const { channel, args } = await new Promise(resolve => w.addEventListener('ipc-message', resolve, { once: true }));
|
||||||
|
|
||||||
|
expect(channel).to.equal('pong');
|
||||||
|
expect(args).to.deep.equal([message]);
|
||||||
|
}, [fixtures]);
|
||||||
|
|
||||||
|
it('works without script tag in page', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
preload: `${fixtures}/module/preload.js`,
|
||||||
|
webpreferences: 'sandbox=no',
|
||||||
|
src: `file://${fixtures}/pages/base-page.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function',
|
||||||
|
module: 'object',
|
||||||
|
process: 'object',
|
||||||
|
Buffer: 'function'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resolves relative URLs', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
preload: '../module/preload.js',
|
||||||
|
webpreferences: 'sandbox=no',
|
||||||
|
src: `file://${fixtures}/pages/e.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function',
|
||||||
|
module: 'object',
|
||||||
|
process: 'object',
|
||||||
|
Buffer: 'function'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
itremote('ignores empty values', async () => {
|
||||||
|
const webview = new WebView();
|
||||||
|
|
||||||
|
for (const emptyValue of ['', null, undefined]) {
|
||||||
|
webview.preload = emptyValue;
|
||||||
|
expect(webview.preload).to.equal('');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('httpreferrer attribute', () => {
|
||||||
|
it('sets the referrer url', async () => {
|
||||||
|
const referrer = 'http://github.com/';
|
||||||
|
const received = await new Promise<string | undefined>((resolve, reject) => {
|
||||||
|
const server = http.createServer((req, res) => {
|
||||||
|
try {
|
||||||
|
resolve(req.headers.referer);
|
||||||
|
} catch (e) {
|
||||||
|
reject(e);
|
||||||
|
} finally {
|
||||||
|
res.end();
|
||||||
|
server.close();
|
||||||
|
}
|
||||||
|
}).listen(0, '127.0.0.1', () => {
|
||||||
|
const port = (server.address() as AddressInfo).port;
|
||||||
|
loadWebView(w, {
|
||||||
|
httpreferrer: referrer,
|
||||||
|
src: `http://127.0.0.1:${port}`
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
expect(received).to.equal(referrer);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('useragent attribute', () => {
|
||||||
|
it('sets the user agent', async () => {
|
||||||
|
const referrer = 'Mozilla/5.0 (Windows NT 6.1; WOW64; Trident/7.0; AS; rv:11.0) like Gecko';
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
src: `file://${fixtures}/pages/useragent.html`,
|
||||||
|
useragent: referrer
|
||||||
|
});
|
||||||
|
expect(message).to.equal(referrer);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('disablewebsecurity attribute', () => {
|
||||||
|
it('does not disable web security when not set', async () => {
|
||||||
|
await loadWebView(w, { src: 'about:blank' });
|
||||||
|
const result = await w.executeJavaScript(`webview.executeJavaScript(\`fetch(${JSON.stringify(blankPageUrl)}).then(() => 'ok', () => 'failed')\`)`);
|
||||||
|
expect(result).to.equal('failed');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('disables web security when set', async () => {
|
||||||
|
await loadWebView(w, { src: 'about:blank', disablewebsecurity: '' });
|
||||||
|
const result = await w.executeJavaScript(`webview.executeJavaScript(\`fetch(${JSON.stringify(blankPageUrl)}).then(() => 'ok', () => 'failed')\`)`);
|
||||||
|
expect(result).to.equal('ok');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not break node integration', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
disablewebsecurity: '',
|
||||||
|
nodeintegration: 'on',
|
||||||
|
webpreferences: 'contextIsolation=no',
|
||||||
|
src: `file://${fixtures}/pages/d.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function',
|
||||||
|
module: 'object',
|
||||||
|
process: 'object'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not break preload script', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
disablewebsecurity: '',
|
||||||
|
preload: `${fixtures}/module/preload.js`,
|
||||||
|
webpreferences: 'sandbox=no',
|
||||||
|
src: `file://${fixtures}/pages/e.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function',
|
||||||
|
module: 'object',
|
||||||
|
process: 'object',
|
||||||
|
Buffer: 'function'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('partition attribute', () => {
|
||||||
|
it('inserts no node symbols when not set', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
partition: 'test1',
|
||||||
|
src: `file://${fixtures}/pages/c.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'undefined',
|
||||||
|
module: 'undefined',
|
||||||
|
process: 'undefined',
|
||||||
|
global: 'undefined'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('inserts node symbols when set', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
nodeintegration: 'on',
|
||||||
|
partition: 'test2',
|
||||||
|
webpreferences: 'contextIsolation=no',
|
||||||
|
src: `file://${fixtures}/pages/d.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function',
|
||||||
|
module: 'object',
|
||||||
|
process: 'object'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('isolates storage for different id', async () => {
|
||||||
|
await w.executeJavaScript('localStorage.setItem(\'test\', \'one\')');
|
||||||
|
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
partition: 'test3',
|
||||||
|
src: `file://${fixtures}/pages/partition/one.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const parsedMessage = JSON.parse(message);
|
||||||
|
expect(parsedMessage).to.include({
|
||||||
|
numberOfEntries: 0,
|
||||||
|
testValue: null
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('uses current session storage when no id is provided', async () => {
|
||||||
|
await w.executeJavaScript('localStorage.setItem(\'test\', \'two\')');
|
||||||
|
const testValue = 'two';
|
||||||
|
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
src: `file://${fixtures}/pages/partition/one.html`
|
||||||
|
});
|
||||||
|
|
||||||
|
const parsedMessage = JSON.parse(message);
|
||||||
|
expect(parsedMessage).to.include({
|
||||||
|
testValue
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('allowpopups attribute', () => {
|
||||||
|
const generateSpecs = (description: string, webpreferences = '') => {
|
||||||
|
describe(description, () => {
|
||||||
|
it('can not open new window when not set', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
webpreferences,
|
||||||
|
src: `file://${fixtures}/pages/window-open-hide.html`
|
||||||
|
});
|
||||||
|
expect(message).to.equal('null');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('can open new window when set', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
webpreferences,
|
||||||
|
allowpopups: 'on',
|
||||||
|
src: `file://${fixtures}/pages/window-open-hide.html`
|
||||||
|
});
|
||||||
|
expect(message).to.equal('window');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
generateSpecs('without sandbox');
|
||||||
|
generateSpecs('with sandbox', 'sandbox=yes');
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('webpreferences attribute', () => {
|
||||||
|
it('can enable nodeintegration', async () => {
|
||||||
|
const message = await loadWebViewAndWaitForMessage(w, {
|
||||||
|
src: `file://${fixtures}/pages/d.html`,
|
||||||
|
webpreferences: 'nodeIntegration,contextIsolation=no'
|
||||||
|
});
|
||||||
|
|
||||||
|
const types = JSON.parse(message);
|
||||||
|
expect(types).to.include({
|
||||||
|
require: 'function',
|
||||||
|
module: 'object',
|
||||||
|
process: 'object'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('can disable web security and enable nodeintegration', async () => {
|
||||||
|
await loadWebView(w, { src: 'about:blank', webpreferences: 'webSecurity=no, nodeIntegration=yes, contextIsolation=no' });
|
||||||
|
const result = await w.executeJavaScript(`webview.executeJavaScript(\`fetch(${JSON.stringify(blankPageUrl)}).then(() => 'ok', () => 'failed')\`)`);
|
||||||
|
expect(result).to.equal('ok');
|
||||||
|
const type = await w.executeJavaScript('webview.executeJavaScript("typeof require")');
|
||||||
|
expect(type).to.equal('function');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -33,28 +33,6 @@ describe('<webview> tag', function () {
|
||||||
return event.message;
|
return event.message;
|
||||||
};
|
};
|
||||||
|
|
||||||
async function loadFileInWebView (webview, attributes = {}) {
|
|
||||||
const thisFile = url.format({
|
|
||||||
pathname: __filename.replace(/\\/g, '/'),
|
|
||||||
protocol: 'file',
|
|
||||||
slashes: true
|
|
||||||
});
|
|
||||||
const src = `<script>
|
|
||||||
function loadFile() {
|
|
||||||
return new Promise((resolve) => {
|
|
||||||
fetch('${thisFile}').then(
|
|
||||||
() => resolve('loaded'),
|
|
||||||
() => resolve('failed')
|
|
||||||
)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
console.log('ok');
|
|
||||||
</script>`;
|
|
||||||
attributes.src = `data:text/html;base64,${btoa(unescape(encodeURIComponent(src)))}`;
|
|
||||||
await startLoadingWebViewAndWaitForMessage(webview, attributes);
|
|
||||||
return await webview.executeJavaScript('loadFile()');
|
|
||||||
}
|
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
webview = new WebView();
|
webview = new WebView();
|
||||||
});
|
});
|
||||||
|
@ -66,459 +44,6 @@ describe('<webview> tag', function () {
|
||||||
webview.remove();
|
webview.remove();
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('src attribute', () => {
|
|
||||||
it('specifies the page to load', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
src: `file://${fixtures}/pages/a.html`
|
|
||||||
});
|
|
||||||
expect(message).to.equal('a');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('navigates to new page when changed', async () => {
|
|
||||||
await loadWebView(webview, {
|
|
||||||
src: `file://${fixtures}/pages/a.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
webview.src = `file://${fixtures}/pages/b.html`;
|
|
||||||
|
|
||||||
const { message } = await waitForEvent(webview, 'console-message');
|
|
||||||
expect(message).to.equal('b');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('resolves relative URLs', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
src: '../fixtures/pages/e.html'
|
|
||||||
});
|
|
||||||
expect(message).to.equal('Window script is loaded before preload script');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('ignores empty values', () => {
|
|
||||||
expect(webview.src).to.equal('');
|
|
||||||
|
|
||||||
for (const emptyValue of ['', null, undefined]) {
|
|
||||||
webview.src = emptyValue;
|
|
||||||
expect(webview.src).to.equal('');
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
it('does not wait until loadURL is resolved', async () => {
|
|
||||||
await loadWebView(webview, { src: 'about:blank' });
|
|
||||||
|
|
||||||
const before = Date.now();
|
|
||||||
webview.src = 'https://github.com';
|
|
||||||
const now = Date.now();
|
|
||||||
|
|
||||||
// Setting src is essentially sending a sync IPC message, which should
|
|
||||||
// not exceed more than a few ms.
|
|
||||||
//
|
|
||||||
// This is for testing #18638.
|
|
||||||
expect(now - before).to.be.below(100);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('nodeintegration attribute', () => {
|
|
||||||
it('inserts no node symbols when not set', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
src: `file://${fixtures}/pages/c.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'undefined',
|
|
||||||
module: 'undefined',
|
|
||||||
process: 'undefined',
|
|
||||||
global: 'undefined'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('inserts node symbols when set', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
nodeintegration: 'on',
|
|
||||||
webpreferences: 'contextIsolation=no',
|
|
||||||
src: `file://${fixtures}/pages/d.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function',
|
|
||||||
module: 'object',
|
|
||||||
process: 'object'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('loads node symbols after POST navigation when set', async function () {
|
|
||||||
// FIXME Figure out why this is timing out on AppVeyor
|
|
||||||
if (process.env.APPVEYOR === 'True') {
|
|
||||||
this.skip();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
nodeintegration: 'on',
|
|
||||||
webpreferences: 'contextIsolation=no',
|
|
||||||
src: `file://${fixtures}/pages/post.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function',
|
|
||||||
module: 'object',
|
|
||||||
process: 'object'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('disables node integration on child windows when it is disabled on the webview', async () => {
|
|
||||||
const src = url.format({
|
|
||||||
pathname: `${fixtures}/pages/webview-opener-no-node-integration.html`,
|
|
||||||
protocol: 'file',
|
|
||||||
query: {
|
|
||||||
p: `${fixtures}/pages/window-opener-node.html`
|
|
||||||
},
|
|
||||||
slashes: true
|
|
||||||
});
|
|
||||||
loadWebView(webview, {
|
|
||||||
allowpopups: 'on',
|
|
||||||
webpreferences: 'contextIsolation=no',
|
|
||||||
src
|
|
||||||
});
|
|
||||||
const { message } = await waitForEvent(webview, 'console-message');
|
|
||||||
expect(JSON.parse(message).isProcessGlobalUndefined).to.be.true();
|
|
||||||
});
|
|
||||||
|
|
||||||
(nativeModulesEnabled ? it : it.skip)('loads native modules when navigation happens', async function () {
|
|
||||||
await loadWebView(webview, {
|
|
||||||
nodeintegration: 'on',
|
|
||||||
webpreferences: 'contextIsolation=no',
|
|
||||||
src: `file://${fixtures}/pages/native-module.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
webview.reload();
|
|
||||||
|
|
||||||
const { message } = await waitForEvent(webview, 'console-message');
|
|
||||||
expect(message).to.equal('function');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('preload attribute', () => {
|
|
||||||
it('loads the script before other scripts in window', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
preload: `${fixtures}/module/preload.js`,
|
|
||||||
src: `file://${fixtures}/pages/e.html`,
|
|
||||||
contextIsolation: false
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(message).to.be.a('string');
|
|
||||||
expect(message).to.be.not.equal('Window script is loaded before preload script');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('preload script can still use "process" and "Buffer" when nodeintegration is off', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
preload: `${fixtures}/module/preload-node-off.js`,
|
|
||||||
src: `file://${fixtures}/api/blank.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
process: 'object',
|
|
||||||
Buffer: 'function'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('runs in the correct scope when sandboxed', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
preload: `${fixtures}/module/preload-context.js`,
|
|
||||||
src: `file://${fixtures}/api/blank.html`,
|
|
||||||
webpreferences: 'sandbox=yes'
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function', // arguments passed to it should be available
|
|
||||||
electron: 'undefined', // objects from the scope it is called from should not be available
|
|
||||||
window: 'object', // the window object should be available
|
|
||||||
localVar: 'undefined' // but local variables should not be exposed to the window
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('preload script can require modules that still use "process" and "Buffer" when nodeintegration is off', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
preload: `${fixtures}/module/preload-node-off-wrapper.js`,
|
|
||||||
webpreferences: 'sandbox=no',
|
|
||||||
src: `file://${fixtures}/api/blank.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
process: 'object',
|
|
||||||
Buffer: 'function'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('receives ipc message in preload script', async () => {
|
|
||||||
await loadWebView(webview, {
|
|
||||||
preload: `${fixtures}/module/preload-ipc.js`,
|
|
||||||
src: `file://${fixtures}/pages/e.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const message = 'boom!';
|
|
||||||
webview.send('ping', message);
|
|
||||||
|
|
||||||
const { channel, args } = await waitForEvent(webview, 'ipc-message');
|
|
||||||
expect(channel).to.equal('pong');
|
|
||||||
expect(args).to.deep.equal([message]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('<webview>.sendToFrame()', async () => {
|
|
||||||
loadWebView(webview, {
|
|
||||||
nodeintegration: 'on',
|
|
||||||
webpreferences: 'contextIsolation=no',
|
|
||||||
preload: `${fixtures}/module/preload-ipc.js`,
|
|
||||||
src: `file://${fixtures}/pages/ipc-message.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const { frameId } = await waitForEvent(webview, 'ipc-message');
|
|
||||||
|
|
||||||
const message = 'boom!';
|
|
||||||
webview.sendToFrame(frameId, 'ping', message);
|
|
||||||
|
|
||||||
const { channel, args } = await waitForEvent(webview, 'ipc-message');
|
|
||||||
expect(channel).to.equal('pong');
|
|
||||||
expect(args).to.deep.equal([message]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('works without script tag in page', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
preload: `${fixtures}/module/preload.js`,
|
|
||||||
webpreferences: 'sandbox=no',
|
|
||||||
src: `file://${fixtures}pages/base-page.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function',
|
|
||||||
module: 'object',
|
|
||||||
process: 'object',
|
|
||||||
Buffer: 'function'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('resolves relative URLs', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
preload: '../fixtures/module/preload.js',
|
|
||||||
webpreferences: 'sandbox=no',
|
|
||||||
src: `file://${fixtures}/pages/e.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function',
|
|
||||||
module: 'object',
|
|
||||||
process: 'object',
|
|
||||||
Buffer: 'function'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('ignores empty values', () => {
|
|
||||||
expect(webview.preload).to.equal('');
|
|
||||||
|
|
||||||
for (const emptyValue of ['', null, undefined]) {
|
|
||||||
webview.preload = emptyValue;
|
|
||||||
expect(webview.preload).to.equal('');
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('httpreferrer attribute', () => {
|
|
||||||
it('sets the referrer url', (done) => {
|
|
||||||
const referrer = 'http://github.com/';
|
|
||||||
const server = http.createServer((req, res) => {
|
|
||||||
try {
|
|
||||||
expect(req.headers.referer).to.equal(referrer);
|
|
||||||
done();
|
|
||||||
} catch (e) {
|
|
||||||
done(e);
|
|
||||||
} finally {
|
|
||||||
res.end();
|
|
||||||
server.close();
|
|
||||||
}
|
|
||||||
}).listen(0, '127.0.0.1', () => {
|
|
||||||
const port = server.address().port;
|
|
||||||
loadWebView(webview, {
|
|
||||||
httpreferrer: referrer,
|
|
||||||
src: `http://127.0.0.1:${port}`
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('useragent attribute', () => {
|
|
||||||
it('sets the user agent', async () => {
|
|
||||||
const referrer = 'Mozilla/5.0 (Windows NT 6.1; WOW64; Trident/7.0; AS; rv:11.0) like Gecko';
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
src: `file://${fixtures}/pages/useragent.html`,
|
|
||||||
useragent: referrer
|
|
||||||
});
|
|
||||||
expect(message).to.equal(referrer);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('disablewebsecurity attribute', () => {
|
|
||||||
it('does not disable web security when not set', async () => {
|
|
||||||
const result = await loadFileInWebView(webview);
|
|
||||||
expect(result).to.equal('failed');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('disables web security when set', async () => {
|
|
||||||
const result = await loadFileInWebView(webview, { disablewebsecurity: '' });
|
|
||||||
expect(result).to.equal('loaded');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('does not break node integration', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
disablewebsecurity: '',
|
|
||||||
nodeintegration: 'on',
|
|
||||||
webpreferences: 'contextIsolation=no',
|
|
||||||
src: `file://${fixtures}/pages/d.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function',
|
|
||||||
module: 'object',
|
|
||||||
process: 'object'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('does not break preload script', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
disablewebsecurity: '',
|
|
||||||
preload: `${fixtures}/module/preload.js`,
|
|
||||||
webpreferences: 'sandbox=no',
|
|
||||||
src: `file://${fixtures}/pages/e.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function',
|
|
||||||
module: 'object',
|
|
||||||
process: 'object',
|
|
||||||
Buffer: 'function'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('partition attribute', () => {
|
|
||||||
it('inserts no node symbols when not set', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
partition: 'test1',
|
|
||||||
src: `file://${fixtures}/pages/c.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'undefined',
|
|
||||||
module: 'undefined',
|
|
||||||
process: 'undefined',
|
|
||||||
global: 'undefined'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('inserts node symbols when set', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
nodeintegration: 'on',
|
|
||||||
partition: 'test2',
|
|
||||||
webpreferences: 'contextIsolation=no',
|
|
||||||
src: `file://${fixtures}/pages/d.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function',
|
|
||||||
module: 'object',
|
|
||||||
process: 'object'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('isolates storage for different id', async () => {
|
|
||||||
window.localStorage.setItem('test', 'one');
|
|
||||||
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
partition: 'test3',
|
|
||||||
src: `file://${fixtures}/pages/partition/one.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const parsedMessage = JSON.parse(message);
|
|
||||||
expect(parsedMessage).to.include({
|
|
||||||
numberOfEntries: 0,
|
|
||||||
testValue: null
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('uses current session storage when no id is provided', async () => {
|
|
||||||
const testValue = 'one';
|
|
||||||
window.localStorage.setItem('test', testValue);
|
|
||||||
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
src: `file://${fixtures}/pages/partition/one.html`
|
|
||||||
});
|
|
||||||
|
|
||||||
const parsedMessage = JSON.parse(message);
|
|
||||||
expect(parsedMessage).to.include({
|
|
||||||
numberOfEntries: 1,
|
|
||||||
testValue
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('allowpopups attribute', () => {
|
|
||||||
const generateSpecs = (description, webpreferences = '') => {
|
|
||||||
describe(description, () => {
|
|
||||||
it('can not open new window when not set', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
webpreferences,
|
|
||||||
src: `file://${fixtures}/pages/window-open-hide.html`
|
|
||||||
});
|
|
||||||
expect(message).to.equal('null');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('can open new window when set', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
webpreferences,
|
|
||||||
allowpopups: 'on',
|
|
||||||
src: `file://${fixtures}/pages/window-open-hide.html`
|
|
||||||
});
|
|
||||||
expect(message).to.equal('window');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
generateSpecs('without sandbox');
|
|
||||||
generateSpecs('with sandbox', 'sandbox=yes');
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('webpreferences attribute', () => {
|
|
||||||
it('can enable nodeintegration', async () => {
|
|
||||||
const message = await startLoadingWebViewAndWaitForMessage(webview, {
|
|
||||||
src: `file://${fixtures}/pages/d.html`,
|
|
||||||
webpreferences: 'nodeIntegration,contextIsolation=no'
|
|
||||||
});
|
|
||||||
|
|
||||||
const types = JSON.parse(message);
|
|
||||||
expect(types).to.include({
|
|
||||||
require: 'function',
|
|
||||||
module: 'object',
|
|
||||||
process: 'object'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('can disables web security and enable nodeintegration', async () => {
|
|
||||||
const result = await loadFileInWebView(webview, { webpreferences: 'webSecurity=no, nodeIntegration=yes, contextIsolation=no' });
|
|
||||||
expect(result).to.equal('loaded');
|
|
||||||
const type = await webview.executeJavaScript('typeof require');
|
|
||||||
expect(type).to.equal('function');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('new-window event', () => {
|
describe('new-window event', () => {
|
||||||
it('emits when window.open is called', async () => {
|
it('emits when window.open is called', async () => {
|
||||||
loadWebView(webview, {
|
loadWebView(webview, {
|
||||||
|
|
Loading…
Reference in a new issue