"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 = require("./formats"); 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 _showWarningForError(error, warningTemplate, ...args) { const message = (0, render_template_1.default)(warningTemplate, ...args); Configuration._showConsoleWarning(message); DEBUG_LOGGER(message); DEBUG_LOGGER(error); } 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; } 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 _hasExtention(filePath, extention) { return !!filePath && (0, path_1.extname)(filePath) === extention; } _isJSConfiguration(filePath = this.filePath) { return Configuration._hasExtention(filePath, formats_1.JS_CONFIGURATION_EXTENSION); } _isJSONConfiguration(filePath = this.filePath) { return Configuration._hasExtention(filePath, formats_1.JSON_CONFIGURATION_EXTENSION); } _readJsConfigurationFileContent(filePath = this.filePath) { if (filePath) { try { delete require.cache[filePath]; return require(filePath); } catch (error) { Configuration._showWarningForError(error, warning_message_1.default.cannotReadConfigFile, filePath); } } return null; } async _readConfigurationFileContent(filePath = this.filePath) { try { return await (0, promisified_functions_1.readFile)(filePath); } catch (error) { Configuration._showWarningForError(error, warning_message_1.default.cannotReadConfigFile, filePath); } return null; } _parseConfigurationFileContent(configurationFileContent, filePath = this.filePath) { try { return json5_1.default.parse(configurationFileContent.toString()); } catch (error) { Configuration._showWarningForError(error, warning_message_1.default.cannotParseConfigFile, filePath); } 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 { JS_CONFIGURATION_EXTENSION, JSON_CONFIGURATION_EXTENSION } from './formats';\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 _showWarningForError (error: Error, warningTemplate: string, ...args: TemplateArguments): void {\n        const message = renderTemplate(warningTemplate, ...args);\n\n        Configuration._showConsoleWarning(message);\n\n        DEBUG_LOGGER(message);\n        DEBUG_LOGGER(error);\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    public 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 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 _hasExtention (filePath: string | undefined, extention: string): boolean {\n        return !!filePath && extname(filePath) === extention;\n    }\n\n    protected _isJSConfiguration (filePath = this.filePath): boolean {\n        return Configuration._hasExtention(filePath, JS_CONFIGURATION_EXTENSION);\n    }\n\n    protected _isJSONConfiguration (filePath = this.filePath): boolean {\n        return Configuration._hasExtention(filePath, JSON_CONFIGURATION_EXTENSION);\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._showWarningForError(error, WARNING_MESSAGES.cannotReadConfigFile, filePath);\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._showWarningForError(error, WARNING_MESSAGES.cannotReadConfigFile, filePath);\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._showWarningForError(error, WARNING_MESSAGES.cannotParseConfigFile, filePath);\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"]}