2020-03-20 20:28:31 +00:00
import { expect } from 'chai' ;
2020-04-07 00:04:09 +00:00
import { app , session , BrowserWindow , ipcMain , WebContents , Extension } from 'electron/main' ;
2020-03-20 20:28:31 +00:00
import { closeAllWindows , closeWindow } from './window-helpers' ;
import * as http from 'http' ;
import { AddressInfo } from 'net' ;
import * as path from 'path' ;
import * as fs from 'fs' ;
import { ifdescribe } from './spec-helpers' ;
import { emittedOnce , emittedNTimes } from './events-helpers' ;
const fixtures = path . join ( __dirname , 'fixtures' ) ;
2019-07-24 23:01:08 +00:00
ifdescribe ( process . electronBinding ( 'features' ) . isExtensionsEnabled ( ) ) ( 'chrome extensions' , ( ) = > {
// NB. extensions are only allowed on http://, https:// and ftp:// (!) urls by default.
2020-03-20 20:28:31 +00:00
let server : http.Server ;
let url : string ;
2019-07-24 23:01:08 +00:00
before ( async ( ) = > {
2020-03-20 20:28:31 +00:00
server = http . createServer ( ( req , res ) = > res . end ( ) ) ;
2019-07-24 23:01:08 +00:00
await new Promise ( resolve = > server . listen ( 0 , '127.0.0.1' , ( ) = > {
2020-03-20 20:28:31 +00:00
url = ` http://127.0.0.1: ${ ( server . address ( ) as AddressInfo ) . port } ` ;
resolve ( ) ;
} ) ) ;
} ) ;
2019-07-24 23:01:08 +00:00
after ( ( ) = > {
2020-03-20 20:28:31 +00:00
server . close ( ) ;
} ) ;
afterEach ( closeAllWindows ) ;
2020-02-10 16:28:03 +00:00
afterEach ( ( ) = > {
session . defaultSession . getAllExtensions ( ) . forEach ( ( e : any ) = > {
2020-03-20 20:28:31 +00:00
session . defaultSession . removeExtension ( e . id ) ;
} ) ;
} ) ;
2020-02-10 16:28:03 +00:00
2019-07-24 23:01:08 +00:00
it ( 'loads an extension' , async ( ) = > {
// NB. we have to use a persist: session (i.e. non-OTR) because the
// extension registry is redirected to the main session. so installing an
// extension in an in-memory session results in it being installed in the
// default session.
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession } } ) ;
await w . loadURL ( url ) ;
const bg = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( bg ) . to . equal ( 'red' ) ;
} ) ;
2019-07-24 23:01:08 +00:00
2020-02-24 03:30:32 +00:00
it ( 'serializes a loaded extension' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const extensionPath = path . join ( fixtures , 'extensions' , 'red-bg' ) ;
const manifest = JSON . parse ( fs . readFileSync ( path . join ( extensionPath , 'manifest.json' ) , 'utf-8' ) ) ;
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
const extension = await customSession . loadExtension ( extensionPath ) ;
expect ( extension . id ) . to . be . a ( 'string' ) ;
expect ( extension . name ) . to . be . a ( 'string' ) ;
expect ( extension . path ) . to . be . a ( 'string' ) ;
expect ( extension . version ) . to . be . a ( 'string' ) ;
expect ( extension . url ) . to . be . a ( 'string' ) ;
expect ( extension . manifest ) . to . deep . equal ( manifest ) ;
} ) ;
2020-02-24 03:30:32 +00:00
2020-01-13 22:56:28 +00:00
it ( 'removes an extension' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
const { id } = await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) ;
2020-01-13 22:56:28 +00:00
{
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession } } ) ;
await w . loadURL ( url ) ;
const bg = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( bg ) . to . equal ( 'red' ) ;
2020-01-13 22:56:28 +00:00
}
2020-03-20 20:28:31 +00:00
customSession . removeExtension ( id ) ;
2020-01-13 22:56:28 +00:00
{
2020-03-20 20:28:31 +00:00
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession } } ) ;
await w . loadURL ( url ) ;
const bg = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( bg ) . to . equal ( '' ) ;
2020-01-13 22:56:28 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2020-01-13 22:56:28 +00:00
it ( 'lists loaded extensions in getAllExtensions' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
const e = await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) ;
expect ( customSession . getAllExtensions ( ) ) . to . deep . equal ( [ e ] ) ;
customSession . removeExtension ( e . id ) ;
expect ( customSession . getAllExtensions ( ) ) . to . deep . equal ( [ ] ) ;
} ) ;
2020-01-13 22:56:28 +00:00
it ( 'gets an extension by id' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
const e = await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) ;
expect ( customSession . getExtension ( e . id ) ) . to . deep . equal ( e ) ;
} ) ;
2020-01-13 22:56:28 +00:00
2019-07-24 23:01:08 +00:00
it ( 'confines an extension to the session it was loaded in' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) ;
const w = new BrowserWindow ( { show : false } ) ; // not in the session
await w . loadURL ( url ) ;
const bg = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( bg ) . to . equal ( '' ) ;
} ) ;
2019-07-24 23:01:08 +00:00
2020-02-10 16:28:03 +00:00
it ( 'loading an extension in a temporary session throws an error' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( require ( 'uuid' ) . v4 ( ) ) ;
await expect ( customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) ) . to . eventually . be . rejectedWith ( 'Extensions cannot be loaded in a temporary session' ) ;
} ) ;
2020-02-10 16:28:03 +00:00
2020-03-05 14:56:21 +00:00
describe ( 'chrome.i18n' , ( ) = > {
2020-03-20 20:28:31 +00:00
let w : BrowserWindow ;
let extension : Extension ;
2020-03-05 14:56:21 +00:00
const exec = async ( name : string ) = > {
2020-03-20 20:28:31 +00:00
const p = emittedOnce ( ipcMain , 'success' ) ;
await w . webContents . executeJavaScript ( ` exec(' ${ name } ') ` ) ;
const [ , result ] = await p ;
return result ;
} ;
2020-03-05 14:56:21 +00:00
beforeEach ( async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
extension = await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'chrome-i18n' ) ) ;
w = new BrowserWindow ( { show : false , webPreferences : { session : customSession , nodeIntegration : true } } ) ;
await w . loadURL ( url ) ;
} ) ;
2020-03-05 14:56:21 +00:00
it ( 'getAcceptLanguages()' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const result = await exec ( 'getAcceptLanguages' ) ;
expect ( result ) . to . be . an ( 'array' ) . and . deep . equal ( [ 'en-US' ] ) ;
} ) ;
2020-03-05 14:56:21 +00:00
it ( 'getMessage()' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const result = await exec ( 'getMessage' ) ;
expect ( result . id ) . to . be . a ( 'string' ) . and . equal ( extension . id ) ;
expect ( result . name ) . to . be . a ( 'string' ) . and . equal ( 'chrome-i18n' ) ;
} ) ;
} ) ;
2020-03-05 14:56:21 +00:00
2019-07-24 23:01:08 +00:00
describe ( 'chrome.runtime' , ( ) = > {
2020-03-20 20:28:31 +00:00
let content : any ;
2019-07-24 23:01:08 +00:00
before ( async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'chrome-runtime' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession } } ) ;
2019-07-24 23:01:08 +00:00
try {
2020-03-20 20:28:31 +00:00
await w . loadURL ( url ) ;
content = JSON . parse ( await w . webContents . executeJavaScript ( 'document.documentElement.textContent' ) ) ;
expect ( content ) . to . be . an ( 'object' ) ;
2019-07-24 23:01:08 +00:00
} finally {
2020-03-20 20:28:31 +00:00
w . destroy ( ) ;
2019-07-24 23:01:08 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2019-07-24 23:01:08 +00:00
it ( 'getManifest()' , ( ) = > {
2020-03-20 20:28:31 +00:00
expect ( content . manifest ) . to . be . an ( 'object' ) . with . property ( 'name' , 'chrome-runtime' ) ;
} ) ;
2019-07-24 23:01:08 +00:00
it ( 'id' , ( ) = > {
2020-03-20 20:28:31 +00:00
expect ( content . id ) . to . be . a ( 'string' ) . with . lengthOf ( 32 ) ;
} ) ;
2019-07-24 23:01:08 +00:00
it ( 'getURL()' , ( ) = > {
2020-03-20 20:28:31 +00:00
expect ( content . url ) . to . be . a ( 'string' ) . and . match ( /^chrome-extension:\/\/.*main.js$/ ) ;
} ) ;
} ) ;
2019-07-24 23:01:08 +00:00
describe ( 'chrome.storage' , ( ) = > {
it ( 'stores and retrieves a key' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'chrome-storage' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession , nodeIntegration : true } } ) ;
2019-07-24 23:01:08 +00:00
try {
2020-03-20 20:28:31 +00:00
const p = emittedOnce ( ipcMain , 'storage-success' ) ;
await w . loadURL ( url ) ;
const [ , v ] = await p ;
expect ( v ) . to . equal ( 'value' ) ;
2019-07-24 23:01:08 +00:00
} finally {
2020-03-20 20:28:31 +00:00
w . destroy ( ) ;
2019-07-24 23:01:08 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2020-01-13 22:55:58 +00:00
2020-01-15 23:11:51 +00:00
describe ( 'chrome.tabs' , ( ) = > {
it ( 'executeScript' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'chrome-api' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession , nodeIntegration : true } } ) ;
await w . loadURL ( url ) ;
2020-01-15 23:11:51 +00:00
2020-03-20 20:28:31 +00:00
const message = { method : 'executeScript' , args : [ '1 + 2' ] } ;
w . webContents . executeJavaScript ( ` window.postMessage(' ${ JSON . stringify ( message ) } ', '*') ` ) ;
2020-01-15 23:11:51 +00:00
2020-03-20 20:28:31 +00:00
const [ , , responseString ] = await emittedOnce ( w . webContents , 'console-message' ) ;
const response = JSON . parse ( responseString ) ;
2020-01-15 23:11:51 +00:00
2020-03-20 20:28:31 +00:00
expect ( response ) . to . equal ( 3 ) ;
} ) ;
2020-01-15 23:11:51 +00:00
2020-03-05 14:59:32 +00:00
it ( 'connect' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'chrome-api' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession , nodeIntegration : true } } ) ;
await w . loadURL ( url ) ;
2020-03-05 14:59:32 +00:00
2020-03-20 20:28:31 +00:00
const portName = require ( 'uuid' ) . v4 ( ) ;
const message = { method : 'connectTab' , args : [ portName ] } ;
w . webContents . executeJavaScript ( ` window.postMessage(' ${ JSON . stringify ( message ) } ', '*') ` ) ;
2020-03-05 14:59:32 +00:00
2020-03-20 20:28:31 +00:00
const [ , , responseString ] = await emittedOnce ( w . webContents , 'console-message' ) ;
const response = responseString . split ( ',' ) ;
expect ( response [ 0 ] ) . to . equal ( portName ) ;
expect ( response [ 1 ] ) . to . equal ( 'howdy' ) ;
} ) ;
2020-03-05 14:59:32 +00:00
2020-01-15 23:11:51 +00:00
it ( 'sendMessage receives the response' , async function ( ) {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'chrome-api' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession , nodeIntegration : true } } ) ;
await w . loadURL ( url ) ;
2020-01-15 23:11:51 +00:00
2020-03-20 20:28:31 +00:00
const message = { method : 'sendMessage' , args : [ 'Hello World!' ] } ;
w . webContents . executeJavaScript ( ` window.postMessage(' ${ JSON . stringify ( message ) } ', '*') ` ) ;
2020-01-15 23:11:51 +00:00
2020-03-20 20:28:31 +00:00
const [ , , responseString ] = await emittedOnce ( w . webContents , 'console-message' ) ;
const response = JSON . parse ( responseString ) ;
2020-01-15 23:11:51 +00:00
2020-03-20 20:28:31 +00:00
expect ( response . message ) . to . equal ( 'Hello World!' ) ;
expect ( response . tabId ) . to . equal ( w . webContents . id ) ;
} ) ;
} ) ;
2020-01-15 23:11:51 +00:00
2020-01-13 22:55:58 +00:00
describe ( 'background pages' , ( ) = > {
it ( 'loads a lazy background page when sending a message' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'lazy-background-page' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession , nodeIntegration : true } } ) ;
2020-01-13 22:55:58 +00:00
try {
2020-03-20 20:28:31 +00:00
w . loadURL ( url ) ;
const [ , resp ] = await emittedOnce ( ipcMain , 'bg-page-message-response' ) ;
expect ( resp . message ) . to . deep . equal ( { some : 'message' } ) ;
expect ( resp . sender . id ) . to . be . a ( 'string' ) ;
expect ( resp . sender . origin ) . to . equal ( url ) ;
expect ( resp . sender . url ) . to . equal ( url + '/' ) ;
2020-01-13 22:55:58 +00:00
} finally {
2020-03-20 20:28:31 +00:00
w . destroy ( ) ;
2020-01-13 22:55:58 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2020-02-06 21:42:34 +00:00
it ( 'can use extension.getBackgroundPage from a ui page' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
const { id } = await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'lazy-background-page' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession } } ) ;
await w . loadURL ( ` chrome-extension:// ${ id } /page-get-background.html ` ) ;
const receivedMessage = await w . webContents . executeJavaScript ( 'window.completionPromise' ) ;
expect ( receivedMessage ) . to . deep . equal ( { some : 'message' } ) ;
} ) ;
2020-02-06 21:42:34 +00:00
it ( 'can use extension.getBackgroundPage from a ui page' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
const { id } = await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'lazy-background-page' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession } } ) ;
await w . loadURL ( ` chrome-extension:// ${ id } /page-get-background.html ` ) ;
const receivedMessage = await w . webContents . executeJavaScript ( 'window.completionPromise' ) ;
expect ( receivedMessage ) . to . deep . equal ( { some : 'message' } ) ;
} ) ;
2020-02-06 21:42:34 +00:00
it ( 'can use runtime.getBackgroundPage from a ui page' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
const { id } = await customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'lazy-background-page' ) ) ;
const w = new BrowserWindow ( { show : false , webPreferences : { session : customSession } } ) ;
await w . loadURL ( ` chrome-extension:// ${ id } /page-runtime-get-background.html ` ) ;
const receivedMessage = await w . webContents . executeJavaScript ( 'window.completionPromise' ) ;
expect ( receivedMessage ) . to . deep . equal ( { some : 'message' } ) ;
} ) ;
} ) ;
2020-01-15 00:20:30 +00:00
describe ( 'devtools extensions' , ( ) = > {
2020-03-20 20:28:31 +00:00
let showPanelTimeoutId : any = null ;
2020-01-15 00:20:30 +00:00
afterEach ( ( ) = > {
2020-03-20 20:28:31 +00:00
if ( showPanelTimeoutId ) clearTimeout ( showPanelTimeoutId ) ;
} ) ;
2020-01-15 00:20:30 +00:00
const showLastDevToolsPanel = ( w : BrowserWindow ) = > {
w . webContents . once ( 'devtools-opened' , ( ) = > {
const show = ( ) = > {
2020-03-20 20:28:31 +00:00
if ( w == null || w . isDestroyed ( ) ) return ;
const { devToolsWebContents } = w as unknown as { devToolsWebContents : WebContents | undefined } ;
2020-01-15 00:20:30 +00:00
if ( devToolsWebContents == null || devToolsWebContents . isDestroyed ( ) ) {
2020-03-20 20:28:31 +00:00
return ;
2020-01-15 00:20:30 +00:00
}
const showLastPanel = ( ) = > {
// this is executed in the devtools context, where UI is a global
2020-03-20 20:28:31 +00:00
const { UI } = ( window as any ) ;
const lastPanelId = UI . inspectorView . _tabbedPane . _tabs . peekLast ( ) . id ;
UI . inspectorView . showPanel ( lastPanelId ) ;
} ;
2020-01-15 00:20:30 +00:00
devToolsWebContents . executeJavaScript ( ` ( ${ showLastPanel } )() ` , false ) . then ( ( ) = > {
2020-03-20 20:28:31 +00:00
showPanelTimeoutId = setTimeout ( show , 100 ) ;
} ) ;
} ;
showPanelTimeoutId = setTimeout ( show , 100 ) ;
} ) ;
} ;
2020-01-15 00:20:30 +00:00
it ( 'loads a devtools extension' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const customSession = session . fromPartition ( ` persist: ${ require ( 'uuid' ) . v4 ( ) } ` ) ;
customSession . loadExtension ( path . join ( fixtures , 'extensions' , 'devtools-extension' ) ) ;
const w = new BrowserWindow ( { show : true , webPreferences : { session : customSession , nodeIntegration : true } } ) ;
2020-04-13 23:39:26 +00:00
await w . loadURL ( url ) ;
2020-03-20 20:28:31 +00:00
w . webContents . openDevTools ( ) ;
showLastDevToolsPanel ( w ) ;
await emittedOnce ( ipcMain , 'winning' ) ;
} ) ;
} ) ;
2020-01-21 17:42:55 +00:00
describe ( 'deprecation shims' , ( ) = > {
it ( 'loads an extension through BrowserWindow.addExtension' , async ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . addExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( url ) ;
const bg = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( bg ) . to . equal ( 'red' ) ;
} ) ;
2020-01-21 17:42:55 +00:00
it ( 'loads an extension through BrowserWindow.addDevToolsExtension' , async ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . addDevToolsExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( url ) ;
const bg = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( bg ) . to . equal ( 'red' ) ;
} ) ;
2020-01-21 17:42:55 +00:00
it ( 'removes an extension through BrowserWindow.removeExtension' , async ( ) = > {
2020-03-20 20:28:31 +00:00
await ( BrowserWindow . addExtension ( path . join ( fixtures , 'extensions' , 'red-bg' ) ) as any ) ;
BrowserWindow . removeExtension ( 'red-bg' ) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( url ) ;
const bg = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( bg ) . to . equal ( '' ) ;
} ) ;
} ) ;
2020-02-03 22:01:10 +00:00
describe ( 'chrome extension content scripts' , ( ) = > {
2020-03-20 20:28:31 +00:00
const fixtures = path . resolve ( __dirname , 'fixtures' ) ;
const extensionPath = path . resolve ( fixtures , 'extensions' ) ;
2020-02-03 22:01:10 +00:00
2020-03-20 20:28:31 +00:00
const addExtension = ( name : string ) = > session . defaultSession . loadExtension ( path . resolve ( extensionPath , name ) ) ;
2020-02-03 22:01:10 +00:00
const removeAllExtensions = ( ) = > {
Object . keys ( session . defaultSession . getAllExtensions ( ) ) . map ( extName = > {
2020-03-20 20:28:31 +00:00
session . defaultSession . removeExtension ( extName ) ;
} ) ;
} ;
2020-02-03 22:01:10 +00:00
2020-03-20 20:28:31 +00:00
let responseIdCounter = 0 ;
2020-02-03 22:01:10 +00:00
const executeJavaScriptInFrame = ( webContents : WebContents , frameRoutingId : number , code : string ) = > {
return new Promise ( resolve = > {
2020-03-20 20:28:31 +00:00
const responseId = responseIdCounter ++ ;
2020-02-03 22:01:10 +00:00
ipcMain . once ( ` executeJavaScriptInFrame_ ${ responseId } ` , ( event , result ) = > {
2020-03-20 20:28:31 +00:00
resolve ( result ) ;
} ) ;
webContents . send ( 'executeJavaScriptInFrame' , frameRoutingId , code , responseId ) ;
} ) ;
} ;
2020-02-03 22:01:10 +00:00
const generateTests = ( sandboxEnabled : boolean , contextIsolationEnabled : boolean ) = > {
describe ( ` with sandbox ${ sandboxEnabled ? 'enabled' : 'disabled' } and context isolation ${ contextIsolationEnabled ? 'enabled' : 'disabled' } ` , ( ) = > {
2020-03-20 20:28:31 +00:00
let w : BrowserWindow ;
2020-02-03 22:01:10 +00:00
describe ( 'supports "run_at" option' , ( ) = > {
beforeEach ( async ( ) = > {
2020-03-20 20:28:31 +00:00
await closeWindow ( w ) ;
2020-02-03 22:01:10 +00:00
w = new BrowserWindow ( {
show : false ,
width : 400 ,
height : 400 ,
webPreferences : {
contextIsolation : contextIsolationEnabled ,
sandbox : sandboxEnabled
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2020-02-03 22:01:10 +00:00
afterEach ( ( ) = > {
2020-03-20 20:28:31 +00:00
removeAllExtensions ( ) ;
return closeWindow ( w ) . then ( ( ) = > { w = null as unknown as BrowserWindow ; } ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'should run content script at document_start' , async ( ) = > {
2020-03-20 20:28:31 +00:00
await addExtension ( 'content-script-document-start' ) ;
2020-02-03 22:01:10 +00:00
w . webContents . once ( 'dom-ready' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const result = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( result ) . to . equal ( 'red' ) ;
} ) ;
w . loadURL ( url ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'should run content script at document_idle' , async ( ) = > {
2020-03-20 20:28:31 +00:00
await addExtension ( 'content-script-document-idle' ) ;
w . loadURL ( url ) ;
const result = await w . webContents . executeJavaScript ( 'document.body.style.backgroundColor' ) ;
expect ( result ) . to . equal ( 'red' ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'should run content script at document_end' , async ( ) = > {
2020-03-20 20:28:31 +00:00
await addExtension ( 'content-script-document-end' ) ;
2020-02-03 22:01:10 +00:00
w . webContents . once ( 'did-finish-load' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const result = await w . webContents . executeJavaScript ( 'document.documentElement.style.backgroundColor' ) ;
expect ( result ) . to . equal ( 'red' ) ;
} ) ;
w . loadURL ( url ) ;
} ) ;
} ) ;
2020-02-03 22:01:10 +00:00
// TODO(nornagon): real extensions don't load on file: urls, so this
// test needs to be updated to serve its content over http.
describe . skip ( 'supports "all_frames" option' , ( ) = > {
2020-03-20 20:28:31 +00:00
const contentScript = path . resolve ( fixtures , 'extensions/content-script' ) ;
2020-02-03 22:01:10 +00:00
// Computed style values
2020-03-20 20:28:31 +00:00
const COLOR_RED = 'rgb(255, 0, 0)' ;
const COLOR_BLUE = 'rgb(0, 0, 255)' ;
const COLOR_TRANSPARENT = 'rgba(0, 0, 0, 0)' ;
2020-02-03 22:01:10 +00:00
before ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . addExtension ( contentScript ) ;
} ) ;
2020-02-03 22:01:10 +00:00
after ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . removeExtension ( 'content-script-test' ) ;
} ) ;
2020-02-03 22:01:10 +00:00
beforeEach ( ( ) = > {
w = new BrowserWindow ( {
show : false ,
webPreferences : {
// enable content script injection in subframes
nodeIntegrationInSubFrames : true ,
preload : path.join ( contentScript , 'all_frames-preload.js' )
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
2020-02-03 22:01:10 +00:00
afterEach ( ( ) = >
closeWindow ( w ) . then ( ( ) = > {
2020-03-20 20:28:31 +00:00
w = null as unknown as BrowserWindow ;
2020-02-03 22:01:10 +00:00
} )
2020-03-20 20:28:31 +00:00
) ;
2020-02-03 22:01:10 +00:00
it ( 'applies matching rules in subframes' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const detailsPromise = emittedNTimes ( w . webContents , 'did-frame-finish-load' , 2 ) ;
w . loadFile ( path . join ( contentScript , 'frame-with-frame.html' ) ) ;
const frameEvents = await detailsPromise ;
2020-02-03 22:01:10 +00:00
await Promise . all (
frameEvents . map ( async frameEvent = > {
2020-03-20 20:28:31 +00:00
const [ , isMainFrame , , frameRoutingId ] = frameEvent ;
2020-02-03 22:01:10 +00:00
const result : any = await executeJavaScriptInFrame (
w . webContents ,
frameRoutingId ,
` (() => {
const a = document . getElementById ( 'all_frames_enabled' )
const b = document . getElementById ( 'all_frames_disabled' )
return {
enabledColor : getComputedStyle ( a ) . backgroundColor ,
disabledColor : getComputedStyle ( b ) . backgroundColor
}
} ) ( ) `
2020-03-20 20:28:31 +00:00
) ;
expect ( result . enabledColor ) . to . equal ( COLOR_RED ) ;
2020-02-03 22:01:10 +00:00
if ( isMainFrame ) {
2020-03-20 20:28:31 +00:00
expect ( result . disabledColor ) . to . equal ( COLOR_BLUE ) ;
2020-02-03 22:01:10 +00:00
} else {
2020-03-20 20:28:31 +00:00
expect ( result . disabledColor ) . to . equal ( COLOR_TRANSPARENT ) ; // null color
2020-02-03 22:01:10 +00:00
}
} )
2020-03-20 20:28:31 +00:00
) ;
} ) ;
} ) ;
} ) ;
} ;
generateTests ( false , false ) ;
generateTests ( false , true ) ;
generateTests ( true , false ) ;
generateTests ( true , true ) ;
} ) ;
2020-02-06 21:42:34 +00:00
describe ( 'extension ui pages' , ( ) = > {
afterEach ( ( ) = > {
session . defaultSession . getAllExtensions ( ) . forEach ( e = > {
2020-03-20 20:28:31 +00:00
session . defaultSession . removeExtension ( e . id ) ;
} ) ;
} ) ;
2020-02-06 21:42:34 +00:00
it ( 'loads a ui page of an extension' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const { id } = await session . defaultSession . loadExtension ( path . join ( fixtures , 'extensions' , 'ui-page' ) ) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` chrome-extension:// ${ id } /bare-page.html ` ) ;
const textContent = await w . webContents . executeJavaScript ( 'document.body.textContent' ) ;
expect ( textContent ) . to . equal ( 'ui page loaded ok\n' ) ;
} ) ;
2020-02-06 21:42:34 +00:00
it ( 'can load resources' , async ( ) = > {
2020-03-20 20:28:31 +00:00
const { id } = await session . defaultSession . loadExtension ( path . join ( fixtures , 'extensions' , 'ui-page' ) ) ;
const w = new BrowserWindow ( { show : false } ) ;
await w . loadURL ( ` chrome-extension:// ${ id } /page-script-load.html ` ) ;
const textContent = await w . webContents . executeJavaScript ( 'document.body.textContent' ) ;
expect ( textContent ) . to . equal ( 'script loaded ok\n' ) ;
} ) ;
} ) ;
} ) ;
2019-09-03 17:10:33 +00:00
ifdescribe ( ! process . electronBinding ( 'features' ) . isExtensionsEnabled ( ) ) ( 'chrome extensions' , ( ) = > {
2020-03-20 20:28:31 +00:00
const fixtures = path . resolve ( __dirname , 'fixtures' ) ;
let w : BrowserWindow ;
2019-09-03 17:10:33 +00:00
before ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . addExtension ( path . join ( fixtures , 'extensions/chrome-api' ) ) ;
} ) ;
2019-09-03 17:10:33 +00:00
after ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . removeExtension ( 'chrome-api' ) ;
} ) ;
2019-09-03 17:10:33 +00:00
beforeEach ( ( ) = > {
2020-03-20 20:28:31 +00:00
w = new BrowserWindow ( { show : false } ) ;
} ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
afterEach ( ( ) = > closeWindow ( w ) . then ( ( ) = > { w = null as unknown as BrowserWindow ; } ) ) ;
2019-09-03 17:10:33 +00:00
2019-10-30 05:46:52 +00:00
it ( 'chrome.runtime.connect parses arguments properly' , async function ( ) {
2020-03-20 20:28:31 +00:00
await w . loadURL ( 'about:blank' ) ;
2019-10-30 05:46:52 +00:00
2020-03-20 20:28:31 +00:00
const promise = emittedOnce ( w . webContents , 'console-message' ) ;
2019-10-30 05:46:52 +00:00
2020-03-20 20:28:31 +00:00
const message = { method : 'connect' } ;
w . webContents . executeJavaScript ( ` window.postMessage(' ${ JSON . stringify ( message ) } ', '*') ` ) ;
2019-10-30 05:46:52 +00:00
2020-03-20 20:28:31 +00:00
const [ , , responseString ] = await promise ;
const response = JSON . parse ( responseString ) ;
2019-10-30 05:46:52 +00:00
2020-03-20 20:28:31 +00:00
expect ( response ) . to . be . true ( ) ;
} ) ;
2019-10-30 05:46:52 +00:00
2019-09-03 17:10:33 +00:00
it ( 'runtime.getManifest returns extension manifest' , async ( ) = > {
const actualManifest = ( ( ) = > {
2020-03-20 20:28:31 +00:00
const data = fs . readFileSync ( path . join ( fixtures , 'extensions/chrome-api/manifest.json' ) , 'utf-8' ) ;
return JSON . parse ( data ) ;
} ) ( ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
await w . loadURL ( 'about:blank' ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const promise = emittedOnce ( w . webContents , 'console-message' ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const message = { method : 'getManifest' } ;
w . webContents . executeJavaScript ( ` window.postMessage(' ${ JSON . stringify ( message ) } ', '*') ` ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const [ , , manifestString ] = await promise ;
const manifest = JSON . parse ( manifestString ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
expect ( manifest . name ) . to . equal ( actualManifest . name ) ;
expect ( manifest . content_scripts ) . to . have . lengthOf ( actualManifest . content_scripts . length ) ;
} ) ;
2019-09-03 17:10:33 +00:00
it ( 'chrome.tabs.sendMessage receives the response' , async function ( ) {
2020-03-20 20:28:31 +00:00
await w . loadURL ( 'about:blank' ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const promise = emittedOnce ( w . webContents , 'console-message' ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const message = { method : 'sendMessage' , args : [ 'Hello World!' ] } ;
w . webContents . executeJavaScript ( ` window.postMessage(' ${ JSON . stringify ( message ) } ', '*') ` ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const [ , , responseString ] = await promise ;
const response = JSON . parse ( responseString ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
expect ( response . message ) . to . equal ( 'Hello World!' ) ;
expect ( response . tabId ) . to . equal ( w . webContents . id ) ;
} ) ;
2019-09-03 17:10:33 +00:00
it ( 'chrome.tabs.executeScript receives the response' , async function ( ) {
2020-03-20 20:28:31 +00:00
await w . loadURL ( 'about:blank' ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const promise = emittedOnce ( w . webContents , 'console-message' ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const message = { method : 'executeScript' , args : [ '1 + 2' ] } ;
w . webContents . executeJavaScript ( ` window.postMessage(' ${ JSON . stringify ( message ) } ', '*') ` ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
const [ , , responseString ] = await promise ;
const response = JSON . parse ( responseString ) ;
2019-09-03 17:10:33 +00:00
2020-03-20 20:28:31 +00:00
expect ( response ) . to . equal ( 3 ) ;
} ) ;
2020-02-03 22:01:10 +00:00
describe ( 'extensions and dev tools extensions' , ( ) = > {
2020-03-20 20:28:31 +00:00
let showPanelTimeoutId : NodeJS.Timeout | null = null ;
2020-02-03 22:01:10 +00:00
const showLastDevToolsPanel = ( w : BrowserWindow ) = > {
w . webContents . once ( 'devtools-opened' , ( ) = > {
const show = ( ) = > {
2020-03-20 20:28:31 +00:00
if ( w == null || w . isDestroyed ( ) ) return ;
const { devToolsWebContents } = w as unknown as { devToolsWebContents : WebContents | undefined } ;
2020-02-03 22:01:10 +00:00
if ( devToolsWebContents == null || devToolsWebContents . isDestroyed ( ) ) {
2020-03-20 20:28:31 +00:00
return ;
2020-02-03 22:01:10 +00:00
}
const showLastPanel = ( ) = > {
// this is executed in the devtools context, where UI is a global
2020-03-20 20:28:31 +00:00
const { UI } = ( window as any ) ;
const lastPanelId = UI . inspectorView . _tabbedPane . _tabs . peekLast ( ) . id ;
UI . inspectorView . showPanel ( lastPanelId ) ;
} ;
2020-02-03 22:01:10 +00:00
devToolsWebContents . executeJavaScript ( ` ( ${ showLastPanel } )() ` , false ) . then ( ( ) = > {
2020-03-20 20:28:31 +00:00
showPanelTimeoutId = setTimeout ( show , 100 ) ;
} ) ;
} ;
showPanelTimeoutId = setTimeout ( show , 100 ) ;
} ) ;
} ;
2020-02-03 22:01:10 +00:00
afterEach ( ( ) = > {
if ( showPanelTimeoutId != null ) {
2020-03-20 20:28:31 +00:00
clearTimeout ( showPanelTimeoutId ) ;
showPanelTimeoutId = null ;
2020-02-03 22:01:10 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2020-02-03 22:01:10 +00:00
describe ( 'BrowserWindow.addDevToolsExtension' , ( ) = > {
describe ( 'for invalid extensions' , ( ) = > {
it ( 'throws errors for missing manifest.json files' , ( ) = > {
2020-03-20 20:28:31 +00:00
const nonexistentExtensionPath = path . join ( __dirname , 'does-not-exist' ) ;
2020-02-03 22:01:10 +00:00
expect ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . addDevToolsExtension ( nonexistentExtensionPath ) ;
} ) . to . throw ( /ENOENT: no such file or directory/ ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'throws errors for invalid manifest.json files' , ( ) = > {
2020-03-20 20:28:31 +00:00
const badManifestExtensionPath = path . join ( __dirname , 'fixtures' , 'devtools-extensions' , 'bad-manifest' ) ;
2020-02-03 22:01:10 +00:00
expect ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . addDevToolsExtension ( badManifestExtensionPath ) ;
} ) . to . throw ( /Unexpected token }/ ) ;
} ) ;
} ) ;
2020-02-03 22:01:10 +00:00
describe ( 'for a valid extension' , ( ) = > {
2020-03-20 20:28:31 +00:00
const extensionName = 'foo' ;
2020-02-03 22:01:10 +00:00
before ( ( ) = > {
2020-03-20 20:28:31 +00:00
const extensionPath = path . join ( __dirname , 'fixtures' , 'devtools-extensions' , 'foo' ) ;
BrowserWindow . addDevToolsExtension ( extensionPath ) ;
expect ( BrowserWindow . getDevToolsExtensions ( ) ) . to . have . property ( extensionName ) ;
} ) ;
2020-02-03 22:01:10 +00:00
after ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . removeDevToolsExtension ( 'foo' ) ;
expect ( BrowserWindow . getDevToolsExtensions ( ) ) . to . not . have . property ( extensionName ) ;
} ) ;
2020-02-03 22:01:10 +00:00
describe ( 'when the devtools is docked' , ( ) = > {
2020-03-20 20:28:31 +00:00
let message : any ;
let w : BrowserWindow ;
2020-02-03 22:01:10 +00:00
before ( async ( ) = > {
2020-03-20 20:28:31 +00:00
w = new BrowserWindow ( { show : false , webPreferences : { nodeIntegration : true } } ) ;
2020-02-03 22:01:10 +00:00
const p = new Promise ( resolve = > ipcMain . once ( 'answer' , ( event , message ) = > {
2020-03-20 20:28:31 +00:00
resolve ( message ) ;
} ) ) ;
showLastDevToolsPanel ( w ) ;
w . loadURL ( 'about:blank' ) ;
w . webContents . openDevTools ( { mode : 'bottom' } ) ;
message = await p ;
} ) ;
after ( closeAllWindows ) ;
2020-02-03 22:01:10 +00:00
describe ( 'created extension info' , function ( ) {
it ( 'has proper "runtimeId"' , async function ( ) {
2020-03-20 20:28:31 +00:00
expect ( message ) . to . have . ownProperty ( 'runtimeId' ) ;
expect ( message . runtimeId ) . to . equal ( extensionName ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'has "tabId" matching webContents id' , function ( ) {
2020-03-20 20:28:31 +00:00
expect ( message ) . to . have . ownProperty ( 'tabId' ) ;
expect ( message . tabId ) . to . equal ( w . webContents . id ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'has "i18nString" with proper contents' , function ( ) {
2020-03-20 20:28:31 +00:00
expect ( message ) . to . have . ownProperty ( 'i18nString' ) ;
expect ( message . i18nString ) . to . equal ( 'foo - bar (baz)' ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'has "storageItems" with proper contents' , function ( ) {
2020-03-20 20:28:31 +00:00
expect ( message ) . to . have . ownProperty ( 'storageItems' ) ;
2020-02-03 22:01:10 +00:00
expect ( message . storageItems ) . to . deep . equal ( {
local : {
set : { hello : 'world' , world : 'hello' } ,
remove : { world : 'hello' } ,
clear : { }
} ,
sync : {
set : { foo : 'bar' , bar : 'foo' } ,
remove : { foo : 'bar' } ,
clear : { }
}
2020-03-20 20:28:31 +00:00
} ) ;
} ) ;
} ) ;
} ) ;
2020-02-03 22:01:10 +00:00
describe ( 'when the devtools is undocked' , ( ) = > {
2020-03-20 20:28:31 +00:00
let message : any ;
let w : BrowserWindow ;
2020-02-03 22:01:10 +00:00
before ( async ( ) = > {
2020-03-20 20:28:31 +00:00
w = new BrowserWindow ( { show : false , webPreferences : { nodeIntegration : true } } ) ;
showLastDevToolsPanel ( w ) ;
w . loadURL ( 'about:blank' ) ;
w . webContents . openDevTools ( { mode : 'undocked' } ) ;
2020-02-03 22:01:10 +00:00
message = await new Promise ( resolve = > ipcMain . once ( 'answer' , ( event , message ) = > {
2020-03-20 20:28:31 +00:00
resolve ( message ) ;
} ) ) ;
} ) ;
after ( closeAllWindows ) ;
2020-02-03 22:01:10 +00:00
describe ( 'created extension info' , function ( ) {
it ( 'has proper "runtimeId"' , function ( ) {
2020-03-20 20:28:31 +00:00
expect ( message ) . to . have . ownProperty ( 'runtimeId' ) ;
expect ( message . runtimeId ) . to . equal ( extensionName ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'has "tabId" matching webContents id' , function ( ) {
2020-03-20 20:28:31 +00:00
expect ( message ) . to . have . ownProperty ( 'tabId' ) ;
expect ( message . tabId ) . to . equal ( w . webContents . id ) ;
} ) ;
} ) ;
} ) ;
} ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'works when used with partitions' , async ( ) = > {
const w = new BrowserWindow ( {
show : false ,
webPreferences : {
nodeIntegration : true ,
partition : 'temp'
}
2020-03-20 20:28:31 +00:00
} ) ;
2020-02-03 22:01:10 +00:00
2020-03-20 20:28:31 +00:00
const extensionPath = path . join ( __dirname , 'fixtures' , 'devtools-extensions' , 'foo' ) ;
BrowserWindow . addDevToolsExtension ( extensionPath ) ;
2020-02-03 22:01:10 +00:00
try {
2020-03-20 20:28:31 +00:00
showLastDevToolsPanel ( w ) ;
2020-02-03 22:01:10 +00:00
const p : Promise < any > = new Promise ( resolve = > ipcMain . once ( 'answer' , function ( event , message ) {
2020-03-20 20:28:31 +00:00
resolve ( message ) ;
} ) ) ;
2020-02-03 22:01:10 +00:00
2020-03-20 20:28:31 +00:00
w . loadURL ( 'about:blank' ) ;
w . webContents . openDevTools ( { mode : 'bottom' } ) ;
const message = await p ;
expect ( message . runtimeId ) . to . equal ( 'foo' ) ;
2020-02-03 22:01:10 +00:00
} finally {
2020-03-20 20:28:31 +00:00
BrowserWindow . removeDevToolsExtension ( 'foo' ) ;
await closeAllWindows ( ) ;
2020-02-03 22:01:10 +00:00
}
2020-03-20 20:28:31 +00:00
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'serializes the registered extensions on quit' , ( ) = > {
2020-03-20 20:28:31 +00:00
const extensionName = 'foo' ;
const extensionPath = path . join ( __dirname , 'fixtures' , 'devtools-extensions' , extensionName ) ;
const serializedPath = path . join ( app . getPath ( 'userData' ) , 'DevTools Extensions' ) ;
2020-02-03 22:01:10 +00:00
2020-03-20 20:28:31 +00:00
BrowserWindow . addDevToolsExtension ( extensionPath ) ;
app . emit ( 'will-quit' ) ;
expect ( JSON . parse ( fs . readFileSync ( serializedPath , 'utf8' ) ) ) . to . deep . equal ( [ extensionPath ] ) ;
2020-02-03 22:01:10 +00:00
2020-03-20 20:28:31 +00:00
BrowserWindow . removeDevToolsExtension ( extensionName ) ;
app . emit ( 'will-quit' ) ;
expect ( fs . existsSync ( serializedPath ) ) . to . be . false ( 'file exists' ) ;
} ) ;
2020-02-03 22:01:10 +00:00
describe ( 'BrowserWindow.addExtension' , ( ) = > {
it ( 'throws errors for missing manifest.json files' , ( ) = > {
expect ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . addExtension ( path . join ( __dirname , 'does-not-exist' ) ) ;
} ) . to . throw ( 'ENOENT: no such file or directory' ) ;
} ) ;
2020-02-03 22:01:10 +00:00
it ( 'throws errors for invalid manifest.json files' , ( ) = > {
expect ( ( ) = > {
2020-03-20 20:28:31 +00:00
BrowserWindow . addExtension ( path . join ( __dirname , 'fixtures' , 'devtools-extensions' , 'bad-manifest' ) ) ;
} ) . to . throw ( 'Unexpected token }' ) ;
} ) ;
} ) ;
} ) ;
} ) ;