From ba752cbe9d3482e7218a20b5c754333488815985 Mon Sep 17 00:00:00 2001 From: Craig Nishina Date: Tue, 3 Jan 2017 14:40:48 -0800 Subject: [PATCH] chore(cli): throw errors on unknown flags clsoes #3216 --- lib/cli.ts | 113 ++++++++++++++++++++++++++++--------------- scripts/errorTest.js | 6 +++ 2 files changed, 80 insertions(+), 39 deletions(-) diff --git a/lib/cli.ts b/lib/cli.ts index c3eb2cde4..af703bffd 100644 --- a/lib/cli.ts +++ b/lib/cli.ts @@ -31,48 +31,83 @@ process.argv.slice(2).forEach(function(arg: string) { } }); -optimist - .usage( - 'Usage: protractor [configFile] [options]\n' + - 'configFile defaults to protractor.conf.js\n' + - 'The [options] object will override values from the config file.\n' + - 'See the reference config for a full list of options.') - .describe('help', 'Print Protractor help menu') - .describe('version', 'Print Protractor version') - .describe('browser', 'Browsername, e.g. chrome or firefox') - .describe('seleniumAddress', 'A running selenium address to use') - .describe('seleniumSessionId', 'Attaching an existing session id') - .describe('seleniumServerJar', 'Location of the standalone selenium jar file') - .describe('seleniumPort', 'Optional port for the selenium standalone server') - .describe('baseUrl', 'URL to prepend to all relative paths') - .describe('rootElement', 'Element housing ng-app, if not html or body') - .describe('specs', 'Comma-separated list of files to test') - .describe('exclude', 'Comma-separated list of files to exclude') - .describe('verbose', 'Print full spec names') - .describe('stackTrace', 'Print stack trace on error') - .describe('params', 'Param object to be passed to the tests') - .describe('framework', 'Test framework to use: jasmine, mocha, or custom') - .describe('resultJsonOutputFile', 'Path to save JSON test result') - .describe('troubleshoot', 'Turn on troubleshooting output') - .describe('elementExplorer', 'Interactively test Protractor commands') - .describe('debuggerServerPort', 'Start a debugger server at specified port instead of repl') - .alias('browser', 'capabilities.browserName') - .alias('name', 'capabilities.name') - .alias('platform', 'capabilities.platform') - .alias('platform-version', 'capabilities.version') - .alias('tags', 'capabilities.tags') - .alias('build', 'capabilities.build') - .alias('grep', 'jasmineNodeOpts.grep') - .alias('invert-grep', 'jasmineNodeOpts.invertGrep') - .alias('explorer', 'elementExplorer') - .string('capabilities.tunnel-identifier') - .check(function(arg: any) { - if (arg._.length > 1) { - throw new Error('Error: more than one config file specified'); +let optimistOptions: any = { + describes: { + help: 'Print Protractor help menu', + version: 'Print Protractor version', + browser: 'Browsername, e.g. chrome or firefox', + seleniumAddress: 'A running selenium address to use', + seleniumSessionId: 'Attaching an existing session id', + seleniumServerJar: 'Location of the standalone selenium jar file', + seleniumPort: 'Optional port for the selenium standalone server', + baseUrl: 'URL to prepend to all relative paths', + rootElement: 'Element housing ng-app, if not html or body', + specs: 'Comma-separated list of files to test', + exclude: 'Comma-separated list of files to exclude', + verbose: 'Print full spec names', + stackTrace: 'Print stack trace on error', + params: 'Param object to be passed to the tests', + framework: 'Test framework to use: jasmine, mocha, or custom', + resultJsonOutputFile: 'Path to save JSON test result', + troubleshoot: 'Turn on troubleshooting output', + elementExplorer: 'Interactively test Protractor commands', + debuggerServerPort: 'Start a debugger server at specified port instead of repl' + }, + aliases: { + browser: 'capabilities.browserName', + name: 'capabilities.name', + platform: 'capabilities.platform', + 'platform-version': 'capabilities.version', + tags: 'capabilities.tags', + build: 'capabilities.build', + grep: 'jasmineNodeOpts.grep', + 'invert-grep': 'jasmineNodeOpts.invertGrep', + explorer: 'elementExplorer' + }, + strings: {'capabilities.tunnel-identifier': ''} +}; + +optimist.usage( + 'Usage: protractor [configFile] [options]\n' + + 'configFile defaults to protractor.conf.js\n' + + 'The [options] object will override values from the config file.\n' + + 'See the reference config for a full list of options.'); +for (let key of Object.keys(optimistOptions.describes)) { + optimist.describe(key, optimistOptions.describes[key]); +} +for (let key of Object.keys(optimistOptions.aliases)) { + optimist.alias(key, optimistOptions.aliases[key]); +} +for (let key of Object.keys(optimistOptions.strings)) { + optimist.string(key); +} +optimist.check(function(arg: any) { + if (arg._.length > 1) { + throw new Error('Error: more than one config file specified'); + } +}); + +let argv: any = optimist.parse(args); + +// Check to see if additional flags were used. +let unknownKeys: string[] = []; +Object.keys(argv) + .filter((element: string, index: number) => { + return index >= 2 && element !== '$0'; + }) + .forEach((currentValue: string) => { + let found = false; + for (let optimistKey of Object.keys(optimistOptions)) { + found = found || (optimistOptions[optimistKey][currentValue] !== undefined); + } + if (!found) { + unknownKeys.push(currentValue); } }); -let argv: any = optimist.parse(args); +if (unknownKeys.length > 0) { + throw new Error('Found extra flags: ' + unknownKeys.join(', ')); +} if (argv.help) { optimist.showHelp(); diff --git a/scripts/errorTest.js b/scripts/errorTest.js index 4a2effe80..99efe5a96 100644 --- a/scripts/errorTest.js +++ b/scripts/errorTest.js @@ -19,6 +19,12 @@ var checkLogs = function(output, messages) { *Below are exit failure tests* ******************************/ +runProtractor = spawn('node', + ['bin/protractor', '--help', '--foobar', 'foobar']); +output = runProtractor.stderr.toString(); +messages = ['Error: Found extra flags: foobar']; +checkLogs(output, messages); + // assert authentication error for sauce labs runProtractor = spawn('node', ['bin/protractor', 'spec/errorTest/sauceLabsAuthentication.js']);