b7b9efa875
By default the Chromedriver will send remote-debugging-port=0 to let the browser choose a free port to listen on. The chosen port is written to a known file in the user data dir that is passed to the app through the CLI. This PR does two things. 1. Correctly passes the USER_DATA_DIR to the remote debugging server so it knows where to write the file 2. Adds support for --user-data-dir as we did not support that CLI argument and Chromedriver relies on being able to tell the "browser" where to write this file. Fixes #17354
199 lines
5.1 KiB
TypeScript
199 lines
5.1 KiB
TypeScript
import { app, dialog } from 'electron'
|
|
|
|
import * as fs from 'fs'
|
|
import * as path from 'path'
|
|
import * as url from 'url'
|
|
|
|
type DefaultAppOptions = {
|
|
file: null | string;
|
|
noHelp: boolean;
|
|
version: boolean;
|
|
webdriver: boolean;
|
|
interactive: boolean;
|
|
abi: boolean;
|
|
modules: string[];
|
|
}
|
|
|
|
const Module = require('module')
|
|
|
|
// Parse command line options.
|
|
const argv = process.argv.slice(1)
|
|
|
|
const option: DefaultAppOptions = {
|
|
file: null,
|
|
noHelp: Boolean(process.env.ELECTRON_NO_HELP),
|
|
version: false,
|
|
webdriver: false,
|
|
interactive: false,
|
|
abi: false,
|
|
modules: []
|
|
}
|
|
|
|
let nextArgIsRequire = false
|
|
|
|
for (const arg of argv) {
|
|
if (nextArgIsRequire) {
|
|
option.modules.push(arg)
|
|
nextArgIsRequire = false
|
|
continue
|
|
} else if (arg === '--version' || arg === '-v') {
|
|
option.version = true
|
|
break
|
|
} else if (arg.match(/^--app=/)) {
|
|
option.file = arg.split('=')[1]
|
|
break
|
|
} else if (arg === '--interactive' || arg === '-i' || arg === '-repl') {
|
|
option.interactive = true
|
|
} else if (arg === '--test-type=webdriver') {
|
|
option.webdriver = true
|
|
} else if (arg === '--require' || arg === '-r') {
|
|
nextArgIsRequire = true
|
|
continue
|
|
} else if (arg === '--abi' || arg === '-a') {
|
|
option.abi = true
|
|
continue
|
|
} else if (arg === '--no-help') {
|
|
option.noHelp = true
|
|
continue
|
|
} else if (arg[0] === '-') {
|
|
continue
|
|
} else {
|
|
option.file = arg
|
|
break
|
|
}
|
|
}
|
|
|
|
if (nextArgIsRequire) {
|
|
console.error('Invalid Usage: --require [file]\n\n"file" is required')
|
|
process.exit(1)
|
|
}
|
|
|
|
// Set up preload modules
|
|
if (option.modules.length > 0) {
|
|
Module._preloadModules(option.modules)
|
|
}
|
|
|
|
function loadApplicationPackage (packagePath: string) {
|
|
// Add a flag indicating app is started from default app.
|
|
Object.defineProperty(process, 'defaultApp', {
|
|
configurable: false,
|
|
enumerable: true,
|
|
value: true
|
|
})
|
|
|
|
try {
|
|
// Override app name and version.
|
|
packagePath = path.resolve(packagePath)
|
|
const packageJsonPath = path.join(packagePath, 'package.json')
|
|
if (fs.existsSync(packageJsonPath)) {
|
|
let packageJson
|
|
try {
|
|
packageJson = require(packageJsonPath)
|
|
} catch (e) {
|
|
showErrorMessage(`Unable to parse ${packageJsonPath}\n\n${e.message}`)
|
|
return
|
|
}
|
|
|
|
if (packageJson.version) {
|
|
app.setVersion(packageJson.version)
|
|
}
|
|
if (packageJson.productName) {
|
|
app.setName(packageJson.productName)
|
|
} else if (packageJson.name) {
|
|
app.setName(packageJson.name)
|
|
}
|
|
app._setDefaultAppPaths(packagePath)
|
|
}
|
|
|
|
try {
|
|
Module._resolveFilename(packagePath, module, true)
|
|
} catch (e) {
|
|
showErrorMessage(`Unable to find Electron app at ${packagePath}\n\n${e.message}`)
|
|
return
|
|
}
|
|
|
|
// Run the app.
|
|
Module._load(packagePath, module, true)
|
|
} catch (e) {
|
|
console.error('App threw an error during load')
|
|
console.error(e.stack || e)
|
|
throw e
|
|
}
|
|
}
|
|
|
|
function showErrorMessage (message: string) {
|
|
app.focus()
|
|
dialog.showErrorBox('Error launching app', message)
|
|
process.exit(1)
|
|
}
|
|
|
|
async function loadApplicationByURL (appUrl: string) {
|
|
const { loadURL } = await import('./default_app')
|
|
loadURL(appUrl)
|
|
}
|
|
|
|
async function loadApplicationByFile (appPath: string) {
|
|
const { loadFile } = await import('./default_app')
|
|
loadFile(appPath)
|
|
}
|
|
|
|
function startRepl () {
|
|
if (process.platform === 'win32') {
|
|
console.error('Electron REPL not currently supported on Windows')
|
|
process.exit(1)
|
|
}
|
|
|
|
// prevent quitting
|
|
app.on('window-all-closed', () => {})
|
|
|
|
const repl = require('repl')
|
|
repl.start('> ').on('exit', () => {
|
|
process.exit(0)
|
|
})
|
|
}
|
|
|
|
// Start the specified app if there is one specified in command line, otherwise
|
|
// start the default app.
|
|
if (option.file && !option.webdriver) {
|
|
const file = option.file
|
|
const protocol = url.parse(file).protocol
|
|
const extension = path.extname(file)
|
|
if (protocol === 'http:' || protocol === 'https:' || protocol === 'file:' || protocol === 'chrome:') {
|
|
loadApplicationByURL(file)
|
|
} else if (extension === '.html' || extension === '.htm') {
|
|
loadApplicationByFile(path.resolve(file))
|
|
} else {
|
|
loadApplicationPackage(file)
|
|
}
|
|
} else if (option.version) {
|
|
console.log('v' + process.versions.electron)
|
|
process.exit(0)
|
|
} else if (option.abi) {
|
|
console.log(process.versions.modules)
|
|
process.exit(0)
|
|
} else if (option.interactive) {
|
|
startRepl()
|
|
} else {
|
|
if (!option.noHelp) {
|
|
const welcomeMessage = `
|
|
Electron ${process.versions.electron} - Build cross platform desktop apps with JavaScript, HTML, and CSS
|
|
Usage: electron [options] [path]
|
|
|
|
A path to an Electron app may be specified. It must be one of the following:
|
|
- index.js file.
|
|
- Folder containing a package.json file.
|
|
- Folder containing an index.js file.
|
|
- .html/.htm file.
|
|
- http://, https://, or file:// URL.
|
|
|
|
Options:
|
|
-i, --interactive Open a REPL to the main process.
|
|
-r, --require Module to preload (option can be repeated).
|
|
-v, --version Print the version.
|
|
-a, --abi Print the Node ABI version.`
|
|
|
|
console.log(welcomeMessage)
|
|
}
|
|
|
|
loadApplicationByFile('index.html')
|
|
}
|