1 | ;
|
2 | var __importDefault = (this && this.__importDefault) || function (mod) {
|
3 | return (mod && mod.__esModule) ? mod : { "default": mod };
|
4 | };
|
5 | Object.defineProperty(exports, "__esModule", { value: true });
|
6 | const path_1 = require("path");
|
7 | const debug_1 = __importDefault(require("debug"));
|
8 | const json5_1 = __importDefault(require("json5"));
|
9 | const lodash_1 = require("lodash");
|
10 | const promisified_functions_1 = require("../utils/promisified-functions");
|
11 | const option_1 = __importDefault(require("./option"));
|
12 | const option_source_1 = __importDefault(require("./option-source"));
|
13 | const resolve_path_relatively_cwd_1 = __importDefault(require("../utils/resolve-path-relatively-cwd"));
|
14 | const render_template_1 = __importDefault(require("../utils/render-template"));
|
15 | const warning_message_1 = __importDefault(require("../notifications/warning-message"));
|
16 | const log_1 = __importDefault(require("../cli/log"));
|
17 | const DEBUG_LOGGER = debug_1.default('testcafe:configuration');
|
18 | class Configuration {
|
19 | constructor(configurationFileName) {
|
20 | this._options = {};
|
21 | this._filePath = Configuration._resolveFilePath(configurationFileName);
|
22 | this._overriddenOptions = [];
|
23 | }
|
24 | static _fromObj(obj) {
|
25 | const result = Object.create(null);
|
26 | Object.entries(obj).forEach(([key, value]) => {
|
27 | const option = new option_1.default(key, value);
|
28 | result[key] = option;
|
29 | });
|
30 | return result;
|
31 | }
|
32 | static _showConsoleWarning(message) {
|
33 | log_1.default.write(message);
|
34 | }
|
35 | static _showWarningForError(error, warningTemplate, ...args) {
|
36 | const message = render_template_1.default(warningTemplate, ...args);
|
37 | Configuration._showConsoleWarning(message);
|
38 | DEBUG_LOGGER(message);
|
39 | DEBUG_LOGGER(error);
|
40 | }
|
41 | static _resolveFilePath(path) {
|
42 | if (!path)
|
43 | return null;
|
44 | return path_1.isAbsolute(path) ? path : resolve_path_relatively_cwd_1.default(path);
|
45 | }
|
46 | async init() {
|
47 | this._overriddenOptions = [];
|
48 | }
|
49 | mergeOptions(options) {
|
50 | Object.entries(options).map(([key, value]) => {
|
51 | const option = this._ensureOption(key, value, option_source_1.default.Input);
|
52 | if (value === void 0)
|
53 | return;
|
54 | this._setOptionValue(option, value);
|
55 | });
|
56 | }
|
57 | mergeDeep(option, source) {
|
58 | lodash_1.mergeWith(option.value, source, (targetValue, sourceValue, property) => {
|
59 | this._addOverriddenOptionIfNecessary(targetValue, sourceValue, option.source, `${option.name}.${property}`);
|
60 | return sourceValue !== void 0 ? sourceValue : targetValue;
|
61 | });
|
62 | }
|
63 | getOption(key) {
|
64 | if (!key)
|
65 | return void 0;
|
66 | const option = this._options[key];
|
67 | if (!option)
|
68 | return void 0;
|
69 | return option.value;
|
70 | }
|
71 | getOptions() {
|
72 | const result = Object.create(null);
|
73 | Object.entries(this._options).forEach(([name, option]) => {
|
74 | result[name] = option.value;
|
75 | });
|
76 | return result;
|
77 | }
|
78 | clone() {
|
79 | return lodash_1.cloneDeep(this);
|
80 | }
|
81 | get filePath() {
|
82 | return this._filePath;
|
83 | }
|
84 | async _load() {
|
85 | if (!this.filePath)
|
86 | return null;
|
87 | if (!await this._isConfigurationFileExists())
|
88 | return null;
|
89 | const configurationFileContent = await this._readConfigurationFileContent();
|
90 | if (!configurationFileContent)
|
91 | return null;
|
92 | return this._parseConfigurationFileContent(configurationFileContent);
|
93 | }
|
94 | async _isConfigurationFileExists() {
|
95 | try {
|
96 | await promisified_functions_1.stat(this.filePath);
|
97 | return true;
|
98 | }
|
99 | catch (error) {
|
100 | DEBUG_LOGGER(render_template_1.default(warning_message_1.default.cannotFindConfigurationFile, this.filePath, error.stack));
|
101 | return false;
|
102 | }
|
103 | }
|
104 | async _readConfigurationFileContent() {
|
105 | try {
|
106 | return await promisified_functions_1.readFile(this.filePath);
|
107 | }
|
108 | catch (error) {
|
109 | Configuration._showWarningForError(error, warning_message_1.default.cannotReadConfigFile);
|
110 | }
|
111 | return null;
|
112 | }
|
113 | _parseConfigurationFileContent(configurationFileContent) {
|
114 | try {
|
115 | return json5_1.default.parse(configurationFileContent.toString());
|
116 | }
|
117 | catch (error) {
|
118 | Configuration._showWarningForError(error, warning_message_1.default.cannotParseConfigFile, this._filePath);
|
119 | }
|
120 | return null;
|
121 | }
|
122 | _ensureArrayOption(name) {
|
123 | const options = this._options[name];
|
124 | if (!options)
|
125 | return;
|
126 | // NOTE: a hack to fix lodash type definitions
|
127 | // @ts-ignore
|
128 | options.value = lodash_1.castArray(options.value);
|
129 | }
|
130 | _ensureOption(name, value, source) {
|
131 | let option = null;
|
132 | if (name in this._options)
|
133 | option = this._options[name];
|
134 | else {
|
135 | option = new option_1.default(name, value, source);
|
136 | this._options[name] = option;
|
137 | }
|
138 | return option;
|
139 | }
|
140 | _ensureOptionWithValue(name, defaultValue, source) {
|
141 | const option = this._ensureOption(name, defaultValue, source);
|
142 | if (option.value !== void 0)
|
143 | return;
|
144 | option.value = defaultValue;
|
145 | option.source = source;
|
146 | }
|
147 | _addOverriddenOptionIfNecessary(value1, value2, source, optionName) {
|
148 | if (value1 === void 0 || value2 === void 0 || value1 === value2 || source !== option_source_1.default.Configuration)
|
149 | return;
|
150 | this._overriddenOptions.push(optionName);
|
151 | }
|
152 | _setOptionValue(option, value) {
|
153 | if (lodash_1.isPlainObject(option.value) && lodash_1.isPlainObject(value))
|
154 | this.mergeDeep(option, value);
|
155 | else {
|
156 | this._addOverriddenOptionIfNecessary(option.value, value, option.source, option.name);
|
157 | option.value = value;
|
158 | }
|
159 | option.source = option_source_1.default.Input;
|
160 | }
|
161 | }
|
162 | exports.default = Configuration;
|
163 | module.exports = exports.default;
|
164 | //# 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 { isAbsolute } from 'path';\nimport debug from 'debug';\nimport JSON5 from 'json5';\nimport { castArray, cloneDeep, isPlainObject, mergeWith } from 'lodash';\nimport { stat, readFile } 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';\n\nconst DEBUG_LOGGER = debug('testcafe:configuration');\n\nexport default class Configuration {\n    protected _options: Dictionary<Option>;\n    protected readonly _filePath: string | null;\n    protected _overriddenOptions: string[];\n\n    public constructor (configurationFileName: string | null) {\n        this._options  = {};\n        this._filePath = Configuration._resolveFilePath(configurationFileName);\n\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            const option = new Option(key, value);\n\n            result[key] = option;\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    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 (): Dictionary<OptionValue> {\n        const result = Object.create(null);\n\n        Object.entries(this._options).forEach(([name, option]) => {\n            result[name] = option.value;\n        });\n\n        return result;\n    }\n\n    public clone (): Configuration {\n        return cloneDeep(this);\n    }\n\n    public get filePath (): string | null {\n        return this._filePath;\n    }\n\n    public async _load (): Promise<null | object> {\n        if (!this.filePath)\n            return null;\n\n        if (!await this._isConfigurationFileExists())\n            return null;\n\n        const configurationFileContent = await this._readConfigurationFileContent();\n\n        if (!configurationFileContent)\n            return null;\n\n        return this._parseConfigurationFileContent(configurationFileContent);\n    }\n\n    protected async _isConfigurationFileExists (): Promise<boolean> {\n        try {\n            await stat(this.filePath);\n\n            return true;\n        }\n        catch (error) {\n            DEBUG_LOGGER(renderTemplate(WARNING_MESSAGES.cannotFindConfigurationFile, this.filePath, error.stack));\n\n            return false;\n        }\n    }\n\n    public async _readConfigurationFileContent (): Promise<Buffer | null> {\n        try {\n            return await readFile(this.filePath);\n        }\n        catch (error) {\n            Configuration._showWarningForError(error, WARNING_MESSAGES.cannotReadConfigFile);\n        }\n\n        return null;\n    }\n\n    private _parseConfigurationFileContent (configurationFileContent: Buffer): object | null {\n        try {\n            return JSON5.parse(configurationFileContent.toString());\n        }\n        catch (error) {\n            Configuration._showWarningForError(error, WARNING_MESSAGES.cannotParseConfigFile, this._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 (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"]} |
\ | No newline at end of file |