2024-10-03 04:36:15 +00:00
import { MediaAccessPermissionRequest } from 'electron' ;
2024-04-10 20:06:47 +00:00
import { clipboard } from 'electron/common' ;
2024-10-03 04:36:15 +00:00
import { BrowserWindow , WebContents , webFrameMain , session , ipcMain , app , protocol , webContents , dialog , MessageBoxOptions } from 'electron/main' ;
import { expect } from 'chai' ;
import * as ws from 'ws' ;
import * as ChildProcess from 'node:child_process' ;
import { EventEmitter , once } from 'node:events' ;
import * as fs from 'node:fs' ;
2023-06-15 14:42:27 +00:00
import * as http from 'node:http' ;
2024-10-03 04:36:15 +00:00
import * as https from 'node:https' ;
import { AddressInfo } from 'node:net' ;
2023-06-15 14:42:27 +00:00
import * as path from 'node:path' ;
2024-10-03 04:36:15 +00:00
import { setTimeout } from 'node:timers/promises' ;
2023-06-15 14:42:27 +00:00
import * as url from 'node:url' ;
2024-10-03 04:36:15 +00:00
2024-10-24 13:21:55 +00:00
import { ifit , ifdescribe , defer , itremote , listen , startRemoteControlApp , waitUntil } from './lib/spec-helpers' ;
2024-10-03 04:36:15 +00:00
import { closeAllWindows } from './lib/window-helpers' ;
2020-05-08 04:27:55 +00:00
import { PipeTransport } from './pipe-transport' ;
2020-03-20 20:28:31 +00:00
2020-06-23 03:32:45 +00:00
const features = process . _linkedBinding ( 'electron_common_features' ) ;
2020-03-20 20:28:31 +00:00
2022-08-16 19:23:13 +00:00
const fixturesPath = path . resolve ( __dirname , 'fixtures' ) ;
2023-04-19 12:24:38 +00:00
const certPath = path . join ( fixturesPath , 'certificates' ) ;
2019-05-29 23:33:19 +00:00
describe ( 'reporting api' , ( ) = > {
2023-04-19 12:24:38 +00:00
it ( 'sends a report for an intervention' , async ( ) = > {
const reporting = new EventEmitter ( ) ;
2019-05-29 23:33:19 +00:00
// The Reporting API only works on https with valid certs. To dodge having
// to set up a trusted certificate, hack the validator.
session . defaultSession . setCertificateVerifyProc ( ( req , cb ) = > {
2020-03-20 20:28:31 +00:00
cb ( 0 ) ;
} ) ;
2023-04-19 12:24:38 +00:00
2019-05-29 23:33:19 +00:00
const options = {
key : fs.readFileSync ( path . join ( certPath , 'server.key' ) ) ,
cert : fs.readFileSync ( path . join ( certPath , 'server.pem' ) ) ,
ca : [
fs . readFileSync ( path . join ( certPath , 'rootCA.pem' ) ) ,
fs . readFileSync ( path . join ( certPath , 'intermediateCA.pem' ) )
] ,
requestCert : true ,
rejectUnauthorized : false
2020-03-20 20:28:31 +00:00
} ;
2019-05-29 23:33:19 +00:00
const server = https . createServer ( options , ( req , res ) = > {
2023-04-19 12:24:38 +00:00
if ( req . url ? . endsWith ( 'report' ) ) {
2020-03-20 20:28:31 +00:00
let data = '' ;
req . on ( 'data' , ( d ) = > { data += d . toString ( 'utf-8' ) ; } ) ;
2019-05-29 23:33:19 +00:00
req . on ( 'end' , ( ) = > {
2023-04-19 12:24:38 +00:00
reporting . emit ( 'report' , JSON . parse ( data ) ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-05-29 23:33:19 +00:00
}
2023-04-19 12:24:38 +00:00
const { port } = server . address ( ) as any ;
res . setHeader ( 'Reporting-Endpoints' , ` default="https://localhost: ${ port } /report" ` ) ;
2020-03-20 20:28:31 +00:00
res . setHeader ( 'Content-Type' , 'text/html' ) ;
2023-04-19 12:24:38 +00:00
res . end ( '<script>window.navigator.vibrate(1)</script>' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2023-04-19 12:24:38 +00:00
2023-09-04 10:33:29 +00:00
await listen ( server ) ;
2023-04-19 12:24:38 +00:00
const bw = new BrowserWindow ( { show : false } ) ;
2019-05-29 23:33:19 +00:00
try {
2023-04-19 12:24:38 +00:00
const reportGenerated = once ( reporting , 'report' ) ;
2023-09-04 10:33:29 +00:00
await bw . loadURL ( ` https://localhost: ${ ( server . address ( ) as AddressInfo ) . port } /a ` ) ;
2023-04-19 12:24:38 +00:00
const [ reports ] = await reportGenerated ;
expect ( reports ) . to . be . an ( 'array' ) . with . lengthOf ( 1 ) ;
const { type , url , body } = reports [ 0 ] ;
expect ( type ) . to . equal ( 'intervention' ) ;
expect ( url ) . to . equal ( url ) ;
expect ( body . id ) . to . equal ( 'NavigatorVibrate' ) ;
expect ( body . message ) . to . match ( /Blocked call to navigator.vibrate because user hasn't tapped on the frame or any embedded frame yet/ ) ;
2019-05-29 23:33:19 +00:00
} finally {
2020-03-20 20:28:31 +00:00
bw . destroy ( ) ;
server . close ( ) ;
2019-05-29 23:33:19 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2019-06-11 23:35:58 +00:00
describe ( 'window.postMessage' , ( ) = > {
afterEach ( async ( ) = > {
2020-03-20 20:28:31 +00:00
await closeAllWindows ( ) ;
} ) ;
2019-06-11 23:35:58 +00:00
2022-01-06 17:28:03 +00:00
it ( 'sets the source and origin correctly' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { nodeIntegration : true , contextIsolation : false } } ) ;
w . loadURL ( ` file:// ${ fixturesPath } /pages/window-open-postMessage-driver.html ` ) ;
2023-02-23 23:53:53 +00:00
const [ , message ] = await once ( ipcMain , 'complete' ) ;
2022-01-06 17:28:03 +00:00
expect ( message . data ) . to . equal ( 'testing' ) ;
expect ( message . origin ) . to . equal ( 'file://' ) ;
expect ( message . sourceEqualsOpener ) . to . equal ( true ) ;
expect ( message . eventOrigin ) . to . equal ( 'file://' ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-06-15 07:15:15 +00:00
describe ( 'focus handling' , ( ) = > {
2023-02-19 09:24:24 +00:00
let webviewContents : WebContents ;
let w : BrowserWindow ;
2019-06-15 07:15:15 +00:00
beforeEach ( async ( ) = > {
w = new BrowserWindow ( {
show : true ,
webPreferences : {
nodeIntegration : true ,
2021-03-01 21:52:29 +00:00
webviewTag : true ,
contextIsolation : false
2019-06-15 07:15:15 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-06-15 07:15:15 +00:00
2023-06-22 18:38:52 +00:00
const webviewReady = once ( w . webContents , 'did-attach-webview' ) as Promise < [ any , WebContents ] > ;
2020-03-20 20:28:31 +00:00
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'tab-focus-loop-elements.html' ) ) ;
const [ , wvContents ] = await webviewReady ;
webviewContents = wvContents ;
2023-02-23 23:53:53 +00:00
await once ( webviewContents , 'did-finish-load' ) ;
2020-03-20 20:28:31 +00:00
w . focus ( ) ;
} ) ;
2019-06-15 07:15:15 +00:00
afterEach ( ( ) = > {
2020-03-20 20:28:31 +00:00
webviewContents = null as unknown as WebContents ;
w . destroy ( ) ;
w = null as unknown as BrowserWindow ;
} ) ;
2019-06-15 07:15:15 +00:00
const expectFocusChange = async ( ) = > {
2023-02-23 23:53:53 +00:00
const [ , focusedElementId ] = await once ( ipcMain , 'focus-changed' ) ;
2020-03-20 20:28:31 +00:00
return focusedElementId ;
} ;
2019-06-15 07:15:15 +00:00
describe ( 'a TAB press' , ( ) = > {
const tabPressEvent : any = {
type : 'keyDown' ,
keyCode : 'Tab'
2020-03-20 20:28:31 +00:00
} ;
2019-06-15 07:15:15 +00:00
it ( 'moves focus to the next focusable item' , async ( ) = > {
2020-03-20 20:28:31 +00:00
let focusChange = expectFocusChange ( ) ;
w . webContents . sendInputEvent ( tabPressEvent ) ;
let focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-element-1' , ` should start focused in element-1, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
w . webContents . sendInputEvent ( tabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-element-2' , ` focus should've moved to element-2, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
w . webContents . sendInputEvent ( tabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-wv-element-1' , ` focus should've moved to the webview's element-1, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
webviewContents . sendInputEvent ( tabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-wv-element-2' , ` focus should've moved to the webview's element-2, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
webviewContents . sendInputEvent ( tabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-element-3' , ` focus should've moved to element-3, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
w . webContents . sendInputEvent ( tabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-element-1' , ` focus should've looped back to element-1, it's instead in ${ focusedElementId } ` ) ;
} ) ;
} ) ;
2019-06-15 07:15:15 +00:00
describe ( 'a SHIFT + TAB press' , ( ) = > {
const shiftTabPressEvent : any = {
type : 'keyDown' ,
modifiers : [ 'Shift' ] ,
keyCode : 'Tab'
2020-03-20 20:28:31 +00:00
} ;
2019-06-15 07:15:15 +00:00
it ( 'moves focus to the previous focusable item' , async ( ) = > {
2020-03-20 20:28:31 +00:00
let focusChange = expectFocusChange ( ) ;
w . webContents . sendInputEvent ( shiftTabPressEvent ) ;
let focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-element-3' , ` should start focused in element-3, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
w . webContents . sendInputEvent ( shiftTabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-wv-element-2' , ` focus should've moved to the webview's element-2, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
webviewContents . sendInputEvent ( shiftTabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-wv-element-1' , ` focus should've moved to the webview's element-1, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
webviewContents . sendInputEvent ( shiftTabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-element-2' , ` focus should've moved to element-2, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
w . webContents . sendInputEvent ( shiftTabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-element-1' , ` focus should've moved to element-1, it's instead in ${ focusedElementId } ` ) ;
focusChange = expectFocusChange ( ) ;
w . webContents . sendInputEvent ( shiftTabPressEvent ) ;
focusedElementId = await focusChange ;
expect ( focusedElementId ) . to . equal ( 'BUTTON-element-3' , ` focus should've looped back to element-3, it's instead in ${ focusedElementId } ` ) ;
} ) ;
} ) ;
} ) ;
2019-08-02 23:56:46 +00:00
describe ( 'web security' , ( ) = > {
2020-03-20 20:28:31 +00:00
afterEach ( closeAllWindows ) ;
let server : http.Server ;
let serverUrl : string ;
2019-08-02 23:56:46 +00:00
before ( async ( ) = > {
server = http . createServer ( ( req , res ) = > {
2020-03-20 20:28:31 +00:00
res . setHeader ( 'Content-Type' , 'text/html' ) ;
res . end ( '<body>' ) ;
} ) ;
2023-02-20 11:30:57 +00:00
serverUrl = ( await listen ( server ) ) . url ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-08-02 23:56:46 +00:00
after ( ( ) = > {
2020-03-20 20:28:31 +00:00
server . close ( ) ;
} ) ;
2019-08-02 23:56:46 +00:00
it ( 'engages CORB when web security is not disabled' , async ( ) = > {
2021-03-01 21:52:29 +00:00
const w = new BrowserWindow ( { show : false , webPreferences : { webSecurity : true , nodeIntegration : true , contextIsolation : false } } ) ;
2023-02-23 23:53:53 +00:00
const p = once ( ipcMain , 'success' ) ;
2019-08-02 23:56:46 +00:00
await w . loadURL ( ` data:text/html,<script>
const s = document . createElement ( 'script' )
s . src = "${serverUrl}"
2024-02-14 17:33:32 +00:00
// The script will not load under ORB, refs https://chromium-review.googlesource.com/c/chromium/src/+/3785025.
// Before ORB an empty response is sent, which is now replaced by a network error.
s . onerror = ( ) = > { require ( 'electron' ) . ipcRenderer . send ( 'success' ) }
2019-08-02 23:56:46 +00:00
document . documentElement . appendChild ( s )
2020-03-20 20:28:31 +00:00
< / script > ` );
await p ;
} ) ;
2019-08-02 23:56:46 +00:00
2022-04-12 10:00:42 +00:00
it ( 'bypasses CORB when web security is disabled' , async ( ) = > {
2021-03-01 21:52:29 +00:00
const w = new BrowserWindow ( { show : false , webPreferences : { webSecurity : false , nodeIntegration : true , contextIsolation : false } } ) ;
2023-02-23 23:53:53 +00:00
const p = once ( ipcMain , 'success' ) ;
2019-08-02 23:56:46 +00:00
await w . loadURL ( ` data:text/html,
< script >
window . onerror = ( e ) = > { require ( 'electron' ) . ipcRenderer . send ( 'success' , e ) }
< / script >
2020-03-20 20:28:31 +00:00
< script src = "${serverUrl}" > < / script > ` );
await p ;
} ) ;
2020-08-28 01:43:08 +00:00
2020-09-16 21:55:53 +00:00
it ( 'engages CORS when web security is not disabled' , async ( ) = > {
2021-03-01 21:52:29 +00:00
const w = new BrowserWindow ( { show : false , webPreferences : { webSecurity : true , nodeIntegration : true , contextIsolation : false } } ) ;
2023-02-23 23:53:53 +00:00
const p = once ( ipcMain , 'response' ) ;
2020-09-16 21:55:53 +00:00
await w . loadURL ( ` data:text/html,<script>
( async function ( ) {
try {
await fetch ( '${serverUrl}' ) ;
require ( 'electron' ) . ipcRenderer . send ( 'response' , 'passed' ) ;
} catch {
require ( 'electron' ) . ipcRenderer . send ( 'response' , 'failed' ) ;
}
} ) ( ) ;
< / script > ` );
const [ , response ] = await p ;
expect ( response ) . to . equal ( 'failed' ) ;
} ) ;
it ( 'bypasses CORS when web security is disabled' , async ( ) = > {
2021-03-01 21:52:29 +00:00
const w = new BrowserWindow ( { show : false , webPreferences : { webSecurity : false , nodeIntegration : true , contextIsolation : false } } ) ;
2023-02-23 23:53:53 +00:00
const p = once ( ipcMain , 'response' ) ;
2020-09-16 21:55:53 +00:00
await w . loadURL ( ` data:text/html,<script>
( async function ( ) {
try {
await fetch ( '${serverUrl}' ) ;
require ( 'electron' ) . ipcRenderer . send ( 'response' , 'passed' ) ;
} catch {
require ( 'electron' ) . ipcRenderer . send ( 'response' , 'failed' ) ;
}
} ) ( ) ;
< / script > ` );
const [ , response ] = await p ;
expect ( response ) . to . equal ( 'passed' ) ;
} ) ;
2021-04-07 01:46:23 +00:00
describe ( 'accessing file://' , ( ) = > {
async function loadFile ( w : BrowserWindow ) {
const thisFile = url . format ( {
2023-09-07 06:50:14 +00:00
pathname : __filename.replaceAll ( '\\' , '/' ) ,
2021-04-07 01:46:23 +00:00
protocol : 'file' ,
slashes : true
} ) ;
await w . loadURL ( ` data:text/html,<script>
function loadFile() {
return new Promise ( ( resolve ) = > {
fetch ( '${thisFile}' ) . then (
( ) = > resolve ( 'loaded' ) ,
( ) = > resolve ( 'failed' )
)
} ) ;
}
< / script > ` );
return await w . webContents . executeJavaScript ( 'loadFile()' ) ;
}
it ( 'is forbidden when web security is enabled' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { webSecurity : true } } ) ;
const result = await loadFile ( w ) ;
expect ( result ) . to . equal ( 'failed' ) ;
} ) ;
it ( 'is allowed when web security is disabled' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { webSecurity : false } } ) ;
const result = await loadFile ( w ) ;
expect ( result ) . to . equal ( 'loaded' ) ;
} ) ;
} ) ;
2021-04-07 20:04:50 +00:00
describe ( 'wasm-eval csp' , ( ) = > {
async function loadWasm ( csp : string ) {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
sandbox : true ,
enableBlinkFeatures : 'WebAssemblyCSP'
}
} ) ;
await w . loadURL ( ` data:text/html,<head>
< meta http-equiv = "Content-Security-Policy" content = "default-src 'self'; script-src 'self' 'unsafe-inline' ${csp}" >
< / head >
< script >
function loadWasm() {
const wasmBin = new Uint8Array ( [ 0 , 97 , 115 , 109 , 1 , 0 , 0 , 0 ] )
return new Promise ( ( resolve ) = > {
WebAssembly . instantiate ( wasmBin ) . then ( ( ) = > {
resolve ( 'loaded' )
} ) . catch ( ( error ) = > {
resolve ( error . message )
} )
} ) ;
}
< / script > ` );
return await w . webContents . executeJavaScript ( 'loadWasm()' ) ;
}
it ( 'wasm codegen is disallowed by default' , async ( ) = > {
const r = await loadWasm ( '' ) ;
2022-07-20 11:03:34 +00:00
expect ( r ) . to . equal ( 'WebAssembly.instantiate(): Refused to compile or instantiate WebAssembly module because \'unsafe-eval\' is not an allowed source of script in the following Content Security Policy directive: "script-src \'self\' \'unsafe-inline\'"' ) ;
2021-04-07 20:04:50 +00:00
} ) ;
2021-06-22 19:17:16 +00:00
it ( 'wasm codegen is allowed with "wasm-unsafe-eval" csp' , async ( ) = > {
const r = await loadWasm ( "'wasm-unsafe-eval'" ) ;
2021-04-07 20:04:50 +00:00
expect ( r ) . to . equal ( 'loaded' ) ;
} ) ;
} ) ;
2023-04-04 08:31:49 +00:00
describe ( 'csp' , ( ) = > {
for ( const sandbox of [ true , false ] ) {
describe ( ` when sandbox: ${ sandbox } ` , ( ) = > {
for ( const contextIsolation of [ true , false ] ) {
describe ( ` when contextIsolation: ${ contextIsolation } ` , ( ) = > {
it ( 'prevents eval from running in an inline script' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : { sandbox , contextIsolation }
} ) ;
w . loadURL ( ` data:text/html,<head>
< meta http-equiv = "Content-Security-Policy" content = "default-src 'self'; script-src 'self' 'unsafe-inline'" >
< / head >
< script >
try {
// We use console.log here because it is easier than making a
// preload script, and the behavior under test changes when
// contextIsolation: false
console . log ( eval ( 'true' ) )
} catch ( e ) {
console . log ( e . message )
}
< / script > ` );
const [ , , message ] = await once ( w . webContents , 'console-message' ) ;
expect ( message ) . to . match ( /Refused to evaluate a string/ ) ;
} ) ;
it ( 'does not prevent eval from running in an inline script when there is no csp' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : { sandbox , contextIsolation }
} ) ;
w . loadURL ( ` data:text/html,
< script >
try {
// We use console.log here because it is easier than making a
// preload script, and the behavior under test changes when
// contextIsolation: false
console . log ( eval ( 'true' ) )
} catch ( e ) {
console . log ( e . message )
}
< / script > ` );
const [ , , message ] = await once ( w . webContents , 'console-message' ) ;
expect ( message ) . to . equal ( 'true' ) ;
} ) ;
it ( 'prevents eval from running in executeJavaScript' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : { sandbox , contextIsolation }
} ) ;
w . loadURL ( 'data:text/html,<head><meta http-equiv="Content-Security-Policy" content="default-src \'self\'; script-src \'self\' \'unsafe-inline\'"></meta></head>' ) ;
await expect ( w . webContents . executeJavaScript ( 'eval("true")' ) ) . to . be . rejected ( ) ;
} ) ;
it ( 'does not prevent eval from running in executeJavaScript when there is no csp' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : { sandbox , contextIsolation }
} ) ;
w . loadURL ( 'data:text/html,' ) ;
expect ( await w . webContents . executeJavaScript ( 'eval("true")' ) ) . to . be . true ( ) ;
} ) ;
} ) ;
}
2022-12-14 18:05:34 +00:00
} ) ;
2023-04-04 08:31:49 +00:00
}
2022-12-14 18:05:34 +00:00
} ) ;
2020-08-28 01:43:08 +00:00
it ( 'does not crash when multiple WebContent are created with web security disabled' , ( ) = > {
2020-09-15 18:48:12 +00:00
const options = { show : false , webPreferences : { webSecurity : false } } ;
2020-08-28 01:43:08 +00:00
const w1 = new BrowserWindow ( options ) ;
w1 . loadURL ( serverUrl ) ;
const w2 = new BrowserWindow ( options ) ;
w2 . loadURL ( serverUrl ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
2021-04-09 07:09:17 +00:00
describe ( 'command line switches' , ( ) = > {
2020-05-08 23:13:32 +00:00
let appProcess : ChildProcess.ChildProcessWithoutNullStreams | undefined ;
afterEach ( ( ) = > {
if ( appProcess && ! appProcess . killed ) {
appProcess . kill ( ) ;
appProcess = undefined ;
}
} ) ;
2019-10-11 20:55:50 +00:00
describe ( '--lang switch' , ( ) = > {
2020-03-20 20:28:31 +00:00
const currentLocale = app . getLocale ( ) ;
2022-09-23 18:50:46 +00:00
const currentSystemLocale = app . getSystemLocale ( ) ;
2022-11-09 15:50:43 +00:00
const currentPreferredLanguages = JSON . stringify ( app . getPreferredSystemLanguages ( ) ) ;
2020-11-18 02:13:01 +00:00
const testLocale = async ( locale : string , result : string , printEnv : boolean = false ) = > {
2020-03-20 20:28:31 +00:00
const appPath = path . join ( fixturesPath , 'api' , 'locale-check' ) ;
2020-11-18 02:13:01 +00:00
const args = [ appPath , ` --set-lang= ${ locale } ` ] ;
if ( printEnv ) {
args . push ( '--print-env' ) ;
}
appProcess = ChildProcess . spawn ( process . execPath , args ) ;
2019-10-11 20:55:50 +00:00
2020-06-30 22:10:36 +00:00
let output = '' ;
2020-03-20 20:28:31 +00:00
appProcess . stdout . on ( 'data' , ( data ) = > { output += data ; } ) ;
2021-03-08 16:23:55 +00:00
let stderr = '' ;
appProcess . stderr . on ( 'data' , ( data ) = > { stderr += data ; } ) ;
2023-02-23 23:53:53 +00:00
const [ code , signal ] = await once ( appProcess , 'exit' ) ;
2021-03-08 16:23:55 +00:00
if ( code !== 0 ) {
throw new Error ( ` Process exited with code " ${ code } " signal " ${ signal } " output " ${ output } " stderr " ${ stderr } " ` ) ;
}
2020-06-30 22:10:36 +00:00
2023-09-07 06:50:14 +00:00
output = output . replaceAll ( /(\r\n|\n|\r)/gm , '' ) ;
2020-06-30 22:10:36 +00:00
expect ( output ) . to . equal ( result ) ;
2020-03-20 20:28:31 +00:00
} ;
2019-10-11 20:55:50 +00:00
2022-11-09 15:50:43 +00:00
it ( 'should set the locale' , async ( ) = > testLocale ( 'fr' , ` fr| ${ currentSystemLocale } | ${ currentPreferredLanguages } ` ) ) ;
it ( 'should set the locale with country code' , async ( ) = > testLocale ( 'zh-CN' , ` zh-CN| ${ currentSystemLocale } | ${ currentPreferredLanguages } ` ) ) ;
it ( 'should not set an invalid locale' , async ( ) = > testLocale ( 'asdfkl' , ` ${ currentLocale } | ${ currentSystemLocale } | ${ currentPreferredLanguages } ` ) ) ;
2020-11-18 02:13:01 +00:00
const lcAll = String ( process . env . LC_ALL ) ;
ifit ( process . platform === 'linux' ) ( 'current process has a valid LC_ALL env' , async ( ) = > {
// The LC_ALL env should not be set to DOM locale string.
expect ( lcAll ) . to . not . equal ( app . getLocale ( ) ) ;
} ) ;
2021-03-10 18:04:57 +00:00
ifit ( process . platform === 'linux' ) ( 'should not change LC_ALL' , async ( ) = > testLocale ( 'fr' , lcAll , true ) ) ;
2020-11-18 02:13:01 +00:00
ifit ( process . platform === 'linux' ) ( 'should not change LC_ALL when setting invalid locale' , async ( ) = > testLocale ( 'asdfkl' , lcAll , true ) ) ;
ifit ( process . platform === 'linux' ) ( 'should not change LC_ALL when --lang is not set' , async ( ) = > testLocale ( '' , lcAll , true ) ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
2021-03-16 21:02:47 +00:00
describe ( '--remote-debugging-pipe switch' , ( ) = > {
2020-05-08 04:27:55 +00:00
it ( 'should expose CDP via pipe' , async ( ) = > {
const electronPath = process . execPath ;
2020-05-08 23:13:32 +00:00
appProcess = ChildProcess . spawn ( electronPath , [ '--remote-debugging-pipe' ] , {
2021-02-23 00:16:17 +00:00
stdio : [ 'inherit' , 'inherit' , 'inherit' , 'pipe' , 'pipe' ]
} ) as ChildProcess . ChildProcessWithoutNullStreams ;
2020-05-08 04:27:55 +00:00
const stdio = appProcess . stdio as unknown as [ NodeJS . ReadableStream , NodeJS . WritableStream , NodeJS . WritableStream , NodeJS . WritableStream , NodeJS . ReadableStream ] ;
const pipe = new PipeTransport ( stdio [ 3 ] , stdio [ 4 ] ) ;
const versionPromise = new Promise ( resolve = > { pipe . onmessage = resolve ; } ) ;
pipe . send ( { id : 1 , method : 'Browser.getVersion' , params : { } } ) ;
const message = ( await versionPromise ) as any ;
expect ( message . id ) . to . equal ( 1 ) ;
expect ( message . result . product ) . to . contain ( 'Chrome' ) ;
expect ( message . result . userAgent ) . to . contain ( 'Electron' ) ;
} ) ;
it ( 'should override --remote-debugging-port switch' , async ( ) = > {
const electronPath = process . execPath ;
2020-05-08 23:13:32 +00:00
appProcess = ChildProcess . spawn ( electronPath , [ '--remote-debugging-pipe' , '--remote-debugging-port=0' ] , {
2021-02-23 00:16:17 +00:00
stdio : [ 'inherit' , 'inherit' , 'pipe' , 'pipe' , 'pipe' ]
} ) as ChildProcess . ChildProcessWithoutNullStreams ;
2020-05-08 04:27:55 +00:00
let stderr = '' ;
2020-05-08 23:13:32 +00:00
appProcess . stderr . on ( 'data' , ( data : string ) = > { stderr += data ; } ) ;
2020-05-08 04:27:55 +00:00
const stdio = appProcess . stdio as unknown as [ NodeJS . ReadableStream , NodeJS . WritableStream , NodeJS . WritableStream , NodeJS . WritableStream , NodeJS . ReadableStream ] ;
const pipe = new PipeTransport ( stdio [ 3 ] , stdio [ 4 ] ) ;
const versionPromise = new Promise ( resolve = > { pipe . onmessage = resolve ; } ) ;
pipe . send ( { id : 1 , method : 'Browser.getVersion' , params : { } } ) ;
const message = ( await versionPromise ) as any ;
expect ( message . id ) . to . equal ( 1 ) ;
expect ( stderr ) . to . not . include ( 'DevTools listening on' ) ;
2020-05-08 23:13:32 +00:00
} ) ;
it ( 'should shut down Electron upon Browser.close CDP command' , async ( ) = > {
const electronPath = process . execPath ;
appProcess = ChildProcess . spawn ( electronPath , [ '--remote-debugging-pipe' ] , {
2021-02-23 00:16:17 +00:00
stdio : [ 'inherit' , 'inherit' , 'inherit' , 'pipe' , 'pipe' ]
} ) as ChildProcess . ChildProcessWithoutNullStreams ;
2020-05-08 23:13:32 +00:00
const stdio = appProcess . stdio as unknown as [ NodeJS . ReadableStream , NodeJS . WritableStream , NodeJS . WritableStream , NodeJS . WritableStream , NodeJS . ReadableStream ] ;
const pipe = new PipeTransport ( stdio [ 3 ] , stdio [ 4 ] ) ;
pipe . send ( { id : 1 , method : 'Browser.close' , params : { } } ) ;
2023-02-23 23:53:53 +00:00
await once ( appProcess , 'exit' ) ;
2020-05-08 04:27:55 +00:00
} ) ;
} ) ;
2021-03-16 21:02:47 +00:00
describe ( '--remote-debugging-port switch' , ( ) = > {
2019-10-11 20:55:50 +00:00
it ( 'should display the discovery page' , ( done ) = > {
2020-03-20 20:28:31 +00:00
const electronPath = process . execPath ;
let output = '' ;
2020-05-08 23:13:32 +00:00
appProcess = ChildProcess . spawn ( electronPath , [ '--remote-debugging-port=' ] ) ;
2021-02-23 00:16:17 +00:00
appProcess . stdout . on ( 'data' , ( data ) = > {
console . log ( data ) ;
} ) ;
2019-10-11 20:55:50 +00:00
appProcess . stderr . on ( 'data' , ( data ) = > {
2021-02-23 00:16:17 +00:00
console . log ( data ) ;
2020-03-20 20:28:31 +00:00
output += data ;
const m = /DevTools listening on ws:\/\/127.0.0.1:(\d+)\// . exec ( output ) ;
2019-10-11 20:55:50 +00:00
if ( m ) {
2020-05-08 23:13:32 +00:00
appProcess ! . stderr . removeAllListeners ( 'data' ) ;
2020-03-20 20:28:31 +00:00
const port = m [ 1 ] ;
2019-10-11 20:55:50 +00:00
http . get ( ` http://127.0.0.1: ${ port } ` , ( res ) = > {
2020-06-30 22:10:36 +00:00
try {
expect ( res . statusCode ) . to . eql ( 200 ) ;
expect ( parseInt ( res . headers [ 'content-length' ] ! ) ) . to . be . greaterThan ( 0 ) ;
done ( ) ;
} catch ( e ) {
done ( e ) ;
} finally {
res . destroy ( ) ;
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
} ) ;
2024-10-18 15:16:53 +00:00
describe ( '--trace-startup switch' , ( ) = > {
const outputFilePath = path . join ( app . getPath ( 'temp' ) , 'trace.json' ) ;
afterEach ( ( ) = > {
if ( fs . existsSync ( outputFilePath ) ) {
fs . unlinkSync ( outputFilePath ) ;
}
} ) ;
// Disable the test on linux arm and arm64 to avoid startup crash
// https://github.com/electron/electron/issues/44293#issuecomment-2420077154
ifit ( process . platform !== 'linux' || ( process . arch !== 'arm' && process . arch !== 'arm64' ) ) ( 'creates startup trace' , async ( ) = > {
const rc = await startRemoteControlApp ( [ '--trace-startup=*' , ` --trace-startup-file= ${ outputFilePath } ` , '--trace-startup-duration=1' , '--enable-logging' ] ) ;
const stderrComplete = new Promise < string > ( resolve = > {
let stderr = '' ;
rc . process . stderr ! . on ( 'data' , ( chunk ) = > {
stderr += chunk . toString ( 'utf8' ) ;
} ) ;
rc . process . on ( 'close' , ( ) = > { resolve ( stderr ) ; } ) ;
} ) ;
rc . remotely ( ( ) = > {
global . setTimeout ( ( ) = > {
require ( 'electron' ) . app . quit ( ) ;
} , 5000 ) ;
} ) ;
const stderr = await stderrComplete ;
expect ( stderr ) . to . match ( /Completed startup tracing to/ ) ;
expect ( fs . existsSync ( outputFilePath ) ) . to . be . true ( 'output exists' ) ;
expect ( fs . statSync ( outputFilePath ) . size ) . to . be . above ( 0 ,
` the trace output file is empty, check " ${ outputFilePath } " ` ) ;
} ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
describe ( 'chromium features' , ( ) = > {
2020-03-20 20:28:31 +00:00
afterEach ( closeAllWindows ) ;
2019-10-11 20:55:50 +00:00
describe ( 'accessing key names also used as Node.js module names' , ( ) = > {
it ( 'does not crash' , ( done ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . webContents . once ( 'did-finish-load' , ( ) = > { done ( ) ; } ) ;
2023-02-16 09:25:41 +00:00
w . webContents . once ( 'render-process-gone' , ( ) = > done ( new Error ( 'WebContents crashed.' ) ) ) ;
2020-03-20 20:28:31 +00:00
w . loadFile ( path . join ( fixturesPath , 'pages' , 'external-string.html' ) ) ;
} ) ;
} ) ;
2019-10-11 20:55:50 +00:00
2022-07-07 19:14:53 +00:00
describe ( 'first party sets' , ( ) = > {
const fps = [
'https://fps-member1.glitch.me' ,
'https://fps-member2.glitch.me' ,
'https://fps-member3.glitch.me'
] ;
it ( 'loads first party sets' , async ( ) = > {
const appPath = path . join ( fixturesPath , 'api' , 'first-party-sets' , 'base' ) ;
const fpsProcess = ChildProcess . spawn ( process . execPath , [ appPath ] ) ;
let output = '' ;
fpsProcess . stdout . on ( 'data' , data = > { output += data ; } ) ;
2023-02-23 23:53:53 +00:00
await once ( fpsProcess , 'exit' ) ;
2022-07-07 19:14:53 +00:00
expect ( output ) . to . include ( fps . join ( ',' ) ) ;
} ) ;
it ( 'loads sets from the command line' , async ( ) = > {
const appPath = path . join ( fixturesPath , 'api' , 'first-party-sets' , 'command-line' ) ;
const args = [ appPath , ` --use-first-party-set= ${ fps } ` ] ;
const fpsProcess = ChildProcess . spawn ( process . execPath , args ) ;
let output = '' ;
fpsProcess . stdout . on ( 'data' , data = > { output += data ; } ) ;
2023-02-23 23:53:53 +00:00
await once ( fpsProcess , 'exit' ) ;
2022-07-07 19:14:53 +00:00
expect ( output ) . to . include ( fps . join ( ',' ) ) ;
} ) ;
} ) ;
2019-10-11 20:55:50 +00:00
describe ( 'loading jquery' , ( ) = > {
it ( 'does not crash' , ( done ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . webContents . once ( 'did-finish-load' , ( ) = > { done ( ) ; } ) ;
2023-02-16 09:25:41 +00:00
w . webContents . once ( 'render-process-gone' , ( ) = > done ( new Error ( 'WebContents crashed.' ) ) ) ;
2020-03-20 20:28:31 +00:00
w . loadFile ( path . join ( __dirname , 'fixtures' , 'pages' , 'jquery.html' ) ) ;
} ) ;
} ) ;
2019-10-11 20:55:50 +00:00
2023-10-31 15:59:39 +00:00
describe ( 'navigator.keyboard' , ( ) = > {
afterEach ( closeAllWindows ) ;
it ( 'getLayoutMap() should return a KeyboardLayoutMap object' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const size = await w . webContents . executeJavaScript ( `
navigator . keyboard . getLayoutMap ( ) . then ( map = > map . size )
` );
expect ( size ) . to . be . a ( 'number' ) ;
} ) ;
it ( 'should lock the keyboard' , async ( ) = > {
2024-06-07 21:18:35 +00:00
const w = new BrowserWindow ( { show : true } ) ;
2023-10-31 15:59:39 +00:00
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'modal.html' ) ) ;
// Test that without lock, with ESC:
// - the window leaves fullscreen
// - the dialog is not closed
const enterFS1 = once ( w , 'enter-full-screen' ) ;
await w . webContents . executeJavaScript ( 'document.body.requestFullscreen()' , true ) ;
await enterFS1 ;
await w . webContents . executeJavaScript ( 'document.getElementById(\'favDialog\').showModal()' , true ) ;
const open1 = await w . webContents . executeJavaScript ( 'document.getElementById(\'favDialog\').open' ) ;
expect ( open1 ) . to . be . true ( ) ;
w . webContents . sendInputEvent ( { type : 'keyDown' , keyCode : 'Escape' } ) ;
await setTimeout ( 1000 ) ;
const openAfter1 = await w . webContents . executeJavaScript ( 'document.getElementById(\'favDialog\').open' ) ;
expect ( openAfter1 ) . to . be . true ( ) ;
expect ( w . isFullScreen ( ) ) . to . be . false ( ) ;
// Test that with lock, with ESC:
// - the window does not leave fullscreen
// - the dialog is closed
const enterFS2 = once ( w , 'enter-full-screen' ) ;
await w . webContents . executeJavaScript ( `
document . body . requestFullscreen ( ) ;
` , true);
await enterFS2 ;
2024-06-07 21:18:35 +00:00
// Request keyboard lock after window has gone fullscreen
// otherwise it will result in blink::kKeyboardLockRequestFailedErrorMsg.
await w . webContents . executeJavaScript ( `
navigator . keyboard . lock ( [ 'Escape' ] ) ;
` , true);
2023-10-31 15:59:39 +00:00
await w . webContents . executeJavaScript ( 'document.getElementById(\'favDialog\').showModal()' , true ) ;
const open2 = await w . webContents . executeJavaScript ( 'document.getElementById(\'favDialog\').open' ) ;
expect ( open2 ) . to . be . true ( ) ;
w . webContents . sendInputEvent ( { type : 'keyDown' , keyCode : 'Escape' } ) ;
await setTimeout ( 1000 ) ;
const openAfter2 = await w . webContents . executeJavaScript ( 'document.getElementById(\'favDialog\').open' ) ;
expect ( openAfter2 ) . to . be . false ( ) ;
expect ( w . isFullScreen ( ) ) . to . be . true ( ) ;
} ) ;
} ) ;
2019-10-11 20:55:50 +00:00
describe ( 'navigator.languages' , ( ) = > {
it ( 'should return the system locale only' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const appLocale = app . getLocale ( ) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( 'about:blank' ) ;
const languages = await w . webContents . executeJavaScript ( 'navigator.languages' ) ;
2020-05-04 17:49:29 +00:00
expect ( languages . length ) . to . be . greaterThan ( 0 ) ;
expect ( languages ) . to . contain ( appLocale ) ;
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2019-10-11 20:55:50 +00:00
describe ( 'navigator.serviceWorker' , ( ) = > {
it ( 'should register for file scheme' , ( done ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
2021-03-01 21:52:29 +00:00
partition : 'sw-file-scheme-spec' ,
contextIsolation : false
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
w . webContents . on ( 'ipc-message' , ( event , channel , message ) = > {
if ( channel === 'reload' ) {
2020-03-20 20:28:31 +00:00
w . webContents . reload ( ) ;
2019-10-11 20:55:50 +00:00
} else if ( channel === 'error' ) {
2020-03-20 20:28:31 +00:00
done ( message ) ;
2019-10-11 20:55:50 +00:00
} else if ( channel === 'response' ) {
2020-03-20 20:28:31 +00:00
expect ( message ) . to . equal ( 'Hello from serviceWorker!' ) ;
2019-10-11 20:55:50 +00:00
session . fromPartition ( 'sw-file-scheme-spec' ) . clearStorageData ( {
storages : [ 'serviceworkers' ]
2020-03-20 20:28:31 +00:00
} ) . then ( ( ) = > done ( ) ) ;
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2023-02-16 09:25:41 +00:00
w . webContents . on ( 'render-process-gone' , ( ) = > done ( new Error ( 'WebContents crashed.' ) ) ) ;
2020-03-20 20:28:31 +00:00
w . loadFile ( path . join ( fixturesPath , 'pages' , 'service-worker' , 'index.html' ) ) ;
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'should register for intercepted file scheme' , ( done ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( 'intercept-file' ) ;
2019-10-11 20:55:50 +00:00
customSession . protocol . interceptBufferProtocol ( 'file' , ( request , callback ) = > {
2024-10-01 20:04:53 +00:00
let file = new URL ( request . url ) . pathname ! ;
2020-03-20 20:28:31 +00:00
if ( file [ 0 ] === '/' && process . platform === 'win32' ) file = file . slice ( 1 ) ;
2019-10-11 20:55:50 +00:00
2020-03-20 20:28:31 +00:00
const content = fs . readFileSync ( path . normalize ( file ) ) ;
const ext = path . extname ( file ) ;
let type = 'text/html' ;
2019-10-11 20:55:50 +00:00
2020-03-20 20:28:31 +00:00
if ( ext === '.js' ) type = 'application/javascript' ;
callback ( { data : content , mimeType : type } as any ) ;
} ) ;
2019-10-11 20:55:50 +00:00
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
2021-03-01 21:52:29 +00:00
session : customSession ,
contextIsolation : false
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
w . webContents . on ( 'ipc-message' , ( event , channel , message ) = > {
if ( channel === 'reload' ) {
2020-03-20 20:28:31 +00:00
w . webContents . reload ( ) ;
2019-10-11 20:55:50 +00:00
} else if ( channel === 'error' ) {
2020-03-20 20:28:31 +00:00
done ( ` unexpected error : ${ message } ` ) ;
2019-10-11 20:55:50 +00:00
} else if ( channel === 'response' ) {
2020-03-20 20:28:31 +00:00
expect ( message ) . to . equal ( 'Hello from serviceWorker!' ) ;
2019-10-11 20:55:50 +00:00
customSession . clearStorageData ( {
storages : [ 'serviceworkers' ]
} ) . then ( ( ) = > {
2020-06-02 16:46:18 +00:00
customSession . protocol . uninterceptProtocol ( 'file' ) ;
done ( ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2023-02-16 09:25:41 +00:00
w . webContents . on ( 'render-process-gone' , ( ) = > done ( new Error ( 'WebContents crashed.' ) ) ) ;
2020-03-20 20:28:31 +00:00
w . loadFile ( path . join ( fixturesPath , 'pages' , 'service-worker' , 'index.html' ) ) ;
} ) ;
2020-07-28 01:48:37 +00:00
2021-06-01 01:45:23 +00:00
it ( 'should register for custom scheme' , ( done ) = > {
const customSession = session . fromPartition ( 'custom-scheme' ) ;
customSession . protocol . registerFileProtocol ( serviceWorkerScheme , ( request , callback ) = > {
2024-10-01 20:04:53 +00:00
let file = new URL ( request . url ) . pathname ! ;
2021-06-01 01:45:23 +00:00
if ( file [ 0 ] === '/' && process . platform === 'win32' ) file = file . slice ( 1 ) ;
callback ( { path : path.normalize ( file ) } as any ) ;
} ) ;
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
session : customSession ,
contextIsolation : false
}
} ) ;
w . webContents . on ( 'ipc-message' , ( event , channel , message ) = > {
if ( channel === 'reload' ) {
w . webContents . reload ( ) ;
} else if ( channel === 'error' ) {
done ( ` unexpected error : ${ message } ` ) ;
} else if ( channel === 'response' ) {
expect ( message ) . to . equal ( 'Hello from serviceWorker!' ) ;
customSession . clearStorageData ( {
storages : [ 'serviceworkers' ]
} ) . then ( ( ) = > {
customSession . protocol . uninterceptProtocol ( serviceWorkerScheme ) ;
done ( ) ;
} ) ;
}
} ) ;
2023-02-16 09:25:41 +00:00
w . webContents . on ( 'render-process-gone' , ( ) = > done ( new Error ( 'WebContents crashed.' ) ) ) ;
2021-06-01 01:45:23 +00:00
w . loadFile ( path . join ( fixturesPath , 'pages' , 'service-worker' , 'custom-scheme-index.html' ) ) ;
} ) ;
2022-10-12 14:36:24 +00:00
it ( 'should not allow nodeIntegrationInWorker' , async ( ) = > {
2020-07-28 01:48:37 +00:00
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
2020-08-18 16:31:24 +00:00
nodeIntegrationInWorker : true ,
2021-03-01 21:52:29 +00:00
partition : 'sw-file-scheme-worker-spec' ,
contextIsolation : false
2020-07-28 01:48:37 +00:00
}
} ) ;
2022-10-12 14:36:24 +00:00
await w . loadURL ( ` file:// ${ fixturesPath } /pages/service-worker/empty.html ` ) ;
2020-07-28 01:48:37 +00:00
2022-10-12 14:36:24 +00:00
const data = await w . webContents . executeJavaScript ( `
navigator . serviceWorker . register ( 'worker-no-node.js' , {
scope : './'
} ) . then ( ( ) = > navigator . serviceWorker . ready )
new Promise ( ( resolve ) = > {
navigator . serviceWorker . onmessage = event = > resolve ( event . data ) ;
} ) ;
` );
expect ( data ) . to . equal ( 'undefined undefined undefined undefined' ) ;
2020-07-28 01:48:37 +00:00
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
2023-05-10 14:47:48 +00:00
describe ( 'navigator.geolocation' , ( ) = > {
ifit ( features . isFakeLocationProviderEnabled ( ) ) ( 'returns error when permission is denied' , async ( ) = > {
2019-10-11 20:55:50 +00:00
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
2021-03-01 21:52:29 +00:00
partition : 'geolocation-spec' ,
contextIsolation : false
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2023-02-23 23:53:53 +00:00
const message = once ( w . webContents , 'ipc-message' ) ;
2019-10-11 20:55:50 +00:00
w . webContents . session . setPermissionRequestHandler ( ( wc , permission , callback ) = > {
if ( permission === 'geolocation' ) {
2020-03-20 20:28:31 +00:00
callback ( false ) ;
2019-10-11 20:55:50 +00:00
} else {
2020-03-20 20:28:31 +00:00
callback ( true ) ;
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'geolocation' , 'index.html' ) ) ;
2020-07-01 19:14:38 +00:00
const [ , channel ] = await message ;
expect ( channel ) . to . equal ( 'success' , 'unexpected response from geolocation api' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2022-07-27 09:10:04 +00:00
2023-05-10 14:47:48 +00:00
ifit ( ! features . isFakeLocationProviderEnabled ( ) ) ( 'returns position when permission is granted' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
partition : 'geolocation-spec'
}
} ) ;
w . webContents . session . setPermissionRequestHandler ( ( _wc , _permission , callback ) = > {
callback ( true ) ;
} ) ;
2022-07-27 09:10:04 +00:00
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const position = await w . webContents . executeJavaScript ( ` new Promise((resolve, reject) =>
navigator . geolocation . getCurrentPosition (
x = > resolve ( { coords : x.coords , timestamp : x.timestamp } ) ,
2023-05-10 14:47:48 +00:00
err = > reject ( new Error ( err . message ) ) ) ) ` );
2022-07-27 09:10:04 +00:00
expect ( position ) . to . have . property ( 'coords' ) ;
expect ( position ) . to . have . property ( 'timestamp' ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
2024-04-10 20:06:47 +00:00
describe ( 'File System API,' , ( ) = > {
afterEach ( closeAllWindows ) ;
afterEach ( ( ) = > {
session . defaultSession . setPermissionRequestHandler ( null ) ;
} ) ;
it ( 'allows access by default to reading an OPFS file' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
partition : 'file-system-spec' ,
contextIsolation : false
}
} ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const result = await w . webContents . executeJavaScript ( `
new Promise ( async ( resolve , reject ) = > {
const root = await navigator . storage . getDirectory ( ) ;
const fileHandle = await root . getFileHandle ( 'test' , { create : true } ) ;
const { name , size } = await fileHandle . getFile ( ) ;
resolve ( { name , size } ) ;
}
) ` , true);
expect ( result ) . to . deep . equal ( { name : 'test' , size : 0 } ) ;
} ) ;
it ( 'fileHandle.queryPermission by default has permission to read and write to OPFS files' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
partition : 'file-system-spec' ,
contextIsolation : false
}
} ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const status = await w . webContents . executeJavaScript ( `
new Promise ( async ( resolve , reject ) = > {
const root = await navigator . storage . getDirectory ( ) ;
const fileHandle = await root . getFileHandle ( 'test' , { create : true } ) ;
const status = await fileHandle . queryPermission ( { mode : 'readwrite' } ) ;
resolve ( status ) ;
}
) ` , true);
expect ( status ) . to . equal ( 'granted' ) ;
} ) ;
it ( 'fileHandle.requestPermission automatically grants permission to read and write to OPFS files' , async ( ) = > {
const w = new BrowserWindow ( {
webPreferences : {
nodeIntegration : true ,
partition : 'file-system-spec' ,
contextIsolation : false
}
} ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const status = await w . webContents . executeJavaScript ( `
new Promise ( async ( resolve , reject ) = > {
const root = await navigator . storage . getDirectory ( ) ;
const fileHandle = await root . getFileHandle ( 'test' , { create : true } ) ;
const status = await fileHandle . requestPermission ( { mode : 'readwrite' } ) ;
resolve ( status ) ;
}
) ` , true);
expect ( status ) . to . equal ( 'granted' ) ;
} ) ;
it ( 'requests permission when trying to create a writable file handle' , ( done ) = > {
const writablePath = path . join ( fixturesPath , 'file-system' , 'test-writable.html' ) ;
const testFile = path . join ( fixturesPath , 'file-system' , 'test.txt' ) ;
const w = new BrowserWindow ( {
webPreferences : {
nodeIntegration : true ,
contextIsolation : false ,
sandbox : false
}
} ) ;
w . webContents . session . setPermissionRequestHandler ( ( wc , permission , callback , details ) = > {
expect ( permission ) . to . equal ( 'fileSystem' ) ;
const { href } = url . pathToFileURL ( writablePath ) ;
expect ( details ) . to . deep . equal ( {
fileAccessType : 'writable' ,
isDirectory : false ,
isMainFrame : true ,
filePath : testFile ,
requestingUrl : href
} ) ;
callback ( true ) ;
} ) ;
ipcMain . once ( 'did-create-file-handle' , async ( ) = > {
const result = await w . webContents . executeJavaScript ( `
new Promise ( ( resolve , reject ) = > {
try {
const writable = fileHandle . createWritable ( ) ;
resolve ( true ) ;
} catch {
resolve ( false ) ;
}
} )
` , true);
expect ( result ) . to . be . true ( ) ;
done ( ) ;
} ) ;
w . loadFile ( writablePath ) ;
w . webContents . once ( 'did-finish-load' , ( ) = > {
// @ts-expect-error Undocumented testing method.
clipboard . _writeFilesForTesting ( [ testFile ] ) ;
w . webContents . paste ( ) ;
} ) ;
} ) ;
} ) ;
2021-07-04 23:48:46 +00:00
describe ( 'web workers' , ( ) = > {
let appProcess : ChildProcess.ChildProcessWithoutNullStreams | undefined ;
afterEach ( ( ) = > {
if ( appProcess && ! appProcess . killed ) {
appProcess . kill ( ) ;
appProcess = undefined ;
}
} ) ;
it ( 'Worker with nodeIntegrationInWorker has access to self.module.paths' , async ( ) = > {
const appPath = path . join ( __dirname , 'fixtures' , 'apps' , 'self-module-paths' ) ;
appProcess = ChildProcess . spawn ( process . execPath , [ appPath ] ) ;
2023-02-23 23:53:53 +00:00
const [ code ] = await once ( appProcess , 'exit' ) ;
2021-07-04 23:48:46 +00:00
expect ( code ) . to . equal ( 0 ) ;
} ) ;
2022-07-27 09:10:04 +00:00
2024-10-31 21:44:00 +00:00
itremote ( 'Worker with nodeIntegrationInWorker has access to EventSource' , ( ) = > {
const es = new EventSource ( 'https://example.com' ) ;
expect ( es ) . to . have . property ( 'url' ) . that . is . a ( 'string' ) ;
expect ( es ) . to . have . property ( 'readyState' ) . that . is . a ( 'number' ) ;
expect ( es ) . to . have . property ( 'withCredentials' ) . that . is . a ( 'boolean' ) ;
} ) ;
itremote ( 'Worker with nodeIntegrationInWorker has access to fetch-dependent interfaces' , async ( fixtures : string ) = > {
const file = require ( 'node:path' ) . join ( fixtures , 'hello.txt' ) ;
expect ( ( ) = > {
fetch ( 'file://' + file ) ;
} ) . to . not . throw ( ) ;
expect ( ( ) = > {
const formData = new FormData ( ) ;
formData . append ( 'username' , 'Groucho' ) ;
} ) . not . to . throw ( ) ;
expect ( ( ) = > {
const request = new Request ( 'https://example.com' , {
method : 'POST' ,
body : JSON.stringify ( { foo : 'bar' } )
} ) ;
expect ( request . method ) . to . equal ( 'POST' ) ;
} ) . not . to . throw ( ) ;
expect ( ( ) = > {
const response = new Response ( 'Hello, world!' ) ;
expect ( response . status ) . to . equal ( 200 ) ;
} ) . not . to . throw ( ) ;
expect ( ( ) = > {
const headers = new Headers ( ) ;
headers . append ( 'Content-Type' , 'text/xml' ) ;
} ) . not . to . throw ( ) ;
} , [ path . join ( __dirname , 'fixtures' ) ] ) ;
2022-07-27 09:10:04 +00:00
it ( 'Worker can work' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const data = await w . webContents . executeJavaScript ( `
const worker = new Worker ( '../workers/worker.js' ) ;
const message = 'ping' ;
const eventPromise = new Promise ( ( resolve ) = > { worker . onmessage = resolve ; } ) ;
worker . postMessage ( message ) ;
eventPromise . then ( t = > t . data )
` );
expect ( data ) . to . equal ( 'ping' ) ;
} ) ;
it ( 'Worker has no node integration by default' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const data = await w . webContents . executeJavaScript ( `
const worker = new Worker ( '../workers/worker_node.js' ) ;
new Promise ( ( resolve ) = > { worker . onmessage = e = > resolve ( e . data ) ; } )
` );
expect ( data ) . to . equal ( 'undefined undefined undefined undefined' ) ;
} ) ;
it ( 'Worker has node integration with nodeIntegrationInWorker' , async ( ) = > {
2024-06-20 20:32:58 +00:00
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
nodeIntegrationInWorker : true ,
contextIsolation : false
}
} ) ;
2022-07-27 09:10:04 +00:00
w . loadURL ( ` file:// ${ fixturesPath } /pages/worker.html ` ) ;
2023-02-23 23:53:53 +00:00
const [ , data ] = await once ( ipcMain , 'worker-result' ) ;
2022-07-27 09:10:04 +00:00
expect ( data ) . to . equal ( 'object function object function' ) ;
} ) ;
2024-06-20 20:32:58 +00:00
it ( 'Worker has access to fetch-dependent interfaces with nodeIntegrationInWorker' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
nodeIntegrationInWorker : true ,
contextIsolation : false
}
} ) ;
w . loadURL ( ` file:// ${ fixturesPath } /pages/worker-fetch.html ` ) ;
const [ , data ] = await once ( ipcMain , 'worker-fetch-result' ) ;
expect ( data ) . to . equal ( 'function function function function function' ) ;
} ) ;
2022-07-27 09:10:04 +00:00
describe ( 'SharedWorker' , ( ) = > {
it ( 'can work' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const data = await w . webContents . executeJavaScript ( `
const worker = new SharedWorker ( '../workers/shared_worker.js' ) ;
const message = 'ping' ;
const eventPromise = new Promise ( ( resolve ) = > { worker . port . onmessage = e = > resolve ( e . data ) ; } ) ;
worker . port . postMessage ( message ) ;
eventPromise
` );
expect ( data ) . to . equal ( 'ping' ) ;
} ) ;
it ( 'has no node integration by default' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const data = await w . webContents . executeJavaScript ( `
const worker = new SharedWorker ( '../workers/shared_worker_node.js' ) ;
new Promise ( ( resolve ) = > { worker . port . onmessage = e = > resolve ( e . data ) ; } )
` );
expect ( data ) . to . equal ( 'undefined undefined undefined undefined' ) ;
} ) ;
2022-10-12 14:36:24 +00:00
it ( 'does not have node integration with nodeIntegrationInWorker' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
nodeIntegrationInWorker : true ,
contextIsolation : false
}
} ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const data = await w . webContents . executeJavaScript ( `
const worker = new SharedWorker ( '../workers/shared_worker_node.js' ) ;
new Promise ( ( resolve ) = > { worker . port . onmessage = e = > resolve ( e . data ) ; } )
` );
expect ( data ) . to . equal ( 'undefined undefined undefined undefined' ) ;
2022-07-27 09:10:04 +00:00
} ) ;
} ) ;
2021-07-04 23:48:46 +00:00
} ) ;
2019-12-11 06:46:25 +00:00
describe ( 'form submit' , ( ) = > {
2020-03-20 20:28:31 +00:00
let server : http.Server ;
let serverUrl : string ;
2019-12-11 06:46:25 +00:00
before ( async ( ) = > {
server = http . createServer ( ( req , res ) = > {
2020-03-20 20:28:31 +00:00
let body = '' ;
2019-12-11 06:46:25 +00:00
req . on ( 'data' , ( chunk ) = > {
2020-03-20 20:28:31 +00:00
body += chunk ;
} ) ;
res . setHeader ( 'Content-Type' , 'application/json' ) ;
2019-12-11 06:46:25 +00:00
req . on ( 'end' , ( ) = > {
2020-03-20 20:28:31 +00:00
res . end ( ` body: ${ body } ` ) ;
} ) ;
} ) ;
2023-02-20 11:30:57 +00:00
serverUrl = ( await listen ( server ) ) . url ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-12-11 06:46:25 +00:00
after ( async ( ) = > {
2020-03-20 20:28:31 +00:00
server . close ( ) ;
await closeAllWindows ( ) ;
2019-12-11 06:46:25 +00:00
} ) ;
2023-08-31 14:36:43 +00:00
for ( const isSandboxEnabled of [ true , false ] ) {
2019-12-11 06:46:25 +00:00
describe ( ` sandbox= ${ isSandboxEnabled } ` , ( ) = > {
2020-03-20 15:12:18 +00:00
it ( 'posts data in the same window' , async ( ) = > {
2019-12-11 06:46:25 +00:00
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
sandbox : isSandboxEnabled
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-12-11 06:46:25 +00:00
2020-03-20 20:28:31 +00:00
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'form-with-data.html' ) ) ;
2019-12-11 06:46:25 +00:00
2023-02-23 23:53:53 +00:00
const loadPromise = once ( w . webContents , 'did-finish-load' ) ;
2019-12-11 06:46:25 +00:00
2020-03-20 15:12:18 +00:00
w . webContents . executeJavaScript ( `
const form = document . querySelector ( 'form' )
form . action = '${serverUrl}' ;
form . submit ( ) ;
2020-03-20 20:28:31 +00:00
` );
2020-03-20 15:12:18 +00:00
2020-03-20 20:28:31 +00:00
await loadPromise ;
2020-03-20 15:12:18 +00:00
2020-03-20 20:28:31 +00:00
const res = await w . webContents . executeJavaScript ( 'document.body.innerText' ) ;
expect ( res ) . to . equal ( 'body:greeting=hello' ) ;
} ) ;
2019-12-11 06:46:25 +00:00
2020-03-20 15:12:18 +00:00
it ( 'posts data to a new window with target=_blank' , async ( ) = > {
2019-12-11 06:46:25 +00:00
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
sandbox : isSandboxEnabled
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-12-11 06:46:25 +00:00
2020-03-20 20:28:31 +00:00
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'form-with-data.html' ) ) ;
2019-12-11 06:46:25 +00:00
2023-06-22 18:38:52 +00:00
const windowCreatedPromise = once ( app , 'browser-window-created' ) as Promise < [ any , BrowserWindow ] > ;
2019-12-11 06:46:25 +00:00
2020-03-20 15:12:18 +00:00
w . webContents . executeJavaScript ( `
const form = document . querySelector ( 'form' )
form . action = '${serverUrl}' ;
form . target = '_blank' ;
form . submit ( ) ;
2020-03-20 20:28:31 +00:00
` );
2020-03-20 15:12:18 +00:00
2020-03-20 20:28:31 +00:00
const [ , newWin ] = await windowCreatedPromise ;
2020-03-20 15:12:18 +00:00
2020-03-20 20:28:31 +00:00
const res = await newWin . webContents . executeJavaScript ( 'document.body.innerText' ) ;
expect ( res ) . to . equal ( 'body:greeting=hello' ) ;
} ) ;
2023-08-31 14:36:43 +00:00
} ) ;
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-12-11 06:46:25 +00:00
2019-10-11 20:55:50 +00:00
describe ( 'window.open' , ( ) = > {
for ( const show of [ true , false ] ) {
2020-11-10 17:06:03 +00:00
it ( ` shows the child regardless of parent visibility when parent {show= ${ show } } ` , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show } ) ;
2019-10-11 20:55:50 +00:00
// toggle visibility
if ( show ) {
2020-03-20 20:28:31 +00:00
w . hide ( ) ;
2019-10-11 20:55:50 +00:00
} else {
2020-03-20 20:28:31 +00:00
w . show ( ) ;
2019-10-11 20:55:50 +00:00
}
2020-06-30 22:10:36 +00:00
defer ( ( ) = > { w . close ( ) ; } ) ;
2023-06-22 18:38:52 +00:00
const promise = once ( app , 'browser-window-created' ) as Promise < [ any , BrowserWindow ] > ;
2020-03-20 20:28:31 +00:00
w . loadFile ( path . join ( fixturesPath , 'pages' , 'window-open.html' ) ) ;
2022-06-13 17:07:47 +00:00
const [ , newWindow ] = await promise ;
expect ( newWindow . isVisible ( ) ) . to . equal ( true ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
}
2022-08-05 00:20:56 +00:00
// FIXME(zcbenz): This test is making the spec runner hang on exit on Windows.
ifit ( process . platform !== 'win32' ) ( 'disables node integration when it is disabled on the parent window' , async ( ) = > {
const windowUrl = url . pathToFileURL ( path . join ( fixturesPath , 'pages' , 'window-opener-no-node-integration.html' ) ) ;
windowUrl . searchParams . set ( 'p' , ` ${ fixturesPath } /pages/window-opener-node.html ` ) ;
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
const { eventData } = await w . webContents . executeJavaScript ( ` (async () => {
const message = new Promise ( resolve = > window . addEventListener ( 'message' , resolve , { once : true } ) ) ;
const b = window . open ( $ { JSON . stringify ( windowUrl ) } , '' , 'show=false' )
const e = await message
b . close ( ) ;
return {
eventData : e.data
}
} ) ( ) ` );
expect ( eventData . isProcessGlobalUndefined ) . to . be . true ( ) ;
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'disables node integration when it is disabled on the parent window for chrome devtools URLs' , async ( ) = > {
2021-04-13 19:36:38 +00:00
// NB. webSecurity is disabled because native window.open() is not
// allowed to load devtools:// URLs.
const w = new BrowserWindow ( { show : false , webPreferences : { nodeIntegration : true , webSecurity : false } } ) ;
2020-03-20 20:28:31 +00:00
w . loadURL ( 'about:blank' ) ;
2019-10-11 20:55:50 +00:00
w . webContents . executeJavaScript ( `
2021-03-01 21:52:29 +00:00
{ b = window . open ( 'devtools://devtools/bundled/inspector.html' , '' , 'nodeIntegration=no,show=no' ) ; null }
2020-03-20 20:28:31 +00:00
` );
2023-06-22 18:38:52 +00:00
const [ , contents ] = await once ( app , 'web-contents-created' ) as [ any , WebContents ] ;
2020-03-20 20:28:31 +00:00
const typeofProcessGlobal = await contents . executeJavaScript ( 'typeof process' ) ;
expect ( typeofProcessGlobal ) . to . equal ( 'undefined' ) ;
} ) ;
2019-10-11 20:55:50 +00:00
2022-01-06 17:28:03 +00:00
it ( 'can disable node integration when it is enabled on the parent window' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { nodeIntegration : true } } ) ;
2021-12-06 03:54:14 +00:00
w . loadURL ( 'about:blank' ) ;
w . webContents . executeJavaScript ( `
{ b = window . open ( 'about:blank' , '' , 'nodeIntegration=no,show=no' ) ; null }
` );
2023-06-22 18:38:52 +00:00
const [ , contents ] = await once ( app , 'web-contents-created' ) as [ any , WebContents ] ;
2021-12-06 03:54:14 +00:00
const typeofProcessGlobal = await contents . executeJavaScript ( 'typeof process' ) ;
expect ( typeofProcessGlobal ) . to . equal ( 'undefined' ) ;
} ) ;
2022-02-25 18:17:35 +00:00
// TODO(jkleinsc) fix this flaky test on WOA
ifit ( process . platform !== 'win32' || process . arch !== 'arm64' ) ( 'disables JavaScript when it is disabled on the parent window' , async ( ) = > {
2021-04-13 19:36:38 +00:00
const w = new BrowserWindow ( { show : true , webPreferences : { nodeIntegration : true } } ) ;
w . webContents . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
2023-06-15 14:42:27 +00:00
const windowUrl = require ( 'node:url' ) . format ( {
2019-10-11 20:55:50 +00:00
pathname : ` ${ fixturesPath } /pages/window-no-javascript.html ` ,
protocol : 'file' ,
slashes : true
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
w . webContents . executeJavaScript ( `
2021-03-01 21:52:29 +00:00
{ b = window . open ( $ { JSON . stringify ( windowUrl ) } , '' , 'javascript=no,show=no' ) ; null }
2020-03-20 20:28:31 +00:00
` );
2023-06-22 18:38:52 +00:00
const [ , contents ] = await once ( app , 'web-contents-created' ) as [ any , WebContents ] ;
2023-02-23 23:53:53 +00:00
await once ( contents , 'did-finish-load' ) ;
2019-10-11 20:55:50 +00:00
// Click link on page
2020-03-20 20:28:31 +00:00
contents . sendInputEvent ( { type : 'mouseDown' , clickCount : 1 , x : 1 , y : 1 } ) ;
contents . sendInputEvent ( { type : 'mouseUp' , clickCount : 1 , x : 1 , y : 1 } ) ;
2023-06-22 18:38:52 +00:00
const [ , window ] = await once ( app , 'browser-window-created' ) as [ any , BrowserWindow ] ;
2020-06-25 17:19:08 +00:00
const preferences = window . webContents . getLastWebPreferences ( ) ;
2023-06-22 18:38:52 +00:00
expect ( preferences ! . javascript ) . to . be . false ( ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'defines a window.location getter' , async ( ) = > {
2020-03-20 20:28:31 +00:00
let targetURL : string ;
2019-10-11 20:55:50 +00:00
if ( process . platform === 'win32' ) {
2023-09-07 06:50:14 +00:00
targetURL = ` file:/// ${ fixturesPath . replaceAll ( '\\' , '/' ) } /pages/base-page.html ` ;
2019-10-11 20:55:50 +00:00
} else {
2020-03-20 20:28:31 +00:00
targetURL = ` file:// ${ fixturesPath } /pages/base-page.html ` ;
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
2021-04-13 19:36:38 +00:00
w . webContents . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
2020-03-20 20:28:31 +00:00
w . webContents . executeJavaScript ( ` { b = window.open( ${ JSON . stringify ( targetURL ) } ); null } ` ) ;
2023-06-22 18:38:52 +00:00
const [ , window ] = await once ( app , 'browser-window-created' ) as [ any , BrowserWindow ] ;
2023-02-23 23:53:53 +00:00
await once ( window . webContents , 'did-finish-load' ) ;
2020-03-20 20:28:31 +00:00
expect ( await w . webContents . executeJavaScript ( 'b.location.href' ) ) . to . equal ( targetURL ) ;
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'defines a window.location setter' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
2021-04-13 19:36:38 +00:00
w . webContents . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
2020-03-20 20:28:31 +00:00
w . webContents . executeJavaScript ( '{ b = window.open("about:blank"); null }' ) ;
2023-06-22 18:38:52 +00:00
const [ , { webContents } ] = await once ( app , 'browser-window-created' ) as [ any , BrowserWindow ] ;
2023-02-23 23:53:53 +00:00
await once ( webContents , 'did-finish-load' ) ;
2019-10-11 20:55:50 +00:00
// When it loads, redirect
2020-03-20 20:28:31 +00:00
w . webContents . executeJavaScript ( ` { b.location = ${ JSON . stringify ( ` file:// ${ fixturesPath } /pages/base-page.html ` ) } ; null } ` ) ;
2023-02-23 23:53:53 +00:00
await once ( webContents , 'did-finish-load' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'defines a window.location.href setter' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
2021-04-13 19:36:38 +00:00
w . webContents . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
2020-03-20 20:28:31 +00:00
w . webContents . executeJavaScript ( '{ b = window.open("about:blank"); null }' ) ;
2023-06-22 18:38:52 +00:00
const [ , { webContents } ] = await once ( app , 'browser-window-created' ) as [ any , BrowserWindow ] ;
2023-02-23 23:53:53 +00:00
await once ( webContents , 'did-finish-load' ) ;
2019-10-11 20:55:50 +00:00
// When it loads, redirect
2020-03-20 20:28:31 +00:00
w . webContents . executeJavaScript ( ` { b.location.href = ${ JSON . stringify ( ` file:// ${ fixturesPath } /pages/base-page.html ` ) } ; null } ` ) ;
2023-02-23 23:53:53 +00:00
await once ( webContents , 'did-finish-load' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'open a blank page when no URL is specified' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
w . webContents . executeJavaScript ( '{ b = window.open(); null }' ) ;
2023-06-22 18:38:52 +00:00
const [ , { webContents } ] = await once ( app , 'browser-window-created' ) as [ any , BrowserWindow ] ;
2023-02-23 23:53:53 +00:00
await once ( webContents , 'did-finish-load' ) ;
2020-03-20 20:28:31 +00:00
expect ( await w . webContents . executeJavaScript ( 'b.location.href' ) ) . to . equal ( 'about:blank' ) ;
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'open a blank page when an empty URL is specified' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
w . webContents . executeJavaScript ( '{ b = window.open(\'\'); null }' ) ;
2023-06-22 18:38:52 +00:00
const [ , { webContents } ] = await once ( app , 'browser-window-created' ) as [ any , BrowserWindow ] ;
2023-02-23 23:53:53 +00:00
await once ( webContents , 'did-finish-load' ) ;
2020-03-20 20:28:31 +00:00
expect ( await w . webContents . executeJavaScript ( 'b.location.href' ) ) . to . equal ( 'about:blank' ) ;
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'does not throw an exception when the frameName is a built-in object property' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
w . webContents . executeJavaScript ( '{ b = window.open(\'\', \'__proto__\'); null }' ) ;
2022-06-13 17:07:47 +00:00
const frameName = await new Promise ( ( resolve ) = > {
w . webContents . setWindowOpenHandler ( details = > {
setImmediate ( ( ) = > resolve ( details . frameName ) ) ;
return { action : 'allow' } ;
} ) ;
} ) ;
2021-02-05 19:20:58 +00:00
expect ( frameName ) . to . equal ( '__proto__' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2022-08-05 00:20:56 +00:00
2023-07-18 08:41:50 +00:00
it ( 'works when used in conjunction with the vm module' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
contextIsolation : false
}
} ) ;
await w . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
const { contextObject } = await w . webContents . executeJavaScript ( ` (async () => {
const vm = require ( 'node:vm' ) ;
const contextObject = { count : 1 , type : 'gecko' } ;
window . open ( '' ) ;
vm . runInNewContext ( 'count += 1; type = "chameleon";' , contextObject ) ;
return { contextObject } ;
} ) ( ) ` );
expect ( contextObject ) . to . deep . equal ( { count : 2 , type : 'chameleon' } ) ;
} ) ;
2023-04-04 13:48:51 +00:00
// FIXME(nornagon): I'm not sure this ... ever was correct?
xit ( 'inherit options of parent window' , async ( ) = > {
2022-08-05 00:20:56 +00:00
const w = new BrowserWindow ( { show : false , width : 123 , height : 456 } ) ;
w . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
const url = ` file:// ${ fixturesPath } /pages/window-open-size.html ` ;
const { width , height , eventData } = await w . webContents . executeJavaScript ( ` (async () => {
const message = new Promise ( resolve = > window . addEventListener ( 'message' , resolve , { once : true } ) ) ;
const b = window . open ( $ { JSON . stringify ( url ) } , '' , 'show=false' )
const e = await message
b . close ( ) ;
const width = outerWidth ;
const height = outerHeight ;
return {
width ,
height ,
eventData : e.data
}
} ) ( ) ` );
expect ( eventData ) . to . equal ( ` size: ${ width } ${ height } ` ) ;
expect ( eventData ) . to . equal ( 'size: 123 456' ) ;
} ) ;
it ( 'does not override child options' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
const windowUrl = ` file:// ${ fixturesPath } /pages/window-open-size.html ` ;
const { eventData } = await w . webContents . executeJavaScript ( ` (async () => {
const message = new Promise ( resolve = > window . addEventListener ( 'message' , resolve , { once : true } ) ) ;
const b = window . open ( $ { JSON . stringify ( windowUrl ) } , '' , 'show=no,width=350,height=450' )
const e = await message
b . close ( ) ;
return { eventData : e.data }
} ) ( ) ` );
expect ( eventData ) . to . equal ( 'size: 350 450' ) ;
} ) ;
2023-06-26 21:04:54 +00:00
it ( 'loads preload script after setting opener to null' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . webContents . setWindowOpenHandler ( ( ) = > ( {
action : 'allow' ,
overrideBrowserWindowOptions : {
webPreferences : {
preload : path.join ( fixturesPath , 'module' , 'preload.js' )
}
}
} ) ) ;
w . loadURL ( 'about:blank' ) ;
w . webContents . executeJavaScript ( 'window.child = window.open(); child.opener = null' ) ;
const [ , { webContents } ] = await once ( app , 'browser-window-created' ) ;
const [ , , message ] = await once ( webContents , 'console-message' ) ;
2023-08-22 04:43:08 +00:00
expect ( message ) . to . equal ( '{"require":"function","module":"object","exports":"object","process":"object","Buffer":"function"}' ) ;
2023-06-26 21:04:54 +00:00
} ) ;
2022-08-05 00:20:56 +00:00
it ( 'disables the <webview> tag when it is disabled on the parent window' , async ( ) = > {
const windowUrl = url . pathToFileURL ( path . join ( fixturesPath , 'pages' , 'window-opener-no-webview-tag.html' ) ) ;
windowUrl . searchParams . set ( 'p' , ` ${ fixturesPath } /pages/window-opener-webview.html ` ) ;
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
const { eventData } = await w . webContents . executeJavaScript ( ` (async () => {
const message = new Promise ( resolve = > window . addEventListener ( 'message' , resolve , { once : true } ) ) ;
const b = window . open ( $ { JSON . stringify ( windowUrl ) } , '' , 'webviewTag=no,contextIsolation=no,nodeIntegration=yes,show=no' )
const e = await message
b . close ( ) ;
return { eventData : e.data }
} ) ( ) ` );
expect ( eventData . isWebViewGlobalUndefined ) . to . be . true ( ) ;
} ) ;
it ( 'throws an exception when the arguments cannot be converted to strings' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
await expect (
w . webContents . executeJavaScript ( 'window.open(\'\', { toString: null })' )
) . to . eventually . be . rejected ( ) ;
await expect (
w . webContents . executeJavaScript ( 'window.open(\'\', \'\', { toString: 3 })' )
) . to . eventually . be . rejected ( ) ;
} ) ;
it ( 'does not throw an exception when the features include webPreferences' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
await expect (
w . webContents . executeJavaScript ( 'window.open(\'\', \'\', \'show=no,webPreferences=\'); null' )
) . to . eventually . be . fulfilled ( ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
describe ( 'window.opener' , ( ) = > {
it ( 'is null for main window' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
2021-03-01 21:52:29 +00:00
nodeIntegration : true ,
contextIsolation : false
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'window-opener.html' ) ) ;
2023-02-23 23:53:53 +00:00
const [ , channel , opener ] = await once ( w . webContents , 'ipc-message' ) ;
2020-03-20 20:28:31 +00:00
expect ( channel ) . to . equal ( 'opener' ) ;
expect ( opener ) . to . equal ( null ) ;
} ) ;
2022-08-05 00:20:56 +00:00
it ( 'is not null for window opened by window.open' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
contextIsolation : false
}
} ) ;
w . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
const windowUrl = ` file:// ${ fixturesPath } /pages/window-opener.html ` ;
const eventData = await w . webContents . executeJavaScript ( `
const b = window . open ( $ { JSON . stringify ( windowUrl ) } , '' , 'show=no' ) ;
new Promise ( resolve = > window . addEventListener ( 'message' , resolve , { once : true } ) ) . then ( e = > e . data ) ;
` );
expect ( eventData ) . to . equal ( 'object' ) ;
} ) ;
} ) ;
describe ( 'window.opener.postMessage' , ( ) = > {
it ( 'sets source and origin correctly' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
const windowUrl = ` file:// ${ fixturesPath } /pages/window-opener-postMessage.html ` ;
const { sourceIsChild , origin } = await w . webContents . executeJavaScript ( `
const b = window . open ( $ { JSON . stringify ( windowUrl ) } , '' , 'show=no' ) ;
new Promise ( resolve = > window . addEventListener ( 'message' , resolve , { once : true } ) ) . then ( e = > ( {
sourceIsChild : e.source === b ,
origin : e.origin
} ) ) ;
` );
expect ( sourceIsChild ) . to . be . true ( ) ;
expect ( origin ) . to . equal ( 'file://' ) ;
} ) ;
it ( 'supports windows opened from a <webview>' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { webviewTag : true } } ) ;
w . loadURL ( 'about:blank' ) ;
const childWindowUrl = url . pathToFileURL ( path . join ( fixturesPath , 'pages' , 'webview-opener-postMessage.html' ) ) ;
childWindowUrl . searchParams . set ( 'p' , ` ${ fixturesPath } /pages/window-opener-postMessage.html ` ) ;
const message = await w . webContents . executeJavaScript ( `
const webview = new WebView ( ) ;
webview . allowpopups = true ;
webview . setAttribute ( 'webpreferences' , 'contextIsolation=no' ) ;
webview . src = $ { JSON . stringify ( childWindowUrl ) }
const consoleMessage = new Promise ( resolve = > webview . addEventListener ( 'console-message' , resolve , { once : true } ) ) ;
document . body . appendChild ( webview ) ;
consoleMessage . then ( e = > e . message )
` );
expect ( message ) . to . equal ( 'message' ) ;
} ) ;
describe ( 'targetOrigin argument' , ( ) = > {
let serverURL : string ;
let server : any ;
2023-02-20 11:30:57 +00:00
beforeEach ( async ( ) = > {
2022-08-05 00:20:56 +00:00
server = http . createServer ( ( req , res ) = > {
res . writeHead ( 200 ) ;
const filePath = path . join ( fixturesPath , 'pages' , 'window-opener-targetOrigin.html' ) ;
res . end ( fs . readFileSync ( filePath , 'utf8' ) ) ;
} ) ;
2023-02-20 11:30:57 +00:00
serverURL = ( await listen ( server ) ) . url ;
2022-08-05 00:20:56 +00:00
} ) ;
afterEach ( ( ) = > {
server . close ( ) ;
} ) ;
it ( 'delivers messages that match the origin' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { nodeIntegration : true , contextIsolation : false } } ) ;
w . loadFile ( path . resolve ( __dirname , 'fixtures' , 'blank.html' ) ) ;
const data = await w . webContents . executeJavaScript ( `
window . open ( $ { JSON . stringify ( serverURL ) } , '' , 'show=no,contextIsolation=no,nodeIntegration=yes' ) ;
new Promise ( resolve = > window . addEventListener ( 'message' , resolve , { once : true } ) ) . then ( e = > e . data )
` );
expect ( data ) . to . equal ( 'deliver' ) ;
} ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
2024-03-27 18:52:24 +00:00
describe ( 'Storage Access API' , ( ) = > {
afterEach ( closeAllWindows ) ;
afterEach ( ( ) = > {
session . defaultSession . setPermissionCheckHandler ( null ) ;
session . defaultSession . setPermissionRequestHandler ( null ) ;
} ) ;
it ( 'can determine if a permission is granted for "storage-access"' , async ( ) = > {
session . defaultSession . setPermissionCheckHandler (
( _wc , permission ) = > permission === 'storage-access'
) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'a.html' ) ) ;
const permission = await w . webContents . executeJavaScript ( `
navigator . permissions . query ( { name : 'storage-access' } )
. then ( permission = > permission . state ) . catch ( err = > err . message ) ;
` , true);
expect ( permission ) . to . eq ( 'granted' ) ;
} ) ;
it ( 'can determine if a permission is denied for "storage-access"' , async ( ) = > {
session . defaultSession . setPermissionCheckHandler (
( _wc , permission ) = > permission !== 'storage-access'
) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'a.html' ) ) ;
const permission = await w . webContents . executeJavaScript ( `
navigator . permissions . query ( { name : 'storage-access' } )
. then ( permission = > permission . state ) . catch ( err = > err . message ) ;
` , true);
expect ( permission ) . to . eq ( 'denied' ) ;
} ) ;
it ( 'can determine if a permission is granted for "top-level-storage-access"' , async ( ) = > {
session . defaultSession . setPermissionCheckHandler (
( _wc , permission ) = > permission === 'top-level-storage-access'
) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'a.html' ) ) ;
const permission = await w . webContents . executeJavaScript ( `
navigator . permissions . query ( {
name : 'top-level-storage-access' ,
requestedOrigin : "https://www.example.com" ,
} ) . then ( permission = > permission . state ) . catch ( err = > err . message ) ;
` , true);
expect ( permission ) . to . eq ( 'granted' ) ;
} ) ;
it ( 'can determine if a permission is denied for "top-level-storage-access"' , async ( ) = > {
session . defaultSession . setPermissionCheckHandler (
( _wc , permission ) = > permission !== 'top-level-storage-access'
) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'a.html' ) ) ;
const permission = await w . webContents . executeJavaScript ( `
navigator . permissions . query ( {
name : 'top-level-storage-access' ,
requestedOrigin : "https://www.example.com" ,
} ) . then ( permission = > permission . state ) . catch ( err = > err . message ) ;
` , true);
expect ( permission ) . to . eq ( 'denied' ) ;
} ) ;
it ( 'can grant a permission request for "top-level-storage-access"' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler (
( _wc , permission , callback ) = > {
callback ( permission === 'top-level-storage-access' ) ;
}
) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'button.html' ) ) ;
// requestStorageAccessFor returns a Promise that fulfills with undefined
// if the access to third-party cookies was granted and rejects if access was denied.
const permission = await w . webContents . executeJavaScript ( `
new Promise ( ( resolve , reject ) = > {
const button = document . getElementById ( 'button' ) ;
button . addEventListener ( "click" , ( ) = > {
document . requestStorageAccessFor ( 'https://myfakesite' ) . then (
( res ) = > { resolve ( 'granted' ) } ,
( err ) = > { resolve ( 'denied' ) } ,
) ;
} ) ;
button . click ( ) ;
} ) ;
` , true);
expect ( permission ) . to . eq ( 'granted' ) ;
} ) ;
it ( 'can deny a permission request for "top-level-storage-access"' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler (
( _wc , permission , callback ) = > {
callback ( permission !== 'top-level-storage-access' ) ;
}
) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'button.html' ) ) ;
// requestStorageAccessFor returns a Promise that fulfills with undefined
// if the access to third-party cookies was granted and rejects if access was denied.
const permission = await w . webContents . executeJavaScript ( `
new Promise ( ( resolve , reject ) = > {
const button = document . getElementById ( 'button' ) ;
button . addEventListener ( "click" , ( ) = > {
document . requestStorageAccessFor ( 'https://myfakesite' ) . then (
( res ) = > { resolve ( 'granted' ) } ,
( err ) = > { resolve ( 'denied' ) } ,
) ;
} ) ;
button . click ( ) ;
} ) ;
` , true);
expect ( permission ) . to . eq ( 'denied' ) ;
} ) ;
} ) ;
2023-07-24 18:57:41 +00:00
describe ( 'IdleDetection' , ( ) = > {
afterEach ( closeAllWindows ) ;
afterEach ( ( ) = > {
session . defaultSession . setPermissionCheckHandler ( null ) ;
session . defaultSession . setPermissionRequestHandler ( null ) ;
} ) ;
it ( 'can grant a permission request' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler (
( _wc , permission , callback ) = > {
callback ( permission === 'idle-detection' ) ;
}
) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'button.html' ) ) ;
const permission = await w . webContents . executeJavaScript ( `
new Promise ( ( resolve , reject ) = > {
const button = document . getElementById ( 'button' ) ;
button . addEventListener ( "click" , async ( ) = > {
const permission = await IdleDetector . requestPermission ( ) ;
resolve ( permission ) ;
} ) ;
button . click ( ) ;
} ) ;
` , true);
expect ( permission ) . to . eq ( 'granted' ) ;
} ) ;
it ( 'can deny a permission request' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler (
( _wc , permission , callback ) = > {
callback ( permission !== 'idle-detection' ) ;
}
) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'button.html' ) ) ;
const permission = await w . webContents . executeJavaScript ( `
new Promise ( ( resolve , reject ) = > {
const button = document . getElementById ( 'button' ) ;
button . addEventListener ( "click" , async ( ) = > {
const permission = await IdleDetector . requestPermission ( ) ;
resolve ( permission ) ;
} ) ;
button . click ( ) ;
} ) ;
` , true);
expect ( permission ) . to . eq ( 'denied' ) ;
} ) ;
it ( 'can allow the IdleDetector to start' , async ( ) = > {
session . defaultSession . setPermissionCheckHandler ( ( wc , permission ) = > {
return permission === 'idle-detection' ;
} ) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const result = await w . webContents . executeJavaScript ( `
const detector = new IdleDetector ( { threshold : 60000 } ) ;
detector . start ( ) . then ( ( ) = > {
return 'success' ;
} ) . catch ( e = > e . message ) ;
` , true);
expect ( result ) . to . eq ( 'success' ) ;
} ) ;
it ( 'can prevent the IdleDetector from starting' , async ( ) = > {
session . defaultSession . setPermissionCheckHandler ( ( wc , permission ) = > {
return permission !== 'idle-detection' ;
} ) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const result = await w . webContents . executeJavaScript ( `
const detector = new IdleDetector ( { threshold : 60000 } ) ;
detector . start ( ) . then ( ( ) = > {
console . log ( 'success' )
} ) . catch ( e = > e . message ) ;
` , true);
expect ( result ) . to . eq ( 'Idle detection permission denied' ) ;
} ) ;
} ) ;
2019-10-11 20:55:50 +00:00
describe ( 'navigator.mediaDevices' , ( ) = > {
2020-03-20 20:28:31 +00:00
afterEach ( closeAllWindows ) ;
2019-10-11 20:55:50 +00:00
afterEach ( ( ) = > {
2020-03-20 20:28:31 +00:00
session . defaultSession . setPermissionCheckHandler ( null ) ;
2021-10-13 21:10:12 +00:00
session . defaultSession . setPermissionRequestHandler ( null ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-11 20:55:50 +00:00
it ( 'can return labels of enumerated devices' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const labels = await w . webContents . executeJavaScript ( 'navigator.mediaDevices.enumerateDevices().then(ds => ds.map(d => d.label))' ) ;
expect ( labels . some ( ( l : any ) = > l ) ) . to . be . true ( ) ;
} ) ;
2019-10-11 20:55:50 +00:00
2024-07-08 14:49:10 +00:00
it ( 'does not return labels of enumerated devices when all permission denied' , async ( ) = > {
2020-03-20 20:28:31 +00:00
session . defaultSession . setPermissionCheckHandler ( ( ) = > false ) ;
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const labels = await w . webContents . executeJavaScript ( 'navigator.mediaDevices.enumerateDevices().then(ds => ds.map(d => d.label))' ) ;
expect ( labels . some ( ( l : any ) = > l ) ) . to . be . false ( ) ;
} ) ;
2019-10-11 20:55:50 +00:00
2024-07-08 14:49:10 +00:00
it ( 'does not return labels of enumerated audio devices when permission denied' , async ( ) = > {
session . defaultSession . setPermissionCheckHandler ( ( wc , permission , origin , { mediaType } ) = > {
return permission !== 'media' || mediaType !== 'audio' ;
} ) ;
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const devices = await w . webContents . executeJavaScript (
` navigator.mediaDevices.enumerateDevices().then(ds => ds.map(d => {
return ( { label : d.label , kind : d.kind } )
} ) ) ;
` );
const audioDevices = devices . filter ( ( d : any ) = > d . kind === 'audioinput' ) ;
expect ( audioDevices . some ( ( d : any ) = > d . label ) ) . to . be . false ( ) ;
const videoDevices = devices . filter ( ( d : any ) = > d . kind === 'videoinput' ) ;
expect ( videoDevices . some ( ( d : any ) = > d . label ) ) . to . be . true ( ) ;
} ) ;
it ( 'does not return labels of enumerated video devices when permission denied' , async ( ) = > {
session . defaultSession . setPermissionCheckHandler ( ( wc , permission , origin , { mediaType } ) = > {
return permission !== 'media' || mediaType !== 'video' ;
} ) ;
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const devices = await w . webContents . executeJavaScript (
` navigator.mediaDevices.enumerateDevices().then(ds => ds.map(d => {
return ( { label : d.label , kind : d.kind } )
} ) ) ;
` );
const audioDevices = devices . filter ( ( d : any ) = > d . kind === 'audioinput' ) ;
expect ( audioDevices . some ( ( d : any ) = > d . label ) ) . to . be . true ( ) ;
const videoDevices = devices . filter ( ( d : any ) = > d . kind === 'videoinput' ) ;
expect ( videoDevices . some ( ( d : any ) = > d . label ) ) . to . be . false ( ) ;
} ) ;
2020-02-27 00:03:55 +00:00
it ( 'returns the same device ids across reloads' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const ses = session . fromPartition ( 'persist:media-device-id' ) ;
2020-02-27 00:03:55 +00:00
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
2021-03-01 21:52:29 +00:00
session : ses ,
contextIsolation : false
2020-02-27 00:03:55 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'media-id-reset.html' ) ) ;
2023-02-23 23:53:53 +00:00
const [ , firstDeviceIds ] = await once ( ipcMain , 'deviceIds' ) ;
w . webContents . reload ( ) ;
const [ , secondDeviceIds ] = await once ( ipcMain , 'deviceIds' ) ;
2020-03-20 20:28:31 +00:00
expect ( firstDeviceIds ) . to . deep . equal ( secondDeviceIds ) ;
} ) ;
2020-02-27 00:03:55 +00:00
2019-10-11 20:55:50 +00:00
it ( 'can return new device id when cookie storage is cleared' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const ses = session . fromPartition ( 'persist:media-device-id' ) ;
2019-10-11 20:55:50 +00:00
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
2021-03-01 21:52:29 +00:00
session : ses ,
contextIsolation : false
2019-10-11 20:55:50 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'media-id-reset.html' ) ) ;
2023-02-23 23:53:53 +00:00
const [ , firstDeviceIds ] = await once ( ipcMain , 'deviceIds' ) ;
2020-03-20 20:28:31 +00:00
await ses . clearStorageData ( { storages : [ 'cookies' ] } ) ;
2023-02-23 23:53:53 +00:00
w . webContents . reload ( ) ;
const [ , secondDeviceIds ] = await once ( ipcMain , 'deviceIds' ) ;
2020-03-20 20:28:31 +00:00
expect ( firstDeviceIds ) . to . not . deep . equal ( secondDeviceIds ) ;
} ) ;
2021-10-13 21:10:12 +00:00
it ( 'provides a securityOrigin to the request handler' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler (
( wc , permission , callback , details ) = > {
2024-04-10 20:06:47 +00:00
if ( ( details as MediaAccessPermissionRequest ) . securityOrigin !== undefined ) {
2021-10-13 21:10:12 +00:00
callback ( true ) ;
} else {
callback ( false ) ;
}
}
) ;
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const labels = await w . webContents . executeJavaScript ( ` navigator.mediaDevices.getUserMedia({
video : {
mandatory : {
chromeMediaSource : "desktop" ,
minWidth : 1280 ,
maxWidth : 1280 ,
minHeight : 720 ,
maxHeight : 720
}
}
} ) . then ( ( stream ) = > stream . getVideoTracks ( ) ) ` );
expect ( labels . some ( ( l : any ) = > l ) ) . to . be . true ( ) ;
} ) ;
2022-07-20 08:09:14 +00:00
it ( 'fails with "not supported" for getDisplayMedia' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
2023-02-03 11:43:42 +00:00
const { ok , err } = await w . webContents . executeJavaScript ( 'navigator.mediaDevices.getDisplayMedia({video: true}).then(s => ({ok: true}), e => ({ok: false, err: e.message}))' , true ) ;
2022-07-20 08:09:14 +00:00
expect ( ok ) . to . be . false ( ) ;
expect ( err ) . to . equal ( 'Not supported' ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-14 16:00:34 +00:00
describe ( 'window.opener access' , ( ) = > {
2020-03-20 20:28:31 +00:00
const scheme = 'app' ;
2019-10-14 16:00:34 +00:00
2020-03-20 20:28:31 +00:00
const fileUrl = ` file:// ${ fixturesPath } /pages/window-opener-location.html ` ;
const httpUrl1 = ` ${ scheme } ://origin1 ` ;
const httpUrl2 = ` ${ scheme } ://origin2 ` ;
const fileBlank = ` file:// ${ fixturesPath } /pages/blank.html ` ;
const httpBlank = ` ${ scheme } ://origin1/blank ` ;
2019-10-14 16:00:34 +00:00
const table = [
2022-01-06 17:28:03 +00:00
{ parent : fileBlank , child : httpUrl1 , nodeIntegration : false , openerAccessible : false } ,
{ parent : fileBlank , child : httpUrl1 , nodeIntegration : true , openerAccessible : false } ,
2019-10-14 16:00:34 +00:00
2022-01-06 17:28:03 +00:00
// {parent: httpBlank, child: fileUrl, nodeIntegration: false, openerAccessible: false}, // can't window.open()
// {parent: httpBlank, child: fileUrl, nodeIntegration: true, openerAccessible: false}, // can't window.open()
2019-10-14 16:00:34 +00:00
// NB. this is different from Chrome's behavior, which isolates file: urls from each other
2022-01-06 17:28:03 +00:00
{ parent : fileBlank , child : fileUrl , nodeIntegration : false , openerAccessible : true } ,
{ parent : fileBlank , child : fileUrl , nodeIntegration : true , openerAccessible : true } ,
{ parent : httpBlank , child : httpUrl1 , nodeIntegration : false , openerAccessible : true } ,
{ parent : httpBlank , child : httpUrl1 , nodeIntegration : true , openerAccessible : true } ,
{ parent : httpBlank , child : httpUrl2 , nodeIntegration : false , openerAccessible : false } ,
{ parent : httpBlank , child : httpUrl2 , nodeIntegration : true , openerAccessible : false }
2020-03-20 20:28:31 +00:00
] ;
const s = ( url : string ) = > url . startsWith ( 'file' ) ? 'file://...' : url ;
2019-10-14 16:00:34 +00:00
2020-06-02 16:46:18 +00:00
before ( ( ) = > {
protocol . registerFileProtocol ( scheme , ( request , callback ) = > {
2019-10-16 15:12:31 +00:00
if ( request . url . includes ( 'blank' ) ) {
2020-03-20 20:28:31 +00:00
callback ( ` ${ fixturesPath } /pages/blank.html ` ) ;
2019-10-16 15:12:31 +00:00
} else {
2020-03-20 20:28:31 +00:00
callback ( ` ${ fixturesPath } /pages/window-opener-location.html ` ) ;
2019-10-16 15:12:31 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2020-06-02 16:46:18 +00:00
after ( ( ) = > {
protocol . unregisterProtocol ( scheme ) ;
2020-03-20 20:28:31 +00:00
} ) ;
afterEach ( closeAllWindows ) ;
2019-10-14 16:00:34 +00:00
2019-10-16 15:12:31 +00:00
describe ( 'when opened from main window' , ( ) = > {
2022-01-06 17:28:03 +00:00
for ( const { parent , child , nodeIntegration , openerAccessible } of table ) {
2020-01-06 21:23:03 +00:00
for ( const sandboxPopup of [ false , true ] ) {
2022-01-06 17:28:03 +00:00
const description = ` when parent= ${ s ( parent ) } opens child= ${ s ( child ) } with nodeIntegration= ${ nodeIntegration } sandboxPopup= ${ sandboxPopup } , child should ${ openerAccessible ? '' : 'not ' } be able to access opener ` ;
2020-01-06 21:23:03 +00:00
it ( description , async ( ) = > {
2022-01-06 17:28:03 +00:00
const w = new BrowserWindow ( { show : true , webPreferences : { nodeIntegration : true , contextIsolation : false } } ) ;
2020-11-10 17:06:03 +00:00
w . webContents . setWindowOpenHandler ( ( ) = > ( {
action : 'allow' ,
overrideBrowserWindowOptions : {
webPreferences : {
sandbox : sandboxPopup
}
}
} ) ) ;
2020-03-20 20:28:31 +00:00
await w . loadURL ( parent ) ;
2020-01-06 21:23:03 +00:00
const childOpenerLocation = await w . webContents . executeJavaScript ( ` new Promise(resolve => {
window . addEventListener ( 'message' , function f ( e ) {
resolve ( e . data )
} )
window . open ( $ { JSON . stringify ( child ) } , "" , "show=no,nodeIntegration=${nodeIntegration ? 'yes' : 'no'}" )
2020-03-20 20:28:31 +00:00
} ) ` );
2020-01-06 21:23:03 +00:00
if ( openerAccessible ) {
2020-03-20 20:28:31 +00:00
expect ( childOpenerLocation ) . to . be . a ( 'string' ) ;
2020-01-06 21:23:03 +00:00
} else {
2020-03-20 20:28:31 +00:00
expect ( childOpenerLocation ) . to . be . null ( ) ;
2020-01-06 21:23:03 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2020-01-06 21:23:03 +00:00
}
2019-10-16 15:12:31 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
describe ( 'when opened from <webview>' , ( ) = > {
2022-01-06 17:28:03 +00:00
for ( const { parent , child , nodeIntegration , openerAccessible } of table ) {
const description = ` when parent= ${ s ( parent ) } opens child= ${ s ( child ) } with nodeIntegration= ${ nodeIntegration } , child should ${ openerAccessible ? '' : 'not ' } be able to access opener ` ;
it ( description , async ( ) = > {
2019-10-16 15:12:31 +00:00
// This test involves three contexts:
// 1. The root BrowserWindow in which the test is run,
// 2. A <webview> belonging to the root window,
// 3. A window opened by calling window.open() from within the <webview>.
// We are testing whether context (3) can access context (2) under various conditions.
// This is context (1), the base window for the test.
2022-01-06 17:28:03 +00:00
const w = new BrowserWindow ( { show : false , webPreferences : { nodeIntegration : true , webviewTag : true , contextIsolation : false } } ) ;
2020-03-20 20:28:31 +00:00
await w . loadURL ( 'about:blank' ) ;
2019-10-16 15:12:31 +00:00
const parentCode = ` new Promise((resolve) => {
// This is context (3), a child window of the WebView.
2021-04-21 17:55:17 +00:00
const child = window . open ( $ { JSON . stringify ( child ) } , "" , "show=no,contextIsolation=no,nodeIntegration=yes" )
2019-10-16 15:12:31 +00:00
window . addEventListener ( "message" , e = > {
resolve ( e . data )
} )
2020-03-20 20:28:31 +00:00
} ) ` ;
2019-10-16 15:12:31 +00:00
const childOpenerLocation = await w . webContents . executeJavaScript ( ` new Promise((resolve, reject) => {
// This is context (2), a WebView which will call window.open()
const webview = new WebView ( )
2019-11-01 20:37:02 +00:00
webview . setAttribute ( 'nodeintegration' , '${nodeIntegration ? ' on ' : ' off '}' )
2022-01-06 17:28:03 +00:00
webview . setAttribute ( 'webpreferences' , 'contextIsolation=no' )
2019-10-16 15:12:31 +00:00
webview . setAttribute ( 'allowpopups' , 'on' )
webview . src = $ { JSON . stringify ( parent + '?p=' + encodeURIComponent ( child ) ) }
webview . addEventListener ( 'dom-ready' , async ( ) = > {
webview . executeJavaScript ( $ { JSON . stringify ( parentCode ) } ) . then ( resolve , reject )
} )
document . body . appendChild ( webview )
2020-03-20 20:28:31 +00:00
} ) ` );
2019-10-16 15:12:31 +00:00
if ( openerAccessible ) {
2020-03-20 20:28:31 +00:00
expect ( childOpenerLocation ) . to . be . a ( 'string' ) ;
2019-10-16 15:12:31 +00:00
} else {
2020-03-20 20:28:31 +00:00
expect ( childOpenerLocation ) . to . be . null ( ) ;
2019-10-16 15:12:31 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2019-10-16 15:12:31 +00:00
describe ( 'storage' , ( ) = > {
describe ( 'custom non standard schemes' , ( ) = > {
2020-03-20 20:28:31 +00:00
const protocolName = 'storage' ;
let contents : WebContents ;
2020-06-02 16:46:18 +00:00
before ( ( ) = > {
2019-10-16 15:12:31 +00:00
protocol . registerFileProtocol ( protocolName , ( request , callback ) = > {
2024-10-01 20:04:53 +00:00
const parsedUrl = new URL ( request . url ) ;
2020-03-20 20:28:31 +00:00
let filename ;
2019-10-16 15:12:31 +00:00
switch ( parsedUrl . pathname ) {
2020-03-20 20:28:31 +00:00
case '/localStorage' : filename = 'local_storage.html' ; break ;
case '/sessionStorage' : filename = 'session_storage.html' ; break ;
case '/indexedDB' : filename = 'indexed_db.html' ; break ;
case '/cookie' : filename = 'cookie.html' ; break ;
default : filename = '' ;
2019-10-16 15:12:31 +00:00
}
2020-03-20 20:28:31 +00:00
callback ( { path : ` ${ fixturesPath } /pages/storage/ ${ filename } ` } ) ;
2020-06-02 16:46:18 +00:00
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
2020-06-02 16:46:18 +00:00
after ( ( ) = > {
protocol . unregisterProtocol ( protocolName ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
beforeEach ( ( ) = > {
2023-02-16 14:41:41 +00:00
contents = ( webContents as typeof ElectronInternal . WebContents ) . create ( {
2021-03-01 21:52:29 +00:00
nodeIntegration : true ,
contextIsolation : false
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2019-10-16 15:12:31 +00:00
afterEach ( ( ) = > {
2022-12-14 21:07:38 +00:00
contents . destroy ( ) ;
2020-03-20 20:28:31 +00:00
contents = null as any ;
} ) ;
2019-10-16 15:12:31 +00:00
2020-06-30 22:10:36 +00:00
it ( 'cannot access localStorage' , async ( ) = > {
2023-02-23 23:53:53 +00:00
const response = once ( ipcMain , 'local-storage-response' ) ;
2020-03-20 20:28:31 +00:00
contents . loadURL ( protocolName + '://host/localStorage' ) ;
2020-06-30 22:10:36 +00:00
const [ , error ] = await response ;
expect ( error ) . to . equal ( 'Failed to read the \'localStorage\' property from \'Window\': Access is denied for this document.' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
2020-06-30 22:10:36 +00:00
it ( 'cannot access sessionStorage' , async ( ) = > {
2023-02-23 23:53:53 +00:00
const response = once ( ipcMain , 'session-storage-response' ) ;
2020-03-20 20:28:31 +00:00
contents . loadURL ( ` ${ protocolName } ://host/sessionStorage ` ) ;
2020-06-30 22:10:36 +00:00
const [ , error ] = await response ;
expect ( error ) . to . equal ( 'Failed to read the \'sessionStorage\' property from \'Window\': Access is denied for this document.' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
2020-06-30 22:10:36 +00:00
it ( 'cannot access indexedDB' , async ( ) = > {
2023-02-23 23:53:53 +00:00
const response = once ( ipcMain , 'indexed-db-response' ) ;
2020-03-20 20:28:31 +00:00
contents . loadURL ( ` ${ protocolName } ://host/indexedDB ` ) ;
2020-06-30 22:10:36 +00:00
const [ , error ] = await response ;
expect ( error ) . to . equal ( 'Failed to execute \'open\' on \'IDBFactory\': access to the Indexed Database API is denied in this context.' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
2020-06-30 22:10:36 +00:00
it ( 'cannot access cookie' , async ( ) = > {
2023-02-23 23:53:53 +00:00
const response = once ( ipcMain , 'cookie-response' ) ;
2020-03-20 20:28:31 +00:00
contents . loadURL ( ` ${ protocolName } ://host/cookie ` ) ;
2020-06-30 22:10:36 +00:00
const [ , error ] = await response ;
expect ( error ) . to . equal ( 'Failed to set the \'cookie\' property on \'Document\': Access is denied for this document.' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2019-10-16 15:12:31 +00:00
describe ( 'can be accessed' , ( ) = > {
2020-03-20 20:28:31 +00:00
let server : http.Server ;
let serverUrl : string ;
let serverCrossSiteUrl : string ;
2023-02-20 11:30:57 +00:00
before ( async ( ) = > {
2019-10-16 15:12:31 +00:00
server = http . createServer ( ( req , res ) = > {
const respond = ( ) = > {
if ( req . url === '/redirect-cross-site' ) {
2020-03-20 20:28:31 +00:00
res . setHeader ( 'Location' , ` ${ serverCrossSiteUrl } /redirected ` ) ;
res . statusCode = 302 ;
res . end ( ) ;
2019-10-16 15:12:31 +00:00
} else if ( req . url === '/redirected' ) {
2020-03-20 20:28:31 +00:00
res . end ( '<html><script>window.localStorage</script></html>' ) ;
2019-10-16 15:12:31 +00:00
} else {
2020-03-20 20:28:31 +00:00
res . end ( ) ;
2019-10-16 15:12:31 +00:00
}
2020-03-20 20:28:31 +00:00
} ;
2023-02-23 23:53:53 +00:00
setTimeout ( ) . then ( respond ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2023-02-20 11:30:57 +00:00
serverUrl = ( await listen ( server ) ) . url ;
serverCrossSiteUrl = serverUrl . replace ( '127.0.0.1' , 'localhost' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
after ( ( ) = > {
2020-03-20 20:28:31 +00:00
server . close ( ) ;
server = null as any ;
} ) ;
2019-10-16 15:12:31 +00:00
2020-03-20 20:28:31 +00:00
afterEach ( closeAllWindows ) ;
2019-10-16 15:12:31 +00:00
const testLocalStorageAfterXSiteRedirect = ( testTitle : string , extraPreferences = { } ) = > {
2020-06-30 22:10:36 +00:00
it ( testTitle , async ( ) = > {
2019-10-16 15:12:31 +00:00
const w = new BrowserWindow ( {
show : false ,
. . . extraPreferences
2020-03-20 20:28:31 +00:00
} ) ;
let redirected = false ;
2023-02-16 09:25:41 +00:00
w . webContents . on ( 'render-process-gone' , ( ) = > {
2020-03-20 20:28:31 +00:00
expect . fail ( 'renderer crashed / was killed' ) ;
} ) ;
2019-10-16 15:12:31 +00:00
w . webContents . on ( 'did-redirect-navigation' , ( event , url ) = > {
2020-03-20 20:28:31 +00:00
expect ( url ) . to . equal ( ` ${ serverCrossSiteUrl } /redirected ` ) ;
redirected = true ;
} ) ;
2020-06-30 22:10:36 +00:00
await w . loadURL ( ` ${ serverUrl } /redirect-cross-site ` ) ;
expect ( redirected ) . to . be . true ( 'didnt redirect' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
} ;
testLocalStorageAfterXSiteRedirect ( 'after a cross-site redirect' ) ;
testLocalStorageAfterXSiteRedirect ( 'after a cross-site redirect in sandbox mode' , { sandbox : true } ) ;
} ) ;
2020-05-06 19:52:59 +00:00
2022-08-05 00:20:56 +00:00
describe ( 'DOM storage quota increase' , ( ) = > {
2023-08-31 14:36:43 +00:00
for ( const storageName of [ 'localStorage' , 'sessionStorage' ] ) {
2022-08-05 00:20:56 +00:00
it ( ` allows saving at least 40MiB in ${ storageName } ` , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
// Although JavaScript strings use UTF-16, the underlying
// storage provider may encode strings differently, muddling the
// translation between character and byte counts. However,
// a string of 40 * 2^20 characters will require at least 40MiB
// and presumably no more than 80MiB, a size guaranteed to
// to exceed the original 10MiB quota yet stay within the
// new 100MiB quota.
// Note that both the key name and value affect the total size.
const testKeyName = '_electronDOMStorageQuotaIncreasedTest' ;
const length = 40 * Math . pow ( 2 , 20 ) - testKeyName . length ;
await w . webContents . executeJavaScript ( `
$ { storageName } . setItem ( $ { JSON . stringify ( testKeyName ) } , 'X' . repeat ( $ { length } ) ) ;
` );
// Wait at least one turn of the event loop to help avoid false positives
// Although not entirely necessary, the previous version of this test case
// failed to detect a real problem (perhaps related to DOM storage data caching)
// wherein calling `getItem` immediately after `setItem` would appear to work
// but then later (e.g. next tick) it would not.
2023-02-23 23:53:53 +00:00
await setTimeout ( 1 ) ;
2022-08-05 00:20:56 +00:00
try {
const storedLength = await w . webContents . executeJavaScript ( ` ${ storageName } .getItem( ${ JSON . stringify ( testKeyName ) } ).length ` ) ;
expect ( storedLength ) . to . equal ( length ) ;
} finally {
await w . webContents . executeJavaScript ( ` ${ storageName } .removeItem( ${ JSON . stringify ( testKeyName ) } ); ` ) ;
}
} ) ;
it ( ` throws when attempting to use more than 128MiB in ${ storageName } ` , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
await expect ( ( async ( ) = > {
const testKeyName = '_electronDOMStorageQuotaStillEnforcedTest' ;
const length = 128 * Math . pow ( 2 , 20 ) - testKeyName . length ;
try {
await w . webContents . executeJavaScript ( `
$ { storageName } . setItem ( $ { JSON . stringify ( testKeyName ) } , 'X' . repeat ( $ { length } ) ) ;
` );
} finally {
await w . webContents . executeJavaScript ( ` ${ storageName } .removeItem( ${ JSON . stringify ( testKeyName ) } ); ` ) ;
}
} ) ( ) ) . to . eventually . be . rejected ( ) ;
} ) ;
2023-08-31 14:36:43 +00:00
}
2022-08-05 00:20:56 +00:00
} ) ;
describe ( 'persistent storage' , ( ) = > {
it ( 'can be requested' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const grantedBytes = await w . webContents . executeJavaScript ( ` new Promise(resolve => {
navigator . webkitPersistentStorage . requestQuota ( 1024 * 1024 , resolve ) ;
} ) ` );
expect ( grantedBytes ) . to . equal ( 1048576 ) ;
} ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
ifdescribe ( features . isPDFViewerEnabled ( ) ) ( 'PDF Viewer' , ( ) = > {
const pdfSource = url . format ( {
2023-09-07 06:50:14 +00:00
pathname : path.join ( __dirname , 'fixtures' , 'cat.pdf' ) . replaceAll ( '\\' , '/' ) ,
2019-10-16 15:12:31 +00:00
protocol : 'file' ,
slashes : true
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
2021-01-28 01:09:55 +00:00
it ( 'successfully loads a PDF file' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( pdfSource ) ;
2023-02-23 23:53:53 +00:00
await once ( w . webContents , 'did-finish-load' ) ;
2021-01-28 01:09:55 +00:00
} ) ;
2020-02-13 00:39:12 +00:00
it ( 'opens when loading a pdf resource as top level navigation' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( pdfSource ) ;
2023-06-22 18:38:52 +00:00
const [ , contents ] = await once ( app , 'web-contents-created' ) as [ any , WebContents ] ;
2023-02-23 23:53:53 +00:00
await once ( contents , 'did-navigate' ) ;
2021-01-13 21:49:35 +00:00
expect ( contents . getURL ( ) ) . to . equal ( 'chrome-extension://mhjfbmdgcfjbbpaeojofohoefgiehjai/index.html' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
2020-02-13 00:39:12 +00:00
it ( 'opens when loading a pdf resource in a iframe' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . loadFile ( path . join ( __dirname , 'fixtures' , 'pages' , 'pdf-in-iframe.html' ) ) ;
2023-06-22 18:38:52 +00:00
const [ , contents ] = await once ( app , 'web-contents-created' ) as [ any , WebContents ] ;
2023-02-23 23:53:53 +00:00
await once ( contents , 'did-navigate' ) ;
2021-01-13 21:49:35 +00:00
expect ( contents . getURL ( ) ) . to . equal ( 'chrome-extension://mhjfbmdgcfjbbpaeojofohoefgiehjai/index.html' ) ;
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2019-10-16 15:12:31 +00:00
describe ( 'window.history' , ( ) = > {
describe ( 'window.history.pushState' , ( ) = > {
2020-01-23 18:51:41 +00:00
it ( 'should push state after calling history.pushState() from the same url' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
2020-01-23 18:51:41 +00:00
// History should have current page by now.
2024-03-21 21:59:23 +00:00
expect ( w . webContents . navigationHistory . length ( ) ) . to . equal ( 1 ) ;
2020-01-23 18:51:41 +00:00
2023-02-23 23:53:53 +00:00
const waitCommit = once ( w . webContents , 'navigation-entry-committed' ) ;
2020-03-20 20:28:31 +00:00
w . webContents . executeJavaScript ( 'window.history.pushState({}, "")' ) ;
await waitCommit ;
2020-01-23 18:51:41 +00:00
// Initial page + pushed state.
2024-03-21 21:59:23 +00:00
expect ( w . webContents . navigationHistory . length ( ) ) . to . equal ( 2 ) ;
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2022-09-09 00:08:56 +00:00
describe ( 'window.history.back' , ( ) = > {
it ( 'should not allow sandboxed iframe to modify main frame state' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'data:text/html,<iframe sandbox="allow-scripts"></iframe>' ) ;
await Promise . all ( [
2023-02-23 23:53:53 +00:00
once ( w . webContents , 'navigation-entry-committed' ) ,
once ( w . webContents , 'did-frame-navigate' ) ,
once ( w . webContents , 'did-navigate' )
2022-09-09 00:08:56 +00:00
] ) ;
w . webContents . executeJavaScript ( 'window.history.pushState(1, "")' ) ;
await Promise . all ( [
2023-02-23 23:53:53 +00:00
once ( w . webContents , 'navigation-entry-committed' ) ,
once ( w . webContents , 'did-navigate-in-page' )
2022-09-09 00:08:56 +00:00
] ) ;
2023-09-05 02:22:41 +00:00
w . webContents . once ( 'navigation-entry-committed' as any , ( ) = > {
2022-09-09 00:08:56 +00:00
expect . fail ( 'Unexpected navigation-entry-committed' ) ;
} ) ;
w . webContents . once ( 'did-navigate-in-page' , ( ) = > {
expect . fail ( 'Unexpected did-navigate-in-page' ) ;
} ) ;
await w . webContents . mainFrame . frames [ 0 ] . executeJavaScript ( 'window.history.back()' ) ;
expect ( await w . webContents . executeJavaScript ( 'window.history.state' ) ) . to . equal ( 1 ) ;
2024-03-21 21:59:23 +00:00
expect ( w . webContents . navigationHistory . getActiveIndex ( ) ) . to . equal ( 1 ) ;
2022-09-09 00:08:56 +00:00
} ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2020-06-11 02:07:49 +00:00
2023-08-30 15:07:41 +00:00
describe ( 'chrome:// pages' , ( ) = > {
const urls = [
'chrome://accessibility' ,
'chrome://gpu' ,
'chrome://media-internals' ,
'chrome://tracing' ,
2024-03-07 14:31:16 +00:00
'chrome://webrtc-internals' ,
'chrome://process-internals'
2023-08-30 15:07:41 +00:00
] ;
2020-06-11 02:07:49 +00:00
2023-08-30 15:07:41 +00:00
for ( const url of urls ) {
describe ( url , ( ) = > {
it ( 'loads the page successfully' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( url ) ;
2024-03-07 14:31:16 +00:00
const host = url . substring ( 'chrome://' . length ) ;
2023-08-30 15:07:41 +00:00
const pageExists = await w . webContents . executeJavaScript (
2024-03-07 14:31:16 +00:00
` window.hasOwnProperty('chrome') && window.location.host === ' ${ host } ' `
2023-08-30 15:07:41 +00:00
) ;
expect ( pageExists ) . to . be . true ( ) ;
} ) ;
} ) ;
}
2020-06-11 02:07:49 +00:00
} ) ;
2021-05-19 09:27:35 +00:00
describe ( 'document.hasFocus' , ( ) = > {
it ( 'has correct value when multiple windows are opened' , async ( ) = > {
const w1 = new BrowserWindow ( { show : true } ) ;
const w2 = new BrowserWindow ( { show : true } ) ;
const w3 = new BrowserWindow ( { show : false } ) ;
await w1 . loadFile ( path . join ( __dirname , 'fixtures' , 'blank.html' ) ) ;
await w2 . loadFile ( path . join ( __dirname , 'fixtures' , 'blank.html' ) ) ;
await w3 . loadFile ( path . join ( __dirname , 'fixtures' , 'blank.html' ) ) ;
2023-04-24 14:28:21 +00:00
expect ( webContents . getFocusedWebContents ( ) ? . id ) . to . equal ( w2 . webContents . id ) ;
2021-05-19 09:27:35 +00:00
let focus = false ;
focus = await w1 . webContents . executeJavaScript (
'document.hasFocus()'
) ;
expect ( focus ) . to . be . false ( ) ;
focus = await w2 . webContents . executeJavaScript (
'document.hasFocus()'
) ;
expect ( focus ) . to . be . true ( ) ;
focus = await w3 . webContents . executeJavaScript (
'document.hasFocus()'
) ;
expect ( focus ) . to . be . false ( ) ;
} ) ;
} ) ;
2022-06-13 16:35:42 +00:00
2023-05-31 15:06:25 +00:00
// https://developer.mozilla.org/en-US/docs/Web/API/NetworkInformation
describe ( 'navigator.connection' , ( ) = > {
it ( 'returns the correct value' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . webContents . session . enableNetworkEmulation ( {
latency : 500 ,
downloadThroughput : 6400 ,
uploadThroughput : 6400
} ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const rtt = await w . webContents . executeJavaScript ( 'navigator.connection.rtt' ) ;
expect ( rtt ) . to . be . a ( 'number' ) ;
const downlink = await w . webContents . executeJavaScript ( 'navigator.connection.downlink' ) ;
expect ( downlink ) . to . be . a ( 'number' ) ;
const effectiveTypes = [ 'slow-2g' , '2g' , '3g' , '4g' ] ;
const effectiveType = await w . webContents . executeJavaScript ( 'navigator.connection.effectiveType' ) ;
expect ( effectiveTypes ) . to . include ( effectiveType ) ;
} ) ;
} ) ;
2022-06-13 16:35:42 +00:00
describe ( 'navigator.userAgentData' , ( ) = > {
// These tests are done on an http server because navigator.userAgentData
// requires a secure context.
let server : http.Server ;
let serverUrl : string ;
before ( async ( ) = > {
server = http . createServer ( ( req , res ) = > {
res . setHeader ( 'Content-Type' , 'text/html' ) ;
res . end ( '' ) ;
} ) ;
2023-02-20 11:30:57 +00:00
serverUrl = ( await listen ( server ) ) . url ;
2022-06-13 16:35:42 +00:00
} ) ;
after ( ( ) = > {
server . close ( ) ;
} ) ;
describe ( 'is not empty' , ( ) = > {
it ( 'by default' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( serverUrl ) ;
const platform = await w . webContents . executeJavaScript ( 'navigator.userAgentData.platform' ) ;
expect ( platform ) . not . to . be . empty ( ) ;
} ) ;
it ( 'when there is a session-wide UA override' , async ( ) = > {
const ses = session . fromPartition ( ` ${ Math . random ( ) } ` ) ;
ses . setUserAgent ( 'foobar' ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : ses } } ) ;
await w . loadURL ( serverUrl ) ;
const platform = await w . webContents . executeJavaScript ( 'navigator.userAgentData.platform' ) ;
expect ( platform ) . not . to . be . empty ( ) ;
} ) ;
it ( 'when there is a WebContents-specific UA override' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . webContents . setUserAgent ( 'foo' ) ;
await w . loadURL ( serverUrl ) ;
const platform = await w . webContents . executeJavaScript ( 'navigator.userAgentData.platform' ) ;
expect ( platform ) . not . to . be . empty ( ) ;
} ) ;
it ( 'when there is a WebContents-specific UA override at load time' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( serverUrl , {
userAgent : 'foo'
} ) ;
const platform = await w . webContents . executeJavaScript ( 'navigator.userAgentData.platform' ) ;
expect ( platform ) . not . to . be . empty ( ) ;
} ) ;
} ) ;
describe ( 'brand list' , ( ) = > {
it ( 'contains chromium' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( serverUrl ) ;
const brands = await w . webContents . executeJavaScript ( 'navigator.userAgentData.brands' ) ;
expect ( brands . map ( ( b : any ) = > b . brand ) ) . to . include ( 'Chromium' ) ;
} ) ;
} ) ;
} ) ;
2022-07-27 09:10:04 +00:00
describe ( 'Badging API' , ( ) = > {
it ( 'does not crash' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
await w . webContents . executeJavaScript ( 'navigator.setAppBadge(42)' ) ;
await w . webContents . executeJavaScript ( 'navigator.setAppBadge()' ) ;
await w . webContents . executeJavaScript ( 'navigator.clearAppBadge()' ) ;
} ) ;
} ) ;
describe ( 'navigator.webkitGetUserMedia' , ( ) = > {
it ( 'calls its callbacks' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
await w . webContents . executeJavaScript ( ` new Promise((resolve) => {
navigator . webkitGetUserMedia ( {
audio : true ,
video : false
} , ( ) = > resolve ( ) ,
( ) = > resolve ( ) ) ;
} ) ` );
} ) ;
} ) ;
describe ( 'navigator.language' , ( ) = > {
it ( 'should not be empty' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( 'about:blank' ) ;
expect ( await w . webContents . executeJavaScript ( 'navigator.language' ) ) . to . not . equal ( '' ) ;
} ) ;
} ) ;
describe ( 'heap snapshot' , ( ) = > {
it ( 'does not crash' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { nodeIntegration : true , contextIsolation : false } } ) ;
w . loadURL ( 'about:blank' ) ;
await w . webContents . executeJavaScript ( 'process._linkedBinding(\'electron_common_v8_util\').takeHeapSnapshot()' ) ;
} ) ;
} ) ;
2023-02-03 11:43:42 +00:00
// This is intentionally disabled on arm macs: https://chromium-review.googlesource.com/c/chromium/src/+/4143761
ifdescribe ( process . platform === 'darwin' && process . arch !== 'arm64' ) ( 'webgl' , ( ) = > {
2022-07-27 09:10:04 +00:00
it ( 'can be gotten as context in canvas' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const canWebglContextBeCreated = await w . webContents . executeJavaScript ( `
document . createElement ( 'canvas' ) . getContext ( 'webgl' ) != null ;
` );
expect ( canWebglContextBeCreated ) . to . be . true ( ) ;
} ) ;
} ) ;
describe ( 'iframe' , ( ) = > {
it ( 'does not have node integration' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const result = await w . webContents . executeJavaScript ( `
const iframe = document . createElement ( 'iframe' )
iframe . src = './set-global.html' ;
document . body . appendChild ( iframe ) ;
new Promise ( resolve = > iframe . onload = e = > resolve ( iframe . contentWindow . test ) )
` );
expect ( result ) . to . equal ( 'undefined undefined undefined' ) ;
} ) ;
} ) ;
describe ( 'websockets' , ( ) = > {
it ( 'has user agent' , async ( ) = > {
const server = http . createServer ( ) ;
2023-02-20 11:30:57 +00:00
const { port } = await listen ( server ) ;
2024-10-01 20:04:53 +00:00
const wss = new ws . Server ( { server } ) ;
2022-07-27 09:10:04 +00:00
const finished = new Promise < string | undefined > ( ( resolve , reject ) = > {
wss . on ( 'error' , reject ) ;
wss . on ( 'connection' , ( ws , upgradeReq ) = > {
resolve ( upgradeReq . headers [ 'user-agent' ] ) ;
} ) ;
} ) ;
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
w . webContents . executeJavaScript ( `
new WebSocket ( 'ws://127.0.0.1:${port}' ) ;
` );
expect ( await finished ) . to . include ( 'Electron' ) ;
} ) ;
} ) ;
describe ( 'fetch' , ( ) = > {
it ( 'does not crash' , async ( ) = > {
const server = http . createServer ( ( req , res ) = > {
res . end ( 'test' ) ;
} ) ;
defer ( ( ) = > server . close ( ) ) ;
2023-02-20 11:30:57 +00:00
const { port } = await listen ( server ) ;
2022-07-27 09:10:04 +00:00
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( ` file:// ${ fixturesPath } /pages/blank.html ` ) ;
const x = await w . webContents . executeJavaScript ( `
fetch ( 'http://127.0.0.1:${port}' ) . then ( ( res ) = > res . body . getReader ( ) )
. then ( ( reader ) = > {
return reader . read ( ) . then ( ( r ) = > {
reader . cancel ( ) ;
return r . value ;
} ) ;
} )
` );
expect ( x ) . to . deep . equal ( new Uint8Array ( [ 116 , 101 , 115 , 116 ] ) ) ;
} ) ;
} ) ;
2022-08-05 00:20:56 +00:00
describe ( 'Promise' , ( ) = > {
before ( ( ) = > {
ipcMain . handle ( 'ping' , ( e , arg ) = > arg ) ;
} ) ;
after ( ( ) = > {
ipcMain . removeHandler ( 'ping' ) ;
} ) ;
itremote ( 'resolves correctly in Node.js calls' , async ( ) = > {
await new Promise < void > ( ( resolve , reject ) = > {
class XElement extends HTMLElement { }
customElements . define ( 'x-element' , XElement ) ;
setImmediate ( ( ) = > {
let called = false ;
Promise . resolve ( ) . then ( ( ) = > {
if ( called ) resolve ( ) ;
else reject ( new Error ( 'wrong sequence' ) ) ;
} ) ;
document . createElement ( 'x-element' ) ;
called = true ;
} ) ;
} ) ;
} ) ;
itremote ( 'resolves correctly in Electron calls' , async ( ) = > {
await new Promise < void > ( ( resolve , reject ) = > {
class YElement extends HTMLElement { }
customElements . define ( 'y-element' , YElement ) ;
require ( 'electron' ) . ipcRenderer . invoke ( 'ping' ) . then ( ( ) = > {
let called = false ;
Promise . resolve ( ) . then ( ( ) = > {
if ( called ) resolve ( ) ;
else reject ( new Error ( 'wrong sequence' ) ) ;
} ) ;
document . createElement ( 'y-element' ) ;
called = true ;
} ) ;
} ) ;
} ) ;
} ) ;
describe ( 'synchronous prompts' , ( ) = > {
describe ( 'window.alert(message, title)' , ( ) = > {
itremote ( 'throws an exception when the arguments cannot be converted to strings' , ( ) = > {
expect ( ( ) = > {
window . alert ( { toString : null } ) ;
} ) . to . throw ( 'Cannot convert object to primitive value' ) ;
} ) ;
2023-12-06 01:36:23 +00:00
it ( 'shows a message box' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
const p = once ( w . webContents , '-run-dialog' ) ;
w . webContents . executeJavaScript ( 'alert("hello")' , true ) ;
const [ info ] = await p ;
expect ( info . frame ) . to . equal ( w . webContents . mainFrame ) ;
expect ( info . messageText ) . to . equal ( 'hello' ) ;
expect ( info . dialogType ) . to . equal ( 'alert' ) ;
} ) ;
it ( 'does not crash if a webContents is destroyed while an alert is showing' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
const p = once ( w . webContents , '-run-dialog' ) ;
w . webContents . executeJavaScript ( 'alert("hello")' , true ) ;
await p ;
w . webContents . close ( ) ;
} ) ;
2022-08-05 00:20:56 +00:00
} ) ;
describe ( 'window.confirm(message, title)' , ( ) = > {
itremote ( 'throws an exception when the arguments cannot be converted to strings' , ( ) = > {
expect ( ( ) = > {
( window . confirm as any ) ( { toString : null } , 'title' ) ;
} ) . to . throw ( 'Cannot convert object to primitive value' ) ;
} ) ;
2023-12-06 01:36:23 +00:00
it ( 'shows a message box' , async ( ) = > {
const w = new BrowserWindow ( { show : false } ) ;
w . loadURL ( 'about:blank' ) ;
const p = once ( w . webContents , '-run-dialog' ) ;
const resultPromise = w . webContents . executeJavaScript ( 'confirm("hello")' , true ) ;
const [ info , cb ] = await p ;
expect ( info . frame ) . to . equal ( w . webContents . mainFrame ) ;
expect ( info . messageText ) . to . equal ( 'hello' ) ;
expect ( info . dialogType ) . to . equal ( 'confirm' ) ;
cb ( true , '' ) ;
const result = await resultPromise ;
expect ( result ) . to . be . true ( ) ;
} ) ;
} ) ;
describe ( 'safeDialogs web preference' , ( ) = > {
const originalShowMessageBox = dialog . showMessageBox ;
afterEach ( ( ) = > {
dialog . showMessageBox = originalShowMessageBox ;
if ( protocol . isProtocolHandled ( 'https' ) ) protocol . unhandle ( 'https' ) ;
if ( protocol . isProtocolHandled ( 'file' ) ) protocol . unhandle ( 'file' ) ;
} ) ;
it ( 'does not show the checkbox if not enabled' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { safeDialogs : false } } ) ;
w . loadURL ( 'about:blank' ) ;
// 1. The first alert() doesn't show the safeDialogs message.
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
let recordedOpts : MessageBoxOptions | undefined ;
dialog . showMessageBox = ( bw , opts? : MessageBoxOptions ) = > {
recordedOpts = opts ;
return Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
} ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
expect ( recordedOpts ? . checkboxLabel ) . to . equal ( '' ) ;
} ) ;
it ( 'is respected' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { safeDialogs : true } } ) ;
w . loadURL ( 'about:blank' ) ;
// 1. The first alert() doesn't show the safeDialogs message.
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
// 2. The second alert() shows the message with a checkbox. Respond that we checked it.
let recordedOpts : MessageBoxOptions | undefined ;
dialog . showMessageBox = ( bw , opts? : MessageBoxOptions ) = > {
recordedOpts = opts ;
return Promise . resolve ( { response : 0 , checkboxChecked : true } ) ;
} ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
expect ( recordedOpts ? . checkboxLabel ) . to . be . a ( 'string' ) . with . length . above ( 0 ) ;
// 3. The third alert() shouldn't show a dialog.
dialog . showMessageBox = ( ) = > Promise . reject ( new Error ( 'unexpected showMessageBox' ) ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
} ) ;
it ( 'shows the safeDialogMessage' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { safeDialogs : true , safeDialogsMessage : 'foo bar' } } ) ;
w . loadURL ( 'about:blank' ) ;
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
let recordedOpts : MessageBoxOptions | undefined ;
dialog . showMessageBox = ( bw , opts? : MessageBoxOptions ) = > {
recordedOpts = opts ;
return Promise . resolve ( { response : 0 , checkboxChecked : true } ) ;
} ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
expect ( recordedOpts ? . checkboxLabel ) . to . equal ( 'foo bar' ) ;
} ) ;
it ( 'has persistent state across navigations' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { safeDialogs : true } } ) ;
w . loadURL ( 'about:blank' ) ;
// 1. The first alert() doesn't show the safeDialogs message.
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
// 2. The second alert() shows the message with a checkbox. Respond that we checked it.
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : true } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
// 3. The third alert() shouldn't show a dialog.
dialog . showMessageBox = ( ) = > Promise . reject ( new Error ( 'unexpected showMessageBox' ) ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
// 4. After navigating to the same origin, message boxes should still be hidden.
w . loadURL ( 'about:blank' ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
} ) ;
it ( 'is separated by origin' , async ( ) = > {
protocol . handle ( 'https' , ( ) = > new Response ( '' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { safeDialogs : true } } ) ;
w . loadURL ( 'https://example1' ) ;
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : true } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
dialog . showMessageBox = ( ) = > Promise . reject ( new Error ( 'unexpected showMessageBox' ) ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
// A different origin is allowed to show message boxes after navigation.
w . loadURL ( 'https://example2' ) ;
let dialogWasShown = false ;
dialog . showMessageBox = ( ) = > {
dialogWasShown = true ;
return Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
} ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
expect ( dialogWasShown ) . to . be . true ( ) ;
// Navigating back to the first origin means alerts are blocked again.
w . loadURL ( 'https://example1' ) ;
dialog . showMessageBox = ( ) = > Promise . reject ( new Error ( 'unexpected showMessageBox' ) ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
} ) ;
it ( 'treats different file: paths as different origins' , async ( ) = > {
protocol . handle ( 'file' , ( ) = > new Response ( '' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { safeDialogs : true } } ) ;
w . loadURL ( 'file:///path/1' ) ;
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
dialog . showMessageBox = ( ) = > Promise . resolve ( { response : 0 , checkboxChecked : true } ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
dialog . showMessageBox = ( ) = > Promise . reject ( new Error ( 'unexpected showMessageBox' ) ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
w . loadURL ( 'file:///path/2' ) ;
let dialogWasShown = false ;
dialog . showMessageBox = ( ) = > {
dialogWasShown = true ;
return Promise . resolve ( { response : 0 , checkboxChecked : false } ) ;
} ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
expect ( dialogWasShown ) . to . be . true ( ) ;
} ) ;
} ) ;
describe ( 'disableDialogs web preference' , ( ) = > {
const originalShowMessageBox = dialog . showMessageBox ;
afterEach ( ( ) = > {
dialog . showMessageBox = originalShowMessageBox ;
if ( protocol . isProtocolHandled ( 'https' ) ) protocol . unhandle ( 'https' ) ;
} ) ;
it ( 'is respected' , async ( ) = > {
const w = new BrowserWindow ( { show : false , webPreferences : { disableDialogs : true } } ) ;
w . loadURL ( 'about:blank' ) ;
dialog . showMessageBox = ( ) = > Promise . reject ( new Error ( 'unexpected message box' ) ) ;
await w . webContents . executeJavaScript ( 'alert("hi")' ) ;
} ) ;
2022-08-05 00:20:56 +00:00
} ) ;
} ) ;
describe ( 'window.history' , ( ) = > {
describe ( 'window.history.go(offset)' , ( ) = > {
itremote ( 'throws an exception when the argument cannot be converted to a string' , ( ) = > {
expect ( ( ) = > {
( window . history . go as any ) ( { toString : null } ) ;
} ) . to . throw ( 'Cannot convert object to primitive value' ) ;
} ) ;
} ) ;
} ) ;
describe ( 'console functions' , ( ) = > {
itremote ( 'should exist' , ( ) = > {
expect ( console . log , 'log' ) . to . be . a ( 'function' ) ;
expect ( console . error , 'error' ) . to . be . a ( 'function' ) ;
expect ( console . warn , 'warn' ) . to . be . a ( 'function' ) ;
expect ( console . info , 'info' ) . to . be . a ( 'function' ) ;
expect ( console . debug , 'debug' ) . to . be . a ( 'function' ) ;
expect ( console . trace , 'trace' ) . to . be . a ( 'function' ) ;
expect ( console . time , 'time' ) . to . be . a ( 'function' ) ;
expect ( console . timeEnd , 'timeEnd' ) . to . be . a ( 'function' ) ;
} ) ;
} ) ;
2023-04-04 13:48:51 +00:00
// FIXME(nornagon): this is broken on CI, it triggers:
// [FATAL:speech_synthesis.mojom-shared.h(237)] The outgoing message will
// trigger VALIDATION_ERROR_UNEXPECTED_NULL_POINTER at the receiving side
// (null text in SpeechSynthesisUtterance struct).
2023-05-24 08:18:43 +00:00
describe ( 'SpeechSynthesis' , ( ) = > {
2022-08-05 00:20:56 +00:00
itremote ( 'should emit lifecycle events' , async ( ) = > {
const sentence = ` long sentence which will take at least a few seconds to
utter so that it ' s possible to pause and resume before the end ` ;
const utter = new SpeechSynthesisUtterance ( sentence ) ;
// Create a dummy utterance so that speech synthesis state
// is initialized for later calls.
speechSynthesis . speak ( new SpeechSynthesisUtterance ( ) ) ;
speechSynthesis . cancel ( ) ;
speechSynthesis . speak ( utter ) ;
// paused state after speak()
expect ( speechSynthesis . paused ) . to . be . false ( ) ;
await new Promise ( ( resolve ) = > { utter . onstart = resolve ; } ) ;
// paused state after start event
expect ( speechSynthesis . paused ) . to . be . false ( ) ;
speechSynthesis . pause ( ) ;
// paused state changes async, right before the pause event
expect ( speechSynthesis . paused ) . to . be . false ( ) ;
await new Promise ( ( resolve ) = > { utter . onpause = resolve ; } ) ;
expect ( speechSynthesis . paused ) . to . be . true ( ) ;
speechSynthesis . resume ( ) ;
await new Promise ( ( resolve ) = > { utter . onresume = resolve ; } ) ;
// paused state after resume event
expect ( speechSynthesis . paused ) . to . be . false ( ) ;
await new Promise ( ( resolve ) = > { utter . onend = resolve ; } ) ;
} ) ;
} ) ;
2024-11-14 11:45:08 +00:00
describe ( 'devtools' , ( ) = > {
it ( 'fetch colors.css' , async ( ) = > {
// <link href="devtools://theme/colors.css?sets=ui,chrome" rel="stylesheet">
const w = new BrowserWindow ( { show : false } ) ;
const devtools = new BrowserWindow ( { show : false } ) ;
const devToolsOpened = once ( w . webContents , 'devtools-opened' ) ;
w . webContents . setDevToolsWebContents ( devtools . webContents ) ;
w . webContents . openDevTools ( ) ;
await devToolsOpened ;
expect ( devtools . webContents . getURL ( ) . startsWith ( 'devtools://devtools' ) ) . to . be . true ( ) ;
const result = await devtools . webContents . executeJavaScript ( `
document . body . querySelector ( 'link[href*=\\' //theme/colors.css\\']')?.getAttribute('href');
` );
expect ( result . startsWith ( 'devtools://theme/colors.css?sets=ui,chrome' ) ) . to . be . true ( ) ;
const colorAccentResult = await devtools . webContents . executeJavaScript ( `
const style = getComputedStyle ( document . body ) ;
style . getPropertyValue ( '--color-accent' ) ;
` );
expect ( colorAccentResult ) . to . not . equal ( '' ) ;
const colorAppMenuHighlightSeverityLow = await devtools . webContents . executeJavaScript ( `
style . getPropertyValue ( '--color-app-menu-highlight-severity-low' ) ;
` );
expect ( colorAppMenuHighlightSeverityLow ) . to . not . equal ( '' ) ;
const rgb = await devtools . webContents . executeJavaScript ( `
style . getPropertyValue ( '--color-accent-rgb' ) ;
` );
expect ( rgb ) . to . equal ( '' ) ;
} ) ;
} ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
describe ( 'font fallback' , ( ) = > {
async function getRenderedFonts ( html : string ) {
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false } ) ;
2019-10-16 15:12:31 +00:00
try {
2020-03-20 20:28:31 +00:00
await w . loadURL ( ` data:text/html, ${ html } ` ) ;
w . webContents . debugger . attach ( ) ;
const sendCommand = ( method : string , commandParams? : any ) = > w . webContents . debugger . sendCommand ( method , commandParams ) ;
const { nodeId } = ( await sendCommand ( 'DOM.getDocument' ) ) . root . children [ 0 ] ;
await sendCommand ( 'CSS.enable' ) ;
const { fonts } = await sendCommand ( 'CSS.getPlatformFontsForNode' , { nodeId } ) ;
return fonts ;
2019-10-16 15:12:31 +00:00
} finally {
2020-03-20 20:28:31 +00:00
w . close ( ) ;
2019-10-16 15:12:31 +00:00
}
}
it ( 'should use Helvetica for sans-serif on Mac, and Arial on Windows and Linux' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const html = '<body style="font-family: sans-serif">test</body>' ;
const fonts = await getRenderedFonts ( html ) ;
expect ( fonts ) . to . be . an ( 'array' ) ;
expect ( fonts ) . to . have . length ( 1 ) ;
2021-04-19 18:27:34 +00:00
if ( process . platform === 'win32' ) {
expect ( fonts [ 0 ] . familyName ) . to . equal ( 'Arial' ) ;
} else if ( process . platform === 'darwin' ) {
expect ( fonts [ 0 ] . familyName ) . to . equal ( 'Helvetica' ) ;
} else if ( process . platform === 'linux' ) {
expect ( fonts [ 0 ] . familyName ) . to . equal ( 'DejaVu Sans' ) ;
} // I think this depends on the distro? We don't specify a default.
2020-03-20 20:28:31 +00:00
} ) ;
2019-10-16 15:12:31 +00:00
ifit ( process . platform !== 'linux' ) ( 'should fall back to Japanese font for sans-serif Japanese script' , async function ( ) {
const html = `
< html lang = "ja-JP" >
< head >
< meta charset = "utf-8" / >
< / head >
< body style = "font-family: sans-serif" > test 智 史 < / body >
< / html >
2020-03-20 20:28:31 +00:00
` ;
const fonts = await getRenderedFonts ( html ) ;
expect ( fonts ) . to . be . an ( 'array' ) ;
expect ( fonts ) . to . have . length ( 1 ) ;
if ( process . platform === 'win32' ) { expect ( fonts [ 0 ] . familyName ) . to . be . oneOf ( [ 'Meiryo' , 'Yu Gothic' ] ) ; } else if ( process . platform === 'darwin' ) { expect ( fonts [ 0 ] . familyName ) . to . equal ( 'Hiragino Kaku Gothic ProN' ) ; }
} ) ;
} ) ;
2019-11-05 21:34:45 +00:00
describe ( 'iframe using HTML fullscreen API while window is OS-fullscreened' , ( ) = > {
2023-11-17 09:44:03 +00:00
const fullscreenChildHtml = fs . promises . readFile (
2019-11-05 21:34:45 +00:00
path . join ( fixturesPath , 'pages' , 'fullscreen-oopif.html' )
2020-03-20 20:28:31 +00:00
) ;
2023-02-20 11:30:57 +00:00
let w : BrowserWindow ;
let server : http.Server ;
let crossSiteUrl : string ;
2019-11-05 21:34:45 +00:00
2023-02-08 20:40:24 +00:00
beforeEach ( async ( ) = > {
2019-11-05 21:34:45 +00:00
server = http . createServer ( async ( _req , res ) = > {
2020-03-20 20:28:31 +00:00
res . writeHead ( 200 , { 'Content-Type' : 'text/html' } ) ;
res . write ( await fullscreenChildHtml ) ;
res . end ( ) ;
} ) ;
2019-11-05 21:34:45 +00:00
2023-02-20 11:30:57 +00:00
const serverUrl = ( await listen ( server ) ) . url ;
crossSiteUrl = serverUrl . replace ( '127.0.0.1' , 'localhost' ) ;
2019-11-05 21:34:45 +00:00
w = new BrowserWindow ( {
show : true ,
fullscreen : true ,
webPreferences : {
nodeIntegration : true ,
2021-03-01 21:52:29 +00:00
nodeIntegrationInSubFrames : true ,
contextIsolation : false
2019-11-05 21:34:45 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2019-11-05 21:34:45 +00:00
afterEach ( async ( ) = > {
2021-04-21 14:56:25 +00:00
await closeAllWindows ( ) ;
( w as any ) = null ;
2020-03-20 20:28:31 +00:00
server . close ( ) ;
} ) ;
2019-11-05 21:34:45 +00:00
2022-06-07 16:59:50 +00:00
ifit ( process . platform !== 'darwin' ) ( 'can fullscreen from out-of-process iframes (non-macOS)' , async ( ) = > {
2023-02-23 23:53:53 +00:00
const fullscreenChange = once ( ipcMain , 'fullscreenChange' ) ;
2020-06-30 22:10:36 +00:00
const html =
2023-02-20 11:30:57 +00:00
` <iframe style="width: 0" frameborder=0 src=" ${ crossSiteUrl } " allowfullscreen></iframe> ` ;
2020-06-30 22:10:36 +00:00
w . loadURL ( ` data:text/html, ${ html } ` ) ;
await fullscreenChange ;
2019-11-05 21:34:45 +00:00
2020-06-30 22:10:36 +00:00
const fullscreenWidth = await w . webContents . executeJavaScript (
"document.querySelector('iframe').offsetWidth"
) ;
expect ( fullscreenWidth > 0 ) . to . be . true ( ) ;
2019-11-05 21:34:45 +00:00
2020-06-30 22:10:36 +00:00
await w . webContents . executeJavaScript (
"document.querySelector('iframe').contentWindow.postMessage('exitFullscreen', '*')"
) ;
2019-11-05 21:34:45 +00:00
2023-02-23 23:53:53 +00:00
await setTimeout ( 500 ) ;
2019-11-05 21:34:45 +00:00
2020-06-30 22:10:36 +00:00
const width = await w . webContents . executeJavaScript (
"document.querySelector('iframe').offsetWidth"
) ;
expect ( width ) . to . equal ( 0 ) ;
2020-03-20 20:28:31 +00:00
} ) ;
2019-11-05 21:34:45 +00:00
2022-06-07 16:59:50 +00:00
ifit ( process . platform === 'darwin' ) ( 'can fullscreen from out-of-process iframes (macOS)' , async ( ) = > {
2023-02-23 23:53:53 +00:00
await once ( w , 'enter-full-screen' ) ;
const fullscreenChange = once ( ipcMain , 'fullscreenChange' ) ;
2022-06-07 16:59:50 +00:00
const html =
2023-02-20 11:30:57 +00:00
` <iframe style="width: 0" frameborder=0 src=" ${ crossSiteUrl } " allowfullscreen></iframe> ` ;
2022-06-07 16:59:50 +00:00
w . loadURL ( ` data:text/html, ${ html } ` ) ;
await fullscreenChange ;
const fullscreenWidth = await w . webContents . executeJavaScript (
"document.querySelector('iframe').offsetWidth"
) ;
expect ( fullscreenWidth > 0 ) . to . be . true ( ) ;
await w . webContents . executeJavaScript (
"document.querySelector('iframe').contentWindow.postMessage('exitFullscreen', '*')"
) ;
2023-02-23 23:53:53 +00:00
await once ( w . webContents , 'leave-html-full-screen' ) ;
2022-06-07 16:59:50 +00:00
2024-10-24 13:21:55 +00:00
await expect ( waitUntil ( async ( ) = > {
const width = await w . webContents . executeJavaScript (
"document.querySelector('iframe').offsetWidth"
) ;
return width === 0 ;
} ) ) . to . eventually . be . fulfilled ( ) ;
2022-06-07 16:59:50 +00:00
w . setFullScreen ( false ) ;
2023-02-23 23:53:53 +00:00
await once ( w , 'leave-full-screen' ) ;
2022-06-07 16:59:50 +00:00
} ) ;
2022-02-25 18:17:35 +00:00
// TODO(jkleinsc) fix this flaky test on WOA
ifit ( process . platform !== 'win32' || process . arch !== 'arm64' ) ( 'can fullscreen from in-process iframes' , async ( ) = > {
2023-02-23 23:53:53 +00:00
if ( process . platform === 'darwin' ) await once ( w , 'enter-full-screen' ) ;
2022-06-07 16:59:50 +00:00
2023-02-23 23:53:53 +00:00
const fullscreenChange = once ( ipcMain , 'fullscreenChange' ) ;
2020-06-30 22:10:36 +00:00
w . loadFile ( path . join ( fixturesPath , 'pages' , 'fullscreen-ipif.html' ) ) ;
await fullscreenChange ;
2019-11-05 21:34:45 +00:00
2020-06-30 22:10:36 +00:00
const fullscreenWidth = await w . webContents . executeJavaScript (
"document.querySelector('iframe').offsetWidth"
) ;
expect ( fullscreenWidth > 0 ) . to . true ( ) ;
2020-03-20 20:28:31 +00:00
2020-06-30 22:10:36 +00:00
await w . webContents . executeJavaScript ( 'document.exitFullscreen()' ) ;
const width = await w . webContents . executeJavaScript (
"document.querySelector('iframe').offsetWidth"
) ;
expect ( width ) . to . equal ( 0 ) ;
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2020-09-28 16:22:03 +00:00
describe ( 'navigator.serial' , ( ) = > {
let w : BrowserWindow ;
before ( async ( ) = > {
w = new BrowserWindow ( {
2021-07-29 23:57:22 +00:00
show : false
2020-09-28 16:22:03 +00:00
} ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
} ) ;
const getPorts : any = ( ) = > {
return w . webContents . executeJavaScript ( `
navigator . serial . requestPort ( ) . then ( port = > port . toString ( ) ) . catch ( err = > err . toString ( ) ) ;
` , true);
} ;
2022-10-27 16:37:04 +00:00
const notFoundError = 'NotFoundError: Failed to execute \'requestPort\' on \'Serial\': No port selected by the user.' ;
2020-09-28 16:22:03 +00:00
after ( closeAllWindows ) ;
afterEach ( ( ) = > {
session . defaultSession . setPermissionCheckHandler ( null ) ;
session . defaultSession . removeAllListeners ( 'select-serial-port' ) ;
} ) ;
it ( 'does not return a port if select-serial-port event is not defined' , async ( ) = > {
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const port = await getPorts ( ) ;
2022-10-27 16:37:04 +00:00
expect ( port ) . to . equal ( notFoundError ) ;
2020-09-28 16:22:03 +00:00
} ) ;
it ( 'does not return a port when permission denied' , async ( ) = > {
w . webContents . session . on ( 'select-serial-port' , ( event , portList , webContents , callback ) = > {
callback ( portList [ 0 ] . portId ) ;
} ) ;
session . defaultSession . setPermissionCheckHandler ( ( ) = > false ) ;
const port = await getPorts ( ) ;
2022-10-27 16:37:04 +00:00
expect ( port ) . to . equal ( notFoundError ) ;
2020-09-28 16:22:03 +00:00
} ) ;
2021-04-12 13:18:39 +00:00
it ( 'does not crash when select-serial-port is called with an invalid port' , async ( ) = > {
w . webContents . session . on ( 'select-serial-port' , ( event , portList , webContents , callback ) = > {
callback ( 'i-do-not-exist' ) ;
} ) ;
const port = await getPorts ( ) ;
2022-10-27 16:37:04 +00:00
expect ( port ) . to . equal ( notFoundError ) ;
2021-04-12 13:18:39 +00:00
} ) ;
2020-09-28 16:22:03 +00:00
it ( 'returns a port when select-serial-port event is defined' , async ( ) = > {
2022-05-23 14:33:39 +00:00
let havePorts = false ;
2020-09-28 16:22:03 +00:00
w . webContents . session . on ( 'select-serial-port' , ( event , portList , webContents , callback ) = > {
2022-05-23 14:33:39 +00:00
if ( portList . length > 0 ) {
havePorts = true ;
callback ( portList [ 0 ] . portId ) ;
} else {
callback ( '' ) ;
}
2020-09-28 16:22:03 +00:00
} ) ;
const port = await getPorts ( ) ;
2022-05-23 14:33:39 +00:00
if ( havePorts ) {
expect ( port ) . to . equal ( '[object SerialPort]' ) ;
} else {
2022-10-27 16:37:04 +00:00
expect ( port ) . to . equal ( notFoundError ) ;
2022-05-23 14:33:39 +00:00
}
} ) ;
it ( 'navigator.serial.getPorts() returns values' , async ( ) = > {
let havePorts = false ;
w . webContents . session . on ( 'select-serial-port' , ( event , portList , webContents , callback ) = > {
if ( portList . length > 0 ) {
havePorts = true ;
callback ( portList [ 0 ] . portId ) ;
} else {
callback ( '' ) ;
}
} ) ;
await getPorts ( ) ;
if ( havePorts ) {
const grantedPorts = await w . webContents . executeJavaScript ( 'navigator.serial.getPorts()' ) ;
expect ( grantedPorts ) . to . not . be . empty ( ) ;
}
2020-09-28 16:22:03 +00:00
} ) ;
2022-10-18 09:22:32 +00:00
it ( 'supports port.forget()' , async ( ) = > {
let forgottenPortFromEvent = { } ;
let havePorts = false ;
w . webContents . session . on ( 'select-serial-port' , ( event , portList , webContents , callback ) = > {
if ( portList . length > 0 ) {
havePorts = true ;
callback ( portList [ 0 ] . portId ) ;
} else {
callback ( '' ) ;
}
} ) ;
w . webContents . session . on ( 'serial-port-revoked' , ( event , details ) = > {
forgottenPortFromEvent = details . port ;
} ) ;
await getPorts ( ) ;
if ( havePorts ) {
const grantedPorts = await w . webContents . executeJavaScript ( 'navigator.serial.getPorts()' ) ;
if ( grantedPorts . length > 0 ) {
const forgottenPort = await w . webContents . executeJavaScript ( `
navigator . serial . getPorts ( ) . then ( async ( ports ) = > {
const portInfo = await ports [ 0 ] . getInfo ( ) ;
await ports [ 0 ] . forget ( ) ;
if ( portInfo . usbVendorId && portInfo . usbProductId ) {
return {
vendorId : '' + portInfo . usbVendorId ,
productId : '' + portInfo . usbProductId
}
} else {
return { } ;
}
} )
` );
const grantedPorts2 = await w . webContents . executeJavaScript ( 'navigator.serial.getPorts()' ) ;
expect ( grantedPorts2 . length ) . to . be . lessThan ( grantedPorts . length ) ;
if ( forgottenPort . vendorId && forgottenPort . productId ) {
expect ( forgottenPortFromEvent ) . to . include ( forgottenPort ) ;
}
}
}
} ) ;
2020-09-28 16:22:03 +00:00
} ) ;
2020-10-29 18:22:32 +00:00
2023-01-10 08:49:04 +00:00
describe ( 'window.getScreenDetails' , ( ) = > {
let w : BrowserWindow ;
before ( async ( ) = > {
w = new BrowserWindow ( {
show : false
} ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
} ) ;
after ( closeAllWindows ) ;
afterEach ( ( ) = > {
session . defaultSession . setPermissionRequestHandler ( null ) ;
} ) ;
const getScreenDetails : any = ( ) = > {
return w . webContents . executeJavaScript ( 'window.getScreenDetails().then(data => data.screens).catch(err => err.message)' , true ) ;
} ;
it ( 'returns screens when a PermissionRequestHandler is not defined' , async ( ) = > {
const screens = await getScreenDetails ( ) ;
expect ( screens ) . to . not . equal ( 'Read permission denied.' ) ;
} ) ;
it ( 'returns an error when permission denied' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler ( ( wc , permission , callback ) = > {
if ( permission === 'window-management' ) {
callback ( false ) ;
} else {
callback ( true ) ;
}
} ) ;
const screens = await getScreenDetails ( ) ;
expect ( screens ) . to . equal ( 'Permission denied.' ) ;
} ) ;
it ( 'returns screens when permission is granted' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler ( ( wc , permission , callback ) = > {
if ( permission === 'window-management' ) {
callback ( true ) ;
} else {
callback ( false ) ;
}
} ) ;
const screens = await getScreenDetails ( ) ;
expect ( screens ) . to . not . equal ( 'Permission denied.' ) ;
} ) ;
} ) ;
2023-02-09 10:38:39 +00:00
describe ( 'navigator.clipboard.read' , ( ) = > {
2020-10-29 18:22:32 +00:00
let w : BrowserWindow ;
before ( async ( ) = > {
2023-02-09 10:38:39 +00:00
w = new BrowserWindow ( ) ;
2020-10-29 18:22:32 +00:00
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
} ) ;
const readClipboard : any = ( ) = > {
return w . webContents . executeJavaScript ( `
navigator . clipboard . read ( ) . then ( clipboard = > clipboard . toString ( ) ) . catch ( err = > err . message ) ;
` , true);
} ;
after ( closeAllWindows ) ;
afterEach ( ( ) = > {
session . defaultSession . setPermissionRequestHandler ( null ) ;
} ) ;
it ( 'returns clipboard contents when a PermissionRequestHandler is not defined' , async ( ) = > {
const clipboard = await readClipboard ( ) ;
2024-01-25 17:46:30 +00:00
expect ( clipboard ) . to . not . contain ( 'Read permission denied.' ) ;
2020-10-29 18:22:32 +00:00
} ) ;
it ( 'returns an error when permission denied' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler ( ( wc , permission , callback ) = > {
if ( permission === 'clipboard-read' ) {
callback ( false ) ;
} else {
callback ( true ) ;
}
} ) ;
const clipboard = await readClipboard ( ) ;
2024-01-25 17:46:30 +00:00
expect ( clipboard ) . to . contain ( 'Read permission denied.' ) ;
2020-10-29 18:22:32 +00:00
} ) ;
it ( 'returns clipboard contents when permission is granted' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler ( ( wc , permission , callback ) = > {
if ( permission === 'clipboard-read' ) {
callback ( true ) ;
} else {
callback ( false ) ;
}
} ) ;
const clipboard = await readClipboard ( ) ;
2024-01-25 17:46:30 +00:00
expect ( clipboard ) . to . not . contain ( 'Read permission denied.' ) ;
2020-10-29 18:22:32 +00:00
} ) ;
} ) ;
2021-01-21 05:45:06 +00:00
2023-02-09 10:38:39 +00:00
describe ( 'navigator.clipboard.write' , ( ) = > {
let w : BrowserWindow ;
before ( async ( ) = > {
w = new BrowserWindow ( ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
} ) ;
const writeClipboard : any = ( ) = > {
return w . webContents . executeJavaScript ( `
navigator . clipboard . writeText ( 'Hello World!' ) . catch ( err = > err . message ) ;
` , true);
} ;
after ( closeAllWindows ) ;
afterEach ( ( ) = > {
session . defaultSession . setPermissionRequestHandler ( null ) ;
} ) ;
it ( 'returns clipboard contents when a PermissionRequestHandler is not defined' , async ( ) = > {
const clipboard = await writeClipboard ( ) ;
2024-01-25 17:46:30 +00:00
expect ( clipboard ) . to . be . undefined ( ) ;
2023-02-09 10:38:39 +00:00
} ) ;
it ( 'returns an error when permission denied' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler ( ( wc , permission , callback ) = > {
if ( permission === 'clipboard-sanitized-write' ) {
callback ( false ) ;
} else {
callback ( true ) ;
}
} ) ;
const clipboard = await writeClipboard ( ) ;
2024-01-25 17:46:30 +00:00
expect ( clipboard ) . to . contain ( 'Write permission denied.' ) ;
2023-02-09 10:38:39 +00:00
} ) ;
it ( 'returns clipboard contents when permission is granted' , async ( ) = > {
session . defaultSession . setPermissionRequestHandler ( ( wc , permission , callback ) = > {
if ( permission === 'clipboard-sanitized-write' ) {
callback ( true ) ;
} else {
callback ( false ) ;
}
} ) ;
const clipboard = await writeClipboard ( ) ;
2024-01-25 17:46:30 +00:00
expect ( clipboard ) . to . be . undefined ( ) ;
2023-02-09 10:38:39 +00:00
} ) ;
} ) ;
2021-01-21 05:45:06 +00:00
ifdescribe ( ( process . platform !== 'linux' || app . isUnityRunning ( ) ) ) ( 'navigator.setAppBadge/clearAppBadge' , ( ) = > {
let w : BrowserWindow ;
const expectedBadgeCount = 42 ;
const fireAppBadgeAction : any = ( action : string , value : any ) = > {
return w . webContents . executeJavaScript ( `
navigator . $ { action } AppBadge ( $ { value } ) . then ( ( ) = > 'success' ) . catch ( err = > err . message ) ` );
} ;
// For some reason on macOS changing the badge count doesn't happen right away, so wait
// until it changes.
async function waitForBadgeCount ( value : number ) {
let badgeCount = app . getBadgeCount ( ) ;
while ( badgeCount !== value ) {
2023-02-23 23:53:53 +00:00
await setTimeout ( 10 ) ;
2021-01-21 05:45:06 +00:00
badgeCount = app . getBadgeCount ( ) ;
}
return badgeCount ;
}
2021-03-05 01:12:03 +00:00
describe ( 'in the renderer' , ( ) = > {
before ( async ( ) = > {
w = new BrowserWindow ( {
show : false
} ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
} ) ;
2021-01-21 05:45:06 +00:00
2021-03-05 01:12:03 +00:00
after ( ( ) = > {
app . badgeCount = 0 ;
closeAllWindows ( ) ;
} ) ;
2021-01-21 05:45:06 +00:00
2021-03-05 01:12:03 +00:00
it ( 'setAppBadge can set a numerical value' , async ( ) = > {
const result = await fireAppBadgeAction ( 'set' , expectedBadgeCount ) ;
expect ( result ) . to . equal ( 'success' ) ;
expect ( waitForBadgeCount ( expectedBadgeCount ) ) . to . eventually . equal ( expectedBadgeCount ) ;
} ) ;
it ( 'setAppBadge can set an empty(dot) value' , async ( ) = > {
const result = await fireAppBadgeAction ( 'set' ) ;
expect ( result ) . to . equal ( 'success' ) ;
expect ( waitForBadgeCount ( 0 ) ) . to . eventually . equal ( 0 ) ;
} ) ;
it ( 'clearAppBadge can clear a value' , async ( ) = > {
let result = await fireAppBadgeAction ( 'set' , expectedBadgeCount ) ;
expect ( result ) . to . equal ( 'success' ) ;
expect ( waitForBadgeCount ( expectedBadgeCount ) ) . to . eventually . equal ( expectedBadgeCount ) ;
result = await fireAppBadgeAction ( 'clear' ) ;
expect ( result ) . to . equal ( 'success' ) ;
expect ( waitForBadgeCount ( 0 ) ) . to . eventually . equal ( 0 ) ;
} ) ;
2021-01-21 05:45:06 +00:00
} ) ;
2021-03-05 01:12:03 +00:00
describe ( 'in a service worker' , ( ) = > {
beforeEach ( async ( ) = > {
w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
partition : 'sw-file-scheme-spec' ,
contextIsolation : false
}
} ) ;
} ) ;
afterEach ( ( ) = > {
app . badgeCount = 0 ;
closeAllWindows ( ) ;
} ) ;
it ( 'setAppBadge can be called in a ServiceWorker' , ( done ) = > {
w . webContents . on ( 'ipc-message' , ( event , channel , message ) = > {
if ( channel === 'reload' ) {
w . webContents . reload ( ) ;
} else if ( channel === 'error' ) {
done ( message ) ;
} else if ( channel === 'response' ) {
expect ( message ) . to . equal ( 'SUCCESS setting app badge' ) ;
expect ( waitForBadgeCount ( expectedBadgeCount ) ) . to . eventually . equal ( expectedBadgeCount ) ;
session . fromPartition ( 'sw-file-scheme-spec' ) . clearStorageData ( {
storages : [ 'serviceworkers' ]
} ) . then ( ( ) = > done ( ) ) ;
}
} ) ;
2023-02-16 09:25:41 +00:00
w . webContents . on ( 'render-process-gone' , ( ) = > done ( new Error ( 'WebContents crashed.' ) ) ) ;
2021-03-05 01:12:03 +00:00
w . loadFile ( path . join ( fixturesPath , 'pages' , 'service-worker' , 'badge-index.html' ) , { search : '?setBadge' } ) ;
} ) ;
it ( 'clearAppBadge can be called in a ServiceWorker' , ( done ) = > {
w . webContents . on ( 'ipc-message' , ( event , channel , message ) = > {
if ( channel === 'reload' ) {
w . webContents . reload ( ) ;
} else if ( channel === 'setAppBadge' ) {
expect ( message ) . to . equal ( 'SUCCESS setting app badge' ) ;
expect ( waitForBadgeCount ( expectedBadgeCount ) ) . to . eventually . equal ( expectedBadgeCount ) ;
} else if ( channel === 'error' ) {
done ( message ) ;
} else if ( channel === 'response' ) {
expect ( message ) . to . equal ( 'SUCCESS clearing app badge' ) ;
expect ( waitForBadgeCount ( expectedBadgeCount ) ) . to . eventually . equal ( expectedBadgeCount ) ;
session . fromPartition ( 'sw-file-scheme-spec' ) . clearStorageData ( {
storages : [ 'serviceworkers' ]
} ) . then ( ( ) = > done ( ) ) ;
}
} ) ;
2023-02-16 09:25:41 +00:00
w . webContents . on ( 'render-process-gone' , ( ) = > done ( new Error ( 'WebContents crashed.' ) ) ) ;
2021-03-05 01:12:03 +00:00
w . loadFile ( path . join ( fixturesPath , 'pages' , 'service-worker' , 'badge-index.html' ) , { search : '?clearBadge' } ) ;
} ) ;
2021-01-21 05:45:06 +00:00
} ) ;
} ) ;
2021-02-26 19:10:27 +00:00
describe ( 'navigator.bluetooth' , ( ) = > {
let w : BrowserWindow ;
before ( async ( ) = > {
w = new BrowserWindow ( {
show : false ,
webPreferences : {
enableBlinkFeatures : 'WebBluetooth'
}
} ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
} ) ;
after ( closeAllWindows ) ;
it ( 'can request bluetooth devices' , async ( ) = > {
const bluetooth = await w . webContents . executeJavaScript ( `
navigator . bluetooth . requestDevice ( { acceptAllDevices : true } ) . then ( device = > "Found a device!" ) . catch ( err = > err . message ) ; ` , true);
expect ( bluetooth ) . to . be . oneOf ( [ 'Found a device!' , 'Bluetooth adapter not available.' , 'User cancelled the requestDevice() chooser.' ] ) ;
} ) ;
} ) ;
2021-09-23 11:00:11 +00:00
describe ( 'navigator.hid' , ( ) = > {
let w : BrowserWindow ;
let server : http.Server ;
let serverUrl : string ;
before ( async ( ) = > {
w = new BrowserWindow ( {
show : false
} ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
server = http . createServer ( ( req , res ) = > {
res . setHeader ( 'Content-Type' , 'text/html' ) ;
res . end ( '<body>' ) ;
} ) ;
2023-02-20 11:30:57 +00:00
serverUrl = ( await listen ( server ) ) . url ;
2021-09-23 11:00:11 +00:00
} ) ;
2022-05-23 19:13:18 +00:00
const requestDevices : any = ( ) = > {
2021-09-23 11:00:11 +00:00
return w . webContents . executeJavaScript ( `
navigator . hid . requestDevice ( { filters : [ ] } ) . then ( device = > device . toString ( ) ) . catch ( err = > err . toString ( ) ) ;
` , true);
} ;
after ( ( ) = > {
server . close ( ) ;
closeAllWindows ( ) ;
} ) ;
afterEach ( ( ) = > {
session . defaultSession . setPermissionCheckHandler ( null ) ;
session . defaultSession . setDevicePermissionHandler ( null ) ;
session . defaultSession . removeAllListeners ( 'select-hid-device' ) ;
} ) ;
it ( 'does not return a device if select-hid-device event is not defined' , async ( ) = > {
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
2022-05-23 19:13:18 +00:00
const device = await requestDevices ( ) ;
2021-09-23 11:00:11 +00:00
expect ( device ) . to . equal ( '' ) ;
} ) ;
it ( 'does not return a device when permission denied' , async ( ) = > {
let selectFired = false ;
w . webContents . session . on ( 'select-hid-device' , ( event , details , callback ) = > {
selectFired = true ;
callback ( ) ;
} ) ;
session . defaultSession . setPermissionCheckHandler ( ( ) = > false ) ;
2022-05-23 19:13:18 +00:00
const device = await requestDevices ( ) ;
2021-09-23 11:00:11 +00:00
expect ( selectFired ) . to . be . false ( ) ;
expect ( device ) . to . equal ( '' ) ;
} ) ;
it ( 'returns a device when select-hid-device event is defined' , async ( ) = > {
let haveDevices = false ;
let selectFired = false ;
w . webContents . session . on ( 'select-hid-device' , ( event , details , callback ) = > {
2023-02-06 20:59:49 +00:00
expect ( details . frame ) . to . have . property ( 'frameTreeNodeId' ) . that . is . a ( 'number' ) ;
2021-09-23 11:00:11 +00:00
selectFired = true ;
if ( details . deviceList . length > 0 ) {
haveDevices = true ;
callback ( details . deviceList [ 0 ] . deviceId ) ;
} else {
callback ( ) ;
}
} ) ;
2022-05-23 19:13:18 +00:00
const device = await requestDevices ( ) ;
2021-09-23 11:00:11 +00:00
expect ( selectFired ) . to . be . true ( ) ;
if ( haveDevices ) {
expect ( device ) . to . contain ( '[object HIDDevice]' ) ;
} else {
expect ( device ) . to . equal ( '' ) ;
}
2022-10-27 16:37:04 +00:00
if ( haveDevices ) {
2021-09-23 11:00:11 +00:00
// Verify that navigation will clear device permissions
const grantedDevices = await w . webContents . executeJavaScript ( 'navigator.hid.getDevices()' ) ;
expect ( grantedDevices ) . to . not . be . empty ( ) ;
w . loadURL ( serverUrl ) ;
2023-02-23 23:53:53 +00:00
const [ , , , , , frameProcessId , frameRoutingId ] = await once ( w . webContents , 'did-frame-navigate' ) ;
2021-09-23 11:00:11 +00:00
const frame = webFrameMain . fromId ( frameProcessId , frameRoutingId ) ;
2023-02-06 20:59:49 +00:00
expect ( ! ! frame ) . to . be . true ( ) ;
2021-09-23 11:00:11 +00:00
if ( frame ) {
const grantedDevicesOnNewPage = await frame . executeJavaScript ( 'navigator.hid.getDevices()' ) ;
expect ( grantedDevicesOnNewPage ) . to . be . empty ( ) ;
}
}
} ) ;
it ( 'returns a device when DevicePermissionHandler is defined' , async ( ) = > {
let haveDevices = false ;
let selectFired = false ;
let gotDevicePerms = false ;
w . webContents . session . on ( 'select-hid-device' , ( event , details , callback ) = > {
selectFired = true ;
if ( details . deviceList . length > 0 ) {
const foundDevice = details . deviceList . find ( ( device ) = > {
if ( device . name && device . name !== '' && device . serialNumber && device . serialNumber !== '' ) {
haveDevices = true ;
return true ;
}
2024-10-01 20:04:53 +00:00
return false ;
2021-09-23 11:00:11 +00:00
} ) ;
if ( foundDevice ) {
callback ( foundDevice . deviceId ) ;
return ;
}
}
callback ( ) ;
} ) ;
session . defaultSession . setDevicePermissionHandler ( ( ) = > {
gotDevicePerms = true ;
return true ;
} ) ;
await w . webContents . executeJavaScript ( 'navigator.hid.getDevices();' , true ) ;
2022-05-23 19:13:18 +00:00
const device = await requestDevices ( ) ;
2021-09-23 11:00:11 +00:00
expect ( selectFired ) . to . be . true ( ) ;
if ( haveDevices ) {
expect ( device ) . to . contain ( '[object HIDDevice]' ) ;
expect ( gotDevicePerms ) . to . be . true ( ) ;
} else {
expect ( device ) . to . equal ( '' ) ;
}
} ) ;
2022-04-12 11:19:14 +00:00
2022-05-23 19:13:18 +00:00
it ( 'excludes a device when a exclusionFilter is specified' , async ( ) = > {
2022-04-12 11:19:14 +00:00
const exclusionFilters = < any > [ ] ;
let haveDevices = false ;
let checkForExcludedDevice = false ;
w . webContents . session . on ( 'select-hid-device' , ( event , details , callback ) = > {
if ( details . deviceList . length > 0 ) {
details . deviceList . find ( ( device ) = > {
if ( device . name && device . name !== '' && device . serialNumber && device . serialNumber !== '' ) {
if ( checkForExcludedDevice ) {
const compareDevice = {
vendorId : device.vendorId ,
productId : device.productId
} ;
expect ( compareDevice ) . to . not . equal ( exclusionFilters [ 0 ] , 'excluded device should not be returned' ) ;
} else {
haveDevices = true ;
exclusionFilters . push ( {
vendorId : device.vendorId ,
productId : device.productId
} ) ;
return true ;
}
}
2024-10-01 20:04:53 +00:00
return false ;
2022-04-12 11:19:14 +00:00
} ) ;
}
callback ( ) ;
} ) ;
2022-05-23 19:13:18 +00:00
await requestDevices ( ) ;
2022-04-12 11:19:14 +00:00
if ( haveDevices ) {
2022-06-16 07:46:11 +00:00
// We have devices to exclude, so check if exclusionFilters work
2022-04-12 11:19:14 +00:00
checkForExcludedDevice = true ;
await w . webContents . executeJavaScript ( `
navigator . hid . requestDevice ( { filters : [ ] , exclusionFilters : $ { JSON . stringify ( exclusionFilters ) } } ) . then ( device = > device . toString ( ) ) . catch ( err = > err . toString ( ) ) ;
2022-05-23 19:13:18 +00:00
2022-04-12 11:19:14 +00:00
` , true);
}
} ) ;
2022-05-23 19:13:18 +00:00
it ( 'supports device.forget()' , async ( ) = > {
let deletedDeviceFromEvent ;
let haveDevices = false ;
w . webContents . session . on ( 'select-hid-device' , ( event , details , callback ) = > {
if ( details . deviceList . length > 0 ) {
haveDevices = true ;
callback ( details . deviceList [ 0 ] . deviceId ) ;
} else {
callback ( ) ;
}
} ) ;
w . webContents . session . on ( 'hid-device-revoked' , ( event , details ) = > {
deletedDeviceFromEvent = details . device ;
} ) ;
await requestDevices ( ) ;
if ( haveDevices ) {
const grantedDevices = await w . webContents . executeJavaScript ( 'navigator.hid.getDevices()' ) ;
if ( grantedDevices . length > 0 ) {
const deletedDevice = await w . webContents . executeJavaScript ( `
navigator . hid . getDevices ( ) . then ( devices = > {
devices [ 0 ] . forget ( ) ;
return {
vendorId : devices [ 0 ] . vendorId ,
productId : devices [ 0 ] . productId ,
name : devices [ 0 ] . productName
}
} )
` );
const grantedDevices2 = await w . webContents . executeJavaScript ( 'navigator.hid.getDevices()' ) ;
expect ( grantedDevices2 . length ) . to . be . lessThan ( grantedDevices . length ) ;
if ( deletedDevice . name !== '' && deletedDevice . productId && deletedDevice . vendorId ) {
expect ( deletedDeviceFromEvent ) . to . include ( deletedDevice ) ;
}
}
}
} ) ;
2021-09-23 11:00:11 +00:00
} ) ;
2022-11-22 21:50:32 +00:00
describe ( 'navigator.usb' , ( ) = > {
let w : BrowserWindow ;
let server : http.Server ;
let serverUrl : string ;
before ( async ( ) = > {
w = new BrowserWindow ( {
show : false
} ) ;
await w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
server = http . createServer ( ( req , res ) = > {
res . setHeader ( 'Content-Type' , 'text/html' ) ;
res . end ( '<body>' ) ;
} ) ;
2023-02-20 11:30:57 +00:00
serverUrl = ( await listen ( server ) ) . url ;
2022-11-22 21:50:32 +00:00
} ) ;
const requestDevices : any = ( ) = > {
return w . webContents . executeJavaScript ( `
navigator . usb . requestDevice ( { filters : [ ] } ) . then ( device = > device . toString ( ) ) . catch ( err = > err . toString ( ) ) ;
` , true);
} ;
2024-06-04 13:24:08 +00:00
const getDevices : any = ( ) = > {
return w . webContents . executeJavaScript ( `
navigator . usb . getDevices ( ) . then ( devices = > devices . map ( device = > device . toString ( ) ) ) . catch ( err = > err . toString ( ) ) ;
` , true);
} ;
2022-11-22 21:50:32 +00:00
const notFoundError = 'NotFoundError: Failed to execute \'requestDevice\' on \'USB\': No device selected.' ;
after ( ( ) = > {
server . close ( ) ;
closeAllWindows ( ) ;
} ) ;
2024-06-04 13:24:08 +00:00
2022-11-22 21:50:32 +00:00
afterEach ( ( ) = > {
session . defaultSession . setPermissionCheckHandler ( null ) ;
session . defaultSession . setDevicePermissionHandler ( null ) ;
session . defaultSession . removeAllListeners ( 'select-usb-device' ) ;
} ) ;
2024-06-04 13:24:08 +00:00
it ( 'does not crash when using in-memory partitions' , async ( ) = > {
const sesWin = new BrowserWindow ( {
webPreferences : {
partition : 'test-partition'
}
} ) ;
await sesWin . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
server = http . createServer ( ( req , res ) = > {
res . setHeader ( 'Content-Type' , 'text/html' ) ;
res . end ( '<body>' ) ;
} ) ;
serverUrl = ( await listen ( server ) ) . url ;
const devices = await getDevices ( ) ;
expect ( devices ) . to . be . an ( 'array' ) . that . is . empty ( ) ;
} ) ;
2022-11-22 21:50:32 +00:00
it ( 'does not return a device if select-usb-device event is not defined' , async ( ) = > {
w . loadFile ( path . join ( fixturesPath , 'pages' , 'blank.html' ) ) ;
const device = await requestDevices ( ) ;
expect ( device ) . to . equal ( notFoundError ) ;
} ) ;
it ( 'does not return a device when permission denied' , async ( ) = > {
let selectFired = false ;
w . webContents . session . on ( 'select-usb-device' , ( event , details , callback ) = > {
selectFired = true ;
callback ( ) ;
} ) ;
session . defaultSession . setPermissionCheckHandler ( ( ) = > false ) ;
const device = await requestDevices ( ) ;
expect ( selectFired ) . to . be . false ( ) ;
expect ( device ) . to . equal ( notFoundError ) ;
} ) ;
it ( 'returns a device when select-usb-device event is defined' , async ( ) = > {
let haveDevices = false ;
let selectFired = false ;
w . webContents . session . on ( 'select-usb-device' , ( event , details , callback ) = > {
2023-02-06 20:59:49 +00:00
expect ( details . frame ) . to . have . property ( 'frameTreeNodeId' ) . that . is . a ( 'number' ) ;
2022-11-22 21:50:32 +00:00
selectFired = true ;
if ( details . deviceList . length > 0 ) {
haveDevices = true ;
callback ( details . deviceList [ 0 ] . deviceId ) ;
} else {
callback ( ) ;
}
} ) ;
const device = await requestDevices ( ) ;
expect ( selectFired ) . to . be . true ( ) ;
if ( haveDevices ) {
expect ( device ) . to . contain ( '[object USBDevice]' ) ;
} else {
expect ( device ) . to . equal ( notFoundError ) ;
}
if ( haveDevices ) {
// Verify that navigation will clear device permissions
const grantedDevices = await w . webContents . executeJavaScript ( 'navigator.usb.getDevices()' ) ;
expect ( grantedDevices ) . to . not . be . empty ( ) ;
w . loadURL ( serverUrl ) ;
2023-02-23 23:53:53 +00:00
const [ , , , , , frameProcessId , frameRoutingId ] = await once ( w . webContents , 'did-frame-navigate' ) ;
2022-11-22 21:50:32 +00:00
const frame = webFrameMain . fromId ( frameProcessId , frameRoutingId ) ;
2023-02-06 20:59:49 +00:00
expect ( ! ! frame ) . to . be . true ( ) ;
2022-11-22 21:50:32 +00:00
if ( frame ) {
const grantedDevicesOnNewPage = await frame . executeJavaScript ( 'navigator.usb.getDevices()' ) ;
expect ( grantedDevicesOnNewPage ) . to . be . empty ( ) ;
}
}
} ) ;
it ( 'returns a device when DevicePermissionHandler is defined' , async ( ) = > {
let haveDevices = false ;
let selectFired = false ;
let gotDevicePerms = false ;
w . webContents . session . on ( 'select-usb-device' , ( event , details , callback ) = > {
selectFired = true ;
if ( details . deviceList . length > 0 ) {
const foundDevice = details . deviceList . find ( ( device ) = > {
if ( device . productName && device . productName !== '' && device . serialNumber && device . serialNumber !== '' ) {
haveDevices = true ;
return true ;
}
2024-10-01 20:04:53 +00:00
return false ;
2022-11-22 21:50:32 +00:00
} ) ;
if ( foundDevice ) {
callback ( foundDevice . deviceId ) ;
return ;
}
}
callback ( ) ;
} ) ;
session . defaultSession . setDevicePermissionHandler ( ( ) = > {
gotDevicePerms = true ;
return true ;
} ) ;
await w . webContents . executeJavaScript ( 'navigator.usb.getDevices();' , true ) ;
const device = await requestDevices ( ) ;
expect ( selectFired ) . to . be . true ( ) ;
if ( haveDevices ) {
expect ( device ) . to . contain ( '[object USBDevice]' ) ;
expect ( gotDevicePerms ) . to . be . true ( ) ;
} else {
expect ( device ) . to . equal ( notFoundError ) ;
}
} ) ;
it ( 'supports device.forget()' , async ( ) = > {
let deletedDeviceFromEvent ;
let haveDevices = false ;
w . webContents . session . on ( 'select-usb-device' , ( event , details , callback ) = > {
if ( details . deviceList . length > 0 ) {
haveDevices = true ;
callback ( details . deviceList [ 0 ] . deviceId ) ;
} else {
callback ( ) ;
}
} ) ;
w . webContents . session . on ( 'usb-device-revoked' , ( event , details ) = > {
deletedDeviceFromEvent = details . device ;
} ) ;
await requestDevices ( ) ;
if ( haveDevices ) {
const grantedDevices = await w . webContents . executeJavaScript ( 'navigator.usb.getDevices()' ) ;
if ( grantedDevices . length > 0 ) {
2023-01-18 21:30:01 +00:00
const deletedDevice : Electron.USBDevice = await w . webContents . executeJavaScript ( `
2022-11-22 21:50:32 +00:00
navigator . usb . getDevices ( ) . then ( devices = > {
devices [ 0 ] . forget ( ) ;
return {
vendorId : devices [ 0 ] . vendorId ,
productId : devices [ 0 ] . productId ,
productName : devices [ 0 ] . productName
}
} )
` );
const grantedDevices2 = await w . webContents . executeJavaScript ( 'navigator.usb.getDevices()' ) ;
expect ( grantedDevices2 . length ) . to . be . lessThan ( grantedDevices . length ) ;
2023-01-18 21:30:01 +00:00
if ( deletedDevice . productName !== '' && deletedDevice . productId && deletedDevice . vendorId ) {
2022-11-22 21:50:32 +00:00
expect ( deletedDeviceFromEvent ) . to . include ( deletedDevice ) ;
}
}
}
} ) ;
} ) ;