mirror of
https://github.com/swc-project/swc.git
synced 2024-12-21 04:32:01 +03:00
321 lines
11 KiB
TypeScript
321 lines
11 KiB
TypeScript
// Loaded from https://deno.land/x/cliffy@v0.12.1/packages/flags/lib/flags.ts
|
|
|
|
|
|
import camelCase from '../../x/camelCase.ts';
|
|
import { normalize } from './normalize.ts';
|
|
import { IFlagArgument, IFlagOptions, IFlags, IFlagsResult, IFlagValue, IFlagValueType, IParseOptions, IType, OptionType } from './types.ts';
|
|
import { boolean } from './types/boolean.ts';
|
|
import { number } from './types/number.ts';
|
|
import { string } from './types/string.ts';
|
|
import { validateFlags } from './validate-flags.ts';
|
|
|
|
const Types: Record<string, IType<any>> = {
|
|
[ OptionType.STRING ]: string,
|
|
[ OptionType.NUMBER ]: number,
|
|
[ OptionType.BOOLEAN ]: boolean
|
|
};
|
|
|
|
/**
|
|
* Parse command line arguments.
|
|
*
|
|
* @param args Command line arguments e.g: `Deno.args`
|
|
* @param opts Parse options.
|
|
*/
|
|
export function parseFlags<O = any>( args: string[], opts: IParseOptions = {} ): IFlagsResult<O> {
|
|
|
|
!opts.flags && ( opts.flags = [] );
|
|
|
|
const normalized = normalize( args );
|
|
|
|
let inLiteral = false;
|
|
let negate = false;
|
|
|
|
const flags: IFlags = {};
|
|
const literal: string[] = [];
|
|
const unknown: string[] = [];
|
|
let stopEarly: boolean = false;
|
|
|
|
opts.flags.forEach( opt => {
|
|
opt.depends?.forEach( flag => {
|
|
if ( !opts.flags || !getOption( opts.flags, flag ) ) {
|
|
throw new Error( `Unknown required option: ${ flag }` );
|
|
}
|
|
} );
|
|
opt.conflicts?.forEach( flag => {
|
|
if ( !opts.flags || !getOption( opts.flags, flag ) ) {
|
|
throw new Error( `Unknown conflicting option: ${ flag }` );
|
|
}
|
|
} );
|
|
} );
|
|
|
|
for ( let i = 0; i < normalized.length; i++ ) {
|
|
|
|
let option: IFlagOptions | undefined;
|
|
let args: IFlagArgument[] | undefined;
|
|
const current = normalized[ i ];
|
|
|
|
// literal args after --
|
|
if ( inLiteral ) {
|
|
literal.push( current );
|
|
continue;
|
|
}
|
|
|
|
if ( current === '--' ) {
|
|
inLiteral = true;
|
|
continue;
|
|
}
|
|
|
|
const isFlag = current.length > 1 && current[ 0 ] === '-';
|
|
const next = () => normalized[ i + 1 ];
|
|
|
|
if ( isFlag && !stopEarly ) {
|
|
|
|
if ( current[ 2 ] === '-' || ( current[ 1 ] === '-' && current.length === 3 ) ) {
|
|
throw new Error( `Invalid flag name: ${ current }` );
|
|
}
|
|
|
|
negate = current.indexOf( '--no-' ) === 0;
|
|
|
|
const name = current.replace( /^-+(no-)?/, '' );
|
|
|
|
option = getOption( opts.flags, name );
|
|
|
|
if ( !option ) {
|
|
|
|
if ( opts.flags.length ) {
|
|
throw new Error( `Unknown option: ${ current }` );
|
|
}
|
|
|
|
option = {
|
|
name,
|
|
optionalValue: true,
|
|
type: OptionType.STRING
|
|
};
|
|
}
|
|
|
|
if ( !option.name ) {
|
|
throw new Error( `Missing name for option: ${ current }` );
|
|
}
|
|
|
|
const friendlyName: string = camelCase( option.name );
|
|
|
|
if ( typeof flags[ friendlyName ] !== 'undefined' && !option.collect ) {
|
|
throw new Error( `Duplicate option: ${ current }` );
|
|
}
|
|
|
|
args = option.args?.length ? option.args : [ {
|
|
type: option.type,
|
|
requiredValue: option.requiredValue,
|
|
optionalValue: option.optionalValue,
|
|
variadic: option.variadic,
|
|
list: option.list,
|
|
separator: option.separator
|
|
} ];
|
|
|
|
let argIndex = 0;
|
|
let inOptionalArg = false;
|
|
const previous = flags[ friendlyName ];
|
|
|
|
parseNext( option, args );
|
|
|
|
if ( typeof flags[ friendlyName ] === 'undefined' ) {
|
|
|
|
if ( typeof option.default !== 'undefined' ) {
|
|
flags[ friendlyName ] = typeof option.default === 'function' ? option.default() : option.default;
|
|
} else if ( args[ argIndex ].requiredValue ) {
|
|
throw new Error( `Missing value for option: --${ option.name }` );
|
|
} else {
|
|
flags[ friendlyName ] = true;
|
|
}
|
|
}
|
|
|
|
if ( typeof option.value !== 'undefined' ) {
|
|
flags[ friendlyName ] = option.value( flags[ friendlyName ], previous );
|
|
} else if ( option.collect ) {
|
|
const value = ( previous || [] ) as IFlagValue[];
|
|
value.push( flags[ friendlyName ] as IFlagValue );
|
|
flags[ friendlyName ] = value;
|
|
}
|
|
|
|
/** Parse next argument for current option. */
|
|
function parseNext( option: IFlagOptions, args: IFlagArgument[] ): void {
|
|
|
|
const arg: IFlagArgument = args[ argIndex ];
|
|
|
|
if ( !arg ) {
|
|
throw new Error( 'Unknown option: ' + next() );
|
|
}
|
|
|
|
if ( !arg.type ) {
|
|
arg.type = OptionType.BOOLEAN;
|
|
}
|
|
|
|
if ( option.args?.length ) {
|
|
// make all value's required per default
|
|
if ( ( typeof arg.optionalValue === 'undefined' || arg.optionalValue === false ) &&
|
|
typeof arg.requiredValue === 'undefined'
|
|
) {
|
|
arg.requiredValue = true;
|
|
}
|
|
} else {
|
|
// make non boolean value required per default
|
|
if ( arg.type !== OptionType.BOOLEAN &&
|
|
( typeof arg.optionalValue === 'undefined' || arg.optionalValue === false ) &&
|
|
typeof arg.requiredValue === 'undefined'
|
|
) {
|
|
arg.requiredValue = true;
|
|
}
|
|
}
|
|
|
|
if ( arg.requiredValue ) {
|
|
if ( inOptionalArg ) {
|
|
throw new Error( `An required argument can not follow an optional argument but found in: ${ option.name }` );
|
|
}
|
|
} else {
|
|
inOptionalArg = true;
|
|
}
|
|
|
|
if ( negate ) {
|
|
if ( arg.type !== OptionType.BOOLEAN && !arg.optionalValue ) {
|
|
throw new Error( `Negate not supported by --${ option.name }. Only optional option or options of type boolean can be negated.` );
|
|
}
|
|
flags[ friendlyName ] = false;
|
|
return;
|
|
}
|
|
|
|
let result: IFlagValue | undefined;
|
|
let increase = false;
|
|
|
|
if ( arg.list && hasNext( arg ) ) {
|
|
|
|
const parsed: IFlagValueType[] = next()
|
|
.split( arg.separator || ',' )
|
|
.map( ( nextValue: string ) => {
|
|
const value = parseValue( option, arg, nextValue );
|
|
if ( typeof value === 'undefined' ) {
|
|
throw new Error( `List item of option --${ option?.name } must be of type ${ arg.type } but got: ${ nextValue }` );
|
|
}
|
|
return value;
|
|
} );
|
|
|
|
if ( parsed?.length ) {
|
|
result = parsed;
|
|
}
|
|
} else {
|
|
if ( hasNext( arg ) ) {
|
|
result = parseValue( option, arg, next() );
|
|
} else if ( arg.optionalValue && arg.type === OptionType.BOOLEAN ) {
|
|
result = true;
|
|
}
|
|
}
|
|
|
|
if ( increase ) {
|
|
i++;
|
|
if ( !arg.variadic ) {
|
|
argIndex++;
|
|
} else if ( args[ argIndex + 1 ] ) {
|
|
throw new Error( 'An argument cannot follow an variadic argument: ' + next() );
|
|
}
|
|
}
|
|
|
|
if ( typeof result !== 'undefined' && ( ( args.length > 1 ) || arg.variadic ) ) {
|
|
|
|
if ( !flags[ friendlyName ] ) {
|
|
flags[ friendlyName ] = [];
|
|
}
|
|
|
|
( flags[ friendlyName ] as IFlagValue[] ).push( result );
|
|
|
|
|
|
if ( hasNext( arg ) ) {
|
|
parseNext( option, args );
|
|
}
|
|
} else {
|
|
flags[ friendlyName ] = result;
|
|
}
|
|
|
|
/** Check if current option should have an argument. */
|
|
function hasNext( arg: IFlagArgument ): boolean {
|
|
|
|
return !!(
|
|
normalized[ i + 1 ] &&
|
|
( arg.optionalValue || arg.requiredValue || arg.variadic ) &&
|
|
( normalized[ i + 1 ][ 0 ] !== '-' ||
|
|
( arg.type === OptionType.NUMBER && !isNaN( normalized[ i + 1 ] as any ) )
|
|
) &&
|
|
arg );
|
|
}
|
|
|
|
/** Parse argument value. */
|
|
function parseValue( option: IFlagOptions, arg: IFlagArgument, nextValue: string ): IFlagValueType {
|
|
|
|
let result: IFlagValueType = opts.parse ?
|
|
opts.parse( arg.type || OptionType.STRING, option, arg, nextValue ) :
|
|
parseFlagValue( option, arg, nextValue );
|
|
|
|
if ( typeof result !== 'undefined' ) {
|
|
increase = true;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
if ( opts.stopEarly ) {
|
|
stopEarly = true;
|
|
}
|
|
unknown.push( current );
|
|
}
|
|
}
|
|
|
|
if ( opts.flags && opts.flags.length ) {
|
|
validateFlags( opts.flags, flags, opts.knownFlaks, opts.allowEmpty );
|
|
}
|
|
|
|
return { flags: flags as any as O, unknown, literal };
|
|
}
|
|
|
|
export function parseFlagValue( option: IFlagOptions, arg: IFlagArgument, nextValue: string ): any {
|
|
|
|
const type = Types[ arg.type || OptionType.STRING ];
|
|
|
|
if ( !type ) {
|
|
throw new Error( `Unknown type ${ arg.type }` );
|
|
}
|
|
|
|
return type( option, arg, nextValue );
|
|
}
|
|
|
|
/**
|
|
* Find option by name.
|
|
*
|
|
* @param flags Source option's array.
|
|
* @param name Name of the option.
|
|
*/
|
|
export function getOption( flags: IFlagOptions[], name: string ): IFlagOptions | undefined {
|
|
|
|
while ( name[ 0 ] === '-' ) {
|
|
name = name.slice( 1 );
|
|
}
|
|
|
|
for ( const flag of flags ) {
|
|
if ( isOption( flag, name ) ) {
|
|
return flag;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Check if option has name or alias.
|
|
*
|
|
* @param option The option to check.
|
|
* @param name The option name or alias.
|
|
*/
|
|
export function isOption( option: IFlagOptions, name: string ) {
|
|
return option.name === name ||
|
|
( option.aliases && option.aliases.indexOf( name ) !== -1 );
|
|
}
|