From af9b265714286c676714e5256b2554afc12bcd52 Mon Sep 17 00:00:00 2001 From: Landon Yarrington Date: Sun, 10 Oct 2021 14:16:26 -0600 Subject: [PATCH 1/8] fix: esm json import --- lib/index.ts | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/lib/index.ts b/lib/index.ts index c0bfac81..0930d4dc 100644 --- a/lib/index.ts +++ b/lib/index.ts @@ -8,11 +8,14 @@ */ import { format } from 'util' -import { readFileSync } from 'fs' import { normalize, resolve } from 'path' import { ArgsInput, Arguments, Parser, Options, DetailedArguments } from './yargs-parser-types.js' import { camelCase, decamelize, looksLikeNumber } from './string-utils.js' import { YargsParser } from './yargs-parser.js' +import { createRequire } from "module"; + +// Addresses: https://github.com/yargs/yargs/issues/2040 +const esmRequire = createRequire(import.meta.url); // See https://github.com/yargs/yargs-parser#supported-nodejs-versions for our // version support policy. The YARGS_MIN_NODE_VERSION is used for testing only. @@ -42,7 +45,7 @@ const parser = new YargsParser({ if (typeof require !== 'undefined') { return require(path) } else if (path.match(/\.json$/)) { - return readFileSync(path, 'utf8') + return esmRequire(path) } else { throw Error('only .json config files are supported in ESM') } From 5161737d310bcf86526a9f080e2f486b85a84bbd Mon Sep 17 00:00:00 2001 From: Landon Yarrington Date: Sun, 10 Oct 2021 14:39:12 -0600 Subject: [PATCH 2/8] fixes --- lib/yargs-parser.ts | 4 ++-- tsconfig.json | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/lib/yargs-parser.ts b/lib/yargs-parser.ts index 1499fc60..ca7daac4 100644 --- a/lib/yargs-parser.ts +++ b/lib/yargs-parser.ts @@ -681,7 +681,7 @@ export class YargsParser { } setConfigObject(config) - } catch (ex) { + } catch (ex: any) { // Deno will receive a PermissionDenied error if an attempt is // made to load config without the --allow-read flag: if (ex.name === 'PermissionDenied') error = ex @@ -759,7 +759,7 @@ export class YargsParser { argv[ali] = value }) } catch (err) { - error = err + error = err as Error } } } diff --git a/tsconfig.json b/tsconfig.json index 9cf3eed3..5fc1b836 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -6,7 +6,7 @@ "sourceMap": false, "target": "es2017", "moduleResolution": "node", - "module": "es2015" + "module": "es2020" }, "include": [ "lib/**/*.ts" From 835c219699d6f24c67102e7945eb9109fbfef003 Mon Sep 17 00:00:00 2001 From: Landon Yarrington Date: Sun, 10 Oct 2021 14:45:51 -0600 Subject: [PATCH 3/8] lint --- lib/index.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/index.ts b/lib/index.ts index 0930d4dc..d875d366 100644 --- a/lib/index.ts +++ b/lib/index.ts @@ -12,10 +12,10 @@ import { normalize, resolve } from 'path' import { ArgsInput, Arguments, Parser, Options, DetailedArguments } from './yargs-parser-types.js' import { camelCase, decamelize, looksLikeNumber } from './string-utils.js' import { YargsParser } from './yargs-parser.js' -import { createRequire } from "module"; +import { createRequire } from 'module' // Addresses: https://github.com/yargs/yargs/issues/2040 -const esmRequire = createRequire(import.meta.url); +const esmRequire = createRequire(import.meta.url) // See https://github.com/yargs/yargs-parser#supported-nodejs-versions for our // version support policy. The YARGS_MIN_NODE_VERSION is used for testing only. From a2a8d15b8df9944df76101b8edeebfc1205e6d76 Mon Sep 17 00:00:00 2001 From: Landon Yarrington Date: Sun, 24 Oct 2021 16:01:19 -0600 Subject: [PATCH 4/8] fixes --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 4d1a6314..82b573d9 100644 --- a/package.json +++ b/package.json @@ -49,7 +49,7 @@ "devDependencies": { "@types/chai": "^4.2.11", "@types/mocha": "^9.0.0", - "@types/node": "^14.0.0", + "@types/node": "^16.11.4", "@typescript-eslint/eslint-plugin": "^3.10.1", "@typescript-eslint/parser": "^3.10.1", "@wessberg/rollup-plugin-ts": "^1.2.28", From c60fe2093795629469df3f8d9f2a41ef9b4accfc Mon Sep 17 00:00:00 2001 From: Landon Yarrington Date: Sun, 24 Oct 2021 16:07:49 -0600 Subject: [PATCH 5/8] fixes --- test/tscc/package.json | 2 +- test/tscc/tsconfig.json | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/test/tscc/package.json b/test/tscc/package.json index 3deb3bfc..5c0815b2 100644 --- a/test/tscc/package.json +++ b/test/tscc/package.json @@ -3,6 +3,6 @@ "version": "0.0.0", "dependencies": { "@tscc/tscc": "^0.7.4", - "@types/node": "^10.0.3" + "@types/node": "^16.11.4" } } diff --git a/test/tscc/tsconfig.json b/test/tscc/tsconfig.json index 854aec18..1a1ba731 100644 --- a/test/tscc/tsconfig.json +++ b/test/tscc/tsconfig.json @@ -1,5 +1,6 @@ { "compilerOptions": { + "module": "es2020", "target": "es2017", "moduleResolution": "node" }, From 9067f06f2d859fc1c6c11f1b4f630deec08b3dc8 Mon Sep 17 00:00:00 2001 From: Landon Yarrington Date: Sun, 14 Nov 2021 14:09:30 -0700 Subject: [PATCH 6/8] fixes --- lib/index.ts | 8 +++----- 1 file changed, 3 insertions(+), 5 deletions(-) diff --git a/lib/index.ts b/lib/index.ts index 65959e36..1b3cad22 100644 --- a/lib/index.ts +++ b/lib/index.ts @@ -12,10 +12,7 @@ import { normalize, resolve } from 'path' import { ArgsInput, Arguments, Parser, Options, DetailedArguments } from './yargs-parser-types.js' import { camelCase, decamelize, looksLikeNumber } from './string-utils.js' import { YargsParser } from './yargs-parser.js' -import { createRequire } from 'module' - -// Addresses: https://github.com/yargs/yargs/issues/2040 -const esmRequire = createRequire(import.meta.url) +import { readFileSync } from 'fs' // See https://github.com/yargs/yargs-parser#supported-nodejs-versions for our // version support policy. The YARGS_MIN_NODE_VERSION is used for testing only. @@ -45,7 +42,8 @@ const parser = new YargsParser({ if (typeof require !== 'undefined') { return require(path) } else if (path.match(/\.json$/)) { - return esmRequire(path) + // Addresses: https://github.com/yargs/yargs/issues/2040 + return JSON.parse(readFileSync(path, 'utf8')); } else { throw Error('only .json config files are supported in ESM') } From ad24db35c3ec6b45c999b0ac7268f897cf7b7cd1 Mon Sep 17 00:00:00 2001 From: Landon Yarrington Date: Sun, 14 Nov 2021 14:13:52 -0700 Subject: [PATCH 7/8] lint --- lib/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/index.ts b/lib/index.ts index 1b3cad22..8c413229 100644 --- a/lib/index.ts +++ b/lib/index.ts @@ -43,7 +43,7 @@ const parser = new YargsParser({ return require(path) } else if (path.match(/\.json$/)) { // Addresses: https://github.com/yargs/yargs/issues/2040 - return JSON.parse(readFileSync(path, 'utf8')); + return JSON.parse(readFileSync(path, 'utf8')) } else { throw Error('only .json config files are supported in ESM') } From abaffe1cc960d118ab8604a653ecb9756e980cbd Mon Sep 17 00:00:00 2001 From: Landon Yarrington Date: Sun, 14 Nov 2021 15:28:55 -0700 Subject: [PATCH 8/8] tests --- package.json | 1 + test/yargs-parser.mjs | 264 ++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 265 insertions(+) create mode 100644 test/yargs-parser.mjs diff --git a/package.json b/package.json index 6e2499e0..ea9a27fc 100644 --- a/package.json +++ b/package.json @@ -19,6 +19,7 @@ "fix": "standardx --fix '**/*.ts' && standardx --fix '**/*.js' && standardx --fix '**/*.cjs'", "pretest": "rimraf build && tsc -p tsconfig.test.json && cross-env NODE_ENV=test npm run build:cjs", "test": "c8 --reporter=text --reporter=html mocha test/*.cjs", + "test:esm": "c8 --reporter=text --reporter=html mocha test/*.mjs", "test:browser": "start-server-and-test 'serve ./ -p 8080' http://127.0.0.1:8080/package.json 'node ./test/browser/yargs-test.cjs'", "pretest:typescript": "npm run pretest", "test:typescript": "c8 mocha ./build/test/typescript/*.js", diff --git a/test/yargs-parser.mjs b/test/yargs-parser.mjs new file mode 100644 index 00000000..5ef41314 --- /dev/null +++ b/test/yargs-parser.mjs @@ -0,0 +1,264 @@ +import { should, expect } from 'chai' +import parser from '../build/lib/index.js' +import path from 'path' +import { fileURLToPath } from 'url' +import { readFileSync } from 'fs' + +should() + +describe('yargs-parser (esm)', function () { + const __dirname = path.dirname(fileURLToPath(import.meta.url)) + const jsonPath = path.resolve(__dirname, './fixtures/config.json') + describe('config', function () { + it('should load options and values from default config if specified', function () { + const argv = parser(['--foo', 'bar'], { + alias: { + z: 'zoom' + }, + default: { + settings: jsonPath + }, + config: 'settings' + }) + + argv.should.have.property('herp', 'derp') + argv.should.have.property('zoom', 55) + argv.should.have.property('foo').and.deep.equal('bar') + }) + + it('should use value from config file, if argv value is using default value', function () { + const argv = parser([], { + alias: { + z: 'zoom' + }, + config: ['settings'], + default: { + settings: jsonPath, + foo: 'banana' + } + }) + + argv.should.have.property('herp', 'derp') + argv.should.have.property('zoom', 55) + argv.should.have.property('foo').and.deep.equal('baz') + }) + + it('should combine values from config file and argv, if argv value is an array', function () { + const argv = parser(['--foo', 'bar'], { + config: ['settings'], + array: ['foo'], + default: { + settings: jsonPath + }, + configuration: { + 'combine-arrays': true + } + }) + + argv.should.have.property('foo').and.deep.equal(['bar', 'baz']) + }) + + it('should use value from config file, if argv key is a boolean', function () { + const argv = parser([], { + config: ['settings'], + default: { + settings: jsonPath + }, + boolean: ['truthy'] + }) + + argv.should.have.property('truthy', true) + }) + + it('should use value from cli, if cli overrides boolean argv key', function () { + const argv = parser(['--no-truthy'], { + config: ['settings'], + default: { + settings: jsonPath + }, + boolean: ['truthy'] + }) + + argv.should.have.property('truthy', false) + }) + + it('should use cli value, if cli value is set and both cli and default value match', function () { + const argv = parser(['--foo', 'banana'], { + alias: { + z: 'zoom' + }, + config: ['settings'], + default: { + settings: jsonPath, + foo: 'banana' + } + }) + + argv.should.have.property('herp', 'derp') + argv.should.have.property('zoom', 55) + argv.should.have.property('foo').and.deep.equal('banana') + }) + + it("should allow config to be set as flag in 'option'", function () { + const argv = parser(['--settings', jsonPath, '--foo', 'bar'], { + alias: { + z: 'zoom' + }, + config: ['settings'] + }) + + argv.should.have.property('herp', 'derp') + argv.should.have.property('zoom', 55) + argv.should.have.property('foo').and.deep.equal('bar') + }) + + // for esm, only support importing json files + it('should fail to load options and values from a JS file when config has .js extention', function () { + const jsPath = path.resolve(__dirname, './fixtures/settings.cjs') + const argv = parser.detailed(['--settings', jsPath, '--foo', 'bar'], { + config: ['settings'] + }) + + argv.error.message.should.include('Invalid JSON config file') + }) + + it('should raise an appropriate error if JSON file is not found', function () { + const argv = parser.detailed(['--settings', 'fake.json', '--foo', 'bar'], { + alias: { + z: 'zoom' + }, + config: ['settings'] + }) + + argv.error.message.should.equal('Invalid JSON config file: fake.json') + }) + + // see: https://github.com/bcoe/yargs/issues/172 + it('should not raise an exception if config file is set as default argument value', function () { + const argv = parser.detailed([], { + default: { + config: 'foo.json' + }, + config: ['config'] + }) + + expect(argv.error).to.equal(null) + }) + + it('should load nested options from config file', function () { + const jsonPath = path.resolve(__dirname, './fixtures/nested_config.json') + const argv = parser(['--settings', jsonPath, '--nested.foo', 'bar'], { + config: ['settings'] + }) + + argv.should.have.property('a', 'a') + argv.should.have.property('b', 'b') + argv.should.have.property('nested').and.deep.equal({ + foo: 'bar', + bar: 'bar' + }) + }) + + it('should use nested value from config file, if argv value is using default value', function () { + const jsonPath = path.resolve(__dirname, './fixtures/nested_config.json') + const argv = parser(['--settings', jsonPath], { + config: ['settings'], + default: { + 'nested.foo': 'banana' + } + }) + + argv.should.have.property('a', 'a') + argv.should.have.property('b', 'b') + argv.should.have.property('nested').and.deep.equal({ + foo: 'baz', + bar: 'bar' + }) + }) + + it('allows a custom parsing function to be provided', function () { + const jsPath = path.resolve(__dirname, './fixtures/config.txt') + const argv = parser(['--settings', jsPath, '--foo', 'bar'], { + config: { + settings: function (configPath) { + // as an example, parse an environment + // variable style config: + // FOO=99 + // BATMAN=grumpy + const config = {} + const txt = readFileSync(configPath, 'utf-8') + txt.split(/\r?\n/).forEach(function (l) { + const kv = l.split('=') + config[kv[0].toLowerCase()] = kv[1] + }) + return config + } + } + }) + + argv.batman.should.equal('grumpy') + argv.awesome.should.equal('banana') + argv.foo.should.equal('bar') + }) + + it('allows a custom parsing function to be provided as an alias', function () { + const jsPath = path.resolve(__dirname, './fixtures/config.json') + const argv = parser(['--settings', jsPath, '--foo', 'bar'], { + config: { + s: function (configPath) { + return JSON.parse(readFileSync(configPath, 'utf-8')) + } + }, + alias: { + s: ['settings'] + } + }) + + argv.should.have.property('herp', 'derp') + argv.should.have.property('foo', 'bar') + }) + + it('outputs an error returned by the parsing function', function () { + const argv = parser.detailed(['--settings=./package.json'], { + config: { + settings: function (configPath) { + return Error('someone set us up the bomb') + } + } + }) + + argv.error.message.should.equal('someone set us up the bomb') + }) + + it('outputs an error if thrown by the parsing function', function () { + const argv = parser.detailed(['--settings=./package.json'], { + config: { + settings: function (configPath) { + throw Error('someone set us up the bomb') + } + } + }) + + argv.error.message.should.equal('someone set us up the bomb') + }) + + it('should not pollute the prototype', function () { + const argv = parser(['--foo', 'bar'], { + alias: { + z: 'zoom' + }, + default: { + settings: jsonPath + }, + config: 'settings' + }) + + argv.should.have.property('herp', 'derp') + argv.should.have.property('zoom', 55) + argv.should.have.property('foo').and.deep.equal('bar') + + expect({}.bbb).to.equal(undefined) + expect({}.aaa).to.equal(undefined) + }) + }) +}) \ No newline at end of file