192 lines
6.1 KiB
TypeScript
192 lines
6.1 KiB
TypeScript
import ConfigurationDefinition, {DefinitionType} from "./configurationDefinition";
|
|
import {parser, ArgCollection} from "args-command-parser";
|
|
|
|
class Configuration {
|
|
|
|
private readonly arguments: ArgCollection;
|
|
private readonly defaultCommand: string;
|
|
private readonly definitions: Map<string, ConfigurationDefinition> = new Map();
|
|
private readonly envPrefix: string;
|
|
private readonly values: Map<string, string[] | undefined> = new Map();
|
|
public constructor (defaultCommand: string, envPrefix: string, ...definitions: ConfigurationDefinition[]) {
|
|
this.arguments = parser();
|
|
for (const definition of definitions) {
|
|
this.definitions.set(definition.getName(), definition);
|
|
this.values.set(definition.getName(), this.getInternalValue(definition));
|
|
}
|
|
this.defaultCommand = defaultCommand;
|
|
this.envPrefix = envPrefix;
|
|
}
|
|
|
|
/**
|
|
* Get the value of a boolean parameter
|
|
* @param {string} name - The name of the parameter.
|
|
* @returns The value of the parameter.
|
|
*/
|
|
public getBooleanValue (name: string): boolean | boolean[] | undefined {
|
|
const def = this.definitions.get(name);
|
|
if(def?.getType() == DefinitionType.Boolean) {
|
|
const value = this.values.get(name) ?? [];
|
|
const result: boolean[] = this.toBoolean(value);
|
|
if(def.getMultiple()) {
|
|
return result;
|
|
}
|
|
return result[0];
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
/**
|
|
* Get the command from the arguments
|
|
* @returns The command that was passed in the arguments.
|
|
*/
|
|
public getCommand (): string {
|
|
if(this.arguments.data.commands.length >= 1)
|
|
return this.arguments.data.commands[0];
|
|
else
|
|
return this.defaultCommand;
|
|
}
|
|
|
|
public getDefinitions (): ConfigurationDefinition[] {
|
|
return Array.from(this.definitions.values());
|
|
}
|
|
|
|
public getNumberValue (name: string): number | number[] | undefined {
|
|
const def = this.definitions.get(name);
|
|
if(def?.getType() == DefinitionType.Number) {
|
|
const value = this.values.get(name) ?? [];
|
|
const result: number[] = this.toInteger(value);
|
|
if(def.getMultiple()) {
|
|
return result;
|
|
}
|
|
return result[0];
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
public getStringValue (name: string): string | string[] | undefined {
|
|
const def = this.definitions.get(name);
|
|
if(def?.getType() == DefinitionType.String) {
|
|
const value = this.values.get(name) ?? [];
|
|
if(def.getMultiple()) {
|
|
return value;
|
|
}
|
|
return value[0];
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
/**
|
|
* Get the value of a parameter
|
|
* @param {string} name - The name of the parameter.
|
|
* @returns The value of the parameter.
|
|
*/
|
|
public getValue (name: string): string | string[] | boolean | boolean[] | number | number[] | undefined {
|
|
const def = this.definitions.get(name);
|
|
const value = this.values.get(name) ?? [];
|
|
switch ( def?.getType() ) {
|
|
case DefinitionType.Boolean: {
|
|
const result = this.toBoolean(value);
|
|
if(def.getMultiple()) {
|
|
return result;
|
|
}
|
|
return result[0];
|
|
}
|
|
case DefinitionType.Number: {
|
|
const result: number[] = this.toInteger(value);
|
|
if(def.getMultiple()) {
|
|
return result;
|
|
}
|
|
return result[0];
|
|
}
|
|
case DefinitionType.String: {
|
|
if(def.getMultiple()) {
|
|
return value;
|
|
}
|
|
return value[0];
|
|
}
|
|
}
|
|
}
|
|
|
|
public validate (): void {
|
|
this.definitions.forEach((v, _k, _m) => {
|
|
const def = v;
|
|
const values = this.values.get(def.getName()) ?? [];
|
|
if(!def.getMultiple() && values.length > 1)
|
|
throw new Error("No more than one value may be assigned to parameter '" + def.getName() + "'.");
|
|
if(def.getRequired() && (def.getCommands().length === 0 || def.getCommands().includes(this.getCommand())) && values.length === 0)
|
|
throw new Error("Parameter '" + def.getName() + "' is required.");
|
|
for (const value of values) {
|
|
if(!def.getValidator().validate(value))
|
|
throw new Error("Parameter '" + def.getName() + "' does not meet the requirements: " + def.getValidator().description());
|
|
}
|
|
});
|
|
}
|
|
|
|
private getArgValue (def: ConfigurationDefinition): string[] | undefined{
|
|
for (const arg of def.getArgName()) {
|
|
if(arg.length >= 2 && this.arguments.hasLongSwitch(arg)) {
|
|
return this.arguments.getLongSwitch(arg).values;
|
|
} else if (this.arguments.hasShortSwitch(arg)) {
|
|
return this.arguments.getShortSwitch(arg).values;
|
|
}
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
/**
|
|
* Convert a string array to a boolean array
|
|
* @param {string[]} value - The value of the parameter.
|
|
* @returns The `toBoolean` function returns an array of booleans.
|
|
*/
|
|
|
|
private getEnvValue (def: ConfigurationDefinition): string[] | undefined {
|
|
const value = process.env[this.envPrefix + "_" + def.getEnvName().toUpperCase()];
|
|
if(value !== undefined)
|
|
return [value];
|
|
else
|
|
return undefined;
|
|
}
|
|
|
|
private getInternalValue (def: ConfigurationDefinition): string[] | undefined {
|
|
let value = this.getArgValue(def);
|
|
value = (value === undefined)?this.getEnvValue(def):value;
|
|
value = (value === undefined)?def.getDefaultValue():value;
|
|
if(value !== undefined) {
|
|
return value;
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
private toBoolean (value: string[]): boolean[] {
|
|
const result: boolean[] = [];
|
|
for (const v of value) {
|
|
result.push(v === "true" || v === "1" || v === "y");
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* It takes an array of strings and returns an array of numbers.
|
|
* @param {string[]} value - The value of the parameter.
|
|
* @returns The `toInteger` function returns an array of numbers.
|
|
*/
|
|
private toInteger (value: string[]): number[] {
|
|
const result: number[] = [];
|
|
for (const v of value) {
|
|
const f = parseFloat(v); const i = parseInt(v);
|
|
result.push((f == i) ? i : f);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
}
|
|
|
|
let configInstance: Configuration;
|
|
|
|
export function initConfiguration (envPrefix: string, ...definitions: ConfigurationDefinition[]): void {
|
|
configInstance = new Configuration("help", envPrefix, ...definitions);
|
|
}
|
|
export default function config (): Configuration {
|
|
return configInstance;
|
|
} |