From 933bb82606f0fd0510f6d86b50658b4e3b499ce0 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Fri, 22 Sep 2023 15:35:18 -0600 Subject: [PATCH 01/27] fix: use ES2022 --- tsconfig.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tsconfig.json b/tsconfig.json index 0aed879ee..6e71ac339 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -10,7 +10,7 @@ "./src" ], "strict": true, - "target": "ES2021", + "target": "ES2022", "allowSyntheticDefaultImports": true, "noErrorTruncation": true, "moduleResolution": "Node16" From 4d14780852e6b4eae000f16170ae06d354986384 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Mon, 25 Sep 2023 09:43:13 -0600 Subject: [PATCH 02/27] test: use --json for config unset --- test/integration/sf.e2e.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/integration/sf.e2e.ts b/test/integration/sf.e2e.ts index 7e423d206..bd2786122 100644 --- a/test/integration/sf.e2e.ts +++ b/test/integration/sf.e2e.ts @@ -115,7 +115,7 @@ describe('Salesforce CLI (sf)', () => { expect(results.failures).to.be.empty - const unset = await executor.executeCommand('config unset disable-telemetry org-api-version --global') + const unset = await executor.executeCommand('config unset disable-telemetry org-api-version --global --json') const unsetParsed = parseJson(unset.stdout!) expect(unsetParsed.status).to.equal(0) }) From 57794c3a506188682b3d06a18095f8b417ba77f0 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Mon, 25 Sep 2023 15:14:58 -0600 Subject: [PATCH 03/27] feat: stop using getters and setters for flags --- src/command.ts | 59 ++++---------- src/config/config.ts | 41 ++++++++-- src/flags.ts | 6 +- src/interfaces/parser.ts | 3 +- test/command/command.test.ts | 147 ++++++++++++++++++++++++----------- 5 files changed, 159 insertions(+), 97 deletions(-) diff --git a/src/command.ts b/src/command.ts index 7a54e2136..45e80dad9 100644 --- a/src/command.ts +++ b/src/command.ts @@ -42,7 +42,7 @@ stdout.on('error', (err: any) => { throw err }) -const jsonFlag = { +export const jsonFlag = { json: boolean({ description: 'Format output as json.', helpGroup: 'GLOBAL', @@ -128,22 +128,7 @@ export abstract class Command { protected static '_--' = false - protected static _enableJsonFlag = false - - public static get enableJsonFlag(): boolean { - return this._enableJsonFlag - } - - public static set enableJsonFlag(value: boolean) { - this._enableJsonFlag = value - if (value === true) { - this.baseFlags = jsonFlag - } else { - delete this.baseFlags?.json - this.flags = {} // force the flags setter to run - delete this.flags?.json - } - } + public static enableJsonFlag = false public static get '--'(): boolean { return Command['_--'] @@ -184,29 +169,10 @@ export abstract class Command { return cmd._run>() } - protected static _baseFlags: FlagInput - - static get baseFlags(): FlagInput { - return this._baseFlags - } - - static set baseFlags(flags: FlagInput) { - // eslint-disable-next-line prefer-object-spread - this._baseFlags = Object.assign({}, this.baseFlags, flags) - this.flags = {} // force the flags setter to run - } + public static baseFlags: FlagInput /** A hash of flags for the command */ - protected static _flags: FlagInput - - public static get flags(): FlagInput { - return this._flags - } - - public static set flags(flags: FlagInput) { - // eslint-disable-next-line prefer-object-spread - this._flags = Object.assign({}, this._flags ?? {}, this.baseFlags, flags) - } + public static flags: FlagInput public id: string | undefined @@ -352,12 +318,19 @@ export abstract class Command { } } - protected async parse(options?: Input, argv = this.argv): Promise> { + protected async parse( + options?: Input, + argv = this.argv, + ): Promise> { if (!options) options = this.ctor as Input - const opts = {context: this, ...options} - // the spread operator doesn't work with getters so we have to manually add it here - opts.flags = options?.flags - opts.args = options?.args + const combinedFlags = {...options.baseFlags, ...options.flags} + + const opts = { + context: this, + ...options, + flags: (options.enableJsonFlag ? {...combinedFlags, ...jsonFlag} : combinedFlags) as FlagInput, + } + const results = await Parser.parse(argv, opts) this.warnIfFlagDeprecated(results.flags ?? {}) diff --git a/src/config/config.ts b/src/config/config.ts index 83428679a..f5f288633 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -7,12 +7,12 @@ import {Hook, Hooks, PJSON, Topic} from '../interfaces' import {Plugin as IPlugin, Options} from '../interfaces/plugin' import {URL, fileURLToPath} from 'node:url' import {arch, userInfo as osUserInfo, release, tmpdir, type} from 'node:os' -import {compact, ensureArgObject, getHomeDir, getPlatform, isProd, requireJson} from '../util' +import {compact, ensureArgObject, getHomeDir, getPlatform, isProd, pickBy, requireJson} from '../util' import {join, sep} from 'node:path' - import {Command} from '../command' import {Performance} from '../performance' import PluginLoader from './plugin-loader' +import {boolean} from '../flags' import {format} from 'node:util' import {getHelpFlagAdditions} from '../help' import {loadWithData} from '../module-loader' @@ -855,10 +855,32 @@ const defaultArgToCached = async (arg: Arg, respectNoCacheDefault: boolean) } } -export async function toCached(c: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { +export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { const flags = {} as {[k: string]: Command.Flag.Cached} - for (const [name, flag] of Object.entries(c.flags || {})) { + // In order to collect static properties up the inheritance chain, we need to recursively + // access the prototypes until there's nothing left. This allows us to combine baseFlags + // and flags as well as add in the json flag if enableJsonFlag is enabled. + const mergePrototype = (result: Command.Class, cmd: Command.Class): Command.Class => { + const proto = Object.getPrototypeOf(cmd) + const filteredProto = pickBy(proto, v => v !== undefined) as Command.Class + return Object.keys(proto).length > 0 ? mergePrototype({...filteredProto, ...result} as Command.Class, proto) : result + } + + const c = mergePrototype(cmd, cmd) + + const cmdFlags = { + ...(c.enableJsonFlag ? { + json: boolean({ + description: 'Format output as json.', + helpGroup: 'GLOBAL', + }), + } : {}), + ...c.flags, + ...c.baseFlags, + } as typeof c['flags'] + + for (const [name, flag] of Object.entries(cmdFlags || {})) { if (flag.type === 'boolean') { flags[name] = { name, @@ -946,7 +968,16 @@ export async function toCached(c: Command.Class, plugin?: IPlugin, respectNoCach } // do not include these properties in manifest - const ignoreCommandProperties = ['plugin', '_flags', '_enableJsonFlag', '_globalFlags', '_baseFlags'] + const ignoreCommandProperties = [ + 'plugin', + '_flags', + '_enableJsonFlag', + '_globalFlags', + '_baseFlags', + 'baseFlags', + '_--', + '_base', + ] const stdKeys = Object.keys(stdProperties) const keysToAdd = Object.keys(c).filter(property => ![...stdKeys, ...ignoreCommandProperties].includes(property)) const additionalProperties: Record = {} diff --git a/src/flags.ts b/src/flags.ts index 61df7db68..699dde32e 100644 --- a/src/flags.ts +++ b/src/flags.ts @@ -9,14 +9,14 @@ type NotArray = T extends Array ? never: T; export function custom( defaults: Partial> & { - multiple: true + multiple: true; } & ( {required: true} | {default: OptionFlag['default']} ), ): FlagDefinition export function custom( - defaults: Partial> & { + defaults: Partial, P>> & { multiple?: false | undefined; } & ( {required: true} | {default: OptionFlag, P>['default']} @@ -24,7 +24,7 @@ export function custom( ): FlagDefinition export function custom( - defaults: Partial> & { + defaults: Partial, P>> & { default?: OptionFlag, P>['default'] | undefined; multiple?: false | undefined; required?: false | undefined; diff --git a/src/interfaces/parser.ts b/src/interfaces/parser.ts index c95bfec7a..ef0b8f690 100644 --- a/src/interfaces/parser.ts +++ b/src/interfaces/parser.ts @@ -243,7 +243,7 @@ export type BooleanFlag = FlagProps & BooleanFlagProps & { } export type OptionFlag = FlagProps & OptionFlagProps & { - parse: FlagParser + parse: FlagParser defaultHelp?: FlagDefaultHelp; input: string[]; default?: FlagDefault; @@ -361,6 +361,7 @@ export type Flag = BooleanFlag | OptionFlag export type Input = { flags?: FlagInput; baseFlags?: FlagInput; + enableJsonFlag?: true | false; args?: ArgInput; strict?: boolean; context?: ParserContext; diff --git a/test/command/command.test.ts b/test/command/command.test.ts index ddbad21bb..c0aa447f2 100644 --- a/test/command/command.test.ts +++ b/test/command/command.test.ts @@ -80,7 +80,6 @@ describe('command', () => { static description = 'test command' static aliases = ['alias1', 'alias2'] static hidden = true - // @ts-ignore static flags = { flaga: Flags.boolean(), flagb: Flags.string({ @@ -98,8 +97,7 @@ describe('command', () => { required: false, description: 'flagc desc', options: ['a', 'b'], - // @ts-expect-error: context is any - default: async context => context.options.min + 1, + default: async context => (context.options.min ?? 1) + 1, }), } @@ -133,6 +131,8 @@ describe('command', () => { deprecateAliases: undefined, summary: undefined, strict: true, + enableJsonFlag: false, + hasDynamicHelp: false, flags: { flaga: { aliases: undefined, @@ -224,51 +224,108 @@ describe('command', () => { .it('converts to cached with everything set') fancy - // .skip() .do(async () => { - // const c = class extends Command { - // }.convertToCached() - // expect(await c.load()).to.have.property('run') - // delete c.load - // expect(c).to.deep.equal({ - // _base: `@oclif/command@${pjson.version}`, - // id: undefined, - // type: undefined, - // hidden: undefined, - // pluginName: undefined, - // description: 'test command', - // aliases: [], - // title: undefined, - // usage: undefined, - // flags: {}, - // args: [], - // }) - }) + class Base extends Command { + public static enableJsonFlag = true + public static baseFlags = { + parentFlag: Flags.boolean(), + } + } + + class Child extends Base { + static flags = { + childFlag: Flags.boolean(), + } + } - .it('adds plugin name') + const cached = await toCached(Child, undefined, false) - fancy - // .skip() - // .do(async () => { - // const c = class extends Command { - // }.convertToCached({pluginName: 'myplugin'}) - // expect(await c.load()).to.have.property('run') - // delete c.load - // expect(c).to.deep.equal({ - // _base: `@oclif/command@${pjson.version}`, - // type: undefined, - // id: undefined, - // hidden: undefined, - // pluginName: 'myplugin', - // description: 'test command', - // aliases: [], - // title: undefined, - // usage: undefined, - // flags: {}, - // args: [], - // }) - // }) - .it('converts to cached with nothing set') + expect(cached).to.deep.equal({ + id: 'command', + summary: undefined, + description: 'test command', + strict: true, + usage: undefined, + pluginName: undefined, + pluginAlias: undefined, + pluginType: undefined, + hidden: undefined, + state: undefined, + aliases: [], + examples: undefined, + deprecationOptions: undefined, + deprecateAliases: undefined, + flags: { + json: { + name: 'json', + type: 'boolean', + char: undefined, + summary: undefined, + description: 'Format output as json.', + hidden: undefined, + required: undefined, + helpLabel: undefined, + helpGroup: 'GLOBAL', + allowNo: false, + dependsOn: undefined, + relationships: undefined, + exclusive: undefined, + deprecated: undefined, + deprecateAliases: undefined, + aliases: undefined, + charAliases: undefined, + delimiter: undefined, + noCacheDefault: undefined, + }, + childFlag: { + name: 'childFlag', + type: 'boolean', + char: undefined, + summary: undefined, + description: undefined, + hidden: undefined, + required: undefined, + helpLabel: undefined, + helpGroup: undefined, + allowNo: false, + dependsOn: undefined, + relationships: undefined, + exclusive: undefined, + deprecated: undefined, + deprecateAliases: undefined, + aliases: undefined, + charAliases: undefined, + delimiter: undefined, + noCacheDefault: undefined, + }, + parentFlag: { + name: 'parentFlag', + type: 'boolean', + char: undefined, + summary: undefined, + description: undefined, + hidden: undefined, + required: undefined, + helpLabel: undefined, + helpGroup: undefined, + allowNo: false, + dependsOn: undefined, + relationships: undefined, + exclusive: undefined, + deprecated: undefined, + deprecateAliases: undefined, + aliases: undefined, + charAliases: undefined, + delimiter: undefined, + noCacheDefault: undefined, + }, + }, + args: {}, + hasDynamicHelp: false, + enableJsonFlag: true, + }) + }) + .it('converts to cached with multiple Command classes in inheritance chain') }) describe('parse', () => { From a893f344fbe33215cb6bfff7a711d12c1c6084d2 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Mon, 25 Sep 2023 15:49:43 -0600 Subject: [PATCH 04/27] chore: clean up --- src/command.ts | 11 ++--------- src/config/config.ts | 9 ++------- src/flags.ts | 5 +++++ src/index.ts | 27 ++++++++++++++++++++++++++- 4 files changed, 35 insertions(+), 17 deletions(-) diff --git a/src/command.ts b/src/command.ts index 45e80dad9..442b9a1f6 100644 --- a/src/command.ts +++ b/src/command.ts @@ -25,9 +25,9 @@ import {LoadOptions} from './interfaces/config' import {PJSON} from './interfaces' import {Plugin} from './interfaces/plugin' import {PrettyPrintableError} from './errors' -import {boolean} from './flags' import chalk from 'chalk' import {fileURLToPath} from 'node:url' +import {json} from './flags' import {ux} from './cli-ux' const pjson = requireJson(__dirname, '..', 'package.json') @@ -42,13 +42,6 @@ stdout.on('error', (err: any) => { throw err }) -export const jsonFlag = { - json: boolean({ - description: 'Format output as json.', - helpGroup: 'GLOBAL', - }), -} - /** * An abstract class which acts as the base for each command * in your project. @@ -328,7 +321,7 @@ export abstract class Command { const opts = { context: this, ...options, - flags: (options.enableJsonFlag ? {...combinedFlags, ...jsonFlag} : combinedFlags) as FlagInput, + flags: (options.enableJsonFlag ? {...combinedFlags, json} : combinedFlags) as FlagInput, } const results = await Parser.parse(argv, opts) diff --git a/src/config/config.ts b/src/config/config.ts index f5f288633..3ff0278f5 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -12,9 +12,9 @@ import {join, sep} from 'node:path' import {Command} from '../command' import {Performance} from '../performance' import PluginLoader from './plugin-loader' -import {boolean} from '../flags' import {format} from 'node:util' import {getHelpFlagAdditions} from '../help' +import {json} from '../flags' import {loadWithData} from '../module-loader' import {settings} from '../settings' import {stdout} from '../cli-ux/stream' @@ -870,12 +870,7 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa const c = mergePrototype(cmd, cmd) const cmdFlags = { - ...(c.enableJsonFlag ? { - json: boolean({ - description: 'Format output as json.', - helpGroup: 'GLOBAL', - }), - } : {}), + ...(c.enableJsonFlag ? {json} : {}), ...c.flags, ...c.baseFlags, } as typeof c['flags'] diff --git a/src/flags.ts b/src/flags.ts index 699dde32e..eb4887ed6 100644 --- a/src/flags.ts +++ b/src/flags.ts @@ -209,3 +209,8 @@ export function option( type: 'option', }) } + +export const json = boolean({ + description: 'Format output as json.', + helpGroup: 'GLOBAL', +}) diff --git a/src/index.ts b/src/index.ts index 4eea78bfa..71b645bb3 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,3 +1,15 @@ +import { + boolean, + custom, + directory, + file, + help, + integer, + option, + string, + url, + version, +} from './flags' import {stderr} from './cli-ux/stream' function checkCWD() { @@ -14,7 +26,20 @@ checkCWD() export * as Args from './args' export * as Errors from './errors' -export * as Flags from './flags' + +export const Flags = { + custom, + option, + boolean, + string, + url, + integer, + directory, + file, + help, + version, +} + export * as Interfaces from './interfaces' export * as Parser from './parser' export * as ux from './cli-ux' From 52f1b12f3c9e690b19ac0341f9d0cbce17592db8 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 09:52:00 -0600 Subject: [PATCH 05/27] feat: expose json flag --- src/command.ts | 7 ++++++- src/config/config.ts | 2 +- src/flags.ts | 15 ++++++++++----- src/index.ts | 26 +------------------------- test/command/command.test.ts | 34 ++++++++++++++++++++++++++++++++++ 5 files changed, 52 insertions(+), 32 deletions(-) diff --git a/src/command.ts b/src/command.ts index 442b9a1f6..7b9263df5 100644 --- a/src/command.ts +++ b/src/command.ts @@ -243,8 +243,13 @@ export abstract class Command { * @returns {boolean} true if the command supports json and the --json flag is present */ public jsonEnabled(): boolean { + const flagOverride = this.config.scopedEnvVarTrue?.('JSON_FLAG_OVERRIDE') + + if (flagOverride) return true + // if the command doesn't support json, return false if (!this.ctor.enableJsonFlag) return false + // if the command parameter pass through is enabled, return true if the --json flag is before the '--' separator if (this.passThroughEnabled) { const ptIndex = this.argv.indexOf('--') @@ -321,7 +326,7 @@ export abstract class Command { const opts = { context: this, ...options, - flags: (options.enableJsonFlag ? {...combinedFlags, json} : combinedFlags) as FlagInput, + flags: (options.enableJsonFlag ? {...combinedFlags, json: json()} : combinedFlags) as FlagInput, } const results = await Parser.parse(argv, opts) diff --git a/src/config/config.ts b/src/config/config.ts index 3ff0278f5..18df7da5e 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -870,7 +870,7 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa const c = mergePrototype(cmd, cmd) const cmdFlags = { - ...(c.enableJsonFlag ? {json} : {}), + ...(c.enableJsonFlag ? {json: json()} : {}), ...c.flags, ...c.baseFlags, } as typeof c['flags'] diff --git a/src/flags.ts b/src/flags.ts index eb4887ed6..f38733570 100644 --- a/src/flags.ts +++ b/src/flags.ts @@ -145,6 +145,16 @@ export const help = (opts: Partial> = {}): BooleanFlag> = {}): BooleanFlag => boolean({ + description: 'Format output as json.', + helpGroup: 'GLOBAL', + ...opts, + async parse(input, cmd) { + if (input) process.env[cmd.config.scopedEnvVarKey('JSON_FLAG_OVERRIDE')] = 'true' + return input + }, +}) + type ElementType> = T[number]; export function option( @@ -209,8 +219,3 @@ export function option( type: 'option', }) } - -export const json = boolean({ - description: 'Format output as json.', - helpGroup: 'GLOBAL', -}) diff --git a/src/index.ts b/src/index.ts index 71b645bb3..b9ce5e1bc 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,15 +1,3 @@ -import { - boolean, - custom, - directory, - file, - help, - integer, - option, - string, - url, - version, -} from './flags' import {stderr} from './cli-ux/stream' function checkCWD() { @@ -27,22 +15,10 @@ checkCWD() export * as Args from './args' export * as Errors from './errors' -export const Flags = { - custom, - option, - boolean, - string, - url, - integer, - directory, - file, - help, - version, -} - export * as Interfaces from './interfaces' export * as Parser from './parser' export * as ux from './cli-ux' +export * as Flags from './flags' export {CommandHelp, HelpBase, Help, loadHelpClass} from './help' export {Config, toCached, Plugin, tsPath} from './config' export {HelpSection, HelpSectionRenderer, HelpSectionKeyValueTable} from './help/formatter' diff --git a/test/command/command.test.ts b/test/command/command.test.ts index c0aa447f2..6bb67ac66 100644 --- a/test/command/command.test.ts +++ b/test/command/command.test.ts @@ -711,5 +711,39 @@ describe('command', () => { expect(cmd.jsonEnabled()).to.equal(false) }) .it('json disabled/pass through enable/--json flag before --/jsonEnabled() should be false') + + fancy + .stdout() + .do(async () => { + class CMD extends Command { + static flags = { + json: Flags.json(), + } + + async run() {} + } + + const cmd = new CMD(['--json'], { + bin: 'FOO', scopedEnvVarTrue: (foo: string) => foo.includes('JSON_FLAG_OVERRIDE'), + } as any) + expect(cmd.jsonEnabled()).to.equal(true) + }) + .it('Flags.json to enable json, enableJsonFlag not set') + + fancy + .stdout() + .do(async () => { + class CMD extends Command { + static flags = { + json: Flags.json(), + } + + async run() {} + } + + const cmd = new CMD([], {} as any) + expect(cmd.jsonEnabled()).to.equal(false) + }) + .it('Flags.json in flags definition but not used, enableJsonFlag not set') }) }) From 178b9a996a279f12d4c5f41dd100c5318bcdf079 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 10:39:24 -0600 Subject: [PATCH 06/27] feat: remove pass through getter and setter --- src/command.ts | 40 +++++++++++++----------------------- test/command/command.test.ts | 14 +++---------- 2 files changed, 17 insertions(+), 37 deletions(-) diff --git a/src/command.ts b/src/command.ts index 7b9263df5..3b60f0b16 100644 --- a/src/command.ts +++ b/src/command.ts @@ -119,22 +119,7 @@ export abstract class Command { public static hasDynamicHelp = false - protected static '_--' = false - public static enableJsonFlag = false - - public static get '--'(): boolean { - return Command['_--'] - } - - public static set '--'(value: boolean) { - Command['_--'] = value - } - - public get passThroughEnabled(): boolean { - return Command['_--'] - } - /** * instantiate and run the command * @@ -243,21 +228,24 @@ export abstract class Command { * @returns {boolean} true if the command supports json and the --json flag is present */ public jsonEnabled(): boolean { - const flagOverride = this.config.scopedEnvVarTrue?.('JSON_FLAG_OVERRIDE') + // If the JSON_FLAG_OVERRIDE env var (set by Flags.json parse) is set to true, return true + // Checking for this first allows commands to define a --json flag using Flags.json() + // without setting enableJsonFlag static property. + if (this.config.scopedEnvVarTrue?.('JSON_FLAG_OVERRIDE')) return true - if (flagOverride) return true - - // if the command doesn't support json, return false + // If the command doesn't support json, return false if (!this.ctor.enableJsonFlag) return false - // if the command parameter pass through is enabled, return true if the --json flag is before the '--' separator - if (this.passThroughEnabled) { - const ptIndex = this.argv.indexOf('--') - const jsonIndex = this.argv.indexOf('--json') - return jsonIndex > -1 && (ptIndex === -1 || jsonIndex < ptIndex) - } + // If the CONTENT_TYPE env var is set to json, return true + if (this.config.scopedEnvVar?.('CONTENT_TYPE')?.toLowerCase() === 'json') return true - return this.argv.includes('--json') || this.config.scopedEnvVar?.('CONTENT_TYPE')?.toLowerCase() === 'json' + const passThroughIndex = this.argv.indexOf('--') + const jsonIndex = this.argv.indexOf('--json') + return passThroughIndex === -1 + // If '--' is not present, then check for `--json` in this.argv + ? jsonIndex > -1 + // If '--' is present, return true only the --json flag exists and is before the '--' + : jsonIndex > -1 && jsonIndex < passThroughIndex } /** diff --git a/test/command/command.test.ts b/test/command/command.test.ts index 6bb67ac66..bb429826a 100644 --- a/test/command/command.test.ts +++ b/test/command/command.test.ts @@ -610,7 +610,6 @@ describe('command', () => { .do(async () => { class CMD extends Command { static enableJsonFlag = true - static '--' = true async run() { const {flags} = await cmd.parse(CMD, ['--json']) @@ -627,16 +626,12 @@ describe('command', () => { .stdout() .do(async () => { class CMD extends Command { - // static initialization block is required whenever using ES2022 - static { - this.enableJsonFlag = true - this['--'] = true - } + static enableJsonFlag = true async run() { const {flags} = await cmd.parse(CMD, ['--', '--json']) expect(flags.json).to.equal(false, 'json flag should be false') - expect(this.passThroughEnabled).to.equal(true, 'pass through should be true') + // expect(this.passThroughEnabled).to.equal(true, 'pass through should be true') } } @@ -650,7 +645,6 @@ describe('command', () => { .do(async () => { class CMD extends Command { static enableJsonFlag = true - static '--' = true async run() { const {flags} = await cmd.parse(CMD, ['--foo', '--json']) @@ -668,12 +662,11 @@ describe('command', () => { .do(async () => { class CMD extends Command { static enableJsonFlag = true - static '--' = true async run() { const {flags} = await cmd.parse(CMD, ['--foo', '--', '--json']) expect(flags.json).to.equal(false, 'json flag should be false') - expect(this.passThroughEnabled).to.equal(true, 'pass through should be true') + // expect(this.passThroughEnabled).to.equal(true, 'pass through should be true') } } @@ -687,7 +680,6 @@ describe('command', () => { .do(async () => { class CMD extends Command { static enableJsonFlag = true - static '--' = true async run() {} } From f50ddf584d1ff5dde06398ec2169a2b9a56a5663 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 10:42:21 -0600 Subject: [PATCH 07/27] fix: correct order of flags in toCached --- src/config/config.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/config/config.ts b/src/config/config.ts index 18df7da5e..cc4422d9f 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -870,9 +870,9 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa const c = mergePrototype(cmd, cmd) const cmdFlags = { - ...(c.enableJsonFlag ? {json: json()} : {}), - ...c.flags, ...c.baseFlags, + ...c.flags, + ...(c.enableJsonFlag ? {json: json()} : {}), } as typeof c['flags'] for (const [name, flag] of Object.entries(cmdFlags || {})) { From ea315ead7b370e1726bac45a3be19ae887f127ce Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 10:43:50 -0600 Subject: [PATCH 08/27] chore: clean up --- src/index.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/index.ts b/src/index.ts index b9ce5e1bc..90d661ba2 100644 --- a/src/index.ts +++ b/src/index.ts @@ -14,11 +14,10 @@ checkCWD() export * as Args from './args' export * as Errors from './errors' - export * as Interfaces from './interfaces' +export * as Flags from './flags' export * as Parser from './parser' export * as ux from './cli-ux' -export * as Flags from './flags' export {CommandHelp, HelpBase, Help, loadHelpClass} from './help' export {Config, toCached, Plugin, tsPath} from './config' export {HelpSection, HelpSectionRenderer, HelpSectionKeyValueTable} from './help/formatter' From f6db50fe165357970314aa3cc5e839195006cd6f Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 10:47:39 -0600 Subject: [PATCH 09/27] fix: flag merge order --- src/config/config.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/config/config.ts b/src/config/config.ts index cc4422d9f..659832416 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -870,9 +870,9 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa const c = mergePrototype(cmd, cmd) const cmdFlags = { + ...(c.enableJsonFlag ? {json: json()} : {}), ...c.baseFlags, ...c.flags, - ...(c.enableJsonFlag ? {json: json()} : {}), } as typeof c['flags'] for (const [name, flag] of Object.entries(cmdFlags || {})) { From 7d80d6f1df2fd8d30bf63ba827f836497d736898 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 11:04:43 -0600 Subject: [PATCH 10/27] chore: documentation --- guides/PRE_CORE_MIGRATION.md | 45 ++++++++++++++++++++++++++++++++++++ guides/V3_MIGRATION.md | 27 +++++++++++++++++++--- src/flags.ts | 8 +++++-- 3 files changed, 75 insertions(+), 5 deletions(-) diff --git a/guides/PRE_CORE_MIGRATION.md b/guides/PRE_CORE_MIGRATION.md index 16a6fdc20..649af2cda 100644 --- a/guides/PRE_CORE_MIGRATION.md +++ b/guides/PRE_CORE_MIGRATION.md @@ -5,6 +5,7 @@ Migrating to `@oclif/core` from the deprecated oclif libraries (`@oclif/config`, - [Migrating to @oclif/core from deprecated libraries](#migrating-to-oclifcore-from-deprecated-libraries) - [Update Imports](#update-imports) + - [Update Command Args](#update-command-args) - [Update your bin scripts](#update-your-bin-scripts) - [Add `main` to your package.json](#add-main-to-your-packagejson) - [Restore `-h`, `-v`, and `version`](#restore--h--v-and-version) @@ -30,6 +31,50 @@ With this import: import {Command, Flags, Topic, Help} from '@oclif/core'; ``` +## Update Command Args + +We updated the `Command.args` to more closely resemble flags + +**Before** + +```typescript +import { Command } from '@oclif/core' + +export default MyCommand extends Command { + static args = [{name: arg1, description: 'an argument', required: true}] + + public async run(): Promise { + const {args} = await this.parse(MyCommand) // args is useless {[name: string]: any} + } +} +``` + +**After** + +```typescript +import { Command, Args } from '@oclif/core' + +export default MyCommand extends Command { + static args = { + arg1: Args.string({description: 'an argument', required: true}) + } + + public async run(): Promise { + const {args} = await this.parse(MyCommand) // args is { arg1: string } + } +} +``` + +These are the available Args: +- string +- integer +- boolean +- url +- file +- directory +- custom + + ## Update your bin scripts `@oclif/core` now supports separate bin scripts for production and development. diff --git a/guides/V3_MIGRATION.md b/guides/V3_MIGRATION.md index 9b83abc6f..cd6331347 100644 --- a/guides/V3_MIGRATION.md +++ b/guides/V3_MIGRATION.md @@ -5,6 +5,7 @@ Migrating to @oclif/core@V3 - [BREAKING CHANGES ❗](#breaking-changes-) - [Dropping node 14 and node 16 support](#dropping-node-14-and-node-16-support) - [Bin Scripts for ESM/CJS Interoperability](#bin-scripts-for-esmcjs-interoperability) + - [Dropped `ts-node` as a dependency](#dropped-ts-node-as-a-dependency) - [`Config.plugins`](#configplugins) - [Readonly properties on `Config`](#readonly-properties-on-config) - [Private methods on `Plugin`](#private-methods-on-plugin) @@ -12,9 +13,10 @@ Migrating to @oclif/core@V3 - [`handle`](#handle) - [`noCacheDefault` flag property replaces `isWritingManifest`](#nocachedefault-flag-property-replaces-iswritingmanifest) - [Features 🎉](#features-) - - [Cache Flexible taxonomy Command Permutations](#cache-flexible-taxonomy-command-permutations) + - [Performance Improvements](#performance-improvements) - [charAliases Flag Property](#charaliases-flag-property) - [Flags.option](#flagsoption) + - [Set spinner styles](#set-spinner-styles) ## BREAKING CHANGES ❗ @@ -36,6 +38,10 @@ In order to support ESM and CommonJS plugin interoperability you will need to up If you'd like to migrate your plugin to ESM, please read our guide [here](https://oclif.io/docs/esm) +### Dropped `ts-node` as a dependency + +We removed `ts-node` as a dependency to reduce the package size. By doing this, it means that linked plugin **must** have `ts-node` as a `devDependency` in order for auto-transpilation to work. + ### `Config.plugins` `Config.plugins` is now a `Map` where the keys are the plugin names and the values are the loaded `Plugin` instances. Previously it was an array of loaded `Plugin` instances. @@ -113,9 +119,12 @@ export const mySensitiveFlag = Flags.string({ ## Features 🎉 -### Cache Flexible taxonomy Command Permutations +### Performance Improvements -The command permutations for flexible taxonomy are now cached in the oclif.manifest.json allowing for quicker startup times. +- Cache command permutations for flexible taxonomy in the `oclif.manifest.json` +- Cache additional command properties (`isESM`, `relativePath`) in the `oclif.manifest.json` +- Improved accuracy in the `DEBUG=perf` output. +- Remove `ts-node` from `dependencies` to reduce the package size. ### charAliases Flag Property @@ -149,3 +158,15 @@ export default class MyCommand extends Command { } } ``` + +### Set spinner styles + +You can now configure the style of the spinner when using `ux.action.start`. See [spinners](https://github.com/oclif/core/blob/main/src/cli-ux/action/spinners.ts) for all the different options. + +```typescript +ux.action.start('starting spinner', 'spinning', {style: 'arc'}) +await ux.wait(2500) +ux.action.status = 'still going' +await ux.wait(2500) +ux.action.stop() +``` diff --git a/src/flags.ts b/src/flags.ts index f38733570..2360500b8 100644 --- a/src/flags.ts +++ b/src/flags.ts @@ -145,12 +145,16 @@ export const help = (opts: Partial> = {}): BooleanFlag> = {}): BooleanFlag => boolean({ description: 'Format output as json.', helpGroup: 'GLOBAL', ...opts, - async parse(input, cmd) { - if (input) process.env[cmd.config.scopedEnvVarKey('JSON_FLAG_OVERRIDE')] = 'true' + async parse(input, {config}) { + if (input) process.env[config.scopedEnvVarKey('JSON_FLAG_OVERRIDE')] = 'true' return input }, }) From ea1a137861e14bf07ba6aec179959600ce1ca556 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 12:08:31 -0600 Subject: [PATCH 11/27] test: use oclif/test v3 --- .github/workflows/automerge.yml | 2 +- package.json | 2 +- test/help/util.test.ts | 23 +++++++------ yarn.lock | 61 +++++++++++++++++++++++++-------- 4 files changed, 61 insertions(+), 27 deletions(-) diff --git a/.github/workflows/automerge.yml b/.github/workflows/automerge.yml index 796eafae2..41d2308a7 100644 --- a/.github/workflows/automerge.yml +++ b/.github/workflows/automerge.yml @@ -6,5 +6,5 @@ on: jobs: automerge: - uses: oclif/github-workflows/.github/workflows/automerge.yml@main + uses: salesforcecli/github-workflows/.github/workflows/automerge.yml@main secrets: inherit diff --git a/package.json b/package.json index 4841001c1..442d93e72 100644 --- a/package.json +++ b/package.json @@ -38,7 +38,7 @@ "@commitlint/config-conventional": "^12.1.4", "@oclif/plugin-help": "^5.2.8", "@oclif/plugin-plugins": "^3.3.0", - "@oclif/test": "^2.4.7", + "@oclif/test": "^3.0.0-beta.1", "@types/ansi-styles": "^3.2.1", "@types/benchmark": "^2.1.2", "@types/chai": "^4.3.4", diff --git a/test/help/util.test.ts b/test/help/util.test.ts index c25aa0b3f..cd02b70ce 100644 --- a/test/help/util.test.ts +++ b/test/help/util.test.ts @@ -4,6 +4,7 @@ import {test} from '@oclif/test' import {loadHelpClass, standardizeIDFromArgv} from '../../src/help' import configuredHelpClass from './_test-help-class' import {expect} from 'chai' +import * as util from '../../src/config/util' describe('util', () => { let config: Interfaces.Config @@ -52,7 +53,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space', () => { config.topicSeparator = ' ' const actual = standardizeIDFromArgv(['foo', 'bar', '--baz'], config) @@ -60,7 +61,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and command is misspelled', () => { config.topicSeparator = ' ' const actual = standardizeIDFromArgv(['foo', 'ba', '--baz'], config) @@ -68,7 +69,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and has args and command is misspelled', () => { config.topicSeparator = ' ' // @ts-expect-error private member @@ -81,7 +82,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and has args', () => { config.topicSeparator = ' ' // @ts-expect-error private member @@ -94,7 +95,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and has variable arguments', () => { config.topicSeparator = ' ' // @ts-expect-error private member @@ -107,7 +108,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and has variable arguments and flags', () => { config.topicSeparator = ' ' // @ts-expect-error private member @@ -120,7 +121,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return full id when topic separator is a space and does not have arguments', () => { config.topicSeparator = ' ' // @ts-expect-error private member @@ -134,7 +135,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and has arg with value', () => { config.topicSeparator = ' ' // @ts-expect-error private member @@ -144,7 +145,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and has variable args with value', () => { config.topicSeparator = ' ' // @ts-expect-error private member @@ -154,7 +155,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and has flags', () => { config.topicSeparator = ' ' // @ts-expect-error private member @@ -164,7 +165,7 @@ describe('util', () => { }) test - .stub(Config.prototype, 'collectUsableIds', () => ['foo', 'foo:bar']) + .stub(util, 'collectUsableIds', stub => stub.returns(new Set(['foo', 'foo:bar']))) .it('should return standardized id when topic separator is a space and has flags, arg, and arg with value', () => { config.topicSeparator = ' ' // @ts-expect-error private member diff --git a/yarn.lock b/yarn.lock index 16a9fccd4..2b13677d9 100644 --- a/yarn.lock +++ b/yarn.lock @@ -524,6 +524,38 @@ wordwrap "^1.0.0" wrap-ansi "^7.0.0" +"@oclif/core@^3.0.0-beta.17": + version "3.0.0-beta.17" + resolved "https://registry.yarnpkg.com/@oclif/core/-/core-3.0.0-beta.17.tgz#a6f0d22703647743aba6c9e53f0560724aba3ccb" + integrity sha512-knoDpfAVYgbVeuoUpX/es8os1GeVp//gNczB3mgNPotENXMALOZKgIv7rBbKpHa1xmpGdEf0n31WrBq3HigKwQ== + dependencies: + "@types/cli-progress" "^3.11.0" + ansi-escapes "^4.3.2" + ansi-styles "^4.3.0" + cardinal "^2.1.1" + chalk "^4.1.2" + clean-stack "^3.0.1" + cli-progress "^3.12.0" + debug "^4.3.4" + ejs "^3.1.8" + get-package-type "^0.1.0" + globby "^11.1.0" + hyperlinker "^1.0.0" + indent-string "^4.0.0" + is-wsl "^2.2.0" + js-yaml "^3.14.1" + natural-orderby "^2.0.3" + object-treeify "^1.1.33" + password-prompt "^1.1.2" + slice-ansi "^4.0.0" + string-width "^4.2.3" + strip-ansi "^6.0.1" + supports-color "^8.1.1" + supports-hyperlinks "^2.2.0" + widest-line "^3.1.0" + wordwrap "^1.0.0" + wrap-ansi "^7.0.0" + "@oclif/plugin-help@^5.2.8": version "5.2.8" resolved "https://registry.yarnpkg.com/@oclif/plugin-help/-/plugin-help-5.2.8.tgz#25821ea8588b59e5554187611ce58b8a95d4e90d" @@ -550,13 +582,13 @@ validate-npm-package-name "^5.0.0" yarn "^1.22.18" -"@oclif/test@^2.4.7": - version "2.4.7" - resolved "https://registry.yarnpkg.com/@oclif/test/-/test-2.4.7.tgz#4372270157aa6598c0d1b61349b2eff4e2003193" - integrity sha512-r18sKGNUm/VGQ8BkSF9Kn7QeMGjGMDUrLxTDPzL5ERaBF5YSi+O9CT3mKhcFdrMwGnCqPVvlAdX4U/6gtYPy1A== +"@oclif/test@^3.0.0-beta.1": + version "3.0.0-beta.1" + resolved "https://registry.yarnpkg.com/@oclif/test/-/test-3.0.0-beta.1.tgz#37beca545657eaf04413652f3b3db4f946c9804a" + integrity sha512-f/cuHO1GboGWBvBCxApkm+EtNjTLIT710XANPChdqPrQ5zF2zLicU6Kt7asH4HPsRKYl+LjJUgF1299t6333LA== dependencies: - "@oclif/core" "^2.11.10" - fancy-test "^2.0.34" + "@oclif/core" "^3.0.0-beta.17" + fancy-test "^3.0.1" "@pkgjs/parseargs@^0.11.0": version "0.11.0" @@ -2254,10 +2286,10 @@ exponential-backoff@^3.1.1: resolved "https://registry.yarnpkg.com/exponential-backoff/-/exponential-backoff-3.1.1.tgz#64ac7526fe341ab18a39016cd22c787d01e00bf6" integrity sha512-dX7e/LHVJ6W3DE1MHWi9S1EYzDESENfLrYohG2G++ovZrYOkm4Knwa0mc1cn84xJOR4KEU0WSchhLbd0UklbHw== -fancy-test@^2.0.34: - version "2.0.35" - resolved "https://registry.yarnpkg.com/fancy-test/-/fancy-test-2.0.35.tgz#18c0ccd767a7332bea186369a7e7a79a3b4582bb" - integrity sha512-XE0L7yAFOKY2jDnkBDDQ3JBD7xbBFwAFl1Z/5WNKBeVNlaEP08wuRTPE3xj2k+fnUp2CMUfD+6aiIS+4pcrjwg== +fancy-test@^3.0.0-beta.2: + version "3.0.0-beta.2" + resolved "https://registry.yarnpkg.com/fancy-test/-/fancy-test-3.0.0-beta.2.tgz#a6b6b4f3ae30200ce64ff8dd79c73fcefba0058f" + integrity sha512-bXffX78q50U/dm9E0RVesZUQ0IZPmRkMhKJdbipuVOm/WuzdoYXZnT/sr0uyzyGaxtjFcatKhZgPRDk49DlXTw== dependencies: "@types/chai" "*" "@types/lodash" "*" @@ -2266,12 +2298,13 @@ fancy-test@^2.0.34: lodash "^4.17.13" mock-stdin "^1.0.0" nock "^13.3.3" + sinon "^16.0.0" stdout-stderr "^0.1.9" -fancy-test@^3.0.0-beta.2: - version "3.0.0-beta.2" - resolved "https://registry.yarnpkg.com/fancy-test/-/fancy-test-3.0.0-beta.2.tgz#a6b6b4f3ae30200ce64ff8dd79c73fcefba0058f" - integrity sha512-bXffX78q50U/dm9E0RVesZUQ0IZPmRkMhKJdbipuVOm/WuzdoYXZnT/sr0uyzyGaxtjFcatKhZgPRDk49DlXTw== +fancy-test@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/fancy-test/-/fancy-test-3.0.1.tgz#8b19ed4ccc8b0625475eabf36fd743e1d0168abb" + integrity sha512-Ke1IFOGEBxP2dNg0X7ZYPUSwKSRr5GNn3xM/2DpHkP86riF3MFDpesXJuD1TGm7gcfwBtYpuSzuw3m704bThVg== dependencies: "@types/chai" "*" "@types/lodash" "*" From a12dbf7f6b193ef608521e3c302eeb3e77c8a1f6 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 14:02:05 -0600 Subject: [PATCH 12/27] feat: set spinner style on windows too --- src/cli-ux/action/spinner.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/cli-ux/action/spinner.ts b/src/cli-ux/action/spinner.ts index d462514b3..7ac05c9ef 100644 --- a/src/cli-ux/action/spinner.ts +++ b/src/cli-ux/action/spinner.ts @@ -64,7 +64,7 @@ export default class SpinnerAction extends ActionBase { } private getFrames(opts?: Options) { - if (opts?.style) return spinners[process.platform === 'win32' ? 'line' : opts.style].frames + if (opts?.style) return spinners[opts.style].frames return spinners[process.platform === 'win32' ? 'line' : 'dots2'].frames } From 45a777f1ba5c21135758aecc39d7b34536e033ae Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 14:18:10 -0600 Subject: [PATCH 13/27] fix: handle cmd with baseFlags but no flags --- src/command.ts | 25 ++++++++++++++++++++++--- 1 file changed, 22 insertions(+), 3 deletions(-) diff --git a/src/command.ts b/src/command.ts index 3b60f0b16..10be2c416 100644 --- a/src/command.ts +++ b/src/command.ts @@ -264,8 +264,13 @@ export abstract class Command { } protected warnIfFlagDeprecated(flags: Record): void { + const allFlags = this.aggregateFlags( + this.ctor.flags, + this.ctor.baseFlags, + this.ctor.enableJsonFlag, + ) for (const flag of Object.keys(flags)) { - const flagDef = this.ctor.flags[flag] + const flagDef = allFlags[flag] const deprecated = flagDef?.deprecated if (deprecated) { this.warn(formatFlagDeprecationWarning(flag, deprecated)) @@ -309,12 +314,15 @@ export abstract class Command { argv = this.argv, ): Promise> { if (!options) options = this.ctor as Input - const combinedFlags = {...options.baseFlags, ...options.flags} const opts = { context: this, ...options, - flags: (options.enableJsonFlag ? {...combinedFlags, json: json()} : combinedFlags) as FlagInput, + flags: this.aggregateFlags( + options.flags, + options.baseFlags, + options.enableJsonFlag, + ), } const results = await Parser.parse(argv, opts) @@ -368,6 +376,17 @@ export abstract class Command { keys.map(key => delete process.env[key]) } + + private aggregateFlags( + flags: FlagInput | undefined, + baseFlags: FlagInput | undefined, + enableJsonFlag: boolean | undefined, + ): FlagInput { + const combinedFlags = {...baseFlags, ...flags} + return (enableJsonFlag + ? {...combinedFlags, json: json()} + : combinedFlags) as FlagInput + } } export namespace Command { From bd8c6c8fe2e6c3715053989a0aa2ee288f0176f2 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 15:12:32 -0600 Subject: [PATCH 14/27] fix: some circular deps --- src/cli-ux/flush.ts | 5 +- src/cli-ux/styled/json.ts | 8 +- src/config/config.ts | 177 +------------------------------------- src/config/index.ts | 1 - src/config/plugin.ts | 2 +- src/errors/handle.ts | 2 +- src/help/index.ts | 2 +- src/index.ts | 3 +- src/to-cached.ts | 177 ++++++++++++++++++++++++++++++++++++++ 9 files changed, 192 insertions(+), 185 deletions(-) create mode 100644 src/to-cached.ts diff --git a/src/cli-ux/flush.ts b/src/cli-ux/flush.ts index c6a0c152d..33abd784f 100644 --- a/src/cli-ux/flush.ts +++ b/src/cli-ux/flush.ts @@ -1,4 +1,5 @@ -import {Errors, stdout} from '..' +import {error} from '../errors' +import {stdout} from './stream' function timeout(p: Promise, ms: number) { function wait(ms: number, unref = false) { @@ -8,7 +9,7 @@ function timeout(p: Promise, ms: number) { }) } - return Promise.race([p, wait(ms, true).then(() => Errors.error('timed out'))]) + return Promise.race([p, wait(ms, true).then(() => error('timed out'))]) } async function _flush() { diff --git a/src/cli-ux/styled/json.ts b/src/cli-ux/styled/json.ts index c7db532d0..cbdc4892a 100644 --- a/src/cli-ux/styled/json.ts +++ b/src/cli-ux/styled/json.ts @@ -1,15 +1,17 @@ import chalk from 'chalk' +import {format} from 'node:util' +import {stdout} from '../stream' -import {ux} from '../../index' +const info = (output: string) => stdout.write(format(output) + '\n') export default function styledJSON(obj: unknown): void { const json = JSON.stringify(obj, null, 2) if (!chalk.level) { - ux.info(json) + info(json) return } const cardinal = require('cardinal') const theme = require('cardinal/themes/jq') - ux.info(cardinal.highlight(json, {json: true, theme})) + info(cardinal.highlight(json, {json: true, theme})) } diff --git a/src/config/config.ts b/src/config/config.ts index 659832416..2ec58b318 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -1,20 +1,19 @@ import * as ejs from 'ejs' import {ArchTypes, Config as IConfig, LoadOptions, PlatformTypes, VersionDetails} from '../interfaces/config' -import {Arg, OptionFlag} from '../interfaces/parser' import {CLIError, error, exit, warn} from '../errors' import {Debug, collectUsableIds, getCommandIdPermutations} from './util' import {Hook, Hooks, PJSON, Topic} from '../interfaces' import {Plugin as IPlugin, Options} from '../interfaces/plugin' import {URL, fileURLToPath} from 'node:url' import {arch, userInfo as osUserInfo, release, tmpdir, type} from 'node:os' -import {compact, ensureArgObject, getHomeDir, getPlatform, isProd, pickBy, requireJson} from '../util' +import {compact, getHomeDir, getPlatform, isProd, requireJson} from '../util' import {join, sep} from 'node:path' import {Command} from '../command' import {Performance} from '../performance' import PluginLoader from './plugin-loader' +import WSL from 'is-wsl' import {format} from 'node:util' import {getHelpFlagAdditions} from '../help' -import {json} from '../flags' import {loadWithData} from '../module-loader' import {settings} from '../settings' import {stdout} from '../cli-ux/stream' @@ -30,8 +29,6 @@ function channelFromVersion(version: string) { return (m && m[1]) || 'stable' } -const WSL = require('is-wsl') - function isConfig(o: any): o is Config { return o && Boolean(o._base) } @@ -812,173 +809,3 @@ export class Config implements IConfig { } } -// when no manifest exists, the default is calculated. This may throw, so we need to catch it -const defaultFlagToCached = async (flag: OptionFlag, respectNoCacheDefault: boolean) => { - if (respectNoCacheDefault && flag.noCacheDefault) return - // Prefer the defaultHelp function (returns a friendly string for complex types) - if (typeof flag.defaultHelp === 'function') { - try { - return await flag.defaultHelp({options: flag, flags: {}}) - } catch { - return - } - } - - // if not specified, try the default function - if (typeof flag.default === 'function') { - try { - return await flag.default({options: flag, flags: {}}) - } catch {} - } else { - return flag.default - } -} - -const defaultArgToCached = async (arg: Arg, respectNoCacheDefault: boolean): Promise => { - if (respectNoCacheDefault && arg.noCacheDefault) return - // Prefer the defaultHelp function (returns a friendly string for complex types) - if (typeof arg.defaultHelp === 'function') { - try { - return await arg.defaultHelp({options: arg, flags: {}}) - } catch { - return - } - } - - // if not specified, try the default function - if (typeof arg.default === 'function') { - try { - return await arg.default({options: arg, flags: {}}) - } catch {} - } else { - return arg.default - } -} - -export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { - const flags = {} as {[k: string]: Command.Flag.Cached} - - // In order to collect static properties up the inheritance chain, we need to recursively - // access the prototypes until there's nothing left. This allows us to combine baseFlags - // and flags as well as add in the json flag if enableJsonFlag is enabled. - const mergePrototype = (result: Command.Class, cmd: Command.Class): Command.Class => { - const proto = Object.getPrototypeOf(cmd) - const filteredProto = pickBy(proto, v => v !== undefined) as Command.Class - return Object.keys(proto).length > 0 ? mergePrototype({...filteredProto, ...result} as Command.Class, proto) : result - } - - const c = mergePrototype(cmd, cmd) - - const cmdFlags = { - ...(c.enableJsonFlag ? {json: json()} : {}), - ...c.baseFlags, - ...c.flags, - } as typeof c['flags'] - - for (const [name, flag] of Object.entries(cmdFlags || {})) { - if (flag.type === 'boolean') { - flags[name] = { - name, - type: flag.type, - char: flag.char, - summary: flag.summary, - description: flag.description, - hidden: flag.hidden, - required: flag.required, - helpLabel: flag.helpLabel, - helpGroup: flag.helpGroup, - allowNo: flag.allowNo, - dependsOn: flag.dependsOn, - relationships: flag.relationships, - exclusive: flag.exclusive, - deprecated: flag.deprecated, - deprecateAliases: c.deprecateAliases, - aliases: flag.aliases, - charAliases: flag.charAliases, - delimiter: flag.delimiter, - noCacheDefault: flag.noCacheDefault, - } - } else { - flags[name] = { - name, - type: flag.type, - char: flag.char, - summary: flag.summary, - description: flag.description, - hidden: flag.hidden, - required: flag.required, - helpLabel: flag.helpLabel, - helpValue: flag.helpValue, - helpGroup: flag.helpGroup, - multiple: flag.multiple, - options: flag.options, - dependsOn: flag.dependsOn, - relationships: flag.relationships, - exclusive: flag.exclusive, - default: await defaultFlagToCached(flag, respectNoCacheDefault), - deprecated: flag.deprecated, - deprecateAliases: c.deprecateAliases, - aliases: flag.aliases, - charAliases: flag.charAliases, - delimiter: flag.delimiter, - noCacheDefault: flag.noCacheDefault, - } - // a command-level placeholder in the manifest so that oclif knows it should regenerate the command during help-time - if (typeof flag.defaultHelp === 'function') { - c.hasDynamicHelp = true - } - } - } - - const args = {} as {[k: string]: Command.Arg.Cached} - for (const [name, arg] of Object.entries(ensureArgObject(c.args))) { - args[name] = { - name, - description: arg.description, - required: arg.required, - options: arg.options, - default: await defaultArgToCached(arg, respectNoCacheDefault), - hidden: arg.hidden, - noCacheDefault: arg.noCacheDefault, - } - } - - const stdProperties = { - id: c.id, - summary: c.summary, - description: c.description, - strict: c.strict, - usage: c.usage, - pluginName: plugin && plugin.name, - pluginAlias: plugin && plugin.alias, - pluginType: plugin && plugin.type, - hidden: c.hidden, - state: c.state, - aliases: c.aliases || [], - examples: c.examples || (c as any).example, - deprecationOptions: c.deprecationOptions, - deprecateAliases: c.deprecateAliases, - flags, - args, - } - - // do not include these properties in manifest - const ignoreCommandProperties = [ - 'plugin', - '_flags', - '_enableJsonFlag', - '_globalFlags', - '_baseFlags', - 'baseFlags', - '_--', - '_base', - ] - const stdKeys = Object.keys(stdProperties) - const keysToAdd = Object.keys(c).filter(property => ![...stdKeys, ...ignoreCommandProperties].includes(property)) - const additionalProperties: Record = {} - for (const key of keysToAdd) { - additionalProperties[key] = (c as any)[key] - } - - return {...stdProperties, ...additionalProperties} -} diff --git a/src/config/index.ts b/src/config/index.ts index a79d4c3bf..748162de7 100644 --- a/src/config/index.ts +++ b/src/config/index.ts @@ -1,5 +1,4 @@ export {Config} from './config' -export {toCached} from './config' export {Plugin} from './plugin' export {tsPath} from './ts-node' diff --git a/src/config/plugin.ts b/src/config/plugin.ts index 8f623db75..8df0822e5 100644 --- a/src/config/plugin.ts +++ b/src/config/plugin.ts @@ -17,7 +17,7 @@ import {Performance} from '../performance' import {Topic} from '../interfaces/topic' import {inspect} from 'node:util' import {sync} from 'globby' -import {toCached} from './config' +import {toCached} from '../to-cached' import {tsPath} from './ts-node' const _pjson = requireJson(__dirname, '..', '..', 'package.json') diff --git a/src/errors/handle.ts b/src/errors/handle.ts index 138e73d8e..ab1ff2c2c 100644 --- a/src/errors/handle.ts +++ b/src/errors/handle.ts @@ -2,7 +2,7 @@ /* eslint-disable unicorn/no-process-exit */ import {OclifError, PrettyPrintableError} from '../interfaces' import {CLIError} from './errors/cli' -import {ExitError} from '.' +import {ExitError} from './errors/exit' import clean from 'clean-stack' import {config} from './config' import prettyPrint from './errors/pretty-print' diff --git a/src/help/index.ts b/src/help/index.ts index 871d6c2f5..f2b819998 100644 --- a/src/help/index.ts +++ b/src/help/index.ts @@ -8,7 +8,7 @@ import RootHelp from './root' import {error} from '../errors' import {format} from 'node:util' import {stdout} from '../cli-ux/stream' -import {toCached} from '../config/config' +import {toCached} from '../to-cached' import stripAnsi = require('strip-ansi') diff --git a/src/index.ts b/src/index.ts index 90d661ba2..b938f0706 100644 --- a/src/index.ts +++ b/src/index.ts @@ -19,7 +19,8 @@ export * as Flags from './flags' export * as Parser from './parser' export * as ux from './cli-ux' export {CommandHelp, HelpBase, Help, loadHelpClass} from './help' -export {Config, toCached, Plugin, tsPath} from './config' +export {Config, Plugin, tsPath} from './config' +export {toCached} from './to-cached' export {HelpSection, HelpSectionRenderer, HelpSectionKeyValueTable} from './help/formatter' export {Settings, settings} from './settings' export {stdout, stderr} from './cli-ux/stream' diff --git a/src/to-cached.ts b/src/to-cached.ts new file mode 100644 index 000000000..bb1469bd2 --- /dev/null +++ b/src/to-cached.ts @@ -0,0 +1,177 @@ +/* eslint-disable no-await-in-loop */ +import {Arg, OptionFlag} from './interfaces/parser' +import {ensureArgObject, pickBy} from './util' +import {Command} from './command' +import {Plugin as IPlugin} from './interfaces/plugin' +import {json} from './flags' + +// when no manifest exists, the default is calculated. This may throw, so we need to catch it +const defaultFlagToCached = async (flag: OptionFlag, respectNoCacheDefault: boolean) => { + if (respectNoCacheDefault && flag.noCacheDefault) return + // Prefer the defaultHelp function (returns a friendly string for complex types) + if (typeof flag.defaultHelp === 'function') { + try { + return await flag.defaultHelp({options: flag, flags: {}}) + } catch { + return + } + } + + // if not specified, try the default function + if (typeof flag.default === 'function') { + try { + return await flag.default({options: flag, flags: {}}) + } catch {} + } else { + return flag.default + } +} + +const defaultArgToCached = async (arg: Arg, respectNoCacheDefault: boolean): Promise => { + if (respectNoCacheDefault && arg.noCacheDefault) return + // Prefer the defaultHelp function (returns a friendly string for complex types) + if (typeof arg.defaultHelp === 'function') { + try { + return await arg.defaultHelp({options: arg, flags: {}}) + } catch { + return + } + } + + // if not specified, try the default function + if (typeof arg.default === 'function') { + try { + return await arg.default({options: arg, flags: {}}) + } catch {} + } else { + return arg.default + } +} + +export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { + const flags = {} as {[k: string]: Command.Flag.Cached} + + // In order to collect static properties up the inheritance chain, we need to recursively + // access the prototypes until there's nothing left. This allows us to combine baseFlags + // and flags as well as add in the json flag if enableJsonFlag is enabled. + const mergePrototype = (result: Command.Class, cmd: Command.Class): Command.Class => { + const proto = Object.getPrototypeOf(cmd) + const filteredProto = pickBy(proto, v => v !== undefined) as Command.Class + return Object.keys(proto).length > 0 ? mergePrototype({...filteredProto, ...result} as Command.Class, proto) : result + } + + const c = mergePrototype(cmd, cmd) + + const cmdFlags = { + ...(c.enableJsonFlag ? {json: json()} : {}), + ...c.baseFlags, + ...c.flags, + } as typeof c['flags'] + + for (const [name, flag] of Object.entries(cmdFlags || {})) { + if (flag.type === 'boolean') { + flags[name] = { + name, + type: flag.type, + char: flag.char, + summary: flag.summary, + description: flag.description, + hidden: flag.hidden, + required: flag.required, + helpLabel: flag.helpLabel, + helpGroup: flag.helpGroup, + allowNo: flag.allowNo, + dependsOn: flag.dependsOn, + relationships: flag.relationships, + exclusive: flag.exclusive, + deprecated: flag.deprecated, + deprecateAliases: c.deprecateAliases, + aliases: flag.aliases, + charAliases: flag.charAliases, + delimiter: flag.delimiter, + noCacheDefault: flag.noCacheDefault, + } + } else { + flags[name] = { + name, + type: flag.type, + char: flag.char, + summary: flag.summary, + description: flag.description, + hidden: flag.hidden, + required: flag.required, + helpLabel: flag.helpLabel, + helpValue: flag.helpValue, + helpGroup: flag.helpGroup, + multiple: flag.multiple, + options: flag.options, + dependsOn: flag.dependsOn, + relationships: flag.relationships, + exclusive: flag.exclusive, + default: await defaultFlagToCached(flag, respectNoCacheDefault), + deprecated: flag.deprecated, + deprecateAliases: c.deprecateAliases, + aliases: flag.aliases, + charAliases: flag.charAliases, + delimiter: flag.delimiter, + noCacheDefault: flag.noCacheDefault, + } + // a command-level placeholder in the manifest so that oclif knows it should regenerate the command during help-time + if (typeof flag.defaultHelp === 'function') { + c.hasDynamicHelp = true + } + } + } + + const args = {} as {[k: string]: Command.Arg.Cached} + for (const [name, arg] of Object.entries(ensureArgObject(c.args))) { + args[name] = { + name, + description: arg.description, + required: arg.required, + options: arg.options, + default: await defaultArgToCached(arg, respectNoCacheDefault), + hidden: arg.hidden, + noCacheDefault: arg.noCacheDefault, + } + } + + const stdProperties = { + id: c.id, + summary: c.summary, + description: c.description, + strict: c.strict, + usage: c.usage, + pluginName: plugin && plugin.name, + pluginAlias: plugin && plugin.alias, + pluginType: plugin && plugin.type, + hidden: c.hidden, + state: c.state, + aliases: c.aliases || [], + examples: c.examples || (c as any).example, + deprecationOptions: c.deprecationOptions, + deprecateAliases: c.deprecateAliases, + flags, + args, + } + + // do not include these properties in manifest + const ignoreCommandProperties = [ + 'plugin', + '_flags', + '_enableJsonFlag', + '_globalFlags', + '_baseFlags', + 'baseFlags', + '_--', + '_base', + ] + const stdKeys = Object.keys(stdProperties) + const keysToAdd = Object.keys(c).filter(property => ![...stdKeys, ...ignoreCommandProperties].includes(property)) + const additionalProperties: Record = {} + for (const key of keysToAdd) { + additionalProperties[key] = (c as any)[key] + } + + return {...stdProperties, ...additionalProperties} +} From fc59ae29467fa68aa7fbfd45925ee911bd3960d5 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 15:18:37 -0600 Subject: [PATCH 15/27] fix: circular deps in help --- src/config/config.ts | 2 +- src/help/index.ts | 30 +++++++++++++++++++++++++++--- src/help/util.ts | 28 +--------------------------- 3 files changed, 29 insertions(+), 31 deletions(-) diff --git a/src/config/config.ts b/src/config/config.ts index 2ec58b318..41795375d 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -13,7 +13,7 @@ import {Performance} from '../performance' import PluginLoader from './plugin-loader' import WSL from 'is-wsl' import {format} from 'node:util' -import {getHelpFlagAdditions} from '../help' +import {getHelpFlagAdditions} from '../help/util' import {loadWithData} from '../module-loader' import {settings} from '../settings' import {stdout} from '../cli-ux/stream' diff --git a/src/help/index.ts b/src/help/index.ts index f2b819998..6bd01d435 100644 --- a/src/help/index.ts +++ b/src/help/index.ts @@ -7,13 +7,13 @@ import {HelpFormatter} from './formatter' import RootHelp from './root' import {error} from '../errors' import {format} from 'node:util' +import {load} from '../module-loader' import {stdout} from '../cli-ux/stream' +import stripAnsi from 'strip-ansi' import {toCached} from '../to-cached' -import stripAnsi = require('strip-ansi') - export {CommandHelp} from './command' -export {standardizeIDFromArgv, loadHelpClass, getHelpFlagAdditions, normalizeArgv} from './util' +export {standardizeIDFromArgv, getHelpFlagAdditions, normalizeArgv} from './util' function getHelpSubject(args: string[], config: Interfaces.Config): string | undefined { // for each help flag that starts with '--' create a new flag with same name sans '--' @@ -327,3 +327,27 @@ export class Help extends HelpBase { stdout.write(format.apply(this, args) + '\n') } } + +interface HelpBaseDerived { + new(config: Interfaces.Config, opts?: Partial): HelpBase; +} + +function extractClass(exported: any): HelpBaseDerived { + return exported && exported.default ? exported.default : exported +} + +export async function loadHelpClass(config: Interfaces.Config): Promise { + const {pjson} = config + const configuredClass = pjson && pjson.oclif && pjson.oclif.helpClass + + if (configuredClass) { + try { + const exported = await load(config, configuredClass) as HelpBaseDerived + return extractClass(exported) as HelpBaseDerived + } catch (error: any) { + throw new Error(`Unable to load configured help class "${configuredClass}", failed with message:\n${error.message}`) + } + } + + return Help +} diff --git a/src/help/util.ts b/src/help/util.ts index 278c26050..71c07573a 100644 --- a/src/help/util.ts +++ b/src/help/util.ts @@ -1,32 +1,6 @@ import * as ejs from 'ejs' -import {Deprecation, HelpOptions, Config as IConfig} from '../interfaces' -import {Help, HelpBase} from '.' +import {Deprecation, Config as IConfig} from '../interfaces' import {collectUsableIds} from '../config/util' -import {load} from '../module-loader' - -interface HelpBaseDerived { - new(config: IConfig, opts?: Partial): HelpBase; -} - -function extractClass(exported: any): HelpBaseDerived { - return exported && exported.default ? exported.default : exported -} - -export async function loadHelpClass(config: IConfig): Promise { - const {pjson} = config - const configuredClass = pjson && pjson.oclif && pjson.oclif.helpClass - - if (configuredClass) { - try { - const exported = await load(config, configuredClass) as HelpBaseDerived - return extractClass(exported) as HelpBaseDerived - } catch (error: any) { - throw new Error(`Unable to load configured help class "${configuredClass}", failed with message:\n${error.message}`) - } - } - - return Help -} export function template(context: any): (t: string) => string { function render(t: string): string { From cdff4b6b3d85e7459244b56c2c27af3f152ba53c Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Tue, 26 Sep 2023 17:12:02 -0600 Subject: [PATCH 16/27] fix: ts-node and config circular deps --- src/config/plugin-loader.ts | 2 +- src/config/plugin.ts | 3 +++ src/config/ts-node.ts | 12 +++++++++--- src/interfaces/plugin.ts | 6 ++++++ src/module-loader.ts | 2 +- test/config/config.flexible.test.ts | 2 ++ test/config/config.test.ts | 2 ++ 7 files changed, 24 insertions(+), 5 deletions(-) diff --git a/src/config/plugin-loader.ts b/src/config/plugin-loader.ts index 7965a08bb..ed008a708 100644 --- a/src/config/plugin-loader.ts +++ b/src/config/plugin-loader.ts @@ -43,7 +43,7 @@ export default class PluginLoader { rootPlugin = plugins.find(p => p.root === this.options.root) ?? plugins[0] } else { const marker = Performance.mark('plugin.load#root') - rootPlugin = new Plugin.Plugin({root: this.options.root}) + rootPlugin = new Plugin.Plugin({root: this.options.root, isRoot: true}) await rootPlugin.load() marker?.addDetails({ hasManifest: rootPlugin.hasManifest ?? false, diff --git a/src/config/plugin.ts b/src/config/plugin.ts index 8df0822e5..1b4941b33 100644 --- a/src/config/plugin.ts +++ b/src/config/plugin.ts @@ -141,6 +141,8 @@ export class Plugin implements IPlugin { hasManifest = false + isRoot = false + private _commandsDir!: string | undefined private flexibleTaxonomy!: boolean @@ -155,6 +157,7 @@ export class Plugin implements IPlugin { public async load(): Promise { this.type = this.options.type || 'core' this.tag = this.options.tag + this.isRoot = this.options.isRoot ?? false if (this.options.parent) this.parent = this.options.parent as Plugin // Linked plugins already have a root so there's no need to search for it. // However there could be child plugins nested inside the linked plugin, in which diff --git a/src/config/ts-node.ts b/src/config/ts-node.ts index 67093f80e..4a0e5f82a 100644 --- a/src/config/ts-node.ts +++ b/src/config/ts-node.ts @@ -2,7 +2,6 @@ import * as TSNode from 'ts-node' import {Plugin, TSConfig} from '../interfaces' import {isProd, readJsonSync} from '../util' import {join, relative as pathRelative} from 'node:path' -import {Config} from './config' import {Debug} from './util' import {existsSync} from 'node:fs' import {memoizedWarn} from '../errors' @@ -13,6 +12,11 @@ const debug = Debug('ts-node') export const TS_CONFIGS: Record = {} const REGISTERED = new Set() +/** + * Cache the root plugin so that we can reference it later when determining if + * we should skip ts-node registration for an ESM plugin. + */ +let ROOT_PLUGIN: Plugin | undefined function loadTSConfig(root: string): TSConfig | undefined { if (TS_CONFIGS[root]) return TS_CONFIGS[root] @@ -111,6 +115,8 @@ function registerTSNode(root: string): TSConfig | undefined { export function tsPath(root: string, orig: string, plugin: Plugin): string export function tsPath(root: string, orig: string | undefined, plugin?: Plugin): string | undefined export function tsPath(root: string, orig: string | undefined, plugin?: Plugin): string | undefined { + if (plugin?.isRoot) ROOT_PLUGIN = plugin + if (!orig) return orig orig = orig.startsWith(root) ? orig : join(root, orig) @@ -134,8 +140,8 @@ export function tsPath(root: string, orig: string | undefined, plugin?: Plugin): * We still register ts-node for ESM plugins when NODE_ENV is "test" or "development" and root plugin is also ESM. * In other words, this allows plugins to be auto-transpiled when developing locally using `bin/dev.js`. */ - if ((isProduction || Config.rootPlugin?.moduleType === 'commonjs') && plugin?.moduleType === 'module') { - debug(`Skipping ts-node registration for ${root} because it's an ESM module (NODE_ENV: ${process.env.NODE_ENV}, root plugin module type: ${Config.rootPlugin?.moduleType})))`) + if ((isProduction || ROOT_PLUGIN?.moduleType === 'commonjs') && plugin?.moduleType === 'module') { + debug(`Skipping ts-node registration for ${root} because it's an ESM module (NODE_ENV: ${process.env.NODE_ENV}, root plugin module type: ${ROOT_PLUGIN?.moduleType})))`) if (plugin.type === 'link') memoizedWarn(`${plugin.name} is a linked ESM module and cannot be auto-transpiled. Existing compiled source will be used instead.`) diff --git a/src/interfaces/plugin.ts b/src/interfaces/plugin.ts index 50207bcba..772fbc10c 100644 --- a/src/interfaces/plugin.ts +++ b/src/interfaces/plugin.ts @@ -13,6 +13,7 @@ export interface PluginOptions { parent?: Plugin; children?: Plugin[]; flexibleTaxonomy?: boolean; + isRoot?: boolean; } export interface Options extends PluginOptions { @@ -73,6 +74,11 @@ export interface Plugin { */ valid: boolean; + /** + * True if the plugin is the root plugin. + */ + isRoot: boolean; + commands: Command.Loadable[]; hooks: { [k: string]: string[] }; readonly commandIDs: string[]; diff --git a/src/module-loader.ts b/src/module-loader.ts index c3105d027..ce715ed8a 100644 --- a/src/module-loader.ts +++ b/src/module-loader.ts @@ -4,7 +4,7 @@ import {extname, join, sep} from 'node:path' import {Command} from './command' import {ModuleLoadError} from './errors' import {pathToFileURL} from 'node:url' -import {tsPath} from './config' +import {tsPath} from './config/ts-node' const getPackageType = require('get-package-type') diff --git a/test/config/config.flexible.test.ts b/test/config/config.flexible.test.ts index 829b00463..6b451da95 100644 --- a/test/config/config.flexible.test.ts +++ b/test/config/config.flexible.test.ts @@ -101,6 +101,7 @@ describe('Config with flexible taxonomy', () => { tag: 'tag', moduleType: 'commonjs', hasManifest: false, + isRoot: false, } const pluginB: IPlugin = { @@ -121,6 +122,7 @@ describe('Config with flexible taxonomy', () => { tag: 'tag', moduleType: 'commonjs', hasManifest: false, + isRoot: false, } const plugins = new Map().set(pluginA.name, pluginA).set(pluginB.name, pluginB) diff --git a/test/config/config.test.ts b/test/config/config.test.ts index cd5aa9971..1b821d793 100644 --- a/test/config/config.test.ts +++ b/test/config/config.test.ts @@ -282,6 +282,7 @@ describe('Config', () => { tag: 'tag', moduleType: 'commonjs', hasManifest: false, + isRoot: false, } const pluginB: IPlugin = { @@ -302,6 +303,7 @@ describe('Config', () => { tag: 'tag', moduleType: 'commonjs', hasManifest: false, + isRoot: false, } const plugins = new Map().set(pluginA.name, pluginA).set(pluginB.name, pluginB) let test = fancy From 8195c03d50fbb9c605c4fda43c7e502a65be1e85 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 09:50:18 -0600 Subject: [PATCH 17/27] fix: toCached circular dep in help --- src/default-flag-to-cached.ts | 23 +++++++++++++++++++++++ src/help/index.ts | 12 +++++++++--- src/index.ts | 1 - src/to-cached.ts | 25 ++----------------------- test/command/command.test.ts | 3 ++- test/help/help-test-utils.ts | 3 ++- 6 files changed, 38 insertions(+), 29 deletions(-) create mode 100644 src/default-flag-to-cached.ts diff --git a/src/default-flag-to-cached.ts b/src/default-flag-to-cached.ts new file mode 100644 index 000000000..285102925 --- /dev/null +++ b/src/default-flag-to-cached.ts @@ -0,0 +1,23 @@ +import {OptionFlag} from './interfaces/parser' + +// when no manifest exists, the default is calculated. This may throw, so we need to catch it +export const defaultFlagToCached = async (flag: OptionFlag, respectNoCacheDefault: boolean) => { + if (respectNoCacheDefault && flag.noCacheDefault) return + // Prefer the defaultHelp function (returns a friendly string for complex types) + if (typeof flag.defaultHelp === 'function') { + try { + return await flag.defaultHelp({options: flag, flags: {}}) + } catch { + return + } + } + + // if not specified, try the default function + if (typeof flag.default === 'function') { + try { + return await flag.default({options: flag, flags: {}}) + } catch {} + } else { + return flag.default + } +} diff --git a/src/help/index.ts b/src/help/index.ts index 6bd01d435..efebb87cc 100644 --- a/src/help/index.ts +++ b/src/help/index.ts @@ -5,12 +5,12 @@ import {Command} from '../command' import {CommandHelp} from './command' import {HelpFormatter} from './formatter' import RootHelp from './root' +import {defaultFlagToCached} from '../default-flag-to-cached' import {error} from '../errors' import {format} from 'node:util' import {load} from '../module-loader' import {stdout} from '../cli-ux/stream' import stripAnsi from 'strip-ansi' -import {toCached} from '../to-cached' export {CommandHelp} from './command' export {standardizeIDFromArgv, getHelpFlagAdditions, normalizeArgv} from './util' @@ -108,8 +108,14 @@ export class Help extends HelpBase { const command = this.config.findCommand(subject) if (command) { if (command.hasDynamicHelp && command.pluginType !== 'jit') { - const dynamicCommand = await toCached(await command.load()) - await this.showCommandHelp(dynamicCommand) + const loaded = await command.load() + for (const flag of Object.values(loaded.flags)) { + if (flag.type === 'boolean') continue + // eslint-disable-next-line no-await-in-loop + flag.default = await defaultFlagToCached(flag, false) + } + + await this.showCommandHelp(command) } else { await this.showCommandHelp(command) } diff --git a/src/index.ts b/src/index.ts index b938f0706..6093911a8 100644 --- a/src/index.ts +++ b/src/index.ts @@ -20,7 +20,6 @@ export * as Parser from './parser' export * as ux from './cli-ux' export {CommandHelp, HelpBase, Help, loadHelpClass} from './help' export {Config, Plugin, tsPath} from './config' -export {toCached} from './to-cached' export {HelpSection, HelpSectionRenderer, HelpSectionKeyValueTable} from './help/formatter' export {Settings, settings} from './settings' export {stdout, stderr} from './cli-ux/stream' diff --git a/src/to-cached.ts b/src/to-cached.ts index bb1469bd2..9920661df 100644 --- a/src/to-cached.ts +++ b/src/to-cached.ts @@ -1,32 +1,11 @@ /* eslint-disable no-await-in-loop */ -import {Arg, OptionFlag} from './interfaces/parser' import {ensureArgObject, pickBy} from './util' +import {Arg} from './interfaces/parser' import {Command} from './command' import {Plugin as IPlugin} from './interfaces/plugin' +import {defaultFlagToCached} from './default-flag-to-cached' import {json} from './flags' -// when no manifest exists, the default is calculated. This may throw, so we need to catch it -const defaultFlagToCached = async (flag: OptionFlag, respectNoCacheDefault: boolean) => { - if (respectNoCacheDefault && flag.noCacheDefault) return - // Prefer the defaultHelp function (returns a friendly string for complex types) - if (typeof flag.defaultHelp === 'function') { - try { - return await flag.defaultHelp({options: flag, flags: {}}) - } catch { - return - } - } - - // if not specified, try the default function - if (typeof flag.default === 'function') { - try { - return await flag.default({options: flag, flags: {}}) - } catch {} - } else { - return flag.default - } -} - const defaultArgToCached = async (arg: Arg, respectNoCacheDefault: boolean): Promise => { if (respectNoCacheDefault && arg.noCacheDefault) return // Prefer the defaultHelp function (returns a friendly string for complex types) diff --git a/test/command/command.test.ts b/test/command/command.test.ts index bb429826a..10dcc965f 100644 --- a/test/command/command.test.ts +++ b/test/command/command.test.ts @@ -1,6 +1,7 @@ import {expect, fancy} from 'fancy-test' // import path = require('path') -import {Args, Command as Base, Flags, toCached} from '../../src' +import {Args, Command as Base, Flags} from '../../src' +import {toCached} from '../../src/to-cached' // import {TestHelpClassConfig} from './helpers/test-help-in-src/src/test-help-plugin' // const pjson = require('../package.json') diff --git a/test/help/help-test-utils.ts b/test/help/help-test-utils.ts index cf94558b2..d2fbecb6d 100644 --- a/test/help/help-test-utils.ts +++ b/test/help/help-test-utils.ts @@ -2,8 +2,9 @@ import {Command} from '../../src/command' import stripAnsi = require('strip-ansi') -import {Interfaces, toCached} from '../../src' +import {Interfaces} from '../../src' import {CommandHelp, Help} from '../../src/help' +import {toCached} from '../../src/to-cached' export class TestCommandHelp extends CommandHelp { protected sections() { From 5a19e538c73495c239888f33cbb98982c34efff5 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 10:02:23 -0600 Subject: [PATCH 18/27] chore: organize utils --- src/config/plugin.ts | 10 ++++------ src/config/util.ts | 12 ------------ src/help/index.ts | 2 +- src/help/root.ts | 2 +- src/{ => util}/default-flag-to-cached.ts | 2 +- src/{util.ts => util/index.ts} | 12 ++++++++++-- src/{ => util}/to-cached.ts | 10 +++++----- test/command/command.test.ts | 2 +- test/help/help-test-utils.ts | 2 +- 9 files changed, 24 insertions(+), 30 deletions(-) rename src/{ => util}/default-flag-to-cached.ts (93%) rename src/{util.ts => util/index.ts} (93%) rename src/{ => util}/to-cached.ts (95%) diff --git a/src/config/plugin.ts b/src/config/plugin.ts index 1b4941b33..d20ddf34e 100644 --- a/src/config/plugin.ts +++ b/src/config/plugin.ts @@ -1,13 +1,11 @@ import {CLIError, error} from '../errors' import { Debug, - flatMap, getCommandIdPermutations, - mapValues, resolvePackage, } from './util' import {Plugin as IPlugin, PluginOptions} from '../interfaces/plugin' -import {compact, exists, isProd, readJson, requireJson} from '../util' +import {compact, exists, isProd, mapValues, readJson, requireJson} from '../util' import {dirname, join, parse, relative, sep} from 'node:path' import {loadWithData, loadWithDataFromManifest} from '../module-loader' import {Command} from '../command' @@ -17,7 +15,7 @@ import {Performance} from '../performance' import {Topic} from '../interfaces/topic' import {inspect} from 'node:util' import {sync} from 'globby' -import {toCached} from '../to-cached' +import {toCached} from '../util/to-cached' import {tsPath} from './ts-node' const _pjson = requireJson(__dirname, '..', '..', 'package.json') @@ -26,10 +24,10 @@ function topicsToArray(input: any, base?: string): Topic[] { if (!input) return [] base = base ? `${base}:` : '' if (Array.isArray(input)) { - return [...input, ...flatMap(input, t => topicsToArray(t.subtopics, `${base}${t.name}`))] + return [...input, input.flatMap(t => topicsToArray(t.subtopics, `${base}${t.name}`))] } - return flatMap(Object.keys(input), k => { + return Object.keys(input).flatMap(k => { input[k].name = k return [{...input[k], name: `${base}${k}`}, ...topicsToArray(input[k].subtopics, `${base}${input[k].name}`)] }) diff --git a/src/config/util.ts b/src/config/util.ts index 182b336a5..482ce4554 100644 --- a/src/config/util.ts +++ b/src/config/util.ts @@ -1,17 +1,5 @@ const debug = require('debug') -export function flatMap(arr: T[], fn: (i: T) => U[]): U[] { - return arr.reduce((arr, i) => [...arr, ...fn(i)], [] as U[]) -} - -export function mapValues, TResult>(obj: {[P in keyof T]: T[P]}, fn: (i: T[keyof T], k: keyof T) => TResult): {[P in keyof T]: TResult} { - return Object.entries(obj) - .reduce((o, [k, v]) => { - o[k] = fn(v as any, k as any) - return o - }, {} as any) -} - export function resolvePackage(id: string, paths: { paths: string[] }): string { return require.resolve(id, paths) } diff --git a/src/help/index.ts b/src/help/index.ts index efebb87cc..069ab4006 100644 --- a/src/help/index.ts +++ b/src/help/index.ts @@ -5,7 +5,7 @@ import {Command} from '../command' import {CommandHelp} from './command' import {HelpFormatter} from './formatter' import RootHelp from './root' -import {defaultFlagToCached} from '../default-flag-to-cached' +import {defaultFlagToCached} from '../util/default-flag-to-cached' import {error} from '../errors' import {format} from 'node:util' import {load} from '../module-loader' diff --git a/src/help/root.ts b/src/help/root.ts index 6aaa587f2..1d1ed0d69 100644 --- a/src/help/root.ts +++ b/src/help/root.ts @@ -1,7 +1,7 @@ import * as Interfaces from '../interfaces' import {HelpFormatter} from './formatter' import {compact} from '../util' -import stripAnsi = require('strip-ansi') +import stripAnsi from 'strip-ansi' export default class RootHelp extends HelpFormatter { constructor(public config: Interfaces.Config, public opts: Interfaces.HelpOptions) { diff --git a/src/default-flag-to-cached.ts b/src/util/default-flag-to-cached.ts similarity index 93% rename from src/default-flag-to-cached.ts rename to src/util/default-flag-to-cached.ts index 285102925..fe3beeeb2 100644 --- a/src/default-flag-to-cached.ts +++ b/src/util/default-flag-to-cached.ts @@ -1,4 +1,4 @@ -import {OptionFlag} from './interfaces/parser' +import {OptionFlag} from '../interfaces/parser' // when no manifest exists, the default is calculated. This may throw, so we need to catch it export const defaultFlagToCached = async (flag: OptionFlag, respectNoCacheDefault: boolean) => { diff --git a/src/util.ts b/src/util/index.ts similarity index 93% rename from src/util.ts rename to src/util/index.ts index a5c0eab5b..f35f3ebf8 100644 --- a/src/util.ts +++ b/src/util/index.ts @@ -1,8 +1,8 @@ import {access, stat} from 'node:fs/promises' import {homedir, platform} from 'node:os' import {readFile, readFileSync} from 'node:fs' -import {ArgInput} from './interfaces/parser' -import {Command} from './command' +import {ArgInput} from '../interfaces/parser' +import {Command} from '../command' import {join} from 'node:path' const debug = require('debug') @@ -189,3 +189,11 @@ export function readJsonSync(path: string, parse = true): T | strin const contents = readFileSync(path, 'utf8') return parse ? JSON.parse(contents) as T : contents } + +export function mapValues, TResult>(obj: {[P in keyof T]: T[P]}, fn: (i: T[keyof T], k: keyof T) => TResult): {[P in keyof T]: TResult} { + return Object.entries(obj) + .reduce((o, [k, v]) => { + o[k] = fn(v as any, k as any) + return o + }, {} as any) +} diff --git a/src/to-cached.ts b/src/util/to-cached.ts similarity index 95% rename from src/to-cached.ts rename to src/util/to-cached.ts index 9920661df..e2bdcf15d 100644 --- a/src/to-cached.ts +++ b/src/util/to-cached.ts @@ -1,10 +1,10 @@ /* eslint-disable no-await-in-loop */ -import {ensureArgObject, pickBy} from './util' -import {Arg} from './interfaces/parser' -import {Command} from './command' -import {Plugin as IPlugin} from './interfaces/plugin' +import {ensureArgObject, pickBy} from './index' +import {Arg} from '../interfaces/parser' +import {Command} from '../command' +import {Plugin as IPlugin} from '../interfaces/plugin' import {defaultFlagToCached} from './default-flag-to-cached' -import {json} from './flags' +import {json} from '../flags' const defaultArgToCached = async (arg: Arg, respectNoCacheDefault: boolean): Promise => { if (respectNoCacheDefault && arg.noCacheDefault) return diff --git a/test/command/command.test.ts b/test/command/command.test.ts index 10dcc965f..e6f75e754 100644 --- a/test/command/command.test.ts +++ b/test/command/command.test.ts @@ -1,7 +1,7 @@ import {expect, fancy} from 'fancy-test' // import path = require('path') import {Args, Command as Base, Flags} from '../../src' -import {toCached} from '../../src/to-cached' +import {toCached} from '../../src/util/to-cached' // import {TestHelpClassConfig} from './helpers/test-help-in-src/src/test-help-plugin' // const pjson = require('../package.json') diff --git a/test/help/help-test-utils.ts b/test/help/help-test-utils.ts index d2fbecb6d..0cf545125 100644 --- a/test/help/help-test-utils.ts +++ b/test/help/help-test-utils.ts @@ -4,7 +4,7 @@ import stripAnsi = require('strip-ansi') import {Interfaces} from '../../src' import {CommandHelp, Help} from '../../src/help' -import {toCached} from '../../src/to-cached' +import {toCached} from '../../src/util/to-cached' export class TestCommandHelp extends CommandHelp { protected sections() { From 38d6d3d7a3056f30245ae10d2a6c3a499c78e27e Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 10:04:52 -0600 Subject: [PATCH 19/27] test: enforce no circular deps --- package.json | 6 +- yarn.lock | 738 +++++++++++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 725 insertions(+), 19 deletions(-) diff --git a/package.json b/package.json index 442d93e72..b7bfeb941 100644 --- a/package.json +++ b/package.json @@ -67,6 +67,7 @@ "fancy-test": "^3.0.0-beta.2", "globby": "^11.1.0", "husky": "6", + "madge": "^6.1.0", "mocha": "^10.2.0", "nock": "^13.3.0", "shx": "^0.3.4", @@ -105,9 +106,10 @@ "commitlint": "commitlint", "compile": "tsc", "lint": "eslint . --ext .ts", - "posttest": "yarn lint", + "posttest": "yarn lint && yarn test:circular-deps", "prepack": "yarn run build", - "pretest": "yarn build --noEmit && tsc -p test --noEmit --skipLibCheck", + "pretest": "yarn build && tsc -p test --skipLibCheck", + "test:circular-deps": "madge lib/ -c", "test:e2e": "mocha --forbid-only \"test/**/*.e2e.ts\" --parallel --timeout 1200000", "test:esm-cjs": "cross-env DEBUG=e2e:* ts-node test/integration/esm-cjs.ts", "test:perf": "ts-node test/perf/parser.perf.ts", diff --git a/yarn.lock b/yarn.lock index 2b13677d9..22e59cfe4 100644 --- a/yarn.lock +++ b/yarn.lock @@ -33,6 +33,11 @@ chalk "^2.0.0" js-tokens "^4.0.0" +"@babel/parser@^7.0.0", "@babel/parser@^7.21.4": + version "7.23.0" + resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.23.0.tgz#da950e622420bf96ca0d0f2909cdddac3acd8719" + integrity sha512-vvPKKdMemU85V9WE/l5wZEmImpCtLqbnTvqDS2U1fJ96KrxoW7KrXhNsNCblQlg8Ck4b85yxdTyelsMUgFUXiw== + "@colors/colors@1.5.0": version "1.5.0" resolved "https://registry.yarnpkg.com/@colors/colors/-/colors-1.5.0.tgz#bb504579c1cae923e6576a4f5da43d25f97bdbd9" @@ -182,6 +187,14 @@ dependencies: "@jridgewell/trace-mapping" "0.3.9" +"@dependents/detective-less@^3.0.1": + version "3.0.2" + resolved "https://registry.yarnpkg.com/@dependents/detective-less/-/detective-less-3.0.2.tgz#c6e46997010fe03a5dc98351a7e99a46d34f5832" + integrity sha512-1YUvQ+e0eeTWAHoN8Uz2x2U37jZs6IGutiIE5LXId7cxfUGhtZjzxE06FdUiuiRrW+UE0vNCdSNPH2lY4dQCOQ== + dependencies: + gonzales-pe "^4.3.0" + node-source-walk "^5.0.1" + "@eslint-community/eslint-utils@^4.2.0", "@eslint-community/eslint-utils@^4.4.0": version "4.4.0" resolved "https://registry.yarnpkg.com/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz#a23514e8fb9af1269d5f7788aa556798d61c6b59" @@ -952,6 +965,16 @@ debug "^4.3.4" ts-api-utils "^1.0.1" +"@typescript-eslint/types@4.33.0": + version "4.33.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-4.33.0.tgz#a1e59036a3b53ae8430ceebf2a919dc7f9af6d72" + integrity sha512-zKp7CjQzLQImXEpLt2BUw1tvOMPfNoTAfb8l51evhYbOEEzdWyQNmHWWGPR6hwKJDAi+1VXSBmnhL9kyVTTOuQ== + +"@typescript-eslint/types@5.62.0": + version "5.62.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-5.62.0.tgz#258607e60effa309f067608931c3df6fed41fd2f" + integrity sha512-87NVngcbVXUahrRTqIK27gD2t5Cu1yuCXxbLcFtCzZGlfyVWWh8mLHkoxzjsB6DDNnvdL+fW8MiwPEJyGJQDgQ== + "@typescript-eslint/types@6.7.2": version "6.7.2" resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-6.7.2.tgz#75a615a6dbeca09cafd102fe7f465da1d8a3c066" @@ -970,6 +993,32 @@ semver "^7.5.4" ts-api-utils "^1.0.1" +"@typescript-eslint/typescript-estree@^4.33.0": + version "4.33.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-4.33.0.tgz#0dfb51c2908f68c5c08d82aefeaf166a17c24609" + integrity sha512-rkWRY1MPFzjwnEVHsxGemDzqqddw2QbTJlICPD9p9I9LfsO8fdmfQPOX3uKfUaGRDFJbfrtm/sXhVXN4E+bzCA== + dependencies: + "@typescript-eslint/types" "4.33.0" + "@typescript-eslint/visitor-keys" "4.33.0" + debug "^4.3.1" + globby "^11.0.3" + is-glob "^4.0.1" + semver "^7.3.5" + tsutils "^3.21.0" + +"@typescript-eslint/typescript-estree@^5.55.0": + version "5.62.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-5.62.0.tgz#7d17794b77fabcac615d6a48fb143330d962eb9b" + integrity sha512-CmcQ6uY7b9y694lKdRB8FEel7JbU/40iSAPomu++SjLMntB+2Leay2LO6i8VnJk58MtE9/nQSFIH6jpyRWyYzA== + dependencies: + "@typescript-eslint/types" "5.62.0" + "@typescript-eslint/visitor-keys" "5.62.0" + debug "^4.3.4" + globby "^11.1.0" + is-glob "^4.0.3" + semver "^7.3.7" + tsutils "^3.21.0" + "@typescript-eslint/utils@6.7.2": version "6.7.2" resolved "https://registry.yarnpkg.com/@typescript-eslint/utils/-/utils-6.7.2.tgz#b9ef0da6f04932167a9222cb4ac59cb187165ebf" @@ -983,6 +1032,22 @@ "@typescript-eslint/typescript-estree" "6.7.2" semver "^7.5.4" +"@typescript-eslint/visitor-keys@4.33.0": + version "4.33.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-4.33.0.tgz#2a22f77a41604289b7a186586e9ec48ca92ef1dd" + integrity sha512-uqi/2aSz9g2ftcHWf8uLPJA70rUv6yuMW5Bohw+bwcuzaxQIHaKFZCKGoGXIrc9vkTJ3+0txM73K0Hq3d5wgIg== + dependencies: + "@typescript-eslint/types" "4.33.0" + eslint-visitor-keys "^2.0.0" + +"@typescript-eslint/visitor-keys@5.62.0": + version "5.62.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-5.62.0.tgz#2174011917ce582875954ffe2f6912d5931e353e" + integrity sha512-07ny+LHRzQXepkGg6w0mFY41fVUNBrL2Roj/++7V1txKugfjm/Ci/qSND03r2RhlJhJYMcTn9AhhSSqQp0Ysyw== + dependencies: + "@typescript-eslint/types" "5.62.0" + eslint-visitor-keys "^3.3.0" + "@typescript-eslint/visitor-keys@6.7.2": version "6.7.2" resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-6.7.2.tgz#4cb2bd786f1f459731b0ad1584c9f73e1c7a4d5c" @@ -1129,6 +1194,11 @@ ansicolors@~0.3.2: resolved "https://registry.yarnpkg.com/ansicolors/-/ansicolors-0.3.2.tgz#665597de86a9ffe3aa9bfbe6cae5c6ea426b4979" integrity sha1-ZlWX3oap/+Oqm/vmyuXG6kJrSXk= +any-promise@^1.1.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/any-promise/-/any-promise-1.3.0.tgz#abc6afeedcea52e809cdc0376aed3ce39635d17f" + integrity sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A== + anymatch@~3.1.2: version "3.1.3" resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.3.tgz#790c58b19ba1720a84205b57c618d5ad8524973e" @@ -1137,6 +1207,11 @@ anymatch@~3.1.2: normalize-path "^3.0.0" picomatch "^2.0.4" +app-module-path@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/app-module-path/-/app-module-path-2.2.0.tgz#641aa55dfb7d6a6f0a8141c4b9c0aa50b6c24dd5" + integrity sha512-gkco+qxENJV+8vFcDiiFhuoSvRXb2a/QPqpSoWhVz829VNJfOTnELbBmPmNKFxf3xdNnw4DWCkzkDaavcX/1YQ== + "aproba@^1.0.3 || ^2.0.0", aproba@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/aproba/-/aproba-2.0.0.tgz#52520b8ae5b569215b354efc0caa3fe1e45a8adc" @@ -1263,6 +1338,21 @@ assertion-error@^1.1.0: resolved "https://registry.yarnpkg.com/assertion-error/-/assertion-error-1.1.0.tgz#e60b6b0e8f301bd97e5375215bda406c85118c0b" integrity sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw== +ast-module-types@^2.7.1: + version "2.7.1" + resolved "https://registry.yarnpkg.com/ast-module-types/-/ast-module-types-2.7.1.tgz#3f7989ef8dfa1fdb82dfe0ab02bdfc7c77a57dd3" + integrity sha512-Rnnx/4Dus6fn7fTqdeLEAn5vUll5w7/vts0RN608yFa6si/rDOUonlIIiwugHBFWjylHjxm9owoSZn71KwG4gw== + +ast-module-types@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/ast-module-types/-/ast-module-types-3.0.0.tgz#9a6d8a80f438b6b8fe4995699d700297f398bf81" + integrity sha512-CMxMCOCS+4D+DkOQfuZf+vLrSEmY/7xtORwdxs4wtcC1wVgvk2MqFFTwQCFhvWsI4KPU9lcWXPI8DgRiz+xetQ== + +ast-module-types@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/ast-module-types/-/ast-module-types-4.0.0.tgz#17e1cadd5b5b108e7295b0cf0cff21ccc226b639" + integrity sha512-Kd0o8r6CDazJGCRzs8Ivpn0xj19oNKrULhoJFzhGjRsLpekF2zyZs9Ukz+JvZhWD6smszfepakTFhAaYpsI12g== + astral-regex@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz#483143c567aeed4785759c0865786dc77d7d2e31" @@ -1316,6 +1406,15 @@ binary-extensions@^2.0.0, binary-extensions@^2.2.0: resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.2.0.tgz#75f502eeaf9ffde42fc98829645be4ea76bd9e2d" integrity sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA== +bl@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/bl/-/bl-4.1.0.tgz#451535264182bec2fbbc83a62ab98cf11d9f7b3a" + integrity sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w== + dependencies: + buffer "^5.5.0" + inherits "^2.0.4" + readable-stream "^3.4.0" + brace-expansion@^1.1.7: version "1.1.11" resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" @@ -1343,6 +1442,14 @@ browser-stdout@1.3.1: resolved "https://registry.yarnpkg.com/browser-stdout/-/browser-stdout-1.3.1.tgz#baa559ee14ced73452229bad7326467c61fabd60" integrity sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw== +buffer@^5.5.0: + version "5.7.1" + resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0" + integrity sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ== + dependencies: + base64-js "^1.3.1" + ieee754 "^1.1.13" + buffer@^6.0.3: version "6.0.3" resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6" @@ -1450,7 +1557,7 @@ chalk@^2.0.0: escape-string-regexp "^1.0.5" supports-color "^5.3.0" -chalk@^4.0.0, chalk@^4.0.2, chalk@^4.1.0, chalk@^4.1.2: +chalk@^4.0.0, chalk@^4.0.2, chalk@^4.1.0, chalk@^4.1.1, chalk@^4.1.2: version "4.1.2" resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== @@ -1534,6 +1641,13 @@ cli-columns@^4.0.0: string-width "^4.2.3" strip-ansi "^6.0.1" +cli-cursor@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-3.1.0.tgz#264305a7ae490d1d03bf0c9ba7c925d1753af307" + integrity sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw== + dependencies: + restore-cursor "^3.1.0" + cli-progress@^3.12.0: version "3.12.0" resolved "https://registry.yarnpkg.com/cli-progress/-/cli-progress-3.12.0.tgz#807ee14b66bcc086258e444ad0f19e7d42577942" @@ -1541,6 +1655,11 @@ cli-progress@^3.12.0: dependencies: string-width "^4.2.3" +cli-spinners@^2.5.0: + version "2.9.1" + resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-2.9.1.tgz#9c0b9dad69a6d47cbb4333c14319b060ed395a35" + integrity sha512-jHgecW0pxkonBJdrKsqxgRX9AcG+u/5k0Q7WPDfi8AogLAdwxEkyYYNWwZ5GvVFoFx2uiY1eNcSK00fh+1+FyQ== + cli-table3@^0.6.3: version "0.6.3" resolved "https://registry.yarnpkg.com/cli-table3/-/cli-table3-0.6.3.tgz#61ab765aac156b52f222954ffc607a6f01dbeeb2" @@ -1588,7 +1707,7 @@ color-name@1.1.3: resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25" integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU= -color-name@~1.1.4: +color-name@^1.1.4, color-name@~1.1.4: version "1.1.4" resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== @@ -1606,6 +1725,21 @@ columnify@^1.6.0: strip-ansi "^6.0.1" wcwidth "^1.0.0" +commander@^2.16.0, commander@^2.20.3, commander@^2.8.1: + version "2.20.3" + resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33" + integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ== + +commander@^7.2.0: + version "7.2.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-7.2.0.tgz#a36cb57d0b501ce108e4d20559a150a391d97ab7" + integrity sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw== + +commander@^9.5.0: + version "9.5.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-9.5.0.tgz#bc08d1eb5cedf7ccb797a96199d41c7bc3e60d30" + integrity sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ== + commitlint@^12.1.4: version "12.1.4" resolved "https://registry.yarnpkg.com/commitlint/-/commitlint-12.1.4.tgz#afc185957afdd07d2ebb6c78b5956d407fc09ad0" @@ -1619,6 +1753,11 @@ common-ancestor-path@^1.0.1: resolved "https://registry.yarnpkg.com/common-ancestor-path/-/common-ancestor-path-1.0.1.tgz#4f7d2d1394d91b7abdf51871c62f71eadb0182a7" integrity sha512-L3sHRo1pXXEqX8VU28kfgUY+YGsk09hPqZiZmLacNib6XNTCM8ubYeT7ryXQw8asB1sKgcU5lkB7ONug08aB8w== +commondir@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/commondir/-/commondir-1.0.1.tgz#ddd800da0c66127393cca5950ea968a3aaf1253b" + integrity sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg== + compare-func@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/compare-func/-/compare-func-2.0.0.tgz#fb65e75edbddfd2e568554e8b5b05fff7a51fcb3" @@ -1730,7 +1869,7 @@ dargs@^7.0.0: resolved "https://registry.yarnpkg.com/dargs/-/dargs-7.0.0.tgz#04015c41de0bcb69ec84050f3d9be0caf8d6d5cc" integrity sha512-2iy1EkLdlBzQGvbweYRFxmFath8+K7+AKB0TlhHWkNuH+TmovaMH/Wp7V7R4u7f4SnX3OgLsU9t1NI9ioDnUpg== -debug@4, debug@4.3.4, debug@^4.1.0, debug@^4.1.1, debug@^4.3.2, debug@^4.3.3, debug@^4.3.4: +debug@4, debug@4.3.4, debug@^4.0.0, debug@^4.1.0, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2, debug@^4.3.3, debug@^4.3.4: version "4.3.4" resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== @@ -1769,6 +1908,11 @@ deep-eql@^4.1.2: dependencies: type-detect "^4.0.0" +deep-extend@^0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.6.0.tgz#c4fa7c95404a17a9c3e8ca7e1537312b736330ac" + integrity sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA== + deep-is@^0.1.3: version "0.1.3" resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.3.tgz#b369d6fb5dbc13eecf524f91b070feedc357cf34" @@ -1804,6 +1948,162 @@ delegates@^1.0.0: resolved "https://registry.yarnpkg.com/delegates/-/delegates-1.0.0.tgz#84c6e159b81904fdca59a0ef44cd870d31250f9a" integrity sha512-bd2L678uiWATM6m5Z1VzNCErI3jiGzt6HGY8OVICs40JQq/HALfbyNJmp0UDakEY4pMMaN0Ly5om/B1VI/+xfQ== +dependency-tree@^9.0.0: + version "9.0.0" + resolved "https://registry.yarnpkg.com/dependency-tree/-/dependency-tree-9.0.0.tgz#9288dd6daf35f6510c1ea30d9894b75369aa50a2" + integrity sha512-osYHZJ1fBSon3lNLw70amAXsQ+RGzXsPvk9HbBgTLbp/bQBmpH5mOmsUvqXU+YEWVU0ZLewsmzOET/8jWswjDQ== + dependencies: + commander "^2.20.3" + debug "^4.3.1" + filing-cabinet "^3.0.1" + precinct "^9.0.0" + typescript "^4.0.0" + +detective-amd@^3.1.0: + version "3.1.2" + resolved "https://registry.yarnpkg.com/detective-amd/-/detective-amd-3.1.2.tgz#bf55eb5291c218b76d6224a3d07932ef13a9a357" + integrity sha512-jffU26dyqJ37JHR/o44La6CxtrDf3Rt9tvd2IbImJYxWKTMdBjctp37qoZ6ZcY80RHg+kzWz4bXn39e4P7cctQ== + dependencies: + ast-module-types "^3.0.0" + escodegen "^2.0.0" + get-amd-module-type "^3.0.0" + node-source-walk "^4.2.0" + +detective-amd@^4.0.1, detective-amd@^4.1.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/detective-amd/-/detective-amd-4.2.0.tgz#21c43465669f06cf894eef047a27e6e72ba6bc13" + integrity sha512-RbuEJHz78A8nW7CklkqTzd8lDCN42En53dgEIsya0DilpkwslamSZDasLg8dJyxbw46OxhSQeY+C2btdSkCvQQ== + dependencies: + ast-module-types "^4.0.0" + escodegen "^2.0.0" + get-amd-module-type "^4.1.0" + node-source-walk "^5.0.1" + +detective-cjs@^3.1.1: + version "3.1.3" + resolved "https://registry.yarnpkg.com/detective-cjs/-/detective-cjs-3.1.3.tgz#50e107d67b37f459b0ec02966ceb7e20a73f268b" + integrity sha512-ljs7P0Yj9MK64B7G0eNl0ThWSYjhAaSYy+fQcpzaKalYl/UoQBOzOeLCSFEY1qEBhziZ3w7l46KG/nH+s+L7BQ== + dependencies: + ast-module-types "^3.0.0" + node-source-walk "^4.0.0" + +detective-cjs@^4.0.0, detective-cjs@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/detective-cjs/-/detective-cjs-4.1.0.tgz#56b1558ca0910933c7fc47c740b957f0280ff302" + integrity sha512-QxzMwt5MfPLwS7mG30zvnmOvHLx5vyVvjsAV6gQOyuMoBR5G1DhS1eJZ4P10AlH+HSnk93mTcrg3l39+24XCtg== + dependencies: + ast-module-types "^4.0.0" + node-source-walk "^5.0.1" + +detective-es6@^2.2.1: + version "2.2.2" + resolved "https://registry.yarnpkg.com/detective-es6/-/detective-es6-2.2.2.tgz#ee5f880981d9fecae9a694007029a2f6f26d8d28" + integrity sha512-eZUKCUsbHm8xoeoCM0z6JFwvDfJ5Ww5HANo+jPR7AzkFpW9Mun3t/TqIF2jjeWa2TFbAiGaWESykf2OQp3oeMw== + dependencies: + node-source-walk "^4.0.0" + +detective-es6@^3.0.0, detective-es6@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/detective-es6/-/detective-es6-3.0.1.tgz#53a15fbb2f298c4a106d9fe7427da8a57162dde6" + integrity sha512-evPeYIEdK1jK3Oji5p0hX4sPV/1vK+o4ihcWZkMQE6voypSW/cIBiynOLxQk5KOOQbdP8oOAsYqouMTYO5l1sw== + dependencies: + node-source-walk "^5.0.0" + +detective-less@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/detective-less/-/detective-less-1.0.2.tgz#a68af9ca5f69d74b7d0aa190218b211d83b4f7e3" + integrity sha512-Rps1xDkEEBSq3kLdsdnHZL1x2S4NGDcbrjmd4q+PykK5aJwDdP5MBgrJw1Xo+kyUHuv3JEzPqxr+Dj9ryeDRTA== + dependencies: + debug "^4.0.0" + gonzales-pe "^4.2.3" + node-source-walk "^4.0.0" + +detective-postcss@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/detective-postcss/-/detective-postcss-4.0.0.tgz#24e69b465e5fefe7a6afd05f7e894e34595dbf51" + integrity sha512-Fwc/g9VcrowODIAeKRWZfVA/EufxYL7XfuqJQFroBKGikKX83d2G7NFw6kDlSYGG3LNQIyVa+eWv1mqre+v4+A== + dependencies: + debug "^4.1.1" + is-url "^1.2.4" + postcss "^8.1.7" + postcss-values-parser "^2.0.1" + +detective-postcss@^6.1.0, detective-postcss@^6.1.1: + version "6.1.3" + resolved "https://registry.yarnpkg.com/detective-postcss/-/detective-postcss-6.1.3.tgz#51a2d4419327ad85d0af071c7054c79fafca7e73" + integrity sha512-7BRVvE5pPEvk2ukUWNQ+H2XOq43xENWbH0LcdCE14mwgTBEAMoAx+Fc1rdp76SmyZ4Sp48HlV7VedUnP6GA1Tw== + dependencies: + is-url "^1.2.4" + postcss "^8.4.23" + postcss-values-parser "^6.0.2" + +detective-sass@^3.0.1: + version "3.0.2" + resolved "https://registry.yarnpkg.com/detective-sass/-/detective-sass-3.0.2.tgz#e0f35aac79a4d2f6409c284d95b8f7ecd5973afd" + integrity sha512-DNVYbaSlmti/eztFGSfBw4nZvwsTaVXEQ4NsT/uFckxhJrNRFUh24d76KzoCC3aarvpZP9m8sC2L1XbLej4F7g== + dependencies: + gonzales-pe "^4.3.0" + node-source-walk "^4.0.0" + +detective-sass@^4.0.1, detective-sass@^4.1.1: + version "4.1.3" + resolved "https://registry.yarnpkg.com/detective-sass/-/detective-sass-4.1.3.tgz#6cdcc27ae8a90d15704e0ba83683048f77f10b75" + integrity sha512-xGRbwGaGte57gvEqM8B9GDiURY3El/H49vA6g9wFkxq9zalmTlTAuqWu+BsH0iwonGPruLt55tZZDEZqPc6lag== + dependencies: + gonzales-pe "^4.3.0" + node-source-walk "^5.0.1" + +detective-scss@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/detective-scss/-/detective-scss-2.0.2.tgz#7d2a642616d44bf677963484fa8754d9558b8235" + integrity sha512-hDWnWh/l0tht/7JQltumpVea/inmkBaanJUcXRB9kEEXVwVUMuZd6z7eusQ6GcBFrfifu3pX/XPyD7StjbAiBg== + dependencies: + gonzales-pe "^4.3.0" + node-source-walk "^4.0.0" + +detective-scss@^3.0.0, detective-scss@^3.0.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/detective-scss/-/detective-scss-3.1.1.tgz#b49f05cadfb0837b04e23baba292581b7c7f65e1" + integrity sha512-FWkfru1jZBhUeuBsOeGKXKAVDrzYFSQFK2o2tuG/nCCFQ0U/EcXC157MNAcR5mmj+mCeneZzlkBOFJTesDjrww== + dependencies: + gonzales-pe "^4.3.0" + node-source-walk "^5.0.1" + +detective-stylus@^1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/detective-stylus/-/detective-stylus-1.0.3.tgz#20a702936c9fd7d4203fd7a903314b5dd43ac713" + integrity sha512-4/bfIU5kqjwugymoxLXXLltzQNeQfxGoLm2eIaqtnkWxqbhap9puDVpJPVDx96hnptdERzS5Cy6p9N8/08A69Q== + +detective-stylus@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/detective-stylus/-/detective-stylus-2.0.1.tgz#d528dfa7ef3c4eb2fbc9a7249d54906ec4e05d09" + integrity sha512-/Tvs1pWLg8eYwwV6kZQY5IslGaYqc/GACxjcaGudiNtN5nKCH6o2WnJK3j0gA3huCnoQcbv8X7oz/c1lnvE3zQ== + +detective-stylus@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/detective-stylus/-/detective-stylus-3.0.0.tgz#c869795a7d6df7043ab6aee8b1a6f3dd66764232" + integrity sha512-1xYTzbrduExqMYmte7Qk99IRA3Aa6oV7PYzd+3yDcQXkmENvyGF/arripri6lxRDdNYEb4fZFuHtNRAXbz3iAA== + +detective-typescript@^7.0.0: + version "7.0.2" + resolved "https://registry.yarnpkg.com/detective-typescript/-/detective-typescript-7.0.2.tgz#c6e00b4c28764741ef719662250e6b014a5f3c8e" + integrity sha512-unqovnhxzvkCz3m1/W4QW4qGsvXCU06aU2BAm8tkza+xLnp9SOFnob2QsTxUv5PdnQKfDvWcv9YeOeFckWejwA== + dependencies: + "@typescript-eslint/typescript-estree" "^4.33.0" + ast-module-types "^2.7.1" + node-source-walk "^4.2.0" + typescript "^3.9.10" + +detective-typescript@^9.0.0, detective-typescript@^9.1.1: + version "9.1.1" + resolved "https://registry.yarnpkg.com/detective-typescript/-/detective-typescript-9.1.1.tgz#b99c0122cbb35b39de2c5f58447f1e93ac28c6d5" + integrity sha512-Uc1yVutTF0RRm1YJ3g//i1Cn2vx1kwHj15cnzQP6ff5koNzQ0idc1zAC73ryaWEulA0ElRXFTq6wOqe8vUQ3MA== + dependencies: + "@typescript-eslint/typescript-estree" "^5.55.0" + ast-module-types "^4.0.0" + node-source-walk "^5.0.1" + typescript "^4.9.5" + diff-sequences@^29.6.3: version "29.6.3" resolved "https://registry.yarnpkg.com/diff-sequences/-/diff-sequences-29.6.3.tgz#4deaf894d11407c51efc8418012f9e70b84ea921" @@ -1881,7 +2181,7 @@ encoding@^0.1.13: dependencies: iconv-lite "^0.6.2" -enhanced-resolve@^5.12.0: +enhanced-resolve@^5.12.0, enhanced-resolve@^5.8.3: version "5.15.0" resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-5.15.0.tgz#1af946c7d93603eb88e9896cee4904dc012e9c35" integrity sha512-LXYT42KJ7lpIKECr2mAXIaMldcNCh/7E0KBKOu4KSfkHmP+mZmSs+8V5gBAqisWBy0OO4W5Oyys0GO1Y8KtdKg== @@ -1996,6 +2296,17 @@ escape-string-regexp@^1.0.5: resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" integrity sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ= +escodegen@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/escodegen/-/escodegen-2.1.0.tgz#ba93bbb7a43986d29d6041f99f5262da773e2e17" + integrity sha512-2NlIDTwUWJN0mRPQOdtQBzbUHvdGY2P1VXSyU83Q3xKxM7WHX2Ql8dKq782Q9TgQUNOLEzEYu9bzLNj1q88I5w== + dependencies: + esprima "^4.0.1" + estraverse "^5.2.0" + esutils "^2.0.2" + optionalDependencies: + source-map "~0.6.1" + eslint-config-oclif-typescript@^2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/eslint-config-oclif-typescript/-/eslint-config-oclif-typescript-2.0.1.tgz#bdaca00f53ee27ff6930673082a00a03d6cf8dd1" @@ -2242,7 +2553,7 @@ espree@^9.6.0, espree@^9.6.1: acorn-jsx "^5.3.2" eslint-visitor-keys "^3.4.1" -esprima@^4.0.0, esprima@~4.0.0: +esprima@^4.0.0, esprima@^4.0.1, esprima@~4.0.0: version "4.0.1" resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71" integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== @@ -2379,6 +2690,25 @@ filelist@^1.0.1: dependencies: minimatch "^3.0.4" +filing-cabinet@^3.0.1: + version "3.3.1" + resolved "https://registry.yarnpkg.com/filing-cabinet/-/filing-cabinet-3.3.1.tgz#45d87bb273a0e0a7dd6ac6bac9111059186e2e9c" + integrity sha512-renEK4Hh6DUl9Vl22Y3cxBq1yh8oNvbAdXnhih0wVpmea+uyKjC9K4QeRjUaybIiIewdzfum+Fg15ZqJ/GyCaA== + dependencies: + app-module-path "^2.2.0" + commander "^2.20.3" + debug "^4.3.3" + enhanced-resolve "^5.8.3" + is-relative-path "^1.0.2" + module-definition "^3.3.1" + module-lookup-amd "^7.0.1" + resolve "^1.21.0" + resolve-dependency-path "^2.0.0" + sass-lookup "^3.0.0" + stylus-lookup "^3.0.1" + tsconfig-paths "^3.10.1" + typescript "^3.9.7" + fill-range@^7.0.1: version "7.0.1" resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40" @@ -2420,6 +2750,11 @@ flatted@^3.1.0: resolved "https://registry.yarnpkg.com/flatted/-/flatted-3.2.2.tgz#64bfed5cb68fe3ca78b3eb214ad97b63bedce561" integrity sha512-JaTY/wtrcSyvXJl4IMFHPKyFur1sE9AUqc0QnhOaJ0CxHtAoIV8pYDzeEfAaNEtGkOfq4gr3LBFmdXW5mOQFnA== +flatten@^1.0.2: + version "1.0.3" + resolved "https://registry.yarnpkg.com/flatten/-/flatten-1.0.3.tgz#c1283ac9f27b368abc1e36d1ff7b04501a30356b" + integrity sha512-dVsPA/UwQ8+2uoFe5GHtiBMu48dWLTdsuEd7CKGlZlD78r1TTWBvDuFaFGKCo/ZfEr95Uk56vZoX86OsHkUeIg== + for-each@^0.3.3: version "0.3.3" resolved "https://registry.yarnpkg.com/for-each/-/for-each-0.3.3.tgz#69b447e88a0a5d32c3e7084f3f1710034b21376e" @@ -2517,6 +2852,22 @@ gauge@^5.0.0: strip-ansi "^6.0.1" wide-align "^1.1.5" +get-amd-module-type@^3.0.0: + version "3.0.2" + resolved "https://registry.yarnpkg.com/get-amd-module-type/-/get-amd-module-type-3.0.2.tgz#46550cee2b8e1fa4c3f2c8a5753c36990aa49ab0" + integrity sha512-PcuKwB8ouJnKuAPn6Hk3UtdfKoUV3zXRqVEvj8XGIXqjWfgd1j7QGdXy5Z9OdQfzVt1Sk29HVe/P+X74ccOuqw== + dependencies: + ast-module-types "^3.0.0" + node-source-walk "^4.2.2" + +get-amd-module-type@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/get-amd-module-type/-/get-amd-module-type-4.1.0.tgz#af1396d02cd935cb6fafdc4a5282395db3422db6" + integrity sha512-0e/eK6vTGCnSfQ6eYs3wtH05KotJYIP7ZIZEueP/KlA+0dIAEs8bYFvOd/U56w1vfjhJqBagUxVMyy9Tr/cViQ== + dependencies: + ast-module-types "^4.0.0" + node-source-walk "^5.0.1" + get-caller-file@^2.0.5: version "2.0.5" resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e" @@ -2537,6 +2888,11 @@ get-intrinsic@^1.0.2, get-intrinsic@^1.1.1, get-intrinsic@^1.1.3, get-intrinsic@ has-proto "^1.0.1" has-symbols "^1.0.3" +get-own-enumerable-property-symbols@^3.0.0: + version "3.0.2" + resolved "https://registry.yarnpkg.com/get-own-enumerable-property-symbols/-/get-own-enumerable-property-symbols-3.0.2.tgz#b5fde77f22cbe35f390b4e089922c50bce6ef664" + integrity sha512-I0UBV/XOz1XkIJHEUDMZAbzCThU/H8DxmSfmdGcKPnVhu2VfFqr34jr9777IyaTYvxjedWhqVIilEDsCdP5G6g== + get-package-type@^0.1.0: version "0.1.0" resolved "https://registry.yarnpkg.com/get-package-type/-/get-package-type-0.1.0.tgz#8de2d803cff44df3bc6c456e6668b36c3926e11a" @@ -2617,7 +2973,7 @@ glob@^7.0.0, glob@^7.1.3: once "^1.3.0" path-is-absolute "^1.0.0" -glob@^7.1.4: +glob@^7.1.4, glob@^7.1.6: version "7.2.3" resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b" integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q== @@ -2650,7 +3006,7 @@ globalthis@^1.0.3: dependencies: define-properties "^1.1.3" -globby@^11.0.1, globby@^11.1.0: +globby@^11.0.1, globby@^11.0.3, globby@^11.1.0: version "11.1.0" resolved "https://registry.yarnpkg.com/globby/-/globby-11.1.0.tgz#bd4be98bb042f83d796f7e3811991fbe82a0d34b" integrity sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g== @@ -2662,6 +3018,13 @@ globby@^11.0.1, globby@^11.1.0: merge2 "^1.4.1" slash "^3.0.0" +gonzales-pe@^4.2.3, gonzales-pe@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/gonzales-pe/-/gonzales-pe-4.3.0.tgz#fe9dec5f3c557eead09ff868c65826be54d067b3" + integrity sha512-otgSPpUmdWJ43VXyiNgEYE4luzHCL2pz4wQ0OnDluC6Eg4Ko3Vexy/SrSynglw/eR+OhkzmqFCZa/OFa/RgAOQ== + dependencies: + minimist "^1.2.5" + gopd@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.0.1.tgz#29ff76de69dac7489b7c0918a5788e56477c332c" @@ -2822,7 +3185,7 @@ iconv-lite@^0.6.2: dependencies: safer-buffer ">= 2.1.2 < 3.0.0" -ieee754@^1.2.1: +ieee754@^1.1.13, ieee754@^1.2.1: version "1.2.1" resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352" integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== @@ -2867,6 +3230,11 @@ indent-string@^4.0.0: resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-4.0.0.tgz#624f8f4497d619b2d9768531d58f4122854d7251" integrity sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg== +indexes-of@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/indexes-of/-/indexes-of-1.0.1.tgz#f30f716c8e2bd346c7b67d3df3915566a7c05607" + integrity sha512-bup+4tap3Hympa+JBJUG7XuOsdNQ6fxt0MHyXMKuLBKn0OqsTfvUxkUrroEX1+B2VsSHvCjiIcZVxRtYa4nllA== + inflight@^1.0.4: version "1.0.6" resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" @@ -2875,12 +3243,12 @@ inflight@^1.0.4: once "^1.3.0" wrappy "1" -inherits@2, inherits@^2.0.3: +inherits@2, inherits@^2.0.3, inherits@^2.0.4: version "2.0.4" resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== -ini@^1.3.4: +ini@^1.3.4, ini@~1.3.0: version "1.3.8" resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz#a29da425b48806f34767a4efce397269af28432c" integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew== @@ -3037,6 +3405,11 @@ is-glob@^4.0.3: dependencies: is-extglob "^2.1.1" +is-interactive@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-interactive/-/is-interactive-1.0.0.tgz#cea6e6ae5c870a7b0a0004070b7b587e0252912e" + integrity sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w== + is-lambda@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/is-lambda/-/is-lambda-1.0.1.tgz#3d9877899e6a53efc0160504cde15f82e6f061d5" @@ -3059,6 +3432,11 @@ is-number@^7.0.0: resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== +is-obj@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-1.0.1.tgz#3e4729ac1f5fde025cd7d83a896dab9f4f67db0f" + integrity sha512-l4RyHgRqGN4Y3+9JHVrNqO+tN0rV5My76uW5/nuO4K1b6vw5G8d/cmFjP9tRfEsdhZNt0IFdZuK/c2Vr4Nb+Qg== + is-obj@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-2.0.0.tgz#473fb05d973705e3fd9620545018ca8e22ef4982" @@ -3087,6 +3465,16 @@ is-regex@^1.1.4: call-bind "^1.0.2" has-tostringtag "^1.0.0" +is-regexp@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-regexp/-/is-regexp-1.0.0.tgz#fd2d883545c46bac5a633e7b9a09e87fa2cb5069" + integrity sha512-7zjFAPO4/gwyQAAgRRmqeEeyIICSdmCqa3tsVHMdBzaXXRiqopZL4Cyghg/XulGWrtABTpbnYYzzIRffLkP4oA== + +is-relative-path@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-relative-path/-/is-relative-path-1.0.2.tgz#091b46a0d67c1ed0fe85f1f8cfdde006bb251d46" + integrity sha512-i1h+y50g+0hRbBD+dbnInl3JlJ702aar58snAeX+MxBAPvzXGej7sYoPMhlnykabt0ZzCJNBEyzMlekuQZN7fA== + is-retry-allowed@^1.1.0: version "1.2.0" resolved "https://registry.yarnpkg.com/is-retry-allowed/-/is-retry-allowed-1.2.0.tgz#d778488bd0a4666a3be8a1482b9f2baafedea8b4" @@ -3137,6 +3525,16 @@ is-unicode-supported@^0.1.0: resolved "https://registry.yarnpkg.com/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz#3f26c76a809593b52bfa2ecb5710ed2779b522a7" integrity sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw== +is-url-superb@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/is-url-superb/-/is-url-superb-4.0.0.tgz#b54d1d2499bb16792748ac967aa3ecb41a33a8c2" + integrity sha512-GI+WjezhPPcbM+tqE9LnmsY5qqjwHzTvjJ36wxYX5ujNXefSUJ/T17r5bqDV8yLhcgB59KTPNOc9O9cmHTPWsA== + +is-url@^1.2.4: + version "1.2.4" + resolved "https://registry.yarnpkg.com/is-url/-/is-url-1.2.4.tgz#04a4df46d28c4cff3d73d01ff06abeb318a1aa52" + integrity sha512-ITvGim8FhRiYe4IQ5uHSkj7pVaPDrCTkNd3yq3cV7iZAcJdHTUMPMEHcqSOy9xZ9qFenQCvi+2wjH9a1nXqHww== + is-weakref@^1.0.2: version "1.0.2" resolved "https://registry.yarnpkg.com/is-weakref/-/is-weakref-1.0.2.tgz#9529f383a9338205e89765e0392efc2f100f06f2" @@ -3472,7 +3870,7 @@ lodash@^4.17.13, lodash@^4.17.15, lodash@^4.17.19, lodash@^4.17.21, lodash@^4.17 resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c" integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== -log-symbols@4.1.0, log-symbols@^4.0.0: +log-symbols@4.1.0, log-symbols@^4.0.0, log-symbols@^4.1.0: version "4.1.0" resolved "https://registry.yarnpkg.com/log-symbols/-/log-symbols-4.1.0.tgz#3fbdbb95b4683ac9fc785111e792e558d4abd503" integrity sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg== @@ -3504,6 +3902,34 @@ lru-cache@^7.4.4, lru-cache@^7.5.1, lru-cache@^7.7.1: resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-10.0.1.tgz#0a3be479df549cca0e5d693ac402ff19537a6b7a" integrity sha512-IJ4uwUTi2qCccrioU6g9g/5rvvVl13bsdczUUcqbciD9iLr095yj8DQKdObriEvuNSx325N1rV1O0sJFszx75g== +madge@^6.1.0: + version "6.1.0" + resolved "https://registry.yarnpkg.com/madge/-/madge-6.1.0.tgz#9835bb53f2e00d184914c2b007b50736a964d297" + integrity sha512-irWhT5RpFOc6lkzGHKLihonCVgM0YtfNUh4IrFeW3EqHpnt/JHUG3z26j8PeJEktCGB4tmGOOOJi1Rl/ACWucQ== + dependencies: + chalk "^4.1.1" + commander "^7.2.0" + commondir "^1.0.1" + debug "^4.3.1" + dependency-tree "^9.0.0" + detective-amd "^4.0.1" + detective-cjs "^4.0.0" + detective-es6 "^3.0.0" + detective-less "^1.0.2" + detective-postcss "^6.1.0" + detective-sass "^4.0.1" + detective-scss "^3.0.0" + detective-stylus "^2.0.1" + detective-typescript "^9.0.0" + ora "^5.4.1" + pluralize "^8.0.0" + precinct "^8.1.0" + pretty-ms "^7.0.1" + rc "^1.2.7" + stream-to-array "^2.3.0" + ts-graphviz "^1.5.0" + walkdir "^0.4.1" + make-error@^1.1.1: version "1.3.6" resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2" @@ -3588,6 +4014,11 @@ micromatch@^4.0.4: braces "^3.0.1" picomatch "^2.2.3" +mimic-fn@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" + integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg== + min-indent@^1.0.0: version "1.0.1" resolved "https://registry.yarnpkg.com/min-indent/-/min-indent-1.0.1.tgz#a63f681673b30571fbe8bc25686ae746eefa9869" @@ -3630,7 +4061,7 @@ minimist-options@4.1.0: is-plain-obj "^1.1.0" kind-of "^6.0.3" -minimist@^1.2.0, minimist@^1.2.6: +minimist@^1.2.0, minimist@^1.2.5, minimist@^1.2.6: version "1.2.8" resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c" integrity sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA== @@ -3749,6 +4180,33 @@ mock-stdin@^1.0.0: resolved "https://registry.yarnpkg.com/mock-stdin/-/mock-stdin-1.0.0.tgz#efcfaf4b18077e14541742fd758b9cae4e5365ea" integrity sha512-tukRdb9Beu27t6dN+XztSRHq9J0B/CoAOySGzHfn8UTfmqipA5yNT/sDUEyYdAV3Hpka6Wx6kOMxuObdOex60Q== +module-definition@^3.3.1: + version "3.4.0" + resolved "https://registry.yarnpkg.com/module-definition/-/module-definition-3.4.0.tgz#953a3861f65df5e43e80487df98bb35b70614c2b" + integrity sha512-XxJ88R1v458pifaSkPNLUTdSPNVGMP2SXVncVmApGO+gAfrLANiYe6JofymCzVceGOMwQE2xogxBSc8uB7XegA== + dependencies: + ast-module-types "^3.0.0" + node-source-walk "^4.0.0" + +module-definition@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/module-definition/-/module-definition-4.1.0.tgz#148ff9348e3401867229dcbe5947f4f6d5ccd3a2" + integrity sha512-rHXi/DpMcD2qcKbPCTklDbX9lBKJrUSl971TW5l6nMpqKCIlzJqmQ8cfEF5M923h2OOLHPDVlh5pJxNyV+AJlw== + dependencies: + ast-module-types "^4.0.0" + node-source-walk "^5.0.1" + +module-lookup-amd@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/module-lookup-amd/-/module-lookup-amd-7.0.1.tgz#d67c1a93f2ff8e38b8774b99a638e9a4395774b2" + integrity sha512-w9mCNlj0S8qviuHzpakaLVc+/7q50jl9a/kmJ/n8bmXQZgDPkQHnPBb8MUOYh3WpAYkXuNc2c+khsozhIp/amQ== + dependencies: + commander "^2.8.1" + debug "^4.1.0" + glob "^7.1.6" + requirejs "^2.3.5" + requirejs-config-file "^4.0.0" + ms@2.1.2: version "2.1.2" resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" @@ -3769,6 +4227,11 @@ nanoid@3.3.3: resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.3.tgz#fd8e8b7aa761fe807dba2d1b98fb7241bb724a25" integrity sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w== +nanoid@^3.3.6: + version "3.3.6" + resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.6.tgz#443380c856d6e9f9824267d960b4236ad583ea4c" + integrity sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA== + natural-compare@^1.4.0: version "1.4.0" resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" @@ -3848,6 +4311,20 @@ node-gyp@^9.0.0, node-gyp@^9.4.0: tar "^6.1.2" which "^2.0.2" +node-source-walk@^4.0.0, node-source-walk@^4.2.0, node-source-walk@^4.2.2: + version "4.3.0" + resolved "https://registry.yarnpkg.com/node-source-walk/-/node-source-walk-4.3.0.tgz#8336b56cfed23ac5180fe98f1e3bb6b11fd5317c" + integrity sha512-8Q1hXew6ETzqKRAs3jjLioSxNfT1cx74ooiF8RlAONwVMcfq+UdzLC2eB5qcPldUxaE5w3ytLkrmV1TGddhZTA== + dependencies: + "@babel/parser" "^7.0.0" + +node-source-walk@^5.0.0, node-source-walk@^5.0.1: + version "5.0.2" + resolved "https://registry.yarnpkg.com/node-source-walk/-/node-source-walk-5.0.2.tgz#0eb439ce378946ce531e07a6a0073d06288396dd" + integrity sha512-Y4jr/8SRS5hzEdZ7SGuvZGwfORvNsSsNRwDXx5WisiqzsVfeftDvRgfeqWNgZvWSJbgubTRVRYBzK6UO+ErqjA== + dependencies: + "@babel/parser" "^7.21.4" + nopt@^6.0.0: version "6.0.0" resolved "https://registry.yarnpkg.com/nopt/-/nopt-6.0.0.tgz#245801d8ebf409c6df22ab9d95b65e1309cdb16d" @@ -4135,6 +4612,13 @@ once@^1.3.0: dependencies: wrappy "1" +onetime@^5.1.0: + version "5.1.2" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-5.1.2.tgz#d0e96ebb56b07476df1dd9c4806e5237985ca45e" + integrity sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg== + dependencies: + mimic-fn "^2.1.0" + optionator@^0.9.3: version "0.9.3" resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.9.3.tgz#007397d44ed1872fdc6ed31360190f81814e2c64" @@ -4147,6 +4631,21 @@ optionator@^0.9.3: prelude-ls "^1.2.1" type-check "^0.4.0" +ora@^5.4.1: + version "5.4.1" + resolved "https://registry.yarnpkg.com/ora/-/ora-5.4.1.tgz#1b2678426af4ac4a509008e5e4ac9e9959db9e18" + integrity sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ== + dependencies: + bl "^4.1.0" + chalk "^4.1.0" + cli-cursor "^3.1.0" + cli-spinners "^2.5.0" + is-interactive "^1.0.0" + is-unicode-supported "^0.1.0" + log-symbols "^4.1.0" + strip-ansi "^6.0.0" + wcwidth "^1.0.1" + p-limit@^2.2.0: version "2.3.0" resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-2.3.0.tgz#3dd33c647a214fdfffd835933eb086da0dc21db1" @@ -4245,6 +4744,11 @@ parse-json@^5.0.0: json-parse-even-better-errors "^2.3.0" lines-and-columns "^1.1.6" +parse-ms@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/parse-ms/-/parse-ms-2.1.0.tgz#348565a753d4391fa524029956b172cb7753097d" + integrity sha512-kHt7kzLoS9VBZfUsiKjv43mr91ea+U05EyKkEtqp7vNbHxmaVuEqN7XxeEVnGrMtYOAxGrDElSi96K7EgO1zCA== + password-prompt@^1.1.2: version "1.1.2" resolved "https://registry.yarnpkg.com/password-prompt/-/password-prompt-1.1.2.tgz#85b2f93896c5bd9e9f2d6ff0627fa5af3dc00923" @@ -4303,6 +4807,11 @@ pathval@^1.1.1: resolved "https://registry.yarnpkg.com/pathval/-/pathval-1.1.1.tgz#8534e77a77ce7ac5a2512ea21e0fdb8fcf6c3d8d" integrity sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ== +picocolors@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.0.tgz#cb5bdc74ff3f51892236eaf79d68bc44564ab81c" + integrity sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ== + picomatch@^2.0.4, picomatch@^2.2.1, picomatch@^2.2.3: version "2.3.0" resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.0.tgz#f1f061de8f6a4bf022892e2d128234fb98302972" @@ -4338,6 +4847,70 @@ postcss-selector-parser@^6.0.10: cssesc "^3.0.0" util-deprecate "^1.0.2" +postcss-values-parser@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/postcss-values-parser/-/postcss-values-parser-2.0.1.tgz#da8b472d901da1e205b47bdc98637b9e9e550e5f" + integrity sha512-2tLuBsA6P4rYTNKCXYG/71C7j1pU6pK503suYOmn4xYrQIzW+opD+7FAFNuGSdZC/3Qfy334QbeMu7MEb8gOxg== + dependencies: + flatten "^1.0.2" + indexes-of "^1.0.1" + uniq "^1.0.1" + +postcss-values-parser@^6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/postcss-values-parser/-/postcss-values-parser-6.0.2.tgz#636edc5b86c953896f1bb0d7a7a6615df00fb76f" + integrity sha512-YLJpK0N1brcNJrs9WatuJFtHaV9q5aAOj+S4DI5S7jgHlRfm0PIbDCAFRYMQD5SHq7Fy6xsDhyutgS0QOAs0qw== + dependencies: + color-name "^1.1.4" + is-url-superb "^4.0.0" + quote-unquote "^1.0.0" + +postcss@^8.1.7, postcss@^8.4.23: + version "8.4.30" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.30.tgz#0e0648d551a606ef2192a26da4cabafcc09c1aa7" + integrity sha512-7ZEao1g4kd68l97aWG/etQKPKq07us0ieSZ2TnFDk11i0ZfDW2AwKHYU8qv4MZKqN2fdBfg+7q0ES06UA73C1g== + dependencies: + nanoid "^3.3.6" + picocolors "^1.0.0" + source-map-js "^1.0.2" + +precinct@^8.1.0: + version "8.3.1" + resolved "https://registry.yarnpkg.com/precinct/-/precinct-8.3.1.tgz#94b99b623df144eed1ce40e0801c86078466f0dc" + integrity sha512-pVppfMWLp2wF68rwHqBIpPBYY8Kd12lDhk8LVQzOwqllifVR15qNFyod43YLyFpurKRZQKnE7E4pofAagDOm2Q== + dependencies: + commander "^2.20.3" + debug "^4.3.3" + detective-amd "^3.1.0" + detective-cjs "^3.1.1" + detective-es6 "^2.2.1" + detective-less "^1.0.2" + detective-postcss "^4.0.0" + detective-sass "^3.0.1" + detective-scss "^2.0.1" + detective-stylus "^1.0.0" + detective-typescript "^7.0.0" + module-definition "^3.3.1" + node-source-walk "^4.2.0" + +precinct@^9.0.0: + version "9.2.1" + resolved "https://registry.yarnpkg.com/precinct/-/precinct-9.2.1.tgz#db0a67abff7b0a9a3b2b1ac33d170e8a5fcac7b2" + integrity sha512-uzKHaTyiVejWW7VJtHInb9KBUq9yl9ojxXGujhjhDmPon2wgZPBKQIKR+6csGqSlUeGXAA4MEFnU6DesxZib+A== + dependencies: + "@dependents/detective-less" "^3.0.1" + commander "^9.5.0" + detective-amd "^4.1.0" + detective-cjs "^4.1.0" + detective-es6 "^3.0.1" + detective-postcss "^6.1.1" + detective-sass "^4.1.1" + detective-scss "^3.0.1" + detective-stylus "^3.0.0" + detective-typescript "^9.1.1" + module-definition "^4.1.0" + node-source-walk "^5.0.1" + prelude-ls@^1.2.1: version "1.2.1" resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396" @@ -4352,6 +4925,13 @@ pretty-format@^29.7.0: ansi-styles "^5.0.0" react-is "^18.0.0" +pretty-ms@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/pretty-ms/-/pretty-ms-7.0.1.tgz#7d903eaab281f7d8e03c66f867e239dc32fb73e8" + integrity sha512-973driJZvxiGOQ5ONsFhOF/DtzPMOMtgC11kCpUrPGMTgqp2q/1gwzCquocrN33is0VZ5GFHXZYMM9l6h67v2Q== + dependencies: + parse-ms "^2.1.0" + proc-log@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/proc-log/-/proc-log-3.0.0.tgz#fb05ef83ccd64fd7b20bbe9c8c1070fc08338dd8" @@ -4422,6 +5002,11 @@ quick-lru@^4.0.1: resolved "https://registry.yarnpkg.com/quick-lru/-/quick-lru-4.0.1.tgz#5b8878f113a58217848c6482026c73e1ba57727f" integrity sha512-ARhCpm70fzdcvNQfPoy49IaanKkTlRWF2JMzqhcJbhSFRZv7nPTvZJdcY7301IPmvW+/p0RgIWnQDLJxifsQ7g== +quote-unquote@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/quote-unquote/-/quote-unquote-1.0.0.tgz#67a9a77148effeaf81a4d428404a710baaac8a0b" + integrity sha512-twwRO/ilhlG/FIgYeKGFqyHhoEhqgnKVkcmqMKi2r524gz3ZbDTcyFt38E9xjJI2vT+KbRNHVbnJ/e0I25Azwg== + rambda@^7.1.0: version "7.5.0" resolved "https://registry.yarnpkg.com/rambda/-/rambda-7.5.0.tgz#1865044c59bc0b16f63026c6e5a97e4b1bbe98fe" @@ -4434,6 +5019,16 @@ randombytes@^2.1.0: dependencies: safe-buffer "^5.1.0" +rc@^1.2.7: + version "1.2.8" + resolved "https://registry.yarnpkg.com/rc/-/rc-1.2.8.tgz#cd924bf5200a075b83c188cd6b9e211b7fc0d3ed" + integrity sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw== + dependencies: + deep-extend "^0.6.0" + ini "~1.3.0" + minimist "^1.2.0" + strip-json-comments "~2.0.1" + react-is@^18.0.0: version "18.2.0" resolved "https://registry.yarnpkg.com/react-is/-/react-is-18.2.0.tgz#199431eeaaa2e09f86427efbb4f1473edb47609b" @@ -4497,7 +5092,7 @@ readable-stream@3, readable-stream@^3.0.0: string_decoder "^1.1.1" util-deprecate "^1.0.1" -readable-stream@^3.6.0: +readable-stream@^3.4.0, readable-stream@^3.6.0: version "3.6.2" resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.2.tgz#56a9b36ea965c00c5a93ef31eb111a0f11056967" integrity sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA== @@ -4577,6 +5172,24 @@ require-directory@^2.1.1: resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42" integrity sha1-jGStX9MNqxyXbiNE/+f3kqam30I= +requirejs-config-file@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/requirejs-config-file/-/requirejs-config-file-4.0.0.tgz#4244da5dd1f59874038cc1091d078d620abb6ebc" + integrity sha512-jnIre8cbWOyvr8a5F2KuqBnY+SDA4NXr/hzEZJG79Mxm2WiFQz2dzhC8ibtPJS7zkmBEl1mxSwp5HhC1W4qpxw== + dependencies: + esprima "^4.0.0" + stringify-object "^3.2.1" + +requirejs@^2.3.5: + version "2.3.6" + resolved "https://registry.yarnpkg.com/requirejs/-/requirejs-2.3.6.tgz#e5093d9601c2829251258c0b9445d4d19fa9e7c9" + integrity sha512-ipEzlWQe6RK3jkzikgCupiTbTvm4S0/CAU5GlgptkN5SO6F3u0UD0K18wy6ErDqiCyP4J4YYe1HuAShvsxePLg== + +resolve-dependency-path@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/resolve-dependency-path/-/resolve-dependency-path-2.0.0.tgz#11700e340717b865d216c66cabeb4a2a3c696736" + integrity sha512-DIgu+0Dv+6v2XwRaNWnumKu7GPufBBOr5I1gRPJHkvghrfCGOooJODFvgFimX/KRxk9j0whD2MnKHzM1jYvk9w== + resolve-from@5.0.0, resolve-from@^5.0.0: version "5.0.0" resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-5.0.0.tgz#c35225843df8f776df21c57557bc087e9dfdfc69" @@ -4607,7 +5220,7 @@ resolve@^1.1.6, resolve@^1.10.0, resolve@^1.10.1, resolve@^1.20.0: is-core-module "^2.2.0" path-parse "^1.0.6" -resolve@^1.22.4: +resolve@^1.21.0, resolve@^1.22.4: version "1.22.6" resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.6.tgz#dd209739eca3aef739c626fea1b4f3c506195362" integrity sha512-njhxM7mV12JfufShqGy3Rz8j11RPdLy4xi15UurGJeoHLfJpVXKdh3ueuOqbYUcDZnffr6X739JBo5LzyahEsw== @@ -4616,6 +5229,14 @@ resolve@^1.22.4: path-parse "^1.0.7" supports-preserve-symlinks-flag "^1.0.0" +restore-cursor@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-3.1.0.tgz#39f67c54b3a7a58cea5236d95cf0034239631f7e" + integrity sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA== + dependencies: + onetime "^5.1.0" + signal-exit "^3.0.2" + retry@^0.12.0: version "0.12.0" resolved "https://registry.yarnpkg.com/retry/-/retry-0.12.0.tgz#1b42a6266a21f07421d1b0b54b7dc167b01c013b" @@ -4669,6 +5290,13 @@ safe-regex-test@^1.0.0: resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== +sass-lookup@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/sass-lookup/-/sass-lookup-3.0.0.tgz#3b395fa40569738ce857bc258e04df2617c48cac" + integrity sha512-TTsus8CfFRn1N44bvdEai1no6PqdmDiQUiqW5DlpmtT+tYnIt1tXtDIph5KA1efC+LmioJXSnCtUVpcK9gaKIg== + dependencies: + commander "^2.16.0" + "semver@2 || 3 || 4 || 5", semver@^5.5.0: version "5.7.1" resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7" @@ -4776,7 +5404,7 @@ side-channel@^1.0.4: get-intrinsic "^1.0.2" object-inspect "^1.9.0" -signal-exit@^3.0.7: +signal-exit@^3.0.2, signal-exit@^3.0.7: version "3.0.7" resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.7.tgz#a9a1767f8af84155114eaabd73f99273c8f59ad9" integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ== @@ -4857,6 +5485,16 @@ socks@^2.6.2: ip "^2.0.0" smart-buffer "^4.2.0" +source-map-js@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.0.2.tgz#adbc361d9c62df380125e7f161f71c826f1e490c" + integrity sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw== + +source-map@~0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" + integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== + spdx-correct@^3.0.0: version "3.1.1" resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-3.1.1.tgz#dece81ac9c1e6713e5f7d1b6f17d468fa53d89a9" @@ -4910,6 +5548,13 @@ stdout-stderr@^0.1.9: debug "^4.1.1" strip-ansi "^6.0.0" +stream-to-array@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/stream-to-array/-/stream-to-array-2.3.0.tgz#bbf6b39f5f43ec30bc71babcb37557acecf34353" + integrity sha512-UsZtOYEn4tWU2RGLOXr/o/xjRBftZRlG3dEWoaHr8j4GuypJ3isitGbVyjQKAuMu+xbiop8q224TjiZWc4XTZA== + dependencies: + any-promise "^1.1.0" + "string-width-cjs@npm:string-width@^4.2.0", "string-width@^1.0.2 || 2 || 3 || 4", string-width@^4.0.0, string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: version "4.2.3" resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" @@ -4962,6 +5607,15 @@ string_decoder@^1.1.1, string_decoder@^1.3.0: dependencies: safe-buffer "~5.2.0" +stringify-object@^3.2.1: + version "3.3.0" + resolved "https://registry.yarnpkg.com/stringify-object/-/stringify-object-3.3.0.tgz#703065aefca19300d3ce88af4f5b3956d7556629" + integrity sha512-rHqiFh1elqCQ9WPLIC8I0Q/g/wj5J1eMkyoiD6eoQApWHP0FtlK7rqnhmabL5VUY9JQCcqwwvlOaSuutekgyrw== + dependencies: + get-own-enumerable-property-symbols "^3.0.0" + is-obj "^1.0.1" + is-regexp "^1.0.0" + "strip-ansi-cjs@npm:strip-ansi@^6.0.1", strip-ansi@^6.0.0, strip-ansi@^6.0.1: version "6.0.1" resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" @@ -5000,6 +5654,19 @@ strip-json-comments@3.1.1, strip-json-comments@^3.1.1: resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006" integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig== +strip-json-comments@~2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz#3c531942e908c2697c0ec344858c286c7ca0a60a" + integrity sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ== + +stylus-lookup@^3.0.1: + version "3.0.2" + resolved "https://registry.yarnpkg.com/stylus-lookup/-/stylus-lookup-3.0.2.tgz#c9eca3ff799691020f30b382260a67355fefdddd" + integrity sha512-oEQGHSjg/AMaWlKe7gqsnYzan8DLcGIHe0dUaFkucZZ14z4zjENRlQMCHT4FNsiWnJf17YN9OvrCfCoi7VvOyg== + dependencies: + commander "^2.8.1" + debug "^4.1.0" + supports-color@8.1.1, supports-color@^8.1.1: version "8.1.1" resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-8.1.1.tgz#cd6fc17e28500cff56c1b86c0a7fd4a54a73005c" @@ -5110,6 +5777,11 @@ ts-api-utils@^1.0.1: resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-1.0.3.tgz#f12c1c781d04427313dbac808f453f050e54a331" integrity sha512-wNMeqtMz5NtwpT/UZGY5alT+VoKdSsOOP/kqHFcUW1P/VRhH2wJ48+DN2WwUliNbQ976ETwDL0Ifd2VVvgonvg== +ts-graphviz@^1.5.0: + version "1.8.1" + resolved "https://registry.yarnpkg.com/ts-graphviz/-/ts-graphviz-1.8.1.tgz#5d95e58120be8b571847331516327d4840cc44f7" + integrity sha512-54/fe5iu0Jb6X0pmDmzsA2UHLfyHjUEUwfHtZcEOR0fZ6Myf+dFoO6eNsyL8CBDMJ9u7WWEewduVaiaXlvjSVw== + ts-node@^10.9.1: version "10.9.1" resolved "https://registry.yarnpkg.com/ts-node/-/ts-node-10.9.1.tgz#e73de9102958af9e1f0b168a6ff320e25adcff4b" @@ -5129,7 +5801,7 @@ ts-node@^10.9.1: v8-compile-cache-lib "^3.0.1" yn "3.1.1" -tsconfig-paths@^3.14.2: +tsconfig-paths@^3.10.1, tsconfig-paths@^3.14.2: version "3.14.2" resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.14.2.tgz#6e32f1f79412decd261f92d633a9dc1cfa99f088" integrity sha512-o/9iXgCYc5L/JxCHPe3Hvh8Q/2xm5Z+p18PESBU6Ff33695QnCHBEjcytY2q19ua7Mbl/DavtBOLq+oG0RCL+g== @@ -5152,6 +5824,11 @@ tsd@^0.29.0: path-exists "^4.0.0" read-pkg-up "^7.0.0" +tslib@^1.8.1: + version "1.14.1" + resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.14.1.tgz#cf2d38bdc34a134bcaf1091c41f6619e2f672d00" + integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg== + tslib@^2.5.0: version "2.5.0" resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.5.0.tgz#42bfed86f5787aeb41d031866c8f402429e0fddf" @@ -5162,6 +5839,13 @@ tslib@^2.6.2: resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.2.tgz#703ac29425e7b37cd6fd456e92404d46d1f3e4ae" integrity sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q== +tsutils@^3.21.0: + version "3.21.0" + resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-3.21.0.tgz#b48717d394cea6c1e096983eed58e9d61715b623" + integrity sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA== + dependencies: + tslib "^1.8.1" + tuf-js@^1.1.7: version "1.1.7" resolved "https://registry.yarnpkg.com/tuf-js/-/tuf-js-1.1.7.tgz#21b7ae92a9373015be77dfe0cb282a80ec3bbe43" @@ -5259,6 +5943,16 @@ typed-array-length@^1.0.4: for-each "^0.3.3" is-typed-array "^1.1.9" +typescript@^3.9.10, typescript@^3.9.7: + version "3.9.10" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-3.9.10.tgz#70f3910ac7a51ed6bef79da7800690b19bf778b8" + integrity sha512-w6fIxVE/H1PkLKcCPsFqKE7Kv7QUwhU8qQY2MueZXWx5cPZdwFupLgKK3vntcK98BtNHZtAF4LA/yl2a7k8R6Q== + +typescript@^4.0.0, typescript@^4.9.5: + version "4.9.5" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.9.5.tgz#095979f9bcc0d09da324d58d03ce8f8374cbe65a" + integrity sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g== + typescript@^5: version "5.2.2" resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.2.2.tgz#5ebb5e5a5b75f085f22bc3f8460fba308310fa78" @@ -5274,6 +5968,11 @@ unbox-primitive@^1.0.2: has-symbols "^1.0.3" which-boxed-primitive "^1.0.2" +uniq@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/uniq/-/uniq-1.0.1.tgz#b31c5ae8254844a3a8281541ce2b04b865a734ff" + integrity sha512-Gw+zz50YNKPDKXs+9d+aKAjVwpjNwqzvNpLigIruT4HA9lMZNdMqs9x07kKHB/L9WRzqp4+DlTU5s4wG2esdoA== + unique-filename@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/unique-filename/-/unique-filename-3.0.0.tgz#48ba7a5a16849f5080d26c760c86cf5cf05770ea" @@ -5330,7 +6029,12 @@ walk-up-path@^3.0.1: resolved "https://registry.yarnpkg.com/walk-up-path/-/walk-up-path-3.0.1.tgz#c8d78d5375b4966c717eb17ada73dbd41490e886" integrity sha512-9YlCL/ynK3CTlrSRrDxZvUauLzAswPCrsaCgilqFevUYpeEW0/3ScEjaa3kbW/T0ghhkEr7mv+fpjqn1Y1YuTA== -wcwidth@^1.0.0: +walkdir@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/walkdir/-/walkdir-0.4.1.tgz#dc119f83f4421df52e3061e514228a2db20afa39" + integrity sha512-3eBwRyEln6E1MSzcxcVpQIhRG8Q1jLvEqRmCZqS3dsfXEDR/AhOF4d+jHg1qvDCpYaVRZjENPQyrVxAkQqxPgQ== + +wcwidth@^1.0.0, wcwidth@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/wcwidth/-/wcwidth-1.0.1.tgz#f0b0dcf915bc5ff1528afadb2c0e17b532da2fe8" integrity sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg== From 6d20d625a1029452d46c6ca40bd28c851c68e8b3 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 10:13:40 -0600 Subject: [PATCH 20/27] chore: remove Flags.json --- src/command.ts | 22 +++------------------- src/flags.ts | 14 -------------- src/util/aggregate-flags.ts | 18 ++++++++++++++++++ src/util/to-cached.ts | 8 ++------ test/command/command.test.ts | 34 ---------------------------------- 5 files changed, 23 insertions(+), 73 deletions(-) create mode 100644 src/util/aggregate-flags.ts diff --git a/src/command.ts b/src/command.ts index 10be2c416..82d9b8798 100644 --- a/src/command.ts +++ b/src/command.ts @@ -25,9 +25,9 @@ import {LoadOptions} from './interfaces/config' import {PJSON} from './interfaces' import {Plugin} from './interfaces/plugin' import {PrettyPrintableError} from './errors' +import {aggregateFlags} from './util/aggregate-flags' import chalk from 'chalk' import {fileURLToPath} from 'node:url' -import {json} from './flags' import {ux} from './cli-ux' const pjson = requireJson(__dirname, '..', 'package.json') @@ -228,11 +228,6 @@ export abstract class Command { * @returns {boolean} true if the command supports json and the --json flag is present */ public jsonEnabled(): boolean { - // If the JSON_FLAG_OVERRIDE env var (set by Flags.json parse) is set to true, return true - // Checking for this first allows commands to define a --json flag using Flags.json() - // without setting enableJsonFlag static property. - if (this.config.scopedEnvVarTrue?.('JSON_FLAG_OVERRIDE')) return true - // If the command doesn't support json, return false if (!this.ctor.enableJsonFlag) return false @@ -264,7 +259,7 @@ export abstract class Command { } protected warnIfFlagDeprecated(flags: Record): void { - const allFlags = this.aggregateFlags( + const allFlags = aggregateFlags( this.ctor.flags, this.ctor.baseFlags, this.ctor.enableJsonFlag, @@ -318,7 +313,7 @@ export abstract class Command { const opts = { context: this, ...options, - flags: this.aggregateFlags( + flags: aggregateFlags( options.flags, options.baseFlags, options.enableJsonFlag, @@ -376,17 +371,6 @@ export abstract class Command { keys.map(key => delete process.env[key]) } - - private aggregateFlags( - flags: FlagInput | undefined, - baseFlags: FlagInput | undefined, - enableJsonFlag: boolean | undefined, - ): FlagInput { - const combinedFlags = {...baseFlags, ...flags} - return (enableJsonFlag - ? {...combinedFlags, json: json()} - : combinedFlags) as FlagInput - } } export namespace Command { diff --git a/src/flags.ts b/src/flags.ts index 2360500b8..699dde32e 100644 --- a/src/flags.ts +++ b/src/flags.ts @@ -145,20 +145,6 @@ export const help = (opts: Partial> = {}): BooleanFlag> = {}): BooleanFlag => boolean({ - description: 'Format output as json.', - helpGroup: 'GLOBAL', - ...opts, - async parse(input, {config}) { - if (input) process.env[config.scopedEnvVarKey('JSON_FLAG_OVERRIDE')] = 'true' - return input - }, -}) - type ElementType> = T[number]; export function option( diff --git a/src/util/aggregate-flags.ts b/src/util/aggregate-flags.ts new file mode 100644 index 000000000..ba96dbc93 --- /dev/null +++ b/src/util/aggregate-flags.ts @@ -0,0 +1,18 @@ +import {FlagInput, FlagOutput} from '../interfaces/parser' +import {boolean} from '../flags' + +export const json = boolean({ + description: 'Format output as json.', + helpGroup: 'GLOBAL', +}) + +export function aggregateFlags( + flags: FlagInput | undefined, + baseFlags: FlagInput | undefined, + enableJsonFlag: boolean | undefined, +): FlagInput { + const combinedFlags = {...baseFlags, ...flags} + return (enableJsonFlag + ? {json, ...combinedFlags} + : combinedFlags) as FlagInput +} diff --git a/src/util/to-cached.ts b/src/util/to-cached.ts index e2bdcf15d..de748284d 100644 --- a/src/util/to-cached.ts +++ b/src/util/to-cached.ts @@ -3,8 +3,8 @@ import {ensureArgObject, pickBy} from './index' import {Arg} from '../interfaces/parser' import {Command} from '../command' import {Plugin as IPlugin} from '../interfaces/plugin' +import {aggregateFlags} from './aggregate-flags' import {defaultFlagToCached} from './default-flag-to-cached' -import {json} from '../flags' const defaultArgToCached = async (arg: Arg, respectNoCacheDefault: boolean): Promise => { if (respectNoCacheDefault && arg.noCacheDefault) return @@ -41,11 +41,7 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa const c = mergePrototype(cmd, cmd) - const cmdFlags = { - ...(c.enableJsonFlag ? {json: json()} : {}), - ...c.baseFlags, - ...c.flags, - } as typeof c['flags'] + const cmdFlags = aggregateFlags(c.flags, c.baseFlags, c.enableJsonFlag) for (const [name, flag] of Object.entries(cmdFlags || {})) { if (flag.type === 'boolean') { diff --git a/test/command/command.test.ts b/test/command/command.test.ts index e6f75e754..9531d565e 100644 --- a/test/command/command.test.ts +++ b/test/command/command.test.ts @@ -704,39 +704,5 @@ describe('command', () => { expect(cmd.jsonEnabled()).to.equal(false) }) .it('json disabled/pass through enable/--json flag before --/jsonEnabled() should be false') - - fancy - .stdout() - .do(async () => { - class CMD extends Command { - static flags = { - json: Flags.json(), - } - - async run() {} - } - - const cmd = new CMD(['--json'], { - bin: 'FOO', scopedEnvVarTrue: (foo: string) => foo.includes('JSON_FLAG_OVERRIDE'), - } as any) - expect(cmd.jsonEnabled()).to.equal(true) - }) - .it('Flags.json to enable json, enableJsonFlag not set') - - fancy - .stdout() - .do(async () => { - class CMD extends Command { - static flags = { - json: Flags.json(), - } - - async run() {} - } - - const cmd = new CMD([], {} as any) - expect(cmd.jsonEnabled()).to.equal(false) - }) - .it('Flags.json in flags definition but not used, enableJsonFlag not set') }) }) From ce3441f79566f3eb31dbea2e8ffe6211fb90d0b7 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 10:14:33 -0600 Subject: [PATCH 21/27] chore: add prettier config --- .prettierrc.json | 1 + package.json | 1 + yarn.lock | 5 +++++ 3 files changed, 7 insertions(+) create mode 100644 .prettierrc.json diff --git a/.prettierrc.json b/.prettierrc.json new file mode 100644 index 000000000..631433570 --- /dev/null +++ b/.prettierrc.json @@ -0,0 +1 @@ +"@oclif/prettier-config" diff --git a/package.json b/package.json index b7bfeb941..0e72309e4 100644 --- a/package.json +++ b/package.json @@ -38,6 +38,7 @@ "@commitlint/config-conventional": "^12.1.4", "@oclif/plugin-help": "^5.2.8", "@oclif/plugin-plugins": "^3.3.0", + "@oclif/prettier-config": "^0.1.1", "@oclif/test": "^3.0.0-beta.1", "@types/ansi-styles": "^3.2.1", "@types/benchmark": "^2.1.2", diff --git a/yarn.lock b/yarn.lock index 22e59cfe4..8370bba9d 100644 --- a/yarn.lock +++ b/yarn.lock @@ -595,6 +595,11 @@ validate-npm-package-name "^5.0.0" yarn "^1.22.18" +"@oclif/prettier-config@^0.1.1": + version "0.1.1" + resolved "https://registry.yarnpkg.com/@oclif/prettier-config/-/prettier-config-0.1.1.tgz#424b7f631b96917efe8b9a925b068cd2218a00f2" + integrity sha512-fxsIwtHCu8V6+WmsN0w/s6QM3z/VJtvTEXTiY8F8l1Oj9S+nOC0eTaQp2YSDFlgpFvDJ5NwC5ssdZAdFE3PyDg== + "@oclif/test@^3.0.0-beta.1": version "3.0.0-beta.1" resolved "https://registry.yarnpkg.com/@oclif/test/-/test-3.0.0-beta.1.tgz#37beca545657eaf04413652f3b3db4f946c9804a" From 811efa1e608cf3c2baeff5519863f8cbb568d67f Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 10:24:29 -0600 Subject: [PATCH 22/27] test: add nyc --- .nycrc.json | 11 + package.json | 3 +- yarn.lock | 676 +++++++++++++++++++++++++++++++++++++++++++++++++-- 3 files changed, 672 insertions(+), 18 deletions(-) create mode 100644 .nycrc.json diff --git a/.nycrc.json b/.nycrc.json new file mode 100644 index 000000000..d6c706920 --- /dev/null +++ b/.nycrc.json @@ -0,0 +1,11 @@ +{ + "check-coverage": true, + "lines": 80, + "statements": 70, + "functions": 70, + "branches": 60, + "reporter": ["lcov", "text"], + "extension": [".ts"], + "include": ["**/*.ts"], + "exclude": ["**/*.d.ts", "test/**"] +} diff --git a/package.json b/package.json index 0e72309e4..674f55568 100644 --- a/package.json +++ b/package.json @@ -71,6 +71,7 @@ "madge": "^6.1.0", "mocha": "^10.2.0", "nock": "^13.3.0", + "nyc": "^15.1.0", "shx": "^0.3.4", "sinon": "^11.1.2", "tsd": "^0.29.0", @@ -114,7 +115,7 @@ "test:e2e": "mocha --forbid-only \"test/**/*.e2e.ts\" --parallel --timeout 1200000", "test:esm-cjs": "cross-env DEBUG=e2e:* ts-node test/integration/esm-cjs.ts", "test:perf": "ts-node test/perf/parser.perf.ts", - "test": "mocha --forbid-only \"test/**/*.test.ts\"" + "test": "nyc mocha --forbid-only \"test/**/*.test.ts\"" }, "types": "lib/index.d.ts" } diff --git a/yarn.lock b/yarn.lock index 8370bba9d..30b5a6612 100644 --- a/yarn.lock +++ b/yarn.lock @@ -7,6 +7,14 @@ resolved "https://registry.yarnpkg.com/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz#bd9154aec9983f77b3a034ecaa015c2e4201f6cf" integrity sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA== +"@ampproject/remapping@^2.2.0": + version "2.2.1" + resolved "https://registry.yarnpkg.com/@ampproject/remapping/-/remapping-2.2.1.tgz#99e8e11851128b8702cd57c33684f1d0f260b630" + integrity sha512-lFMjJTrFL3j7L9yBxwYfCq2k6qqwHyzuUl/XBnif78PWTJYyL/dfowQHWE3sp6U6ZzqWiiIZnpTMO96zhkjwtg== + dependencies: + "@jridgewell/gen-mapping" "^0.3.0" + "@jridgewell/trace-mapping" "^0.3.9" + "@babel/code-frame@^7.0.0": version "7.14.5" resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.14.5.tgz#23b08d740e83f49c5e59945fbf1b43e80bbf4edb" @@ -14,16 +22,142 @@ dependencies: "@babel/highlight" "^7.14.5" +"@babel/code-frame@^7.22.13": + version "7.22.13" + resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.22.13.tgz#e3c1c099402598483b7a8c46a721d1038803755e" + integrity sha512-XktuhWlJ5g+3TJXc5upd9Ks1HutSArik6jf2eAjYFyIOf4ej3RN+184cZbzDvbPnuTJIUhPKKJE3cIsYTiAT3w== + dependencies: + "@babel/highlight" "^7.22.13" + chalk "^2.4.2" + +"@babel/compat-data@^7.22.9": + version "7.22.20" + resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.22.20.tgz#8df6e96661209623f1975d66c35ffca66f3306d0" + integrity sha512-BQYjKbpXjoXwFW5jGqiizJQQT/aC7pFm9Ok1OWssonuguICi264lbgMzRp2ZMmRSlfkX6DsWDDcsrctK8Rwfiw== + +"@babel/core@^7.7.5": + version "7.23.0" + resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.23.0.tgz#f8259ae0e52a123eb40f552551e647b506a94d83" + integrity sha512-97z/ju/Jy1rZmDxybphrBuI+jtJjFVoz7Mr9yUQVVVi+DNZE333uFQeMOqcCIy1x3WYBIbWftUSLmbNXNT7qFQ== + dependencies: + "@ampproject/remapping" "^2.2.0" + "@babel/code-frame" "^7.22.13" + "@babel/generator" "^7.23.0" + "@babel/helper-compilation-targets" "^7.22.15" + "@babel/helper-module-transforms" "^7.23.0" + "@babel/helpers" "^7.23.0" + "@babel/parser" "^7.23.0" + "@babel/template" "^7.22.15" + "@babel/traverse" "^7.23.0" + "@babel/types" "^7.23.0" + convert-source-map "^2.0.0" + debug "^4.1.0" + gensync "^1.0.0-beta.2" + json5 "^2.2.3" + semver "^6.3.1" + +"@babel/generator@^7.23.0": + version "7.23.0" + resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.23.0.tgz#df5c386e2218be505b34837acbcb874d7a983420" + integrity sha512-lN85QRR+5IbYrMWM6Y4pE/noaQtg4pNiqeNGX60eqOfo6gtEj6uw/JagelB8vVztSd7R6M5n1+PQkDbHbBRU4g== + dependencies: + "@babel/types" "^7.23.0" + "@jridgewell/gen-mapping" "^0.3.2" + "@jridgewell/trace-mapping" "^0.3.17" + jsesc "^2.5.1" + +"@babel/helper-compilation-targets@^7.22.15": + version "7.22.15" + resolved "https://registry.yarnpkg.com/@babel/helper-compilation-targets/-/helper-compilation-targets-7.22.15.tgz#0698fc44551a26cf29f18d4662d5bf545a6cfc52" + integrity sha512-y6EEzULok0Qvz8yyLkCvVX+02ic+By2UdOhylwUOvOn9dvYc9mKICJuuU1n1XBI02YWsNsnrY1kc6DVbjcXbtw== + dependencies: + "@babel/compat-data" "^7.22.9" + "@babel/helper-validator-option" "^7.22.15" + browserslist "^4.21.9" + lru-cache "^5.1.1" + semver "^6.3.1" + +"@babel/helper-environment-visitor@^7.22.20": + version "7.22.20" + resolved "https://registry.yarnpkg.com/@babel/helper-environment-visitor/-/helper-environment-visitor-7.22.20.tgz#96159db61d34a29dba454c959f5ae4a649ba9167" + integrity sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA== + +"@babel/helper-function-name@^7.23.0": + version "7.23.0" + resolved "https://registry.yarnpkg.com/@babel/helper-function-name/-/helper-function-name-7.23.0.tgz#1f9a3cdbd5b2698a670c30d2735f9af95ed52759" + integrity sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw== + dependencies: + "@babel/template" "^7.22.15" + "@babel/types" "^7.23.0" + +"@babel/helper-hoist-variables@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-hoist-variables/-/helper-hoist-variables-7.22.5.tgz#c01a007dac05c085914e8fb652b339db50d823bb" + integrity sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-module-imports@^7.22.15": + version "7.22.15" + resolved "https://registry.yarnpkg.com/@babel/helper-module-imports/-/helper-module-imports-7.22.15.tgz#16146307acdc40cc00c3b2c647713076464bdbf0" + integrity sha512-0pYVBnDKZO2fnSPCrgM/6WMc7eS20Fbok+0r88fp+YtWVLZrp4CkafFGIp+W0VKw4a22sgebPT99y+FDNMdP4w== + dependencies: + "@babel/types" "^7.22.15" + +"@babel/helper-module-transforms@^7.23.0": + version "7.23.0" + resolved "https://registry.yarnpkg.com/@babel/helper-module-transforms/-/helper-module-transforms-7.23.0.tgz#3ec246457f6c842c0aee62a01f60739906f7047e" + integrity sha512-WhDWw1tdrlT0gMgUJSlX0IQvoO1eN279zrAUbVB+KpV2c3Tylz8+GnKOLllCS6Z/iZQEyVYxhZVUdPTqs2YYPw== + dependencies: + "@babel/helper-environment-visitor" "^7.22.20" + "@babel/helper-module-imports" "^7.22.15" + "@babel/helper-simple-access" "^7.22.5" + "@babel/helper-split-export-declaration" "^7.22.6" + "@babel/helper-validator-identifier" "^7.22.20" + +"@babel/helper-simple-access@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-simple-access/-/helper-simple-access-7.22.5.tgz#4938357dc7d782b80ed6dbb03a0fba3d22b1d5de" + integrity sha512-n0H99E/K+Bika3++WNL17POvo4rKWZ7lZEp1Q+fStVbUi8nxPQEBOlTmCOxW/0JsS56SKKQ+ojAe2pHKJHN35w== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-split-export-declaration@^7.22.6": + version "7.22.6" + resolved "https://registry.yarnpkg.com/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.22.6.tgz#322c61b7310c0997fe4c323955667f18fcefb91c" + integrity sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g== + dependencies: + "@babel/types" "^7.22.5" + +"@babel/helper-string-parser@^7.22.5": + version "7.22.5" + resolved "https://registry.yarnpkg.com/@babel/helper-string-parser/-/helper-string-parser-7.22.5.tgz#533f36457a25814cf1df6488523ad547d784a99f" + integrity sha512-mM4COjgZox8U+JcXQwPijIZLElkgEpO5rsERVDJTc2qfCDfERyob6k5WegS14SX18IIjv+XD+GrqNumY5JRCDw== + "@babel/helper-validator-identifier@^7.14.5": version "7.15.7" resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.15.7.tgz#220df993bfe904a4a6b02ab4f3385a5ebf6e2389" integrity sha512-K4JvCtQqad9OY2+yTU8w+E82ywk/fe+ELNlt1G8z3bVGlZfn/hOcQQsUhGhW/N+tb3fxK800wLtKOE/aM0m72w== -"@babel/helper-validator-identifier@^7.22.5": +"@babel/helper-validator-identifier@^7.22.20", "@babel/helper-validator-identifier@^7.22.5": version "7.22.20" resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.20.tgz#c4ae002c61d2879e724581d96665583dbc1dc0e0" integrity sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A== +"@babel/helper-validator-option@^7.22.15": + version "7.22.15" + resolved "https://registry.yarnpkg.com/@babel/helper-validator-option/-/helper-validator-option-7.22.15.tgz#694c30dfa1d09a6534cdfcafbe56789d36aba040" + integrity sha512-bMn7RmyFjY/mdECUbgn9eoSY4vqvacUnS9i9vGAGttgFWesO6B4CYWA7XlpbWgBt71iv/hfbPlynohStqnu5hA== + +"@babel/helpers@^7.23.0": + version "7.23.1" + resolved "https://registry.yarnpkg.com/@babel/helpers/-/helpers-7.23.1.tgz#44e981e8ce2b9e99f8f0b703f3326a4636c16d15" + integrity sha512-chNpneuK18yW5Oxsr+t553UZzzAs3aZnFm4bxhebsNTeshrC95yA7l5yl7GBAG+JG1rF0F7zzD2EixK9mWSDoA== + dependencies: + "@babel/template" "^7.22.15" + "@babel/traverse" "^7.23.0" + "@babel/types" "^7.23.0" + "@babel/highlight@^7.14.5": version "7.14.5" resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.14.5.tgz#6861a52f03966405001f6aa534a01a24d99e8cd9" @@ -33,11 +167,54 @@ chalk "^2.0.0" js-tokens "^4.0.0" -"@babel/parser@^7.0.0", "@babel/parser@^7.21.4": +"@babel/highlight@^7.22.13": + version "7.22.20" + resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.22.20.tgz#4ca92b71d80554b01427815e06f2df965b9c1f54" + integrity sha512-dkdMCN3py0+ksCgYmGG8jKeGA/8Tk+gJwSYYlFGxG5lmhfKNoAy004YpLxpS1W2J8m/EK2Ew+yOs9pVRwO89mg== + dependencies: + "@babel/helper-validator-identifier" "^7.22.20" + chalk "^2.4.2" + js-tokens "^4.0.0" + +"@babel/parser@^7.0.0", "@babel/parser@^7.21.4", "@babel/parser@^7.22.15", "@babel/parser@^7.23.0": version "7.23.0" resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.23.0.tgz#da950e622420bf96ca0d0f2909cdddac3acd8719" integrity sha512-vvPKKdMemU85V9WE/l5wZEmImpCtLqbnTvqDS2U1fJ96KrxoW7KrXhNsNCblQlg8Ck4b85yxdTyelsMUgFUXiw== +"@babel/template@^7.22.15": + version "7.22.15" + resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.22.15.tgz#09576efc3830f0430f4548ef971dde1350ef2f38" + integrity sha512-QPErUVm4uyJa60rkI73qneDacvdvzxshT3kksGqlGWYdOTIUOwJ7RDUL8sGqslY1uXWSL6xMFKEXDS3ox2uF0w== + dependencies: + "@babel/code-frame" "^7.22.13" + "@babel/parser" "^7.22.15" + "@babel/types" "^7.22.15" + +"@babel/traverse@^7.23.0": + version "7.23.0" + resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.23.0.tgz#18196ddfbcf4ccea324b7f6d3ada00d8c5a99c53" + integrity sha512-t/QaEvyIoIkwzpiZ7aoSKK8kObQYeF7T2v+dazAYCb8SXtp58zEVkWW7zAnju8FNKNdr4ScAOEDmMItbyOmEYw== + dependencies: + "@babel/code-frame" "^7.22.13" + "@babel/generator" "^7.23.0" + "@babel/helper-environment-visitor" "^7.22.20" + "@babel/helper-function-name" "^7.23.0" + "@babel/helper-hoist-variables" "^7.22.5" + "@babel/helper-split-export-declaration" "^7.22.6" + "@babel/parser" "^7.23.0" + "@babel/types" "^7.23.0" + debug "^4.1.0" + globals "^11.1.0" + +"@babel/types@^7.22.15", "@babel/types@^7.22.5", "@babel/types@^7.23.0": + version "7.23.0" + resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.23.0.tgz#8c1f020c9df0e737e4e247c0619f58c68458aaeb" + integrity sha512-0oIyUfKoI3mSqMvsxBdclDwxXKXAUA8v/apZbc+iSyARYou1o8ZGDxbUYyLFoW2arqS2jDGqJuZvv1d/io1axg== + dependencies: + "@babel/helper-string-parser" "^7.22.5" + "@babel/helper-validator-identifier" "^7.22.20" + to-fast-properties "^2.0.0" + "@colors/colors@1.5.0": version "1.5.0" resolved "https://registry.yarnpkg.com/@colors/colors/-/colors-1.5.0.tgz#bb504579c1cae923e6576a4f5da43d25f97bdbd9" @@ -263,6 +440,22 @@ resolved "https://registry.yarnpkg.com/@isaacs/string-locale-compare/-/string-locale-compare-1.1.0.tgz#291c227e93fd407a96ecd59879a35809120e432b" integrity sha512-SQ7Kzhh9+D+ZW9MA0zkYv3VXhIDNx+LzM6EJ+/65I3QY+enU6Itte7E5XX7EWrqLW2FN4n06GWzBnPoC3th2aQ== +"@istanbuljs/load-nyc-config@^1.0.0": + version "1.1.0" + resolved "https://registry.yarnpkg.com/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz#fd3db1d59ecf7cf121e80650bb86712f9b55eced" + integrity sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ== + dependencies: + camelcase "^5.3.1" + find-up "^4.1.0" + get-package-type "^0.1.0" + js-yaml "^3.13.1" + resolve-from "^5.0.0" + +"@istanbuljs/schema@^0.1.2": + version "0.1.3" + resolved "https://registry.yarnpkg.com/@istanbuljs/schema/-/schema-0.1.3.tgz#e45e384e4b8ec16bce2fd903af78450f6bf7ec98" + integrity sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA== + "@jest/schemas@^29.6.3": version "29.6.3" resolved "https://registry.yarnpkg.com/@jest/schemas/-/schemas-29.6.3.tgz#430b5ce8a4e0044a7e3819663305a7b3091c8e03" @@ -270,16 +463,40 @@ dependencies: "@sinclair/typebox" "^0.27.8" +"@jridgewell/gen-mapping@^0.3.0", "@jridgewell/gen-mapping@^0.3.2": + version "0.3.3" + resolved "https://registry.yarnpkg.com/@jridgewell/gen-mapping/-/gen-mapping-0.3.3.tgz#7e02e6eb5df901aaedb08514203b096614024098" + integrity sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ== + dependencies: + "@jridgewell/set-array" "^1.0.1" + "@jridgewell/sourcemap-codec" "^1.4.10" + "@jridgewell/trace-mapping" "^0.3.9" + "@jridgewell/resolve-uri@^3.0.3": version "3.1.0" resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.1.0.tgz#2203b118c157721addfe69d47b70465463066d78" integrity sha512-F2msla3tad+Mfht5cJq7LSXcdudKTWCVYUgw6pLFOOHSTtZlj6SWNYAp+AhuqLmWdBO2X5hPrLcu8cVP8fy28w== +"@jridgewell/resolve-uri@^3.1.0": + version "3.1.1" + resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz#c08679063f279615a3326583ba3a90d1d82cc721" + integrity sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA== + +"@jridgewell/set-array@^1.0.1": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@jridgewell/set-array/-/set-array-1.1.2.tgz#7c6cf998d6d20b914c0a55a91ae928ff25965e72" + integrity sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw== + "@jridgewell/sourcemap-codec@^1.4.10": version "1.4.14" resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.14.tgz#add4c98d341472a289190b424efbdb096991bb24" integrity sha512-XPSJHWmi394fuUuzDnGz1wiKqWfo1yXecHQMRf2l6hztTO+nPru658AyDngaBe7isIxEkRsPR3FZh+s7iVa4Uw== +"@jridgewell/sourcemap-codec@^1.4.14": + version "1.4.15" + resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz#d7c6e6755c78567a951e04ab52ef0fd26de59f32" + integrity sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg== + "@jridgewell/trace-mapping@0.3.9": version "0.3.9" resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz#6534fd5933a53ba7cbf3a17615e273a0d1273ff9" @@ -288,6 +505,14 @@ "@jridgewell/resolve-uri" "^3.0.3" "@jridgewell/sourcemap-codec" "^1.4.10" +"@jridgewell/trace-mapping@^0.3.17", "@jridgewell/trace-mapping@^0.3.9": + version "0.3.19" + resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.19.tgz#f8a3249862f91be48d3127c3cfe992f79b4b8811" + integrity sha512-kf37QtfW+Hwx/buWGMPcR60iF9ziHa6r/CZJIHbmcm4+0qrXiVdxegAH0F6yddEVQ7zdkjcGCgCzUu+BcbhQxw== + dependencies: + "@jridgewell/resolve-uri" "^3.1.0" + "@jridgewell/sourcemap-codec" "^1.4.14" + "@nodelib/fs.scandir@2.1.5": version "2.1.5" resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" @@ -1217,12 +1442,19 @@ app-module-path@^2.2.0: resolved "https://registry.yarnpkg.com/app-module-path/-/app-module-path-2.2.0.tgz#641aa55dfb7d6a6f0a8141c4b9c0aa50b6c24dd5" integrity sha512-gkco+qxENJV+8vFcDiiFhuoSvRXb2a/QPqpSoWhVz829VNJfOTnELbBmPmNKFxf3xdNnw4DWCkzkDaavcX/1YQ== +append-transform@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/append-transform/-/append-transform-2.0.0.tgz#99d9d29c7b38391e6f428d28ce136551f0b77e12" + integrity sha512-7yeyCEurROLQJFv5Xj4lEGTy0borxepjFv1g22oAdqFu//SrAlDl1O1Nxx15SH1RoliUml6p8dwJW9jvZughhg== + dependencies: + default-require-extensions "^3.0.0" + "aproba@^1.0.3 || ^2.0.0", aproba@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/aproba/-/aproba-2.0.0.tgz#52520b8ae5b569215b354efc0caa3fe1e45a8adc" integrity sha512-lYe4Gx7QT+MKGbDsA+Z+he/Wtef0BiwDOlK/XkBrdfsh9J/jPPXbX0tE9x9cl27Tmu5gg3QUbUrQYa/y+KOHPQ== -archy@~1.0.0: +archy@^1.0.0, archy@~1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/archy/-/archy-1.0.0.tgz#f9c8c13757cc1dd7bc379ac77b2c62a5c2868c40" integrity sha512-Xg+9RwCg/0p32teKdGMPTPnVXKD0w3DfHnFTficozsAgsvq2XenPJq/MYpzzQ/v8zrOyJn6Ds39VA4JIDwFfqw== @@ -1447,6 +1679,16 @@ browser-stdout@1.3.1: resolved "https://registry.yarnpkg.com/browser-stdout/-/browser-stdout-1.3.1.tgz#baa559ee14ced73452229bad7326467c61fabd60" integrity sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw== +browserslist@^4.21.9: + version "4.22.0" + resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.22.0.tgz#6adc8116589ccea8a99d0df79c5de2436199abdb" + integrity sha512-v+Jcv64L2LbfTC6OnRcaxtqJNJuQAVhZKSJfR/6hn7lhnChUXl4amwVviqN1k411BB+3rRoKMitELRn1CojeRA== + dependencies: + caniuse-lite "^1.0.30001539" + electron-to-chromium "^1.4.530" + node-releases "^2.0.13" + update-browserslist-db "^1.0.13" + buffer@^5.5.0: version "5.7.1" resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0" @@ -1493,6 +1735,16 @@ cacache@^17.0.0, cacache@^17.0.4, cacache@^17.1.3: tar "^6.1.11" unique-filename "^3.0.0" +caching-transform@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/caching-transform/-/caching-transform-4.0.0.tgz#00d297a4206d71e2163c39eaffa8157ac0651f0f" + integrity sha512-kpqOvwXnjjN44D89K5ccQC+RUrsy7jB/XLlRrx0D7/2HNcTPqzsb6XgYoErwko6QsV184CA2YgS1fxDiiDZMWA== + dependencies: + hasha "^5.0.0" + make-dir "^3.0.0" + package-hash "^4.0.0" + write-file-atomic "^3.0.0" + call-bind@^1.0.0, call-bind@^1.0.2: version "1.0.2" resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.2.tgz#b1d4e89e688119c3c9a903ad30abb2f6a919be3c" @@ -1515,7 +1767,7 @@ camelcase-keys@^6.2.2: map-obj "^4.0.0" quick-lru "^4.0.1" -camelcase@^5.3.1: +camelcase@^5.0.0, camelcase@^5.3.1: version "5.3.1" resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.3.1.tgz#e3c9b31569e106811df242f715725a1f4c494320" integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg== @@ -1525,6 +1777,11 @@ camelcase@^6.0.0: resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.2.0.tgz#924af881c9d525ac9d87f40d964e5cea982a1809" integrity sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg== +caniuse-lite@^1.0.30001539: + version "1.0.30001540" + resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001540.tgz#a316ca4f2ae673ab02ff0ec533334016d56ff658" + integrity sha512-9JL38jscuTJBTcuETxm8QLsFr/F6v0CYYTEU6r5+qSM98P2Q0Hmu0eG1dTG5GBUmywU3UlcVOUSIJYY47rdFSw== + cardinal@^2.1.1: version "2.1.1" resolved "https://registry.yarnpkg.com/cardinal/-/cardinal-2.1.1.tgz#7cc1055d822d212954d07b085dea251cc7bc5505" @@ -1553,7 +1810,7 @@ chai@^4.3.7: pathval "^1.1.1" type-detect "^4.0.5" -chalk@^2.0.0: +chalk@^2.0.0, chalk@^2.4.2: version "2.4.2" resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424" integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ== @@ -1674,6 +1931,15 @@ cli-table3@^0.6.3: optionalDependencies: "@colors/colors" "1.5.0" +cliui@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/cliui/-/cliui-6.0.0.tgz#511d702c0c4e41ca156d7d0e96021f23e13225b1" + integrity sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ== + dependencies: + string-width "^4.2.0" + strip-ansi "^6.0.0" + wrap-ansi "^6.2.0" + cliui@^7.0.2: version "7.0.4" resolved "https://registry.yarnpkg.com/cliui/-/cliui-7.0.4.tgz#a0265ee655476fc807aea9df3df8df7783808b4f" @@ -1821,6 +2087,16 @@ conventional-commits-parser@^3.0.0: through2 "^4.0.0" trim-off-newlines "^1.0.0" +convert-source-map@^1.7.0: + version "1.9.0" + resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-1.9.0.tgz#7faae62353fb4213366d0ca98358d22e8368b05f" + integrity sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A== + +convert-source-map@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-2.0.0.tgz#4b560f649fc4e918dd0ab75cf4961e8bc882d82a" + integrity sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg== + cosmiconfig@^7.0.0: version "7.0.0" resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-7.0.0.tgz#ef9b44d773959cae63ddecd122de23853b60f8d3" @@ -1855,7 +2131,7 @@ cross-spawn@^6.0.5: shebang-command "^1.2.0" which "^1.2.9" -cross-spawn@^7.0.0, cross-spawn@^7.0.1, cross-spawn@^7.0.2: +cross-spawn@^7.0.0, cross-spawn@^7.0.1, cross-spawn@^7.0.2, cross-spawn@^7.0.3: version "7.0.3" resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w== @@ -1923,6 +2199,13 @@ deep-is@^0.1.3: resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.3.tgz#b369d6fb5dbc13eecf524f91b070feedc357cf34" integrity sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ= +default-require-extensions@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/default-require-extensions/-/default-require-extensions-3.0.1.tgz#bfae00feeaeada68c2ae256c62540f60b80625bd" + integrity sha512-eXTJmRbm2TIt9MgWTsOH1wEuhew6XGZcMeGKCtLedIg/NCsg1iBePXkceTdK4Fii7pzmN9tGsZhKzZ4h7O/fxw== + dependencies: + strip-bom "^4.0.0" + defaults@^1.0.3: version "1.0.4" resolved "https://registry.yarnpkg.com/defaults/-/defaults-1.0.4.tgz#b0b02062c1e2aa62ff5d9528f0f98baa90978d7a" @@ -2169,6 +2452,11 @@ ejs@^3.1.8: dependencies: jake "^10.8.5" +electron-to-chromium@^1.4.530: + version "1.4.531" + resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.531.tgz#22966d894c4680726c17cf2908ee82ff5d26ac25" + integrity sha512-H6gi5E41Rn3/mhKlPaT1aIMg/71hTAqn0gYEllSuw9igNWtvQwu185jiCZoZD29n7Zukgh7GVZ3zGf0XvkhqjQ== + emoji-regex@^8.0.0: version "8.0.0" resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" @@ -2281,6 +2569,11 @@ es-to-primitive@^1.2.1: is-date-object "^1.0.1" is-symbol "^1.0.2" +es6-error@^4.0.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/es6-error/-/es6-error-4.1.1.tgz#9e3af407459deed47e9a91f9b885a84eb05c561d" + integrity sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg== + escalade@^3.1.1: version "3.1.1" resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.1.1.tgz#d8cfdc7000965c5a0174b4a82eaa5c0552742e40" @@ -2721,6 +3014,15 @@ fill-range@^7.0.1: dependencies: to-regex-range "^5.0.1" +find-cache-dir@^3.2.0: + version "3.3.2" + resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-3.3.2.tgz#b30c5b6eff0730731aea9bbd9dbecbd80256d64b" + integrity sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig== + dependencies: + commondir "^1.0.1" + make-dir "^3.0.2" + pkg-dir "^4.1.0" + find-up@5.0.0, find-up@^5.0.0: version "5.0.0" resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc" @@ -2729,7 +3031,7 @@ find-up@5.0.0, find-up@^5.0.0: locate-path "^6.0.0" path-exists "^4.0.0" -find-up@^4.1.0: +find-up@^4.0.0, find-up@^4.1.0: version "4.1.0" resolved "https://registry.yarnpkg.com/find-up/-/find-up-4.1.0.tgz#97afe7d6cdc0bc5928584b7c8d7b16e8a9aa5d19" integrity sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw== @@ -2767,6 +3069,14 @@ for-each@^0.3.3: dependencies: is-callable "^1.1.3" +foreground-child@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/foreground-child/-/foreground-child-2.0.0.tgz#71b32800c9f15aa8f2f83f4a6bd9bff35d861a53" + integrity sha512-dCIq9FpEcyQyXKCkyzmlPTFNgrCzPudOe+mhvJU5zAtlBnGVy2yKxtfsxK2tQBThwq225jcvBjpw1Gr40uzZCA== + dependencies: + cross-spawn "^7.0.0" + signal-exit "^3.0.2" + foreground-child@^3.1.0: version "3.1.1" resolved "https://registry.yarnpkg.com/foreground-child/-/foreground-child-3.1.1.tgz#1d173e776d75d2772fed08efe4a0de1ea1b12d0d" @@ -2775,6 +3085,11 @@ foreground-child@^3.1.0: cross-spawn "^7.0.0" signal-exit "^4.0.1" +fromentries@^1.2.0: + version "1.3.2" + resolved "https://registry.yarnpkg.com/fromentries/-/fromentries-1.3.2.tgz#e4bca6808816bf8f93b52750f1127f5a6fd86e3a" + integrity sha512-cHEpEQHUg0f8XdtZCc2ZAhrHzKzT0MrFUTcvx+hfxYu7rGMDc5SKoXFh+n4YigxsHXRzc6OrCshdR1bWH6HHyg== + fs-extra@^9.0, fs-extra@^9.0.0, fs-extra@^9.1.0: version "9.1.0" resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-9.1.0.tgz#5954460c764a8da2094ba3554bf839e6b9a7c86d" @@ -2857,6 +3172,11 @@ gauge@^5.0.0: strip-ansi "^6.0.1" wide-align "^1.1.5" +gensync@^1.0.0-beta.2: + version "1.0.0-beta.2" + resolved "https://registry.yarnpkg.com/gensync/-/gensync-1.0.0-beta.2.tgz#32a6ee76c3d7f52d46b2b1ae5d93fea8580a25e0" + integrity sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg== + get-amd-module-type@^3.0.0: version "3.0.2" resolved "https://registry.yarnpkg.com/get-amd-module-type/-/get-amd-module-type-3.0.2.tgz#46550cee2b8e1fa4c3f2c8a5753c36990aa49ab0" @@ -2873,7 +3193,7 @@ get-amd-module-type@^4.1.0: ast-module-types "^4.0.0" node-source-walk "^5.0.1" -get-caller-file@^2.0.5: +get-caller-file@^2.0.1, get-caller-file@^2.0.5: version "2.0.5" resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e" integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg== @@ -2997,6 +3317,11 @@ global-dirs@^0.1.1: dependencies: ini "^1.3.4" +globals@^11.1.0: + version "11.12.0" + resolved "https://registry.yarnpkg.com/globals/-/globals-11.12.0.tgz#ab8795338868a0babd8525758018c2a7eb95c42e" + integrity sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA== + globals@^13.19.0: version "13.21.0" resolved "https://registry.yarnpkg.com/globals/-/globals-13.21.0.tgz#163aae12f34ef502f5153cfbdd3600f36c63c571" @@ -3108,6 +3433,14 @@ has@^1.0.3: dependencies: function-bind "^1.1.1" +hasha@^5.0.0: + version "5.2.2" + resolved "https://registry.yarnpkg.com/hasha/-/hasha-5.2.2.tgz#a48477989b3b327aea3c04f53096d816d97522a1" + integrity sha512-Hrp5vIK/xr5SkeN2onO32H0MgNZ0f17HRNH39WfL0SYUNOTZ5Lz1TJ8Pajo/87dYGEFlLMm7mIc/k/s6Bvz9HQ== + dependencies: + is-stream "^2.0.0" + type-fest "^0.8.0" + he@1.2.0: version "1.2.0" resolved "https://registry.yarnpkg.com/he/-/he-1.2.0.tgz#84ae65fa7eafb165fddb61566ae14baf05664f0f" @@ -3132,6 +3465,11 @@ hosted-git-info@^6.0.0, hosted-git-info@^6.1.1: dependencies: lru-cache "^7.5.1" +html-escaper@^2.0.0: + version "2.0.2" + resolved "https://registry.yarnpkg.com/html-escaper/-/html-escaper-2.0.2.tgz#dfd60027da36a36dfcbe236262c00a5822681453" + integrity sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg== + http-cache-semantics@^4.1.1: version "4.1.1" resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz#abe02fcb2985460bf0323be664436ec3476a6d5a" @@ -3525,6 +3863,11 @@ is-typed-array@^1.1.10, is-typed-array@^1.1.12, is-typed-array@^1.1.9: dependencies: which-typed-array "^1.1.11" +is-typedarray@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a" + integrity sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA== + is-unicode-supported@^0.1.0: version "0.1.0" resolved "https://registry.yarnpkg.com/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz#3f26c76a809593b52bfa2ecb5710ed2779b522a7" @@ -3547,6 +3890,11 @@ is-weakref@^1.0.2: dependencies: call-bind "^1.0.2" +is-windows@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-windows/-/is-windows-1.0.2.tgz#d1850eb9791ecd18e6182ce12a30f396634bb19d" + integrity sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA== + is-wsl@^2.2.0: version "2.2.0" resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-2.2.0.tgz#74a4c76e77ca9fd3f932f290c17ea326cd157271" @@ -3569,6 +3917,66 @@ isexe@^2.0.0: resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" integrity sha1-6PvzdNxVb/iUehDcsFctYz8s+hA= +istanbul-lib-coverage@^3.0.0, istanbul-lib-coverage@^3.2.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.0.tgz#189e7909d0a39fa5a3dfad5b03f71947770191d3" + integrity sha512-eOeJ5BHCmHYvQK7xt9GkdHuzuCGS1Y6g9Gvnx3Ym33fz/HpLRYxiS0wHNr+m/MBC8B647Xt608vCDEvhl9c6Mw== + +istanbul-lib-hook@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/istanbul-lib-hook/-/istanbul-lib-hook-3.0.0.tgz#8f84c9434888cc6b1d0a9d7092a76d239ebf0cc6" + integrity sha512-Pt/uge1Q9s+5VAZ+pCo16TYMWPBIl+oaNIjgLQxcX0itS6ueeaA+pEfThZpH8WxhFgCiEb8sAJY6MdUKgiIWaQ== + dependencies: + append-transform "^2.0.0" + +istanbul-lib-instrument@^4.0.0: + version "4.0.3" + resolved "https://registry.yarnpkg.com/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.3.tgz#873c6fff897450118222774696a3f28902d77c1d" + integrity sha512-BXgQl9kf4WTCPCCpmFGoJkz/+uhvm7h7PFKUYxh7qarQd3ER33vHG//qaE8eN25l07YqZPpHXU9I09l/RD5aGQ== + dependencies: + "@babel/core" "^7.7.5" + "@istanbuljs/schema" "^0.1.2" + istanbul-lib-coverage "^3.0.0" + semver "^6.3.0" + +istanbul-lib-processinfo@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/istanbul-lib-processinfo/-/istanbul-lib-processinfo-2.0.3.tgz#366d454cd0dcb7eb6e0e419378e60072c8626169" + integrity sha512-NkwHbo3E00oybX6NGJi6ar0B29vxyvNwoC7eJ4G4Yq28UfY758Hgn/heV8VRFhevPED4LXfFz0DQ8z/0kw9zMg== + dependencies: + archy "^1.0.0" + cross-spawn "^7.0.3" + istanbul-lib-coverage "^3.2.0" + p-map "^3.0.0" + rimraf "^3.0.0" + uuid "^8.3.2" + +istanbul-lib-report@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz#908305bac9a5bd175ac6a74489eafd0fc2445a7d" + integrity sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw== + dependencies: + istanbul-lib-coverage "^3.0.0" + make-dir "^4.0.0" + supports-color "^7.1.0" + +istanbul-lib-source-maps@^4.0.0: + version "4.0.1" + resolved "https://registry.yarnpkg.com/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz#895f3a709fcfba34c6de5a42939022f3e4358551" + integrity sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw== + dependencies: + debug "^4.1.1" + istanbul-lib-coverage "^3.0.0" + source-map "^0.6.1" + +istanbul-reports@^3.0.2: + version "3.1.6" + resolved "https://registry.yarnpkg.com/istanbul-reports/-/istanbul-reports-3.1.6.tgz#2544bcab4768154281a2f0870471902704ccaa1a" + integrity sha512-TLgnMkKg3iTDsQ9PbPTdpfAK2DzjF9mqUG7RMgcQl8oFjad8ob4laGxv5XV5U9MAfx8D6tSJiUyuAwzLicaxlg== + dependencies: + html-escaper "^2.0.0" + istanbul-lib-report "^3.0.0" + jackspeak@^2.0.3: version "2.3.0" resolved "https://registry.yarnpkg.com/jackspeak/-/jackspeak-2.3.0.tgz#aa228a94de830f31d4e4f0184427ce91c4ff1493" @@ -3615,7 +4023,7 @@ js-yaml@4.1.0, js-yaml@^4.1.0: dependencies: argparse "^2.0.1" -js-yaml@^3.14.1: +js-yaml@^3.13.1, js-yaml@^3.14.1: version "3.14.1" resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.14.1.tgz#dae812fdb3825fa306609a8717383c50c36a0537" integrity sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g== @@ -3623,6 +4031,11 @@ js-yaml@^3.14.1: argparse "^1.0.7" esprima "^4.0.0" +jsesc@^2.5.1: + version "2.5.2" + resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-2.5.2.tgz#80564d2e483dacf6e8ef209650a67df3f0c283a4" + integrity sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA== + jsesc@^3.0.2: version "3.0.2" resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-3.0.2.tgz#bb8b09a6597ba426425f2e4a07245c3d00b9343e" @@ -3675,6 +4088,11 @@ json5@^1.0.2: dependencies: minimist "^1.2.0" +json5@^2.2.3: + version "2.2.3" + resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.3.tgz#78cd6f1a19bdc12b73db5ad0c61efd66c1e29283" + integrity sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg== + jsonfile@^6.0.1: version "6.1.0" resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-6.1.0.tgz#bc55b2634793c679ec6403094eb13698a6ec0aae" @@ -3860,6 +4278,11 @@ locate-path@^6.0.0: dependencies: p-locate "^5.0.0" +lodash.flattendeep@^4.4.0: + version "4.4.0" + resolved "https://registry.yarnpkg.com/lodash.flattendeep/-/lodash.flattendeep-4.4.0.tgz#fb030917f86a3134e5bc9bec0d69e0013ddfedb2" + integrity sha512-uHaJFihxmJcEX3kT4I23ABqKKalJ/zDrDg0lsFtc1h+3uw49SIJ5beyhx5ExVRti3AvKoOJngIj7xz3oylPdWQ== + lodash.get@^4.4.2: version "4.4.2" resolved "https://registry.yarnpkg.com/lodash.get/-/lodash.get-4.4.2.tgz#2d177f652fa31e939b4438d5341499dfa3825e99" @@ -3890,6 +4313,13 @@ loupe@^2.3.1: dependencies: get-func-name "^2.0.0" +lru-cache@^5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-5.1.1.tgz#1da27e6710271947695daf6848e847f01d84b920" + integrity sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w== + dependencies: + yallist "^3.0.2" + lru-cache@^6.0.0: version "6.0.0" resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" @@ -3935,6 +4365,20 @@ madge@^6.1.0: ts-graphviz "^1.5.0" walkdir "^0.4.1" +make-dir@^3.0.0, make-dir@^3.0.2: + version "3.1.0" + resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-3.1.0.tgz#415e967046b3a7f1d185277d84aa58203726a13f" + integrity sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw== + dependencies: + semver "^6.0.0" + +make-dir@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-4.0.0.tgz#c3c2307a771277cd9638305f915c29ae741b614e" + integrity sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw== + dependencies: + semver "^7.5.3" + make-error@^1.1.1: version "1.3.6" resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2" @@ -4316,6 +4760,18 @@ node-gyp@^9.0.0, node-gyp@^9.4.0: tar "^6.1.2" which "^2.0.2" +node-preload@^0.2.1: + version "0.2.1" + resolved "https://registry.yarnpkg.com/node-preload/-/node-preload-0.2.1.tgz#c03043bb327f417a18fee7ab7ee57b408a144301" + integrity sha512-RM5oyBy45cLEoHqCeh+MNuFAxO0vTFBLskvQbOKnEE7YTTSN4tbN8QWDIPQ6L+WvKsB/qLEGpYe2ZZ9d4W9OIQ== + dependencies: + process-on-spawn "^1.0.0" + +node-releases@^2.0.13: + version "2.0.13" + resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.13.tgz#d5ed1627c23e3461e819b02e57b75e4899b1c81d" + integrity sha512-uYr7J37ae/ORWdZeQ1xxMJe3NtdmqMC/JZK+geofDrkLUApKRHPd18/TxtBOJ4A0/+uUIliorNrfYV6s1b02eQ== + node-source-walk@^4.0.0, node-source-walk@^4.2.0, node-source-walk@^4.2.2: version "4.3.0" resolved "https://registry.yarnpkg.com/node-source-walk/-/node-source-walk-4.3.0.tgz#8336b56cfed23ac5180fe98f1e3bb6b11fd5317c" @@ -4557,6 +5013,39 @@ npmlog@^7.0.1: gauge "^5.0.0" set-blocking "^2.0.0" +nyc@^15.1.0: + version "15.1.0" + resolved "https://registry.yarnpkg.com/nyc/-/nyc-15.1.0.tgz#1335dae12ddc87b6e249d5a1994ca4bdaea75f02" + integrity sha512-jMW04n9SxKdKi1ZMGhvUTHBN0EICCRkHemEoE5jm6mTYcqcdas0ATzgUgejlQUHMvpnOZqGB5Xxsv9KxJW1j8A== + dependencies: + "@istanbuljs/load-nyc-config" "^1.0.0" + "@istanbuljs/schema" "^0.1.2" + caching-transform "^4.0.0" + convert-source-map "^1.7.0" + decamelize "^1.2.0" + find-cache-dir "^3.2.0" + find-up "^4.1.0" + foreground-child "^2.0.0" + get-package-type "^0.1.0" + glob "^7.1.6" + istanbul-lib-coverage "^3.0.0" + istanbul-lib-hook "^3.0.0" + istanbul-lib-instrument "^4.0.0" + istanbul-lib-processinfo "^2.0.2" + istanbul-lib-report "^3.0.0" + istanbul-lib-source-maps "^4.0.0" + istanbul-reports "^3.0.2" + make-dir "^3.0.0" + node-preload "^0.2.1" + p-map "^3.0.0" + process-on-spawn "^1.0.0" + resolve-from "^5.0.0" + rimraf "^3.0.0" + signal-exit "^3.0.2" + spawn-wrap "^2.0.0" + test-exclude "^6.0.0" + yargs "^15.0.2" + object-inspect@^1.12.3, object-inspect@^1.9.0: version "1.12.3" resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.12.3.tgz#ba62dffd67ee256c8c086dfae69e016cd1f198b9" @@ -4679,6 +5168,13 @@ p-locate@^5.0.0: dependencies: p-limit "^3.0.2" +p-map@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/p-map/-/p-map-3.0.0.tgz#d704d9af8a2ba684e2600d9a215983d4141a979d" + integrity sha512-d3qXVTF/s+W+CdJ5A29wywV2n8CQQYahlgz2bFiA+4eVNJbHJodPZ+/gXwPGh0bOqA+j8S+6+ckmvLGPk1QpxQ== + dependencies: + aggregate-error "^3.0.0" + p-map@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/p-map/-/p-map-4.0.0.tgz#bb2f95a5eda2ec168ec9274e06a747c3e2904d2b" @@ -4691,6 +5187,16 @@ p-try@^2.0.0: resolved "https://registry.yarnpkg.com/p-try/-/p-try-2.2.0.tgz#cb2868540e313d61de58fafbe35ce9004d5540e6" integrity sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ== +package-hash@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/package-hash/-/package-hash-4.0.0.tgz#3537f654665ec3cc38827387fc904c163c54f506" + integrity sha512-whdkPIooSu/bASggZ96BWVvZTRMOFxnyUG5PnTSGKoJE2gd5mbVNmR2Nj20QFzxYYgAXpoqC+AiXzl+UMRh7zQ== + dependencies: + graceful-fs "^4.1.15" + hasha "^5.0.0" + lodash.flattendeep "^4.4.0" + release-zalgo "^1.0.0" + pacote@^15.0.0, pacote@^15.0.8, pacote@^15.2.0: version "15.2.0" resolved "https://registry.yarnpkg.com/pacote/-/pacote-15.2.0.tgz#0f0dfcc3e60c7b39121b2ac612bf8596e95344d3" @@ -4827,6 +5333,13 @@ pify@^4.0.1: resolved "https://registry.yarnpkg.com/pify/-/pify-4.0.1.tgz#4b2cd25c50d598735c50292224fd8c6df41e3231" integrity sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g== +pkg-dir@^4.1.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-4.2.0.tgz#f099133df7ede422e81d1d8448270eeb3e4261f3" + integrity sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ== + dependencies: + find-up "^4.0.0" + platform@^1.3.3: version "1.3.6" resolved "https://registry.yarnpkg.com/platform/-/platform-1.3.6.tgz#48b4ce983164b209c2d45a107adb31f473a6e7a7" @@ -4942,6 +5455,13 @@ proc-log@^3.0.0: resolved "https://registry.yarnpkg.com/proc-log/-/proc-log-3.0.0.tgz#fb05ef83ccd64fd7b20bbe9c8c1070fc08338dd8" integrity sha512-++Vn7NS4Xf9NacaU9Xq3URUuqZETPsf8L4j5/ckhaRYsfPeRyzGw+iDjFhV/Jr3uNmTvvddEJFWh5R1gRgUH8A== +process-on-spawn@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/process-on-spawn/-/process-on-spawn-1.0.0.tgz#95b05a23073d30a17acfdc92a440efd2baefdc93" + integrity sha512-1WsPDsUSMmZH5LeMLegqkPDrsGgsWwk1Exipy2hvB0o/F0ASzbpIctSCcZIK1ykJvtTJULEH+20WOFjMvGnCTg== + dependencies: + fromentries "^1.2.0" + process@^0.11.10: version "0.11.10" resolved "https://registry.yarnpkg.com/process/-/process-0.11.10.tgz#7332300e840161bda3e69a1d1d91a7d4bc16f182" @@ -5172,11 +5692,23 @@ regjsparser@^0.10.0: dependencies: jsesc "~0.5.0" +release-zalgo@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/release-zalgo/-/release-zalgo-1.0.0.tgz#09700b7e5074329739330e535c5a90fb67851730" + integrity sha512-gUAyHVHPPC5wdqX/LG4LWtRYtgjxyX78oanFNTMMyFEfOqdC54s3eE82imuWKbOeqYht2CrNf64Qb8vgmmtZGA== + dependencies: + es6-error "^4.0.1" + require-directory@^2.1.1: version "2.1.1" resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42" integrity sha1-jGStX9MNqxyXbiNE/+f3kqam30I= +require-main-filename@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/require-main-filename/-/require-main-filename-2.0.0.tgz#d0b329ecc7cc0f61649f62215be69af54aa8989b" + integrity sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg== + requirejs-config-file@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/requirejs-config-file/-/requirejs-config-file-4.0.0.tgz#4244da5dd1f59874038cc1091d078d620abb6ebc" @@ -5252,7 +5784,7 @@ reusify@^1.0.4: resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76" integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw== -rimraf@^3.0.2: +rimraf@^3.0.0, rimraf@^3.0.2: version "3.0.2" resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz#f1a5402ba6220ad52cc1282bac1ae3aa49fd061a" integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA== @@ -5314,16 +5846,16 @@ semver@7.3.5: dependencies: lru-cache "^6.0.0" +semver@^6.0.0, semver@^6.3.0, semver@^6.3.1: + version "6.3.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" + integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== + semver@^6.1.0: version "6.3.0" resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== -semver@^6.3.1: - version "6.3.1" - resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" - integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== - semver@^7.0.0, semver@^7.1.1, semver@^7.5.4: version "7.5.4" resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.4.tgz#483986ec4ed38e1c6c48c34894a9182dbff68a6e" @@ -5495,11 +6027,23 @@ source-map-js@^1.0.2: resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.0.2.tgz#adbc361d9c62df380125e7f161f71c826f1e490c" integrity sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw== -source-map@~0.6.1: +source-map@^0.6.1, source-map@~0.6.1: version "0.6.1" resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== +spawn-wrap@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/spawn-wrap/-/spawn-wrap-2.0.0.tgz#103685b8b8f9b79771318827aa78650a610d457e" + integrity sha512-EeajNjfN9zMnULLwhZZQU3GWBoFNkbngTUPfaawT4RkMiviTxcX0qfhVbGey39mfctfDHkWtuecgQ8NJcyQWHg== + dependencies: + foreground-child "^2.0.0" + is-windows "^1.0.2" + make-dir "^3.0.0" + rimraf "^3.0.0" + signal-exit "^3.0.2" + which "^2.0.1" + spdx-correct@^3.0.0: version "3.1.1" resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-3.1.1.tgz#dece81ac9c1e6713e5f7d1b6f17d468fa53d89a9" @@ -5647,6 +6191,11 @@ strip-bom@^3.0.0: resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3" integrity sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM= +strip-bom@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-4.0.0.tgz#9c3505c1db45bcedca3d9cf7a16f5c5aa3901878" + integrity sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w== + strip-indent@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/strip-indent/-/strip-indent-3.0.0.tgz#c32e1cee940b6b3432c771bc2c54bcce73cd3001" @@ -5728,6 +6277,15 @@ tar@^6.1.11, tar@^6.1.13, tar@^6.1.15, tar@^6.1.2: mkdirp "^1.0.3" yallist "^4.0.0" +test-exclude@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/test-exclude/-/test-exclude-6.0.0.tgz#04a8698661d805ea6fa293b6cb9e63ac044ef15e" + integrity sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w== + dependencies: + "@istanbuljs/schema" "^0.1.2" + glob "^7.1.4" + minimatch "^3.0.4" + text-extensions@^1.0.0: version "1.9.0" resolved "https://registry.yarnpkg.com/text-extensions/-/text-extensions-1.9.0.tgz#1853e45fee39c945ce6f6c36b2d659b5aabc2a26" @@ -5755,6 +6313,11 @@ tiny-relative-date@^1.3.0: resolved "https://registry.yarnpkg.com/tiny-relative-date/-/tiny-relative-date-1.3.0.tgz#fa08aad501ed730f31cc043181d995c39a935e07" integrity sha512-MOQHpzllWxDCHHaDno30hhLfbouoYlOI8YlMNtvKe1zXbjEVhbcEovQxvZrPvtiYW630GQDoMMarCnjfyfHA+A== +to-fast-properties@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/to-fast-properties/-/to-fast-properties-2.0.0.tgz#dc5e698cbd079265bc73e0377681a4e4e83f616e" + integrity sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog== + to-regex-range@^5.0.1: version "5.0.1" resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" @@ -5904,7 +6467,7 @@ type-fest@^0.6.0: resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.6.0.tgz#8d2a2370d3df886eb5c90ada1c5bf6188acf838b" integrity sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg== -type-fest@^0.8.1: +type-fest@^0.8.0, type-fest@^0.8.1: version "0.8.1" resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.8.1.tgz#09e249ebde851d3b1e48d27c105444667f17b83d" integrity sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA== @@ -5948,6 +6511,13 @@ typed-array-length@^1.0.4: for-each "^0.3.3" is-typed-array "^1.1.9" +typedarray-to-buffer@^3.1.5: + version "3.1.5" + resolved "https://registry.yarnpkg.com/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz#a97ee7a9ff42691b9f783ff1bc5112fe3fca9080" + integrity sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q== + dependencies: + is-typedarray "^1.0.0" + typescript@^3.9.10, typescript@^3.9.7: version "3.9.10" resolved "https://registry.yarnpkg.com/typescript/-/typescript-3.9.10.tgz#70f3910ac7a51ed6bef79da7800690b19bf778b8" @@ -5997,6 +6567,14 @@ universalify@^2.0.0: resolved "https://registry.yarnpkg.com/universalify/-/universalify-2.0.0.tgz#75a4984efedc4b08975c5aeb73f530d02df25717" integrity sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ== +update-browserslist-db@^1.0.13: + version "1.0.13" + resolved "https://registry.yarnpkg.com/update-browserslist-db/-/update-browserslist-db-1.0.13.tgz#3c5e4f5c083661bd38ef64b6328c26ed6c8248c4" + integrity sha512-xebP81SNcPuNpPP3uzeW1NYXxI3rxyJzF3pD6sH4jE7o/IX+WtSpwnVU+qIsDPyk0d3hmFQ7mjqc6AtV604hbg== + dependencies: + escalade "^3.1.1" + picocolors "^1.0.0" + uri-js@^4.2.2: version "4.4.1" resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" @@ -6009,6 +6587,11 @@ util-deprecate@^1.0.1, util-deprecate@^1.0.2: resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" integrity sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8= +uuid@^8.3.2: + version "8.3.2" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-8.3.2.tgz#80d5b5ced271bb9af6c445f21a1a04c606cefbe2" + integrity sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg== + v8-compile-cache-lib@^3.0.1: version "3.0.1" resolved "https://registry.yarnpkg.com/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz#6336e8d71965cb3d35a1bbb7868445a7c05264bf" @@ -6057,6 +6640,11 @@ which-boxed-primitive@^1.0.2: is-string "^1.0.5" is-symbol "^1.0.3" +which-module@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/which-module/-/which-module-2.0.1.tgz#776b1fe35d90aebe99e8ac15eb24093389a4a409" + integrity sha512-iBdZ57RDvnOR9AGBhML2vFZf7h8vmBjhoaZqODJBFWHVtKkDmKuHai3cx5PgVMrX5YDNp27AofYbAwctSS+vhQ== + which-typed-array@^1.1.11: version "1.1.11" resolved "https://registry.yarnpkg.com/which-typed-array/-/which-typed-array-1.1.11.tgz#99d691f23c72aab6768680805a271b69761ed61a" @@ -6122,6 +6710,15 @@ workerpool@6.2.1: string-width "^4.1.0" strip-ansi "^6.0.0" +wrap-ansi@^6.2.0: + version "6.2.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-6.2.0.tgz#e9393ba07102e6c91a3b221478f0257cd2856e53" + integrity sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA== + dependencies: + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" + wrap-ansi@^8.1.0: version "8.1.0" resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-8.1.0.tgz#56dc22368ee570face1b49819975d9b9a5ead214" @@ -6136,6 +6733,16 @@ wrappy@1: resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8= +write-file-atomic@^3.0.0: + version "3.0.3" + resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-3.0.3.tgz#56bd5c5a5c70481cd19c571bd39ab965a5de56e8" + integrity sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q== + dependencies: + imurmurhash "^0.1.4" + is-typedarray "^1.0.0" + signal-exit "^3.0.2" + typedarray-to-buffer "^3.1.5" + write-file-atomic@^5.0.0, write-file-atomic@^5.0.1: version "5.0.1" resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-5.0.1.tgz#68df4717c55c6fa4281a7860b4c2ba0a6d2b11e7" @@ -6144,11 +6751,21 @@ write-file-atomic@^5.0.0, write-file-atomic@^5.0.1: imurmurhash "^0.1.4" signal-exit "^4.0.1" +y18n@^4.0.0: + version "4.0.3" + resolved "https://registry.yarnpkg.com/y18n/-/y18n-4.0.3.tgz#b5f259c82cd6e336921efd7bfd8bf560de9eeedf" + integrity sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ== + y18n@^5.0.5: version "5.0.8" resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.8.tgz#7f4934d0f7ca8c56f95314939ddcd2dd91ce1d55" integrity sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA== +yallist@^3.0.2: + version "3.1.1" + resolved "https://registry.yarnpkg.com/yallist/-/yallist-3.1.1.tgz#dbb7daf9bfd8bac9ab45ebf602b8cbad0d5d08fd" + integrity sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g== + yallist@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" @@ -6164,6 +6781,14 @@ yargs-parser@20.2.4: resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.4.tgz#b42890f14566796f85ae8e3a25290d205f154a54" integrity sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA== +yargs-parser@^18.1.2: + version "18.1.3" + resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-18.1.3.tgz#be68c4975c6b2abf469236b0c870362fab09a7b0" + integrity sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ== + dependencies: + camelcase "^5.0.0" + decamelize "^1.2.0" + yargs-parser@^20.2.2, yargs-parser@^20.2.3: version "20.2.9" resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.9.tgz#2eb7dc3b0289718fc295f362753845c41a0c94ee" @@ -6192,6 +6817,23 @@ yargs@16.2.0, yargs@^16.2.0: y18n "^5.0.5" yargs-parser "^20.2.2" +yargs@^15.0.2: + version "15.4.1" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-15.4.1.tgz#0d87a16de01aee9d8bec2bfbf74f67851730f4f8" + integrity sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A== + dependencies: + cliui "^6.0.0" + decamelize "^1.2.0" + find-up "^4.1.0" + get-caller-file "^2.0.1" + require-directory "^2.1.1" + require-main-filename "^2.0.0" + set-blocking "^2.0.0" + string-width "^4.2.0" + which-module "^2.0.0" + y18n "^4.0.0" + yargs-parser "^18.1.2" + yarn@^1.22.18: version "1.22.19" resolved "https://registry.yarnpkg.com/yarn/-/yarn-1.22.19.tgz#4ba7fc5c6e704fce2066ecbfb0b0d8976fe62447" From e365ad1834727dc1527d437eef88c4b26da4cb81 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 11:58:54 -0600 Subject: [PATCH 23/27] test: improve test coverage --- package.json | 2 +- src/help/index.ts | 4 +- src/util/default-flag-to-cached.ts | 23 -- src/util/default-to-cached.ts | 23 ++ src/util/index.ts | 18 +- src/util/to-cached.ts | 30 +-- test/command/command.test.ts | 262 +------------------ test/util.test.ts | 393 ++++++++++++++++++++++++++++- 8 files changed, 428 insertions(+), 327 deletions(-) delete mode 100644 src/util/default-flag-to-cached.ts create mode 100644 src/util/default-to-cached.ts diff --git a/package.json b/package.json index 674f55568..adf4bd810 100644 --- a/package.json +++ b/package.json @@ -110,7 +110,7 @@ "lint": "eslint . --ext .ts", "posttest": "yarn lint && yarn test:circular-deps", "prepack": "yarn run build", - "pretest": "yarn build && tsc -p test --skipLibCheck", + "pretest": "yarn build && tsc -p test --noEmit --skipLibCheck", "test:circular-deps": "madge lib/ -c", "test:e2e": "mocha --forbid-only \"test/**/*.e2e.ts\" --parallel --timeout 1200000", "test:esm-cjs": "cross-env DEBUG=e2e:* ts-node test/integration/esm-cjs.ts", diff --git a/src/help/index.ts b/src/help/index.ts index 069ab4006..aa5416b00 100644 --- a/src/help/index.ts +++ b/src/help/index.ts @@ -5,7 +5,7 @@ import {Command} from '../command' import {CommandHelp} from './command' import {HelpFormatter} from './formatter' import RootHelp from './root' -import {defaultFlagToCached} from '../util/default-flag-to-cached' +import {defaultToCached} from '../util/default-to-cached' import {error} from '../errors' import {format} from 'node:util' import {load} from '../module-loader' @@ -112,7 +112,7 @@ export class Help extends HelpBase { for (const flag of Object.values(loaded.flags)) { if (flag.type === 'boolean') continue // eslint-disable-next-line no-await-in-loop - flag.default = await defaultFlagToCached(flag, false) + flag.default = await defaultToCached(flag, false) } await this.showCommandHelp(command) diff --git a/src/util/default-flag-to-cached.ts b/src/util/default-flag-to-cached.ts deleted file mode 100644 index fe3beeeb2..000000000 --- a/src/util/default-flag-to-cached.ts +++ /dev/null @@ -1,23 +0,0 @@ -import {OptionFlag} from '../interfaces/parser' - -// when no manifest exists, the default is calculated. This may throw, so we need to catch it -export const defaultFlagToCached = async (flag: OptionFlag, respectNoCacheDefault: boolean) => { - if (respectNoCacheDefault && flag.noCacheDefault) return - // Prefer the defaultHelp function (returns a friendly string for complex types) - if (typeof flag.defaultHelp === 'function') { - try { - return await flag.defaultHelp({options: flag, flags: {}}) - } catch { - return - } - } - - // if not specified, try the default function - if (typeof flag.default === 'function') { - try { - return await flag.default({options: flag, flags: {}}) - } catch {} - } else { - return flag.default - } -} diff --git a/src/util/default-to-cached.ts b/src/util/default-to-cached.ts new file mode 100644 index 000000000..038a583fa --- /dev/null +++ b/src/util/default-to-cached.ts @@ -0,0 +1,23 @@ +import {Arg, OptionFlag} from '../interfaces/parser' + +// when no manifest exists, the default is calculated. This may throw, so we need to catch it +export const defaultToCached = async (flagOrArg: OptionFlag | Arg, respectNoCacheDefault: boolean) => { + if (respectNoCacheDefault && flagOrArg.noCacheDefault) return + // Prefer the defaultHelp function (returns a friendly string for complex types) + if (typeof flagOrArg.defaultHelp === 'function') { + try { + return await flagOrArg.defaultHelp({options: flagOrArg, flags: {}}) + } catch { + return + } + } + + // if not specified, try the default function + if (typeof flagOrArg.default === 'function') { + try { + return await flagOrArg.default({options: flagOrArg, flags: {}}) + } catch {} + } else { + return flagOrArg.default + } +} diff --git a/src/util/index.ts b/src/util/index.ts index f35f3ebf8..df4a90ec4 100644 --- a/src/util/index.ts +++ b/src/util/index.ts @@ -1,9 +1,9 @@ -import {access, stat} from 'node:fs/promises' +import {access, readFile, stat} from 'node:fs/promises' import {homedir, platform} from 'node:os' -import {readFile, readFileSync} from 'node:fs' import {ArgInput} from '../interfaces/parser' import {Command} from '../command' import {join} from 'node:path' +import {readFileSync} from 'node:fs' const debug = require('debug') @@ -169,18 +169,10 @@ export function getPlatform(): NodeJS.Platform { return platform() } -export function readJson(path: string): Promise { +export async function readJson(path: string): Promise { debug('config')('readJson %s', path) - return new Promise((resolve, reject) => { - readFile(path, 'utf8', (err: any, d: any) => { - try { - if (err) reject(err) - else resolve(JSON.parse(d) as T) - } catch (error: any) { - reject(error) - } - }) - }) + const contents = await readFile(path, 'utf8') + return JSON.parse(contents) as T } export function readJsonSync(path: string, parse: false): string diff --git a/src/util/to-cached.ts b/src/util/to-cached.ts index de748284d..a2887e14f 100644 --- a/src/util/to-cached.ts +++ b/src/util/to-cached.ts @@ -1,31 +1,9 @@ /* eslint-disable no-await-in-loop */ import {ensureArgObject, pickBy} from './index' -import {Arg} from '../interfaces/parser' import {Command} from '../command' import {Plugin as IPlugin} from '../interfaces/plugin' import {aggregateFlags} from './aggregate-flags' -import {defaultFlagToCached} from './default-flag-to-cached' - -const defaultArgToCached = async (arg: Arg, respectNoCacheDefault: boolean): Promise => { - if (respectNoCacheDefault && arg.noCacheDefault) return - // Prefer the defaultHelp function (returns a friendly string for complex types) - if (typeof arg.defaultHelp === 'function') { - try { - return await arg.defaultHelp({options: arg, flags: {}}) - } catch { - return - } - } - - // if not specified, try the default function - if (typeof arg.default === 'function') { - try { - return await arg.default({options: arg, flags: {}}) - } catch {} - } else { - return arg.default - } -} +import {defaultToCached} from './default-to-cached' export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { const flags = {} as {[k: string]: Command.Flag.Cached} @@ -43,7 +21,7 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa const cmdFlags = aggregateFlags(c.flags, c.baseFlags, c.enableJsonFlag) - for (const [name, flag] of Object.entries(cmdFlags || {})) { + for (const [name, flag] of Object.entries(cmdFlags)) { if (flag.type === 'boolean') { flags[name] = { name, @@ -83,7 +61,7 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa dependsOn: flag.dependsOn, relationships: flag.relationships, exclusive: flag.exclusive, - default: await defaultFlagToCached(flag, respectNoCacheDefault), + default: await defaultToCached(flag, respectNoCacheDefault), deprecated: flag.deprecated, deprecateAliases: c.deprecateAliases, aliases: flag.aliases, @@ -105,7 +83,7 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa description: arg.description, required: arg.required, options: arg.options, - default: await defaultArgToCached(arg, respectNoCacheDefault), + default: await defaultToCached(arg, respectNoCacheDefault), hidden: arg.hidden, noCacheDefault: arg.noCacheDefault, } diff --git a/test/command/command.test.ts b/test/command/command.test.ts index 9531d565e..ee8991892 100644 --- a/test/command/command.test.ts +++ b/test/command/command.test.ts @@ -1,7 +1,6 @@ import {expect, fancy} from 'fancy-test' // import path = require('path') -import {Args, Command as Base, Flags} from '../../src' -import {toCached} from '../../src/util/to-cached' +import {Command as Base, Flags} from '../../src' // import {TestHelpClassConfig} from './helpers/test-help-in-src/src/test-help-plugin' // const pjson = require('../package.json') @@ -70,265 +69,6 @@ describe('command', () => { .catch(/EEXIT: 0/) .it('exits with 0') - describe('toCached', () => { - fancy - .do(async () => { - class C extends Command { - static id = 'foo:bar' - static title = 'cmd title' - static type = 'mytype' - static usage = ['$ usage'] - static description = 'test command' - static aliases = ['alias1', 'alias2'] - static hidden = true - static flags = { - flaga: Flags.boolean(), - flagb: Flags.string({ - char: 'b', - hidden: true, - required: false, - description: 'flagb desc', - options: ['a', 'b'], - default: async () => 'a', - }), - flagc: Flags.integer({ - char: 'c', - min: 1, - max: 10, - required: false, - description: 'flagc desc', - options: ['a', 'b'], - default: async context => (context.options.min ?? 1) + 1, - }), - } - - static args = { - arg1: Args.string({ - description: 'arg1 desc', - required: true, - hidden: false, - options: ['af', 'b'], - default: async () => 'a', - }), - } - } - - const c = await toCached(C, undefined, false) - - expect(c).to.deep.equal({ - id: 'foo:bar', - type: 'mytype', - hidden: true, - pluginName: undefined, - pluginAlias: undefined, - pluginType: undefined, - state: undefined, - description: 'test command', - aliases: ['alias1', 'alias2'], - title: 'cmd title', - usage: ['$ usage'], - examples: undefined, - deprecationOptions: undefined, - deprecateAliases: undefined, - summary: undefined, - strict: true, - enableJsonFlag: false, - hasDynamicHelp: false, - flags: { - flaga: { - aliases: undefined, - char: undefined, - charAliases: undefined, - description: undefined, - dependsOn: undefined, - deprecateAliases: undefined, - deprecated: undefined, - exclusive: undefined, - helpGroup: undefined, - helpLabel: undefined, - summary: undefined, - name: 'flaga', - hidden: undefined, - required: undefined, - relationships: undefined, - allowNo: false, - type: 'boolean', - delimiter: undefined, - noCacheDefault: undefined, - }, - flagb: { - aliases: undefined, - char: 'b', - charAliases: undefined, - description: 'flagb desc', - dependsOn: undefined, - deprecateAliases: undefined, - deprecated: undefined, - exclusive: undefined, - helpGroup: undefined, - helpLabel: undefined, - summary: undefined, - name: 'flagb', - hidden: true, - required: false, - multiple: false, - relationships: undefined, - type: 'option', - helpValue: undefined, - default: 'a', - options: ['a', 'b'], - delimiter: undefined, - noCacheDefault: undefined, - }, - flagc: { - aliases: undefined, - char: 'c', - charAliases: undefined, - default: 2, - delimiter: undefined, - dependsOn: undefined, - deprecateAliases: undefined, - deprecated: undefined, - description: 'flagc desc', - exclusive: undefined, - helpGroup: undefined, - helpLabel: undefined, - helpValue: undefined, - hidden: undefined, - multiple: false, - name: 'flagc', - options: [ - 'a', - 'b', - ], - relationships: undefined, - required: false, - summary: undefined, - type: 'option', - noCacheDefault: undefined, - }, - - }, - args: { - arg1: { - description: 'arg1 desc', - name: 'arg1', - hidden: false, - required: true, - options: ['af', 'b'], - default: 'a', - noCacheDefault: undefined, - }, - }, - }) - }) - .it('converts to cached with everything set') - - fancy - .do(async () => { - class Base extends Command { - public static enableJsonFlag = true - public static baseFlags = { - parentFlag: Flags.boolean(), - } - } - - class Child extends Base { - static flags = { - childFlag: Flags.boolean(), - } - } - - const cached = await toCached(Child, undefined, false) - - expect(cached).to.deep.equal({ - id: 'command', - summary: undefined, - description: 'test command', - strict: true, - usage: undefined, - pluginName: undefined, - pluginAlias: undefined, - pluginType: undefined, - hidden: undefined, - state: undefined, - aliases: [], - examples: undefined, - deprecationOptions: undefined, - deprecateAliases: undefined, - flags: { - json: { - name: 'json', - type: 'boolean', - char: undefined, - summary: undefined, - description: 'Format output as json.', - hidden: undefined, - required: undefined, - helpLabel: undefined, - helpGroup: 'GLOBAL', - allowNo: false, - dependsOn: undefined, - relationships: undefined, - exclusive: undefined, - deprecated: undefined, - deprecateAliases: undefined, - aliases: undefined, - charAliases: undefined, - delimiter: undefined, - noCacheDefault: undefined, - }, - childFlag: { - name: 'childFlag', - type: 'boolean', - char: undefined, - summary: undefined, - description: undefined, - hidden: undefined, - required: undefined, - helpLabel: undefined, - helpGroup: undefined, - allowNo: false, - dependsOn: undefined, - relationships: undefined, - exclusive: undefined, - deprecated: undefined, - deprecateAliases: undefined, - aliases: undefined, - charAliases: undefined, - delimiter: undefined, - noCacheDefault: undefined, - }, - parentFlag: { - name: 'parentFlag', - type: 'boolean', - char: undefined, - summary: undefined, - description: undefined, - hidden: undefined, - required: undefined, - helpLabel: undefined, - helpGroup: undefined, - allowNo: false, - dependsOn: undefined, - relationships: undefined, - exclusive: undefined, - deprecated: undefined, - deprecateAliases: undefined, - aliases: undefined, - charAliases: undefined, - delimiter: undefined, - noCacheDefault: undefined, - }, - }, - args: {}, - hasDynamicHelp: false, - enableJsonFlag: true, - }) - }) - .it('converts to cached with multiple Command classes in inheritance chain') - }) - describe('parse', () => { fancy .stdout() diff --git a/test/util.test.ts b/test/util.test.ts index 4820f23a6..11921299a 100644 --- a/test/util.test.ts +++ b/test/util.test.ts @@ -1,5 +1,8 @@ import {expect} from 'chai' -import {capitalize, ensureArgObject, last, maxBy, sumBy} from '../src/util' +import {Args, Command, Flags} from '../src/index' +import {capitalize, castArray, ensureArgObject, getHomeDir, isNotFalsy, isTruthy, last, maxBy, readJson, sumBy} from '../src/util' +import {defaultToCached} from '../src/util/default-to-cached' +import {toCached} from '../src/util/to-cached' describe('capitalize', () => { it('capitalizes the string', () => { @@ -68,3 +71,391 @@ describe('ensureArgObject', () => { expect(ensureArgObject(args)).to.deep.equal(args) }) }) + +describe('isNotFalsy', () => { + it('should return true for truthy values', () => { + expect(isNotFalsy('true')).to.be.true + expect(isNotFalsy('1')).to.be.true + expect(isNotFalsy('yes')).to.be.true + expect(isNotFalsy('y')).to.be.true + }) + + it('should return false for falsy values', () => { + expect(isNotFalsy('false')).to.be.false + expect(isNotFalsy('0')).to.be.false + expect(isNotFalsy('no')).to.be.false + expect(isNotFalsy('n')).to.be.false + }) +}) + +describe('isTruthy', () => { + it('should return true for truthy values', () => { + expect(isTruthy('true')).to.be.true + expect(isTruthy('1')).to.be.true + expect(isTruthy('yes')).to.be.true + expect(isTruthy('y')).to.be.true + }) + + it('should return false for falsy values', () => { + expect(isTruthy('false')).to.be.false + expect(isTruthy('0')).to.be.false + expect(isTruthy('no')).to.be.false + expect(isTruthy('n')).to.be.false + }) +}) + +describe('getHomeDir', () => { + it('should return the home directory', () => { + expect(getHomeDir()).to.equal(process.env.HOME) + }) +}) + +describe('readJson', () => { + it('should return parsed JSON', async () => { + const json = await readJson<{name: string}>('package.json') + expect(json.name).to.equal('@oclif/core') + }) + + it('should throw an error if the file does not exist', async () => { + try { + await readJson('does-not-exist.json') + throw new Error('Expected an error to be thrown') + } catch (error) { + const err = error as Error + expect(err.message).to.equal('ENOENT: no such file or directory, open \'does-not-exist.json\'') + } + }) +}) + +describe('castArray', () => { + it('should cast a value to an array', () => { + expect(castArray('foo')).to.deep.equal(['foo']) + }) + + it('should return an array if the value is an array', () => { + expect(castArray(['foo'])).to.deep.equal(['foo']) + }) + + it('should return an empty array if the value is undefined', () => { + expect(castArray()).to.deep.equal([]) + }) +}) + +describe('defaultToCached', () => { + it('should do nothing if noCacheDefault is true', async () => { + const flag = {noCacheDefault: true} + const result = await defaultToCached(flag as any, true) + expect(result).to.be.undefined + }) + + it('should do nothing if respectNoCacheDefault is true', async () => { + const result = await defaultToCached({} as any, true) + expect(result).to.be.undefined + }) + + it('should return the result of defaultHelp if it exists', async () => { + const flag = {defaultHelp: async () => 'foo'} + const result = await defaultToCached(flag as any, false) + expect(result).to.equal('foo') + }) + + it('should return undefined if defaultHelp throws', async () => { + const flag = {async defaultHelp() { + throw new Error('foo') + }} + const result = await defaultToCached(flag as any, false) + expect(result).to.be.undefined + }) + + it('should return the result of the default if it\'s a function', async () => { + const flag = {default: async () => 'foo'} + const result = await defaultToCached(flag as any, false) + expect(result).to.equal('foo') + }) + + it('should return the result of the default if it\'s a simple value', async () => { + const flag = {default: 'foo'} + const result = await defaultToCached(flag as any, false) + expect(result).to.equal('foo') + }) +}) + +describe('toCached', () => { + it('should return a cached command with every thing set', async () => { + class C extends Command { + static id = 'foo:bar' + static title = 'cmd title' + static type = 'mytype' + static usage = ['$ usage'] + static description = 'test command' + static aliases = ['alias1', 'alias2'] + static hidden = true + static flags = { + flaga: Flags.boolean(), + flagb: Flags.string({ + char: 'b', + hidden: true, + required: false, + description: 'flagb desc', + options: ['a', 'b'], + default: async () => 'a', + }), + flagc: Flags.integer({ + char: 'c', + min: 1, + max: 10, + required: false, + description: 'flagc desc', + options: ['a', 'b'], + default: async context => (context.options.min ?? 1) + 1, + }), + } + + static args = { + arg1: Args.string({ + description: 'arg1 desc', + required: true, + hidden: false, + options: ['af', 'b'], + default: async () => 'a', + }), + } + + public async run(): Promise {} + } + const c = await toCached(C, undefined, false) + expect(c).to.deep.equal({ + id: 'foo:bar', + type: 'mytype', + hidden: true, + pluginName: undefined, + pluginAlias: undefined, + pluginType: undefined, + state: undefined, + description: 'test command', + aliases: ['alias1', 'alias2'], + title: 'cmd title', + usage: ['$ usage'], + examples: undefined, + deprecationOptions: undefined, + deprecateAliases: undefined, + summary: undefined, + strict: true, + enableJsonFlag: false, + hasDynamicHelp: false, + flags: { + flaga: { + aliases: undefined, + char: undefined, + charAliases: undefined, + description: undefined, + dependsOn: undefined, + deprecateAliases: undefined, + deprecated: undefined, + exclusive: undefined, + helpGroup: undefined, + helpLabel: undefined, + summary: undefined, + name: 'flaga', + hidden: undefined, + required: undefined, + relationships: undefined, + allowNo: false, + type: 'boolean', + delimiter: undefined, + noCacheDefault: undefined, + }, + flagb: { + aliases: undefined, + char: 'b', + charAliases: undefined, + description: 'flagb desc', + dependsOn: undefined, + deprecateAliases: undefined, + deprecated: undefined, + exclusive: undefined, + helpGroup: undefined, + helpLabel: undefined, + summary: undefined, + name: 'flagb', + hidden: true, + required: false, + multiple: false, + relationships: undefined, + type: 'option', + helpValue: undefined, + default: 'a', + options: ['a', 'b'], + delimiter: undefined, + noCacheDefault: undefined, + }, + flagc: { + aliases: undefined, + char: 'c', + charAliases: undefined, + default: 2, + delimiter: undefined, + dependsOn: undefined, + deprecateAliases: undefined, + deprecated: undefined, + description: 'flagc desc', + exclusive: undefined, + helpGroup: undefined, + helpLabel: undefined, + helpValue: undefined, + hidden: undefined, + multiple: false, + name: 'flagc', + options: [ + 'a', + 'b', + ], + relationships: undefined, + required: false, + summary: undefined, + type: 'option', + noCacheDefault: undefined, + }, + + }, + args: { + arg1: { + description: 'arg1 desc', + name: 'arg1', + hidden: false, + required: true, + options: ['af', 'b'], + default: 'a', + noCacheDefault: undefined, + }, + }, + }) + }) + + it('should return a cached command with multiple Command classes in inheritance chain', async () => { + class Base extends Command { + public static enableJsonFlag = true + public static aliases = ['base'] + public static state = 'beta' + public static summary = 'base summary' + public static baseFlags = { + parentFlag: Flags.boolean(), + } + + public async run(): Promise {} + } + + class Child extends Base { + static id = 'command' + public static summary = 'child summary' + static flags = { + childFlag: Flags.boolean(), + } + + public async run(): Promise {} + } + + const cached = await toCached(Child, undefined, false) + + expect(cached).to.deep.equal({ + id: 'command', + summary: 'child summary', + description: undefined, + strict: true, + usage: undefined, + pluginName: undefined, + pluginAlias: undefined, + pluginType: undefined, + hidden: undefined, + state: 'beta', + aliases: ['base'], + examples: undefined, + deprecationOptions: undefined, + deprecateAliases: undefined, + flags: { + json: { + name: 'json', + type: 'boolean', + char: undefined, + summary: undefined, + description: 'Format output as json.', + hidden: undefined, + required: undefined, + helpLabel: undefined, + helpGroup: 'GLOBAL', + allowNo: false, + dependsOn: undefined, + relationships: undefined, + exclusive: undefined, + deprecated: undefined, + deprecateAliases: undefined, + aliases: undefined, + charAliases: undefined, + delimiter: undefined, + noCacheDefault: undefined, + }, + childFlag: { + name: 'childFlag', + type: 'boolean', + char: undefined, + summary: undefined, + description: undefined, + hidden: undefined, + required: undefined, + helpLabel: undefined, + helpGroup: undefined, + allowNo: false, + dependsOn: undefined, + relationships: undefined, + exclusive: undefined, + deprecated: undefined, + deprecateAliases: undefined, + aliases: undefined, + charAliases: undefined, + delimiter: undefined, + noCacheDefault: undefined, + }, + parentFlag: { + name: 'parentFlag', + type: 'boolean', + char: undefined, + summary: undefined, + description: undefined, + hidden: undefined, + required: undefined, + helpLabel: undefined, + helpGroup: undefined, + allowNo: false, + dependsOn: undefined, + relationships: undefined, + exclusive: undefined, + deprecated: undefined, + deprecateAliases: undefined, + aliases: undefined, + charAliases: undefined, + delimiter: undefined, + noCacheDefault: undefined, + }, + }, + args: {}, + hasDynamicHelp: false, + enableJsonFlag: true, + }) + }) + + it('should set dynamicHelp to true if defaultHelp is a function', async () => { + class C extends Command { + static id = 'foo:bar' + static flags = { + flaga: Flags.boolean(), + flagb: Flags.string({ + defaultHelp: async () => 'foo', + }), + } + + public async run(): Promise {} + } + const c = await toCached(C, undefined, false) + expect(c.hasDynamicHelp).to.be.true + }) +}) From 920ea0e0913d3f7d0571172c0dbbfceeba74db43 Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 12:48:33 -0600 Subject: [PATCH 24/27] test: windows unit tests --- test/util.test.ts | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/test/util.test.ts b/test/util.test.ts index 11921299a..889fa6464 100644 --- a/test/util.test.ts +++ b/test/util.test.ts @@ -1,4 +1,5 @@ import {expect} from 'chai' +import {homedir} from 'node:os' import {Args, Command, Flags} from '../src/index' import {capitalize, castArray, ensureArgObject, getHomeDir, isNotFalsy, isTruthy, last, maxBy, readJson, sumBy} from '../src/util' import {defaultToCached} from '../src/util/default-to-cached' @@ -106,7 +107,7 @@ describe('isTruthy', () => { describe('getHomeDir', () => { it('should return the home directory', () => { - expect(getHomeDir()).to.equal(process.env.HOME) + expect(getHomeDir()).to.equal(homedir()) }) }) @@ -122,7 +123,7 @@ describe('readJson', () => { throw new Error('Expected an error to be thrown') } catch (error) { const err = error as Error - expect(err.message).to.equal('ENOENT: no such file or directory, open \'does-not-exist.json\'') + expect(err.message).to.include('ENOENT: no such file or directory') } }) }) From 42282e4946f7b2c2441ea94d08711b813fb25d3e Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Wed, 27 Sep 2023 16:28:54 -0600 Subject: [PATCH 25/27] chore: revert change to automerge.yml --- .github/workflows/automerge.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/automerge.yml b/.github/workflows/automerge.yml index 41d2308a7..796eafae2 100644 --- a/.github/workflows/automerge.yml +++ b/.github/workflows/automerge.yml @@ -6,5 +6,5 @@ on: jobs: automerge: - uses: salesforcecli/github-workflows/.github/workflows/automerge.yml@main + uses: oclif/github-workflows/.github/workflows/automerge.yml@main secrets: inherit From 09c9b667a7f3e52d6b8cd507caeed7f4ac9dab1c Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Thu, 28 Sep 2023 09:31:31 -0600 Subject: [PATCH 26/27] chore: code review --- guides/V3_MIGRATION.md | 4 + src/config/plugin.ts | 6 +- src/help/index.ts | 6 +- src/util/aggregate-flags.ts | 2 +- src/util/{to-cached.ts => cache-command.ts} | 8 +- ...lt-to-cached.ts => cache-default-value.ts} | 2 +- test/help/help-test-utils.ts | 4 +- .../cache-command.test.ts} | 191 +----------------- test/util/cache-default-value.test.ts | 42 ++++ test/util/index.test.ts | 141 +++++++++++++ 10 files changed, 207 insertions(+), 199 deletions(-) rename src/util/{to-cached.ts => cache-command.ts} (92%) rename src/util/{default-to-cached.ts => cache-default-value.ts} (85%) rename test/{util.test.ts => util/cache-command.test.ts} (56%) create mode 100644 test/util/cache-default-value.test.ts create mode 100644 test/util/index.test.ts diff --git a/guides/V3_MIGRATION.md b/guides/V3_MIGRATION.md index cd6331347..82eaa7ba1 100644 --- a/guides/V3_MIGRATION.md +++ b/guides/V3_MIGRATION.md @@ -12,6 +12,7 @@ Migrating to @oclif/core@V3 - [`global['cli-ux']` -\> `global.ux`](#globalcli-ux---globalux) - [`handle`](#handle) - [`noCacheDefault` flag property replaces `isWritingManifest`](#nocachedefault-flag-property-replaces-iswritingmanifest) + - [Removed `toCached` export](#removed-tocached-export) - [Features 🎉](#features-) - [Performance Improvements](#performance-improvements) - [charAliases Flag Property](#charaliases-flag-property) @@ -116,6 +117,9 @@ export const mySensitiveFlag = Flags.string({ }); ``` +### Removed `toCached` export + +We removed the `toCached` export since there's no need for consumers of `@oclif/core` to use this function. ## Features 🎉 diff --git a/src/config/plugin.ts b/src/config/plugin.ts index d20ddf34e..cd229e86a 100644 --- a/src/config/plugin.ts +++ b/src/config/plugin.ts @@ -13,9 +13,9 @@ import {Manifest} from '../interfaces/manifest' import {PJSON} from '../interfaces/pjson' import {Performance} from '../performance' import {Topic} from '../interfaces/topic' +import {cacheCommand} from '../util/cache-command' import {inspect} from 'node:util' import {sync} from 'globby' -import {toCached} from '../util/to-cached' import {tsPath} from './ts-node' const _pjson = requireJson(__dirname, '..', '..', 'package.json') @@ -305,7 +305,7 @@ export class Plugin implements IPlugin { version: this.version, commands: (await Promise.all(this.commandIDs.map(async id => { try { - const cached = await toCached(await this.findCommand(id, {must: true}), this, respectNoCacheDefault) + const cached = await cacheCommand(await this.findCommand(id, {must: true}), this, respectNoCacheDefault) if (this.flexibleTaxonomy) { const permutations = getCommandIdPermutations(id) const aliasPermutations = cached.aliases.flatMap(a => getCommandIdPermutations(a)) @@ -314,7 +314,7 @@ export class Plugin implements IPlugin { return [id, cached] } catch (error: any) { - const scope = 'toCached' + const scope = 'cacheCommand' if (Boolean(errorOnManifestCreate) === false) this.warn(error, scope) else throw this.addErrorScope(error, scope) } diff --git a/src/help/index.ts b/src/help/index.ts index aa5416b00..7abaf3d01 100644 --- a/src/help/index.ts +++ b/src/help/index.ts @@ -5,7 +5,7 @@ import {Command} from '../command' import {CommandHelp} from './command' import {HelpFormatter} from './formatter' import RootHelp from './root' -import {defaultToCached} from '../util/default-to-cached' +import {cacheDefaultValue} from '../util/cache-default-value' import {error} from '../errors' import {format} from 'node:util' import {load} from '../module-loader' @@ -112,7 +112,7 @@ export class Help extends HelpBase { for (const flag of Object.values(loaded.flags)) { if (flag.type === 'boolean') continue // eslint-disable-next-line no-await-in-loop - flag.default = await defaultToCached(flag, false) + flag.default = await cacheDefaultValue(flag, false) } await this.showCommandHelp(command) @@ -344,7 +344,7 @@ function extractClass(exported: any): HelpBaseDerived { export async function loadHelpClass(config: Interfaces.Config): Promise { const {pjson} = config - const configuredClass = pjson && pjson.oclif && pjson.oclif.helpClass + const configuredClass = pjson.oclif?.helpClass if (configuredClass) { try { diff --git a/src/util/aggregate-flags.ts b/src/util/aggregate-flags.ts index ba96dbc93..bd13b9fd7 100644 --- a/src/util/aggregate-flags.ts +++ b/src/util/aggregate-flags.ts @@ -1,7 +1,7 @@ import {FlagInput, FlagOutput} from '../interfaces/parser' import {boolean} from '../flags' -export const json = boolean({ +const json = boolean({ description: 'Format output as json.', helpGroup: 'GLOBAL', }) diff --git a/src/util/to-cached.ts b/src/util/cache-command.ts similarity index 92% rename from src/util/to-cached.ts rename to src/util/cache-command.ts index a2887e14f..3c8c5f5bf 100644 --- a/src/util/to-cached.ts +++ b/src/util/cache-command.ts @@ -3,9 +3,9 @@ import {ensureArgObject, pickBy} from './index' import {Command} from '../command' import {Plugin as IPlugin} from '../interfaces/plugin' import {aggregateFlags} from './aggregate-flags' -import {defaultToCached} from './default-to-cached' +import {cacheDefaultValue} from './cache-default-value' -export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { +export async function cacheCommand(cmd: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { const flags = {} as {[k: string]: Command.Flag.Cached} // In order to collect static properties up the inheritance chain, we need to recursively @@ -61,7 +61,7 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa dependsOn: flag.dependsOn, relationships: flag.relationships, exclusive: flag.exclusive, - default: await defaultToCached(flag, respectNoCacheDefault), + default: await cacheDefaultValue(flag, respectNoCacheDefault), deprecated: flag.deprecated, deprecateAliases: c.deprecateAliases, aliases: flag.aliases, @@ -83,7 +83,7 @@ export async function toCached(cmd: Command.Class, plugin?: IPlugin, respectNoCa description: arg.description, required: arg.required, options: arg.options, - default: await defaultToCached(arg, respectNoCacheDefault), + default: await cacheDefaultValue(arg, respectNoCacheDefault), hidden: arg.hidden, noCacheDefault: arg.noCacheDefault, } diff --git a/src/util/default-to-cached.ts b/src/util/cache-default-value.ts similarity index 85% rename from src/util/default-to-cached.ts rename to src/util/cache-default-value.ts index 038a583fa..990eb5179 100644 --- a/src/util/default-to-cached.ts +++ b/src/util/cache-default-value.ts @@ -1,7 +1,7 @@ import {Arg, OptionFlag} from '../interfaces/parser' // when no manifest exists, the default is calculated. This may throw, so we need to catch it -export const defaultToCached = async (flagOrArg: OptionFlag | Arg, respectNoCacheDefault: boolean) => { +export const cacheDefaultValue = async (flagOrArg: OptionFlag | Arg, respectNoCacheDefault: boolean) => { if (respectNoCacheDefault && flagOrArg.noCacheDefault) return // Prefer the defaultHelp function (returns a friendly string for complex types) if (typeof flagOrArg.defaultHelp === 'function') { diff --git a/test/help/help-test-utils.ts b/test/help/help-test-utils.ts index 0cf545125..76db5c203 100644 --- a/test/help/help-test-utils.ts +++ b/test/help/help-test-utils.ts @@ -4,7 +4,7 @@ import stripAnsi = require('strip-ansi') import {Interfaces} from '../../src' import {CommandHelp, Help} from '../../src/help' -import {toCached} from '../../src/util/to-cached' +import {cacheCommand} from '../../src/util/cache-command' export class TestCommandHelp extends CommandHelp { protected sections() { @@ -48,7 +48,7 @@ export class TestHelp extends Help { export const commandHelp = (command?: any) => ({ async run(ctx: {help: TestHelp; commandHelp: string; expectation: string}) { - const cached = await toCached(command!, {} as any, false) + const cached = await cacheCommand(command!, {} as any, false) const help = ctx.help.formatCommand(cached) if (process.env.TEST_OUTPUT === '1') { console.log(help) diff --git a/test/util.test.ts b/test/util/cache-command.test.ts similarity index 56% rename from test/util.test.ts rename to test/util/cache-command.test.ts index 889fa6464..7244d0777 100644 --- a/test/util.test.ts +++ b/test/util/cache-command.test.ts @@ -1,187 +1,8 @@ import {expect} from 'chai' -import {homedir} from 'node:os' -import {Args, Command, Flags} from '../src/index' -import {capitalize, castArray, ensureArgObject, getHomeDir, isNotFalsy, isTruthy, last, maxBy, readJson, sumBy} from '../src/util' -import {defaultToCached} from '../src/util/default-to-cached' -import {toCached} from '../src/util/to-cached' +import {Args, Command, Flags} from '../../src/index' +import {cacheCommand} from '../../src/util/cache-command' -describe('capitalize', () => { - it('capitalizes the string', () => { - expect(capitalize('dominik')).to.equal('Dominik') - }) - it('works with an empty string', () => { - expect(capitalize('')).to.equal('') - }) -}) - -type Item = { x: number } - -describe('sumBy', () => { - it('returns zero for empty array', () => { - const arr: Item[] = [] - expect(sumBy(arr, i => i.x)).to.equal(0) - }) - it('returns sum for non-empty array', () => { - const arr: Item[] = [{x: 1}, {x: 2}, {x: 3}] - expect(sumBy(arr, i => i.x)).to.equal(6) - }) -}) - -describe('maxBy', () => { - it('returns undefined for empty array', () => { - const arr: Item[] = [] - expect(maxBy(arr, i => i.x)).to.be.undefined - }) - it('returns max value in the array', () => { - const arr: Item[] = [{x: 1}, {x: 3}, {x: 2}] - expect(maxBy(arr, i => i.x)).to.equal(arr[1]) - }) -}) - -describe('last', () => { - it('returns undefined for empty array', () => { - expect(last([])).to.be.undefined - }) - it('returns undefined for undefined', () => { - expect(last()).to.be.undefined - }) - it('returns last value in the array', () => { - const arr: Item[] = [{x: 1}, {x: 3}, {x: 2}] - expect(last(arr)).to.equal(arr[2]) - }) - it('returns only item in array', () => { - expect(last([6])).to.equal(6) - }) -}) - -describe('ensureArgObject', () => { - it('should convert array of arguments to an object', () => { - const args = [ - {name: 'foo', description: 'foo desc', required: true}, - {name: 'bar', description: 'bar desc'}, - ] - const expected = {foo: args[0], bar: args[1]} - expect(ensureArgObject(args)).to.deep.equal(expected) - }) - - it('should do nothing to an arguments object', () => { - const args = { - foo: {name: 'foo', description: 'foo desc', required: true}, - bar: {name: 'bar', description: 'bar desc'}, - } - expect(ensureArgObject(args)).to.deep.equal(args) - }) -}) - -describe('isNotFalsy', () => { - it('should return true for truthy values', () => { - expect(isNotFalsy('true')).to.be.true - expect(isNotFalsy('1')).to.be.true - expect(isNotFalsy('yes')).to.be.true - expect(isNotFalsy('y')).to.be.true - }) - - it('should return false for falsy values', () => { - expect(isNotFalsy('false')).to.be.false - expect(isNotFalsy('0')).to.be.false - expect(isNotFalsy('no')).to.be.false - expect(isNotFalsy('n')).to.be.false - }) -}) - -describe('isTruthy', () => { - it('should return true for truthy values', () => { - expect(isTruthy('true')).to.be.true - expect(isTruthy('1')).to.be.true - expect(isTruthy('yes')).to.be.true - expect(isTruthy('y')).to.be.true - }) - - it('should return false for falsy values', () => { - expect(isTruthy('false')).to.be.false - expect(isTruthy('0')).to.be.false - expect(isTruthy('no')).to.be.false - expect(isTruthy('n')).to.be.false - }) -}) - -describe('getHomeDir', () => { - it('should return the home directory', () => { - expect(getHomeDir()).to.equal(homedir()) - }) -}) - -describe('readJson', () => { - it('should return parsed JSON', async () => { - const json = await readJson<{name: string}>('package.json') - expect(json.name).to.equal('@oclif/core') - }) - - it('should throw an error if the file does not exist', async () => { - try { - await readJson('does-not-exist.json') - throw new Error('Expected an error to be thrown') - } catch (error) { - const err = error as Error - expect(err.message).to.include('ENOENT: no such file or directory') - } - }) -}) - -describe('castArray', () => { - it('should cast a value to an array', () => { - expect(castArray('foo')).to.deep.equal(['foo']) - }) - - it('should return an array if the value is an array', () => { - expect(castArray(['foo'])).to.deep.equal(['foo']) - }) - - it('should return an empty array if the value is undefined', () => { - expect(castArray()).to.deep.equal([]) - }) -}) - -describe('defaultToCached', () => { - it('should do nothing if noCacheDefault is true', async () => { - const flag = {noCacheDefault: true} - const result = await defaultToCached(flag as any, true) - expect(result).to.be.undefined - }) - - it('should do nothing if respectNoCacheDefault is true', async () => { - const result = await defaultToCached({} as any, true) - expect(result).to.be.undefined - }) - - it('should return the result of defaultHelp if it exists', async () => { - const flag = {defaultHelp: async () => 'foo'} - const result = await defaultToCached(flag as any, false) - expect(result).to.equal('foo') - }) - - it('should return undefined if defaultHelp throws', async () => { - const flag = {async defaultHelp() { - throw new Error('foo') - }} - const result = await defaultToCached(flag as any, false) - expect(result).to.be.undefined - }) - - it('should return the result of the default if it\'s a function', async () => { - const flag = {default: async () => 'foo'} - const result = await defaultToCached(flag as any, false) - expect(result).to.equal('foo') - }) - - it('should return the result of the default if it\'s a simple value', async () => { - const flag = {default: 'foo'} - const result = await defaultToCached(flag as any, false) - expect(result).to.equal('foo') - }) -}) - -describe('toCached', () => { +describe('cacheCommand', () => { it('should return a cached command with every thing set', async () => { class C extends Command { static id = 'foo:bar' @@ -224,7 +45,7 @@ describe('toCached', () => { public async run(): Promise {} } - const c = await toCached(C, undefined, false) + const c = await cacheCommand(C, undefined, false) expect(c).to.deep.equal({ id: 'foo:bar', type: 'mytype', @@ -356,7 +177,7 @@ describe('toCached', () => { public async run(): Promise {} } - const cached = await toCached(Child, undefined, false) + const cached = await cacheCommand(Child, undefined, false) expect(cached).to.deep.equal({ id: 'command', @@ -456,7 +277,7 @@ describe('toCached', () => { public async run(): Promise {} } - const c = await toCached(C, undefined, false) + const c = await cacheCommand(C, undefined, false) expect(c.hasDynamicHelp).to.be.true }) }) diff --git a/test/util/cache-default-value.test.ts b/test/util/cache-default-value.test.ts new file mode 100644 index 000000000..9f7a2b6f9 --- /dev/null +++ b/test/util/cache-default-value.test.ts @@ -0,0 +1,42 @@ +import {expect} from 'chai' +import {cacheDefaultValue} from '../../src/util/cache-default-value' + +describe('cacheDefaultValue', () => { + it('should do nothing if noCacheDefault is true', async () => { + const flag = {noCacheDefault: true} + const result = await cacheDefaultValue(flag as any, true) + expect(result).to.be.undefined + }) + + it('should do nothing if respectNoCacheDefault is true', async () => { + const result = await cacheDefaultValue({} as any, true) + expect(result).to.be.undefined + }) + + it('should return the result of defaultHelp if it exists', async () => { + const flag = {defaultHelp: async () => 'foo'} + const result = await cacheDefaultValue(flag as any, false) + expect(result).to.equal('foo') + }) + + it('should return undefined if defaultHelp throws', async () => { + const flag = {async defaultHelp() { + throw new Error('foo') + }} + const result = await cacheDefaultValue(flag as any, false) + expect(result).to.be.undefined + }) + + it('should return the result of the default if it\'s a function', async () => { + const flag = {default: async () => 'foo'} + const result = await cacheDefaultValue(flag as any, false) + expect(result).to.equal('foo') + }) + + it('should return the result of the default if it\'s a simple value', async () => { + const flag = {default: 'foo'} + const result = await cacheDefaultValue(flag as any, false) + expect(result).to.equal('foo') + }) +}) + diff --git a/test/util/index.test.ts b/test/util/index.test.ts new file mode 100644 index 000000000..815e45f39 --- /dev/null +++ b/test/util/index.test.ts @@ -0,0 +1,141 @@ +import {expect} from 'chai' +import {homedir} from 'node:os' +import {capitalize, castArray, ensureArgObject, getHomeDir, isNotFalsy, isTruthy, last, maxBy, readJson, sumBy} from '../../src/util/index' + +describe('capitalize', () => { + it('capitalizes the string', () => { + expect(capitalize('dominik')).to.equal('Dominik') + }) + it('works with an empty string', () => { + expect(capitalize('')).to.equal('') + }) +}) + +type Item = { x: number } + +describe('sumBy', () => { + it('returns zero for empty array', () => { + const arr: Item[] = [] + expect(sumBy(arr, i => i.x)).to.equal(0) + }) + it('returns sum for non-empty array', () => { + const arr: Item[] = [{x: 1}, {x: 2}, {x: 3}] + expect(sumBy(arr, i => i.x)).to.equal(6) + }) +}) + +describe('maxBy', () => { + it('returns undefined for empty array', () => { + const arr: Item[] = [] + expect(maxBy(arr, i => i.x)).to.be.undefined + }) + it('returns max value in the array', () => { + const arr: Item[] = [{x: 1}, {x: 3}, {x: 2}] + expect(maxBy(arr, i => i.x)).to.equal(arr[1]) + }) +}) + +describe('last', () => { + it('returns undefined for empty array', () => { + expect(last([])).to.be.undefined + }) + it('returns undefined for undefined', () => { + expect(last()).to.be.undefined + }) + it('returns last value in the array', () => { + const arr: Item[] = [{x: 1}, {x: 3}, {x: 2}] + expect(last(arr)).to.equal(arr[2]) + }) + it('returns only item in array', () => { + expect(last([6])).to.equal(6) + }) +}) + +describe('ensureArgObject', () => { + it('should convert array of arguments to an object', () => { + const args = [ + {name: 'foo', description: 'foo desc', required: true}, + {name: 'bar', description: 'bar desc'}, + ] + const expected = {foo: args[0], bar: args[1]} + expect(ensureArgObject(args)).to.deep.equal(expected) + }) + + it('should do nothing to an arguments object', () => { + const args = { + foo: {name: 'foo', description: 'foo desc', required: true}, + bar: {name: 'bar', description: 'bar desc'}, + } + expect(ensureArgObject(args)).to.deep.equal(args) + }) +}) + +describe('isNotFalsy', () => { + it('should return true for truthy values', () => { + expect(isNotFalsy('true')).to.be.true + expect(isNotFalsy('1')).to.be.true + expect(isNotFalsy('yes')).to.be.true + expect(isNotFalsy('y')).to.be.true + }) + + it('should return false for falsy values', () => { + expect(isNotFalsy('false')).to.be.false + expect(isNotFalsy('0')).to.be.false + expect(isNotFalsy('no')).to.be.false + expect(isNotFalsy('n')).to.be.false + }) +}) + +describe('isTruthy', () => { + it('should return true for truthy values', () => { + expect(isTruthy('true')).to.be.true + expect(isTruthy('1')).to.be.true + expect(isTruthy('yes')).to.be.true + expect(isTruthy('y')).to.be.true + }) + + it('should return false for falsy values', () => { + expect(isTruthy('false')).to.be.false + expect(isTruthy('0')).to.be.false + expect(isTruthy('no')).to.be.false + expect(isTruthy('n')).to.be.false + }) +}) + +describe('getHomeDir', () => { + it('should return the home directory', () => { + expect(getHomeDir()).to.equal(homedir()) + }) +}) + +describe('readJson', () => { + it('should return parsed JSON', async () => { + const json = await readJson<{name: string}>('package.json') + expect(json.name).to.equal('@oclif/core') + }) + + it('should throw an error if the file does not exist', async () => { + try { + await readJson('does-not-exist.json') + throw new Error('Expected an error to be thrown') + } catch (error) { + const err = error as Error + expect(err.message).to.include('ENOENT: no such file or directory') + } + }) +}) + +describe('castArray', () => { + it('should cast a value to an array', () => { + expect(castArray('foo')).to.deep.equal(['foo']) + }) + + it('should return an array if the value is an array', () => { + expect(castArray(['foo'])).to.deep.equal(['foo']) + }) + + it('should return an empty array if the value is undefined', () => { + expect(castArray()).to.deep.equal([]) + }) +}) + From f51da6b85354f7684da4db4bef976d1c800070eb Mon Sep 17 00:00:00 2001 From: Mike Donnalley Date: Thu, 28 Sep 2023 10:40:21 -0600 Subject: [PATCH 27/27] perf: parallelize cacheCommand --- src/command.ts | 2 +- src/help/index.ts | 6 +- src/interfaces/parser.ts | 12 +-- src/parser/parse.ts | 7 +- src/util/cache-command.ts | 175 ++++++++++++++------------------ test/util/cache-command.test.ts | 23 ++++- 6 files changed, 111 insertions(+), 114 deletions(-) diff --git a/src/command.ts b/src/command.ts index 82d9b8798..9b365af69 100644 --- a/src/command.ts +++ b/src/command.ts @@ -411,7 +411,7 @@ export namespace Command { export type Flag = IFlag export namespace Flag { - export type Cached = Omit & (BooleanFlagProps | OptionFlagProps) + export type Cached = Omit & (BooleanFlagProps | OptionFlagProps) & {hasDynamicHelp?: boolean} export type Any = Flag | Cached } diff --git a/src/help/index.ts b/src/help/index.ts index 7abaf3d01..1a8d1f076 100644 --- a/src/help/index.ts +++ b/src/help/index.ts @@ -109,10 +109,10 @@ export class Help extends HelpBase { if (command) { if (command.hasDynamicHelp && command.pluginType !== 'jit') { const loaded = await command.load() - for (const flag of Object.values(loaded.flags)) { - if (flag.type === 'boolean') continue + for (const [name, flag] of Object.entries(loaded.flags)) { + if (flag.type === 'boolean' || !command.flags[name].hasDynamicHelp) continue // eslint-disable-next-line no-await-in-loop - flag.default = await cacheDefaultValue(flag, false) + command.flags[name].default = await cacheDefaultValue(flag, false) } await this.showCommandHelp(command) diff --git a/src/interfaces/parser.ts b/src/interfaces/parser.ts index ef0b8f690..a201323d4 100644 --- a/src/interfaces/parser.ts +++ b/src/interfaces/parser.ts @@ -161,12 +161,6 @@ export type FlagProps = { * Emit deprecation warning when a flag alias is provided */ deprecateAliases?: boolean - /** - * Delimiter to separate the values for a multiple value flag. - * Only respected if multiple is set to true. Default behavior is to - * separate on spaces. - */ - delimiter?: ',', /** * If true, the value returned by defaultHelp will not be cached in the oclif.manifest.json. * This is helpful if the default value contains sensitive data that shouldn't be published to npm. @@ -210,6 +204,12 @@ export type OptionFlagProps = FlagProps & { helpValue?: string; options?: readonly string[]; multiple?: boolean; + /** + * Delimiter to separate the values for a multiple value flag. + * Only respected if multiple is set to true. Default behavior is to + * separate on spaces. + */ + delimiter?: ',', } export type FlagParserContext = Command & {token: FlagToken} diff --git a/src/parser/parse.ts b/src/parser/parse.ts index 6376ec80c..fffcd68a7 100644 --- a/src/parser/parse.ts +++ b/src/parser/parse.ts @@ -250,7 +250,7 @@ export class Parser parseFlagOrThrowError( + valueFunction: async i => parseFlagOrThrowError( last(i.tokens)?.input !== `--no-${i.inputFlag.name}`, i.inputFlag.flag, this.context, @@ -262,8 +262,9 @@ export class Parser (await Promise.all( - ((i.tokens ?? []).flatMap(token => (token.input as string).split(i.inputFlag.flag.delimiter as string))) + ...fws, + valueFunction: async i => (await Promise.all( + ((i.tokens ?? []).flatMap(token => token.input.split((i.inputFlag.flag as OptionFlag).delimiter ?? ','))) // trim, and remove surrounding doubleQuotes (which would hav been needed if the elements contain spaces) .map(v => v.trim().replace(/^"(.*)"$/, '$1').replace(/^'(.*)'$/, '$1')) .map(async v => parseFlagOrThrowError(v, i.inputFlag.flag, this.context, {...last(i.tokens) as FlagToken, input: v})), diff --git a/src/util/cache-command.ts b/src/util/cache-command.ts index 3c8c5f5bf..176f98516 100644 --- a/src/util/cache-command.ts +++ b/src/util/cache-command.ts @@ -1,111 +1,93 @@ -/* eslint-disable no-await-in-loop */ +import {ArgInput, FlagInput} from '../interfaces/parser' import {ensureArgObject, pickBy} from './index' import {Command} from '../command' import {Plugin as IPlugin} from '../interfaces/plugin' import {aggregateFlags} from './aggregate-flags' import {cacheDefaultValue} from './cache-default-value' -export async function cacheCommand(cmd: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { - const flags = {} as {[k: string]: Command.Flag.Cached} +// In order to collect static properties up the inheritance chain, we need to recursively +// access the prototypes until there's nothing left. This allows us to combine baseFlags +// and flags as well as add in the json flag if enableJsonFlag is enabled. +function mergePrototype(result: Command.Class, cmd: Command.Class): Command.Class { + const proto = Object.getPrototypeOf(cmd) + const filteredProto = pickBy(proto, v => v !== undefined) as Command.Class + return Object.keys(proto).length > 0 ? mergePrototype({...filteredProto, ...result} as Command.Class, proto) : result +} - // In order to collect static properties up the inheritance chain, we need to recursively - // access the prototypes until there's nothing left. This allows us to combine baseFlags - // and flags as well as add in the json flag if enableJsonFlag is enabled. - const mergePrototype = (result: Command.Class, cmd: Command.Class): Command.Class => { - const proto = Object.getPrototypeOf(cmd) - const filteredProto = pickBy(proto, v => v !== undefined) as Command.Class - return Object.keys(proto).length > 0 ? mergePrototype({...filteredProto, ...result} as Command.Class, proto) : result - } +async function cacheFlags(cmdFlags: FlagInput, respectNoCacheDefault: boolean): Promise> { + const promises = Object.entries(cmdFlags).map(async ([name, flag]) => ([name, { + name, + char: flag.char, + summary: flag.summary, + hidden: flag.hidden, + required: flag.required, + helpLabel: flag.helpLabel, + helpGroup: flag.helpGroup, + description: flag.description, + dependsOn: flag.dependsOn, + relationships: flag.relationships, + exclusive: flag.exclusive, + deprecated: flag.deprecated, + deprecateAliases: flag.deprecateAliases, + aliases: flag.aliases, + charAliases: flag.charAliases, + noCacheDefault: flag.noCacheDefault, + ...flag.type === 'boolean' ? { + allowNo: flag.allowNo, + type: flag.type, + } : { + type: flag.type, + helpValue: flag.helpValue, + multiple: flag.multiple, + options: flag.options, + delimiter: flag.delimiter, + default: await cacheDefaultValue(flag, respectNoCacheDefault), + hasDynamicHelp: typeof flag.defaultHelp === 'function', + }, + }])) + return Object.fromEntries(await Promise.all(promises)) +} - const c = mergePrototype(cmd, cmd) +async function cacheArgs(cmdArgs: ArgInput, respectNoCacheDefault: boolean): Promise> { + const promises = Object.entries(cmdArgs).map(async ([name, arg]) => ([name, { + name, + description: arg.description, + required: arg.required, + options: arg.options, + default: await cacheDefaultValue(arg, respectNoCacheDefault), + hidden: arg.hidden, + noCacheDefault: arg.noCacheDefault, + }])) + return Object.fromEntries(await Promise.all(promises)) +} - const cmdFlags = aggregateFlags(c.flags, c.baseFlags, c.enableJsonFlag) +export async function cacheCommand(uncachedCmd: Command.Class, plugin?: IPlugin, respectNoCacheDefault = false): Promise { + const cmd = mergePrototype(uncachedCmd, uncachedCmd) - for (const [name, flag] of Object.entries(cmdFlags)) { - if (flag.type === 'boolean') { - flags[name] = { - name, - type: flag.type, - char: flag.char, - summary: flag.summary, - description: flag.description, - hidden: flag.hidden, - required: flag.required, - helpLabel: flag.helpLabel, - helpGroup: flag.helpGroup, - allowNo: flag.allowNo, - dependsOn: flag.dependsOn, - relationships: flag.relationships, - exclusive: flag.exclusive, - deprecated: flag.deprecated, - deprecateAliases: c.deprecateAliases, - aliases: flag.aliases, - charAliases: flag.charAliases, - delimiter: flag.delimiter, - noCacheDefault: flag.noCacheDefault, - } - } else { - flags[name] = { - name, - type: flag.type, - char: flag.char, - summary: flag.summary, - description: flag.description, - hidden: flag.hidden, - required: flag.required, - helpLabel: flag.helpLabel, - helpValue: flag.helpValue, - helpGroup: flag.helpGroup, - multiple: flag.multiple, - options: flag.options, - dependsOn: flag.dependsOn, - relationships: flag.relationships, - exclusive: flag.exclusive, - default: await cacheDefaultValue(flag, respectNoCacheDefault), - deprecated: flag.deprecated, - deprecateAliases: c.deprecateAliases, - aliases: flag.aliases, - charAliases: flag.charAliases, - delimiter: flag.delimiter, - noCacheDefault: flag.noCacheDefault, - } - // a command-level placeholder in the manifest so that oclif knows it should regenerate the command during help-time - if (typeof flag.defaultHelp === 'function') { - c.hasDynamicHelp = true - } - } - } - - const args = {} as {[k: string]: Command.Arg.Cached} - for (const [name, arg] of Object.entries(ensureArgObject(c.args))) { - args[name] = { - name, - description: arg.description, - required: arg.required, - options: arg.options, - default: await cacheDefaultValue(arg, respectNoCacheDefault), - hidden: arg.hidden, - noCacheDefault: arg.noCacheDefault, - } - } + const flags = await cacheFlags( + aggregateFlags(cmd.flags, cmd.baseFlags, cmd.enableJsonFlag), + respectNoCacheDefault, + ) + const args = await cacheArgs(ensureArgObject(cmd.args), respectNoCacheDefault) const stdProperties = { - id: c.id, - summary: c.summary, - description: c.description, - strict: c.strict, - usage: c.usage, + id: cmd.id, + summary: cmd.summary, + description: cmd.description, + strict: cmd.strict, + usage: cmd.usage, pluginName: plugin && plugin.name, pluginAlias: plugin && plugin.alias, pluginType: plugin && plugin.type, - hidden: c.hidden, - state: c.state, - aliases: c.aliases || [], - examples: c.examples || (c as any).example, - deprecationOptions: c.deprecationOptions, - deprecateAliases: c.deprecateAliases, + hidden: cmd.hidden, + state: cmd.state, + aliases: cmd.aliases || [], + examples: cmd.examples || (cmd as any).example, + deprecationOptions: cmd.deprecationOptions, + deprecateAliases: cmd.deprecateAliases, flags, args, + hasDynamicHelp: Object.values(flags).some(f => f.hasDynamicHelp), } // do not include these properties in manifest @@ -119,12 +101,11 @@ export async function cacheCommand(cmd: Command.Class, plugin?: IPlugin, respect '_--', '_base', ] - const stdKeys = Object.keys(stdProperties) - const keysToAdd = Object.keys(c).filter(property => ![...stdKeys, ...ignoreCommandProperties].includes(property)) - const additionalProperties: Record = {} - for (const key of keysToAdd) { - additionalProperties[key] = (c as any)[key] - } + + // Add in any additional properties that are not standard command properties. + const stdKeysAndIgnored = new Set([...Object.keys(stdProperties), ...ignoreCommandProperties]) + const keysToAdd = Object.keys(cmd).filter(property => !stdKeysAndIgnored.has(property)) + const additionalProperties = Object.fromEntries(keysToAdd.map(key => [key, (cmd as any)[key]])) return {...stdProperties, ...additionalProperties} } diff --git a/test/util/cache-command.test.ts b/test/util/cache-command.test.ts index 7244d0777..b46a878fe 100644 --- a/test/util/cache-command.test.ts +++ b/test/util/cache-command.test.ts @@ -84,7 +84,6 @@ describe('cacheCommand', () => { relationships: undefined, allowNo: false, type: 'boolean', - delimiter: undefined, noCacheDefault: undefined, }, flagb: { @@ -110,6 +109,7 @@ describe('cacheCommand', () => { options: ['a', 'b'], delimiter: undefined, noCacheDefault: undefined, + hasDynamicHelp: false, }, flagc: { aliases: undefined, @@ -137,6 +137,7 @@ describe('cacheCommand', () => { summary: undefined, type: 'option', noCacheDefault: undefined, + hasDynamicHelp: false, }, }, @@ -213,7 +214,6 @@ describe('cacheCommand', () => { deprecateAliases: undefined, aliases: undefined, charAliases: undefined, - delimiter: undefined, noCacheDefault: undefined, }, childFlag: { @@ -234,7 +234,6 @@ describe('cacheCommand', () => { deprecateAliases: undefined, aliases: undefined, charAliases: undefined, - delimiter: undefined, noCacheDefault: undefined, }, parentFlag: { @@ -255,7 +254,6 @@ describe('cacheCommand', () => { deprecateAliases: undefined, aliases: undefined, charAliases: undefined, - delimiter: undefined, noCacheDefault: undefined, }, }, @@ -279,5 +277,22 @@ describe('cacheCommand', () => { } const c = await cacheCommand(C, undefined, false) expect(c.hasDynamicHelp).to.be.true + expect(c.flags.flagb.hasDynamicHelp).to.be.true + }) + + it('should add additional command properties', async () => { + class C extends Command { + static id = 'foo:bar' + static flags = { + flaga: Flags.boolean(), + } + + static envVars = ['FOO_BAR'] + + public async run(): Promise {} + } + + const c = await cacheCommand(C, undefined, false) + expect(c.envVars).to.deep.equal(['FOO_BAR']) }) })