import { ActionType, ParamDefinitionAny, ParamDefinitionsMap, TaskArguments, TaskDefinition } from "../../../types"; import { PolarError } from "../errors"; import { ErrorDescriptor, ERRORS } from "../errors-list"; import * as types from "../params/argument-types"; import { POLAR_PARAM_DEFINITIONS } from "../params/polar-params"; /** * This class creates a task definition, which consists of: * * a name, that should be unique and will be used to call the task. * * a description. This is optional. * * the action that the task will execute. * * a set of parameters that can be used by the action. * */ export class SimpleTaskDefinition implements TaskDefinition { get description (): (string | undefined) { return this._description; } public readonly paramDefinitions: ParamDefinitionsMap = {}; public readonly positionalParamDefinitions: ParamDefinitionAny[] = []; public action: ActionType; private readonly _positionalParamNames: Set; private _hasVariadicParam: boolean; private _hasOptionalPositionalParam: boolean; private _description?: string; /** * Creates an empty task definition. * * This definition will have no params, and will throw a POLAR_ERROR205 if executed. * * @param name The task's name. * @param isInternal `true` if the task is internal, `false` otherwise. */ constructor ( public readonly name: string, public readonly isInternal: boolean = false ) { this._positionalParamNames = new Set(); this._hasVariadicParam = false; this._hasOptionalPositionalParam = false; this.action = () => { throw new PolarError(ERRORS.TASK_DEFINITIONS.ACTION_NOT_SET, { taskName: name }); }; } /** * Sets the task's description. * @param description The description. */ public setDescription (description: string): this { this._description = description; return this; } /** * Sets the task's action. * @param action The action. */ public setAction(action: ActionType): this { // TODO: There's probably something bad here. See types.ts for more info. this.action = action; return this; } /** * Adds a paramater to the task's definition. * * @remarks This will throw if the `name` is already used by this task or * by Polar's global parameters. * * @param name The parameter's name. * @param description The parameter's description. * @param defaultValue A default value. This must be `undefined` if `isOptional` is `true`. * @param type The param's `ArgumentType`. It will parse and validate the user's input. * @param isOptional `true` if the parameter is optional. It's default value is `true` if `defaultValue` is not `undefined`. */ public addParam( name: string, description?: string, defaultValue?: T, type?: types.ArgumentType, isOptional: boolean = defaultValue !== undefined ): this { if (type === undefined) { if (defaultValue === undefined) { return this.addParam( name, description, undefined, types.string, isOptional ); } if (typeof defaultValue !== "string") { throw new PolarError( ERRORS.TASK_DEFINITIONS.DEFAULT_VALUE_WRONG_TYPE, { paramName: name, taskName: this.name } ); } return this.addParam( name, description, defaultValue, types.string, isOptional ); } this._validateParamNameCasing(name); this._validateNameNotUsed(name); this._validateNoDefaultValueForMandatoryParam( defaultValue, isOptional, name ); this.paramDefinitions[name] = { name, defaultValue, type, description, isOptional, isFlag: false, isVariadic: false }; return this; } /** * Adds an optional paramater to the task's definition. * * @see addParam. * * @param name the parameter's name. * @param description the parameter's description. * @param defaultValue a default value. * @param type param's type. */ public addOptionalParam( name: string, description?: string, defaultValue?: T, type?: types.ArgumentType ): this { return this.addParam(name, description, defaultValue, type, true); } /** * Adds a boolean paramater or flag to the task's definition. * * Flags are params with default value set to `false`, and that don't expect * values to be set in the CLI. A normal boolean param must be called with * `--param true`, while a flag is called with `--flag`. * * @param name the parameter's name. * @param description the parameter's description. */ public addFlag (name: string, description?: string): this { this._validateParamNameCasing(name); this._validateNameNotUsed(name); this.paramDefinitions[name] = { name, defaultValue: false, type: types.boolean, description, isFlag: true, isOptional: true, isVariadic: false }; return this; } /** * Adds a positional paramater to the task's definition. * * @remarks This will throw if the `name` is already used by this task or * by Polar's global parameters. * @remarks This will throw if `isOptional` is `false` and an optional positional * param was already set. * @remarks This will throw if a variadic positional param is already set. * * @param name The parameter's name. * @param description The parameter's description. * @param defaultValue A default value. This must be `undefined` if `isOptional` is `true`. * @param type The param's `ArgumentType`. It will parse and validate the user's input. * @param isOptional `true` if the parameter is optional. It's default value is `true` if `defaultValue` is not `undefined`. */ public addPositionalParam( name: string, description?: string, defaultValue?: T, type?: types.ArgumentType, isOptional = defaultValue !== undefined ): this { if (type === undefined) { if (defaultValue === undefined) { return this.addPositionalParam( name, description, undefined, types.string, isOptional ); } if (typeof defaultValue !== "string") { throw new PolarError( ERRORS.TASK_DEFINITIONS.DEFAULT_VALUE_WRONG_TYPE, { paramName: name, taskName: this.name } ); } return this.addPositionalParam( name, description, defaultValue, types.string, isOptional ); } this._validateParamNameCasing(name); this._validateNameNotUsed(name); this._validateNotAfterVariadicParam(name); this._validateNoMandatoryParamAfterOptionalOnes(name, isOptional); this._validateNoDefaultValueForMandatoryParam( defaultValue, isOptional, name ); const definition = { name, defaultValue, type, description, isVariadic: false, isOptional, isFlag: false }; this._addPositionalParamDefinition(definition); return this; } /** * Adds an optional positional paramater to the task's definition. * * @see addPositionalParam. * * @param name the parameter's name. * @param description the parameter's description. * @param defaultValue a default value. * @param type param's type. */ public addOptionalPositionalParam( name: string, description?: string, defaultValue?: T, type?: types.ArgumentType ): this { return this.addPositionalParam(name, description, defaultValue, type, true); } /** * Adds a variadic positional paramater to the task's definition. Variadic * positional params act as `...rest` parameters in JavaScript. * * @param name The parameter's name. * @param description The parameter's description. * @param defaultValue A default value. This must be `undefined` if `isOptional` is `true`. * @param type The param's `ArgumentType`. It will parse and validate the user's input. * @param isOptional `true` if the parameter is optional. It's default value is `true` if `defaultValue` is not `undefined`. */ public addVariadicPositionalParam( name: string, description?: string, defaultValue?: T[] | T, type?: types.ArgumentType, isOptional = defaultValue !== undefined ): this { if (defaultValue !== undefined && !Array.isArray(defaultValue)) { defaultValue = [defaultValue]; } if (type === undefined) { if (defaultValue === undefined) { return this.addVariadicPositionalParam( name, description, undefined, types.string, isOptional ); } if (!this._isStringArray(defaultValue)) { throw new PolarError( ERRORS.TASK_DEFINITIONS.DEFAULT_VALUE_WRONG_TYPE, { paramName: name, taskName: this.name } ); } return this.addVariadicPositionalParam( name, description, defaultValue, types.string, isOptional ); } this._validateParamNameCasing(name); this._validateNameNotUsed(name); this._validateNotAfterVariadicParam(name); this._validateNoMandatoryParamAfterOptionalOnes(name, isOptional); this._validateNoDefaultValueForMandatoryParam( defaultValue, isOptional, name ); const definition = { name, defaultValue, type, description, isVariadic: true, isOptional, isFlag: false }; this._addPositionalParamDefinition(definition); return this; } /** * Adds a positional paramater to the task's definition. * * This will check if the `name` is already used and * if the parameter is being added after a varidic argument. * * @param name the parameter's name. * @param description the parameter's description. * @param defaultValue a default value. * @param type param's type. */ public addOptionalVariadicPositionalParam( name: string, description?: string, defaultValue?: T[] | T, type?: types.ArgumentType ): this { return this.addVariadicPositionalParam( name, description, defaultValue, type, true ); } /** * Adds a positional paramater to the task's definition. * * @param definition the param's definition */ private _addPositionalParamDefinition (definition: ParamDefinitionAny): void { if (definition.isVariadic) { this._hasVariadicParam = true; } if (definition.isOptional) { this._hasOptionalPositionalParam = true; } this._positionalParamNames.add(definition.name); this.positionalParamDefinitions.push(definition); } /** * Validates if the given param's name is after a variadic parameter. * @param name the param's name. * @throws POLAR_ERROR200 */ private _validateNotAfterVariadicParam (name: string): void { if (this._hasVariadicParam) { throw new PolarError(ERRORS.TASK_DEFINITIONS.PARAM_AFTER_VARIADIC, { paramName: name, taskName: this.name }); } } /** * Validates if the param's name is already used. * @param name the param's name. * * @throws POLAR_ERROR201 if `name` is already used as a param. * @throws POLAR_ERROR202 if `name` is already used as a param by Polar */ private _validateNameNotUsed (name: string): void { if (this._hasParamDefined(name)) { throw new PolarError(ERRORS.TASK_DEFINITIONS.PARAM_ALREADY_DEFINED, { paramName: name, taskName: this.name }); } if (Object.keys(POLAR_PARAM_DEFINITIONS).includes(name)) { throw new PolarError( ERRORS.TASK_DEFINITIONS.PARAM_CLASHES_WITH_POLAR_PARAM, { paramName: name, taskName: this.name } ); } } /** * Checks if the given name is already used. * @param name the param's name. */ private _hasParamDefined (name: string): boolean { return ( this.paramDefinitions[name] !== undefined || this._positionalParamNames.has(name) ); } /** * Validates if a mandatory param is being added after optional params. * * @param name the param's name to be added. * @param isOptional true if the new param is optional, false otherwise. * * @throws POLAR_ERROR203 if validation fail */ private _validateNoMandatoryParamAfterOptionalOnes ( name: string, isOptional: boolean ): void { if (!isOptional && this._hasOptionalPositionalParam) { throw new PolarError( ERRORS.TASK_DEFINITIONS.MANDATORY_PARAM_AFTER_OPTIONAL, { paramName: name, taskName: this.name } ); } } private _validateParamNameCasing (name: string): void { const pattern = /^[a-z]+([a-zA-Z0-9])*$/; const match = name.match(pattern); if (match === null) { throw new PolarError( ERRORS.TASK_DEFINITIONS.INVALID_PARAM_NAME_CASING, { paramName: name, taskName: this.name } ); } } private _validateNoDefaultValueForMandatoryParam ( defaultValue: any | undefined, // eslint-disable-line @typescript-eslint/no-explicit-any isOptional: boolean, name: string ): void { if (defaultValue !== undefined && !isOptional) { throw new PolarError( ERRORS.TASK_DEFINITIONS.DEFAULT_IN_MANDATORY_PARAM, { paramName: name, taskName: this.name } ); } } private _isStringArray (values: any): values is string[] { // eslint-disable-line @typescript-eslint/no-explicit-any return Array.isArray(values) && values.every((v) => typeof v === "string"); } } /** * Allows you to override a previously defined task. * * When overriding a task you can: * * flag it as internal * * set a new description * * set a new action * */ export class OverriddenTaskDefinition implements TaskDefinition { private _description?: string; private _action?: ActionType; constructor ( public readonly parentTaskDefinition: TaskDefinition, public readonly isInternal: boolean = false ) { this.isInternal = isInternal; this.parentTaskDefinition = parentTaskDefinition; } public setDescription (description: string): this { this._description = description; return this; } /** * Overrides the parent task's action. * @param action the action. */ public setAction(action: ActionType): this { // TODO: There's probably something bad here. See types.ts for more info. this._action = action; return this; } /** * Retrieves the parent task's name. */ get name (): string { return this.parentTaskDefinition.name; } /** * Retrieves, if defined, the description of the overriden task, * otherwise retrieves the description of the parent task. */ get description (): string { if (this._description !== undefined) { return this._description; } return this.parentTaskDefinition.description ?? ""; } /** * Retrieves, if defined, the action of the overriden task, * otherwise retrieves the action of the parent task. */ get action (): ActionType { if (this._action !== undefined) { return this._action; } return this.parentTaskDefinition.action; } /** * Retrieves the parent task's param definitions. */ get paramDefinitions (): ParamDefinitionsMap { return this.parentTaskDefinition.paramDefinitions; } /** * Retrieves the parent task's positional param definitions. */ get positionalParamDefinitions (): ParamDefinitionAny[] { return this.parentTaskDefinition.positionalParamDefinitions; } /** * Overriden tasks can't add new parameters. */ public addParam( name: string, description?: string, defaultValue?: T, type?: types.ArgumentType, isOptional?: boolean ): this { if (isOptional === undefined || !isOptional) { return this._throwNoParamsOverrideError( ERRORS.TASK_DEFINITIONS.OVERRIDE_NO_MANDATORY_PARAMS ); } return this.addOptionalParam(name, description, defaultValue, type); } /** * Overriden tasks can't add new parameters. */ public addOptionalParam( name: string, description?: string, defaultValue?: T, type?: types.ArgumentType ): this { this.parentTaskDefinition.addOptionalParam( name, description, defaultValue, type ); return this; } /** * Overriden tasks can't add new parameters. */ public addPositionalParam( name: string, description?: string, defaultValue?: T, type?: types.ArgumentType, isOptional?: boolean ): this { return this._throwNoParamsOverrideError( ERRORS.TASK_DEFINITIONS.OVERRIDE_NO_POSITIONAL_PARAMS ); } /* eslint-disable sonarjs/no-identical-functions */ /** * Overriden tasks can't add new parameters. */ public addOptionalPositionalParam( name: string, description?: string, defaultValue?: T, type?: types.ArgumentType ): this { return this._throwNoParamsOverrideError( ERRORS.TASK_DEFINITIONS.OVERRIDE_NO_POSITIONAL_PARAMS ); } /** * Overriden tasks can't add new parameters. */ public addVariadicPositionalParam( name: string, description?: string, defaultValue?: T[], type?: types.ArgumentType, isOptional?: boolean ): this { return this._throwNoParamsOverrideError( ERRORS.TASK_DEFINITIONS.OVERRIDE_NO_VARIADIC_PARAMS ); } /** * Overriden tasks can't add new parameters. */ public addOptionalVariadicPositionalParam( name: string, description?: string, defaultValue?: T[], type?: types.ArgumentType ): this { return this._throwNoParamsOverrideError( ERRORS.TASK_DEFINITIONS.OVERRIDE_NO_VARIADIC_PARAMS ); } /** * Add a flag param to the overridden task. * @throws POLAR_ERROR201 if param name was already defined in any parent task. * @throws POLAR_ERROR209 if param name is not in camelCase. */ public addFlag (name: string, description?: string): this { this.parentTaskDefinition.addFlag(name, description); return this; } private _throwNoParamsOverrideError (errorDescriptor: ErrorDescriptor): never { throw new PolarError(errorDescriptor, { taskName: this.name }); } }