testcafe
Version:
Automated browser testing for the modern web development stack.
232 lines • 33.4 kB
JavaScript
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const path_1 = require("path");
const debug_1 = __importDefault(require("debug"));
const json5_1 = __importDefault(require("json5"));
const lodash_1 = require("lodash");
const promisified_functions_1 = require("../utils/promisified-functions");
const option_1 = __importDefault(require("./option"));
const option_source_1 = __importDefault(require("./option-source"));
const resolve_path_relatively_cwd_1 = __importDefault(require("../utils/resolve-path-relatively-cwd"));
const render_template_1 = __importDefault(require("../utils/render-template"));
const warning_message_1 = __importDefault(require("../notifications/warning-message"));
const log_1 = __importDefault(require("../cli/log"));
const formats_1 = __importDefault(require("./formats"));
const runtime_1 = require("../errors/runtime");
const types_1 = require("../errors/types");
const DEBUG_LOGGER = (0, debug_1.default)('testcafe:configuration');
class Configuration {
constructor(configurationFilesNames) {
var _a;
this._options = {};
this._defaultPaths = this._resolveFilePaths(configurationFilesNames);
this._filePath = (_a = this._defaultPaths) === null || _a === void 0 ? void 0 : _a[0];
this._overriddenOptions = [];
}
static _fromObj(obj) {
const result = Object.create(null);
Object.entries(obj).forEach(([key, value]) => {
result[key] = new option_1.default(key, value);
});
return result;
}
static _showConsoleWarning(message) {
log_1.default.write(message);
}
static _throwReadConfigError(code, error, path, renderCallsite) {
const readConfigError = new runtime_1.ReadConfigFileError(code, error, path, renderCallsite);
DEBUG_LOGGER(readConfigError.message);
DEBUG_LOGGER(error);
throw readConfigError;
}
static _resolveFilePath(path) {
if (!path)
return null;
return (0, path_1.isAbsolute)(path) ? path : (0, resolve_path_relatively_cwd_1.default)(path);
}
_resolveFilePaths(filesNames) {
if (!filesNames)
return void 0;
return (0, lodash_1.castArray)(filesNames).reduce((result, name) => {
const resolveFilePath = Configuration._resolveFilePath(name);
if (resolveFilePath)
result.push(resolveFilePath);
return result;
}, []);
}
async init() {
this._overriddenOptions = [];
}
mergeOptions(options) {
Object.entries(options).map(([key, value]) => {
const option = this._ensureOption(key, value, option_source_1.default.Input);
if (value === void 0)
return;
this._setOptionValue(option, value);
});
}
mergeDeep(option, source) {
(0, lodash_1.mergeWith)(option.value, source, (targetValue, sourceValue, property) => {
this._addOverriddenOptionIfNecessary(targetValue, sourceValue, option.source, `${option.name}.${property}`);
return sourceValue !== void 0 ? sourceValue : targetValue;
});
}
_getOption(key) {
if (!key)
return void 0;
const option = this._options[key];
if (!option)
return void 0;
return option.value;
}
getOption(key) {
return this._getOption(key);
}
getOptions(predicate) {
const result = Object.create(null);
let includeInResult = true;
Object.entries(this._options).forEach(([name, option]) => {
includeInResult = predicate ? predicate(name, option) : true;
if (includeInResult)
result[name] = option.value;
});
return result;
}
clone(nonClonedOptions) {
const configuration = (0, lodash_1.cloneDeep)(this);
if (nonClonedOptions) {
(0, lodash_1.castArray)(nonClonedOptions).forEach(key => {
if (configuration._options[key])
configuration._options[key].value = this._options[key].value;
});
}
return configuration;
}
get filePath() {
return this._filePath;
}
get defaultPaths() {
return this._defaultPaths;
}
async _load() {
var _a;
if (!((_a = this.defaultPaths) === null || _a === void 0 ? void 0 : _a.length))
return null;
const configs = await Promise.all(this.defaultPaths.map(async (filePath) => {
if (!await this._isConfigurationFileExists(filePath))
return { filePath, options: null };
let options = null;
if (this._isJSConfiguration(filePath))
options = this._readJsConfigurationFileContent(filePath);
else {
const configurationFileContent = await this._readConfigurationFileContent(filePath);
if (configurationFileContent)
options = this._parseConfigurationFileContent(configurationFileContent, filePath);
}
return { filePath, options };
}));
const existedConfigs = configs.filter(config => !!config.options);
if (!existedConfigs.length)
return null;
this._filePath = existedConfigs[0].filePath;
if (existedConfigs.length > 1)
Configuration._showConsoleWarning((0, render_template_1.default)(warning_message_1.default.multipleConfigurationFilesFound, this._filePath));
return existedConfigs[0].options;
}
async _isConfigurationFileExists(filePath = this.filePath) {
try {
await (0, promisified_functions_1.stat)(filePath);
return true;
}
catch (error) {
DEBUG_LOGGER((0, render_template_1.default)(warning_message_1.default.cannotFindConfigurationFile, filePath, error.stack));
return false;
}
}
static _hasExtension(filePath, extention) {
return !!filePath && (0, path_1.extname)(filePath) === extention;
}
_isJSConfiguration(filePath = this.filePath) {
return Configuration._hasExtension(filePath, formats_1.default.js) || Configuration._hasExtension(filePath, formats_1.default.cjs);
}
_isJSONConfiguration(filePath = this.filePath) {
return Configuration._hasExtension(filePath, formats_1.default.json);
}
_readJsConfigurationFileContent(filePath = this.filePath) {
if (filePath) {
try {
delete require.cache[filePath];
return require(filePath);
}
catch (error) {
Configuration._throwReadConfigError(types_1.RUNTIME_ERRORS.cannotReadConfigFile, error, filePath, true);
}
}
return null;
}
async _readConfigurationFileContent(filePath = this.filePath) {
try {
return await (0, promisified_functions_1.readFile)(filePath);
}
catch (error) {
Configuration._throwReadConfigError(types_1.RUNTIME_ERRORS.cannotReadConfigFile, error, filePath || '', false);
}
return null;
}
_parseConfigurationFileContent(configurationFileContent, filePath = this.filePath) {
try {
return json5_1.default.parse(configurationFileContent.toString());
}
catch (error) {
Configuration._throwReadConfigError(types_1.RUNTIME_ERRORS.cannotParseConfigFile, error, filePath || '', false);
}
return null;
}
_ensureArrayOption(name) {
const options = this._options[name];
if (!options)
return;
// NOTE: a hack to fix lodash type definitions
// @ts-ignore
options.value = (0, lodash_1.castArray)(options.value);
}
_ensureOption(name, value, source) {
let option = null;
if (name in this._options)
option = this._options[name];
else {
option = new option_1.default(name, value, source);
this._options[name] = option;
}
return option;
}
_ensureOptionWithValue(name, defaultValue, source) {
const option = this._ensureOption(name, defaultValue, source);
if (option.value !== void 0)
return;
option.value = defaultValue;
option.source = source;
}
_addOverriddenOptionIfNecessary(value1, value2, source, optionName) {
if (source === option_source_1.default.Default)
return;
if (value1 === void 0 || value2 === void 0 || value1 === value2 || source !== option_source_1.default.Configuration)
return;
this._overriddenOptions.push(optionName);
}
_setOptionValue(option, value) {
if ((0, lodash_1.isPlainObject)(option.value) && (0, lodash_1.isPlainObject)(value))
this.mergeDeep(option, value);
else {
this._addOverriddenOptionIfNecessary(option.value, value, option.source, option.name);
option.value = value;
}
option.source = option_source_1.default.Input;
}
}
exports.default = Configuration;
module.exports = exports.default;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"configuration-base.js","sourceRoot":"","sources":["../../src/configuration/configuration-base.ts"],"names":[],"mappings":"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","sourcesContent":["import { extname, isAbsolute } from 'path';\nimport debug from 'debug';\nimport JSON5 from 'json5';\n\nimport {\n    castArray,\n    cloneDeep,\n    isPlainObject,\n    mergeWith,\n} from 'lodash';\n\nimport { readFile, stat } from '../utils/promisified-functions';\nimport Option from './option';\nimport OptionSource from './option-source';\nimport resolvePathRelativelyCwd from '../utils/resolve-path-relatively-cwd';\nimport renderTemplate from '../utils/render-template';\nimport WARNING_MESSAGES from '../notifications/warning-message';\nimport log from '../cli/log';\nimport { Dictionary } from './interfaces';\nimport Extensions from './formats';\nimport { ReadConfigFileError } from '../errors/runtime';\nimport { RUNTIME_ERRORS } from '../errors/types';\n\nconst DEBUG_LOGGER = debug('testcafe:configuration');\n\nexport default class Configuration {\n    protected _options: Dictionary<Option>;\n    protected _filePath?: string;\n    protected readonly _defaultPaths?: string[];\n    protected _overriddenOptions: string[];\n\n    public constructor (configurationFilesNames: string | null | string[]) {\n        this._options           = {};\n        this._defaultPaths      = this._resolveFilePaths(configurationFilesNames);\n        this._filePath          = this._defaultPaths?.[0];\n        this._overriddenOptions = [];\n    }\n\n    protected static _fromObj (obj: object): Dictionary<Option> {\n        const result = Object.create(null);\n\n        Object.entries(obj).forEach(([key, value]) => {\n            result[key] = new Option(key, value);\n        });\n\n        return result;\n    }\n\n    protected static _showConsoleWarning (message: string): void {\n        log.write(message);\n    }\n\n    private static _throwReadConfigError (code: string, error: Error, path: string, renderCallsite: boolean): void {\n        const readConfigError = new ReadConfigFileError(code, error, path, renderCallsite);\n\n        DEBUG_LOGGER(readConfigError.message);\n        DEBUG_LOGGER(error);\n\n        throw readConfigError;\n    }\n\n    private static _resolveFilePath (path: string | null): string | null {\n        if (!path)\n            return null;\n\n        return isAbsolute(path) ? path : resolvePathRelativelyCwd(path);\n    }\n\n    private _resolveFilePaths (filesNames: string | null | string[]): string[] | undefined {\n        if (!filesNames)\n            return void 0;\n\n        return castArray(filesNames).reduce((result, name) => {\n            const resolveFilePath = Configuration._resolveFilePath(name);\n\n            if (resolveFilePath)\n                result.push(resolveFilePath);\n\n            return result;\n        }, [] as string[]);\n    }\n\n    public async init (): Promise<void> {\n        this._overriddenOptions = [];\n    }\n\n    public mergeOptions (options: object): void {\n        Object.entries(options).map(([key, value]) => {\n            const option = this._ensureOption(key, value, OptionSource.Input);\n\n            if (value === void 0)\n                return;\n\n            this._setOptionValue(option, value);\n        });\n    }\n\n    protected mergeDeep (option: Option, source: object): void {\n        mergeWith(option.value, source, (targetValue: OptionValue, sourceValue: OptionValue, property: string) => {\n            this._addOverriddenOptionIfNecessary(targetValue, sourceValue, option.source, `${option.name}.${property}`);\n\n            return sourceValue !== void 0 ? sourceValue : targetValue;\n        });\n    }\n\n    protected _getOption (key: string): OptionValue {\n        if (!key)\n            return void 0;\n\n        const option = this._options[key];\n\n        if (!option)\n            return void 0;\n\n        return option.value;\n    }\n\n    public getOption<Type> (key: string): Type {\n        return this._getOption(key) as Type;\n    }\n\n    public getOptions (predicate?: (name: string, option: Option) => boolean): Dictionary<OptionValue> {\n        const result        = Object.create(null);\n        let includeInResult = true;\n\n        Object.entries(this._options).forEach(([name, option]) => {\n            includeInResult = predicate ? predicate(name, option) : true;\n\n            if (includeInResult)\n                result[name] = option.value;\n        });\n\n        return result;\n    }\n\n    public clone (nonClonedOptions?: string | string[]): Configuration {\n        const configuration = cloneDeep(this);\n\n        if (nonClonedOptions) {\n            castArray(nonClonedOptions).forEach(key => {\n                if (configuration._options[key])\n                    configuration._options[key].value = this._options[key].value;\n            });\n        }\n\n        return configuration;\n    }\n\n    public get filePath (): string | undefined {\n        return this._filePath;\n    }\n\n    public get defaultPaths (): string[] | undefined {\n        return this._defaultPaths;\n    }\n\n    public async _load (): Promise<null | object> {\n        if (!this.defaultPaths?.length)\n            return null;\n\n        const configs = await Promise.all(this.defaultPaths.map(async filePath => {\n            if (!await this._isConfigurationFileExists(filePath))\n                return { filePath, options: null };\n\n            let options = null as object | null;\n\n            if (this._isJSConfiguration(filePath))\n                options = this._readJsConfigurationFileContent(filePath);\n            else {\n                const configurationFileContent = await this._readConfigurationFileContent(filePath);\n\n                if (configurationFileContent)\n                    options = this._parseConfigurationFileContent(configurationFileContent, filePath);\n            }\n\n            return { filePath, options };\n        }));\n\n        const existedConfigs = configs.filter(config => !!config.options);\n\n        if (!existedConfigs.length)\n            return null;\n\n        this._filePath = existedConfigs[0].filePath;\n\n        if (existedConfigs.length > 1)\n            Configuration._showConsoleWarning(renderTemplate(WARNING_MESSAGES.multipleConfigurationFilesFound, this._filePath));\n\n        return existedConfigs[0].options;\n    }\n\n    protected async _isConfigurationFileExists (filePath = this.filePath): Promise<boolean> {\n        try {\n            await stat(filePath);\n\n            return true;\n        }\n        catch (error: any) {\n            DEBUG_LOGGER(renderTemplate(WARNING_MESSAGES.cannotFindConfigurationFile, filePath, error.stack));\n\n            return false;\n        }\n    }\n\n    private static _hasExtension (filePath: string | undefined, extention: string): boolean {\n        return !!filePath && extname(filePath) === extention;\n    }\n\n    protected _isJSConfiguration (filePath = this.filePath): boolean {\n        return Configuration._hasExtension(filePath, Extensions.js) || Configuration._hasExtension(filePath, Extensions.cjs);\n    }\n\n    protected _isJSONConfiguration (filePath = this.filePath): boolean {\n        return Configuration._hasExtension(filePath, Extensions.json);\n    }\n\n    public _readJsConfigurationFileContent (filePath = this.filePath): object | null {\n        if (filePath) {\n            try {\n                delete require.cache[filePath];\n\n                return require(filePath);\n            }\n            catch (error: any) {\n                Configuration._throwReadConfigError(RUNTIME_ERRORS.cannotReadConfigFile, error, filePath, true);\n            }\n        }\n\n        return null;\n    }\n\n    public async _readConfigurationFileContent (filePath = this.filePath): Promise<Buffer | null> {\n        try {\n            return await readFile(filePath);\n        }\n        catch (error: any) {\n            Configuration._throwReadConfigError(RUNTIME_ERRORS.cannotReadConfigFile, error, filePath || '', false);\n        }\n\n        return null;\n    }\n\n    private _parseConfigurationFileContent (configurationFileContent: Buffer, filePath = this.filePath): object | null {\n        try {\n            return JSON5.parse(configurationFileContent.toString());\n        }\n        catch (error: any) {\n            Configuration._throwReadConfigError(RUNTIME_ERRORS.cannotParseConfigFile, error, filePath || '', false);\n        }\n\n        return null;\n    }\n\n    protected _ensureArrayOption (name: string): void {\n        const options = this._options[name];\n\n        if (!options)\n            return;\n\n        // NOTE: a hack to fix lodash type definitions\n        // @ts-ignore\n        options.value = castArray(options.value);\n    }\n\n    protected _ensureOption (name: string, value: OptionValue, source: OptionSource): Option {\n        let option = null;\n\n        if (name in this._options)\n            option = this._options[name];\n        else {\n            option = new Option(name, value, source);\n\n            this._options[name] = option;\n        }\n\n        return option;\n    }\n\n    protected _ensureOptionWithValue (name: string, defaultValue: OptionValue, source: OptionSource): void {\n        const option = this._ensureOption(name, defaultValue, source);\n\n        if (option.value !== void 0)\n            return;\n\n        option.value  = defaultValue;\n        option.source = source;\n    }\n\n    protected _addOverriddenOptionIfNecessary (value1: OptionValue, value2: OptionValue, source: OptionSource, optionName: string): void {\n        if (source === OptionSource.Default)\n            return;\n\n        if (value1 === void 0 || value2 === void 0 || value1 === value2 || source !== OptionSource.Configuration)\n            return;\n\n        this._overriddenOptions.push(optionName);\n    }\n\n    protected _setOptionValue (option: Option, value: OptionValue): void {\n        if (isPlainObject(option.value) && isPlainObject(value))\n            this.mergeDeep(option, value as object);\n        else {\n            this._addOverriddenOptionIfNecessary(option.value, value, option.source, option.name);\n\n            option.value = value;\n        }\n\n        option.source = OptionSource.Input;\n    }\n}\n"]}