@rosen-bridge/config 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.eslintignore +1 -0
- package/CHANGELOG.md +8 -0
- package/README.md +24 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +102 -0
- package/dist/config.d.ts +131 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +578 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +2 -0
- package/dist/schema/Validators/fieldProperties.d.ts +29 -0
- package/dist/schema/Validators/fieldProperties.d.ts.map +1 -0
- package/dist/schema/Validators/fieldProperties.js +254 -0
- package/dist/schema/types/fields.d.ts +37 -0
- package/dist/schema/types/fields.d.ts.map +1 -0
- package/dist/schema/types/fields.js +2 -0
- package/dist/schema/types/validations.d.ts +46 -0
- package/dist/schema/types/validations.d.ts.map +1 -0
- package/dist/schema/types/validations.js +2 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/dist/utils.d.ts +36 -0
- package/dist/utils.d.ts.map +1 -0
- package/dist/utils.js +59 -0
- package/dist/value/validators.d.ts +3 -0
- package/dist/value/validators.d.ts.map +1 -0
- package/dist/value/validators.js +188 -0
- package/lib/cli.ts +113 -0
- package/lib/config.ts +664 -0
- package/lib/index.ts +1 -0
- package/lib/schema/Validators/fieldProperties.ts +273 -0
- package/lib/schema/types/fields.ts +46 -0
- package/lib/schema/types/validations.ts +63 -0
- package/lib/utils.ts +68 -0
- package/lib/value/validators.ts +268 -0
- package/package.json +48 -0
- package/tests/.gitkeep +0 -0
- package/tests/config.spec.ts +895 -0
- package/tests/configEnvSetup.ts +34 -0
- package/tests/configTestData.ts +977 -0
- package/tests/configTestFiles/custom-environment-variables.json +5 -0
- package/tests/configTestFiles/default.json +12 -0
- package/tests/configTestFiles/local.json +5 -0
- package/tests/utils.spec.ts +117 -0
- package/tests/utilsTestData.ts +26 -0
- package/tsconfig.build.json +7 -0
- package/tsconfig.build.tsbuildinfo +1 -0
- package/tsconfig.json +7 -0
- package/vitest.config.ts +11 -0
package/dist/config.js
ADDED
|
@@ -0,0 +1,578 @@
|
|
|
1
|
+
import * as fs from 'fs';
|
|
2
|
+
import * as yaml from 'js-yaml';
|
|
3
|
+
import JsonBigIntFactory from 'json-bigint';
|
|
4
|
+
import path from 'path';
|
|
5
|
+
import {
|
|
6
|
+
propertyValidators,
|
|
7
|
+
supportedTypes,
|
|
8
|
+
} from './schema/Validators/fieldProperties';
|
|
9
|
+
import { getSourceName, getValueFromConfigSources } from './utils';
|
|
10
|
+
import { valueValidations, valueValidators } from './value/validators';
|
|
11
|
+
export class ConfigValidator {
|
|
12
|
+
schema;
|
|
13
|
+
constructor(schema) {
|
|
14
|
+
this.schema = schema;
|
|
15
|
+
this.validateSchema();
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* validates the passed config against the instance's schema
|
|
19
|
+
*
|
|
20
|
+
* @param {Record<string, any>} config
|
|
21
|
+
*/
|
|
22
|
+
validateConfig(config) {
|
|
23
|
+
const errorPreamble = (path) =>
|
|
24
|
+
`config validation failed for "${path.join('.')}" field`;
|
|
25
|
+
const stack = [
|
|
26
|
+
{
|
|
27
|
+
subSchema: this.schema,
|
|
28
|
+
subConfig: config,
|
|
29
|
+
parentPath: [],
|
|
30
|
+
},
|
|
31
|
+
];
|
|
32
|
+
this.validateValue(
|
|
33
|
+
config,
|
|
34
|
+
{ type: 'object', children: this.schema },
|
|
35
|
+
config
|
|
36
|
+
);
|
|
37
|
+
// Traverses the schema object tree depth first in coordination with config
|
|
38
|
+
// object tree and validate config using the schema
|
|
39
|
+
while (stack.length > 0) {
|
|
40
|
+
const { subSchema, subConfig, parentPath } = stack.pop();
|
|
41
|
+
// Process children of current field
|
|
42
|
+
for (const name of Object.keys(subSchema)) {
|
|
43
|
+
const path = parentPath.concat([name]);
|
|
44
|
+
try {
|
|
45
|
+
const field = subSchema[name];
|
|
46
|
+
let value = undefined;
|
|
47
|
+
if (subConfig != undefined && Object.hasOwn(subConfig, name)) {
|
|
48
|
+
value = subConfig[name];
|
|
49
|
+
}
|
|
50
|
+
this.validateValue(value, field, config);
|
|
51
|
+
// if a node/field is of type object and thus is a subtree, add it to
|
|
52
|
+
// the stack to be traversed later
|
|
53
|
+
if (field.type === 'object') {
|
|
54
|
+
stack.push({
|
|
55
|
+
subSchema: field.children,
|
|
56
|
+
subConfig: value,
|
|
57
|
+
parentPath: path,
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
} catch (error) {
|
|
61
|
+
throw new Error(`${errorPreamble(path)}: ${error.message}`);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* validates a value in config object
|
|
68
|
+
*
|
|
69
|
+
* @private
|
|
70
|
+
* @param {*} value
|
|
71
|
+
* @param {ConfigField} field the field specification in schema
|
|
72
|
+
* @param {Record<string, any>} config the config object
|
|
73
|
+
*/
|
|
74
|
+
validateValue = (value, field, config) => {
|
|
75
|
+
if (value != undefined) {
|
|
76
|
+
if (field.type === 'bigint') {
|
|
77
|
+
value = BigInt(value);
|
|
78
|
+
} else if (field.type === 'number' && value && !isNaN(value)) {
|
|
79
|
+
value = Number(value);
|
|
80
|
+
}
|
|
81
|
+
valueValidators[field.type](value, field);
|
|
82
|
+
}
|
|
83
|
+
if (field.type != 'object' && field.validations) {
|
|
84
|
+
for (const validation of field.validations) {
|
|
85
|
+
const name = Object.keys(validation).filter(
|
|
86
|
+
(key) => key !== 'when' && key !== 'error'
|
|
87
|
+
)[0];
|
|
88
|
+
if (Object.hasOwn(valueValidations[field.type], name)) {
|
|
89
|
+
try {
|
|
90
|
+
valueValidations[field.type][name](value, validation, config, this);
|
|
91
|
+
} catch (error) {
|
|
92
|
+
if (validation.error != undefined) {
|
|
93
|
+
throw new Error(validation.error);
|
|
94
|
+
}
|
|
95
|
+
throw error;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
};
|
|
101
|
+
/**
|
|
102
|
+
* determines if a when clause in validations section of a schema field is
|
|
103
|
+
* satisfied
|
|
104
|
+
*
|
|
105
|
+
* @param {When} when
|
|
106
|
+
* @param {Record<string, any>} config
|
|
107
|
+
* @return {boolean}
|
|
108
|
+
*/
|
|
109
|
+
isWhenTrue = (when, config) => {
|
|
110
|
+
const pathParts = when.path.split('.');
|
|
111
|
+
const value = ConfigValidator.valueAt(config, pathParts);
|
|
112
|
+
return value != undefined && value === when.value;
|
|
113
|
+
};
|
|
114
|
+
/**
|
|
115
|
+
* returns the value at specified path in config object
|
|
116
|
+
*
|
|
117
|
+
* @static
|
|
118
|
+
* @param {Record<string, any>} config
|
|
119
|
+
* @param {string[]} path
|
|
120
|
+
* @return {*}
|
|
121
|
+
*/
|
|
122
|
+
static valueAt = (config, path) => {
|
|
123
|
+
let value = config;
|
|
124
|
+
for (const key of path) {
|
|
125
|
+
if (value != undefined && Object.hasOwn(value, key)) {
|
|
126
|
+
value = value[key];
|
|
127
|
+
} else {
|
|
128
|
+
return undefined;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
return value;
|
|
132
|
+
};
|
|
133
|
+
/**
|
|
134
|
+
* validates this.schema and throws exception if any errors found
|
|
135
|
+
*/
|
|
136
|
+
validateSchema = () => {
|
|
137
|
+
const errorPreamble = (path) =>
|
|
138
|
+
`Schema validation failed for "${path.join('.')}" field`;
|
|
139
|
+
const stack = [
|
|
140
|
+
{
|
|
141
|
+
subSchema: this.schema,
|
|
142
|
+
parentPath: [],
|
|
143
|
+
},
|
|
144
|
+
];
|
|
145
|
+
// Traverses the schema object tree depth first and validate fields
|
|
146
|
+
while (stack.length > 0) {
|
|
147
|
+
const { subSchema, parentPath } = stack.pop();
|
|
148
|
+
// process children of current object field
|
|
149
|
+
for (const name of Object.keys(subSchema).reverse()) {
|
|
150
|
+
const path = parentPath.concat([name]);
|
|
151
|
+
try {
|
|
152
|
+
this.validateConfigName(name);
|
|
153
|
+
const field = subSchema[name];
|
|
154
|
+
if (!Object.hasOwn(field, 'type') || typeof field.type !== 'string') {
|
|
155
|
+
throw new Error(
|
|
156
|
+
`every schema field must have a "type" property of type "string"`
|
|
157
|
+
);
|
|
158
|
+
}
|
|
159
|
+
if (!supportedTypes.includes(field.type)) {
|
|
160
|
+
throw new Error(`unsupported field type "${field.type}"`);
|
|
161
|
+
}
|
|
162
|
+
this.validateSchemaField(field);
|
|
163
|
+
// if the child is an object field itself add it to stack for
|
|
164
|
+
// processing
|
|
165
|
+
if (field.type === 'object') {
|
|
166
|
+
stack.push({
|
|
167
|
+
subSchema: field.children,
|
|
168
|
+
parentPath: path,
|
|
169
|
+
});
|
|
170
|
+
}
|
|
171
|
+
} catch (error) {
|
|
172
|
+
throw new Error(`${errorPreamble(path)}: ${error.message}`);
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
};
|
|
177
|
+
/**
|
|
178
|
+
* validates config key name
|
|
179
|
+
*
|
|
180
|
+
* @param {string} name
|
|
181
|
+
*/
|
|
182
|
+
validateConfigName = (name) => {
|
|
183
|
+
if (name.includes('.')) {
|
|
184
|
+
throw new Error(`config key name can not contain the '.' character`);
|
|
185
|
+
}
|
|
186
|
+
};
|
|
187
|
+
/**
|
|
188
|
+
* validates passed schema field structure
|
|
189
|
+
*
|
|
190
|
+
* @param {ConfigField} field
|
|
191
|
+
*/
|
|
192
|
+
validateSchemaField = (field) => {
|
|
193
|
+
for (const key of Object.keys(field)) {
|
|
194
|
+
if (
|
|
195
|
+
!Object.hasOwn(propertyValidators.all, key) &&
|
|
196
|
+
!(
|
|
197
|
+
field.type !== 'object' &&
|
|
198
|
+
Object.hasOwn(propertyValidators.primitive, key)
|
|
199
|
+
) &&
|
|
200
|
+
!Object.hasOwn(propertyValidators[field.type], key)
|
|
201
|
+
) {
|
|
202
|
+
throw new Error(`schema field has unknown property "${key}"`);
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
for (const validator of Object.values(propertyValidators.all)) {
|
|
206
|
+
validator(field, this);
|
|
207
|
+
}
|
|
208
|
+
if (field.type !== 'object') {
|
|
209
|
+
for (const validator of Object.values(propertyValidators.primitive)) {
|
|
210
|
+
validator(field, this);
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
for (const validator of Object.values(propertyValidators[field.type])) {
|
|
214
|
+
validator(field, this);
|
|
215
|
+
}
|
|
216
|
+
};
|
|
217
|
+
/**
|
|
218
|
+
* returns a field corresponding to a path in schema tree
|
|
219
|
+
*
|
|
220
|
+
* @param {string[]} path
|
|
221
|
+
* @return {(ConfigField | undefined)} returns undefined if field is not found
|
|
222
|
+
*/
|
|
223
|
+
getSchemaField = (path) => {
|
|
224
|
+
let subTree = this.schema;
|
|
225
|
+
let field = undefined;
|
|
226
|
+
for (const part of path) {
|
|
227
|
+
if (subTree != undefined && Object.hasOwn(subTree, part)) {
|
|
228
|
+
field = subTree[part];
|
|
229
|
+
subTree = 'children' in field ? field.children : undefined;
|
|
230
|
+
} else {
|
|
231
|
+
return undefined;
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
return field;
|
|
235
|
+
};
|
|
236
|
+
/**
|
|
237
|
+
* extracts default values from a schema
|
|
238
|
+
*
|
|
239
|
+
* @return {Record<string, any>} object of default values
|
|
240
|
+
*/
|
|
241
|
+
generateDefault = () => {
|
|
242
|
+
const valueTree = Object.create(null);
|
|
243
|
+
const stack = [
|
|
244
|
+
{
|
|
245
|
+
schema: this.schema,
|
|
246
|
+
parentValue: undefined,
|
|
247
|
+
fieldName: '',
|
|
248
|
+
children: Object.keys(this.schema).reverse(),
|
|
249
|
+
},
|
|
250
|
+
];
|
|
251
|
+
// Traverses the schema object tree depth first
|
|
252
|
+
while (stack.length > 0) {
|
|
253
|
+
const { schema, parentValue, fieldName, children } = stack.at(-1);
|
|
254
|
+
// if a subtree's processing is finished go to the previous level
|
|
255
|
+
if (children.length === 0) {
|
|
256
|
+
// if a subtree is empty (has no values) remove it from the result
|
|
257
|
+
if (
|
|
258
|
+
parentValue != undefined &&
|
|
259
|
+
Object.keys(parentValue[fieldName]).length === 0
|
|
260
|
+
) {
|
|
261
|
+
delete parentValue[fieldName];
|
|
262
|
+
}
|
|
263
|
+
stack.pop();
|
|
264
|
+
continue;
|
|
265
|
+
}
|
|
266
|
+
const childName = children.pop();
|
|
267
|
+
const value =
|
|
268
|
+
parentValue != undefined ? parentValue[fieldName] : valueTree;
|
|
269
|
+
const field = schema[childName];
|
|
270
|
+
// if a node/field is of type object and thus is a subtree, add it both to
|
|
271
|
+
// value tree and to the stack to be traversed later. Otherwise it's a
|
|
272
|
+
// leaf and needs no traversal, so add it only to the value tree.
|
|
273
|
+
if (field.type === 'object') {
|
|
274
|
+
value[childName] = Object.create(null);
|
|
275
|
+
stack.push({
|
|
276
|
+
schema: field.children,
|
|
277
|
+
parentValue: value,
|
|
278
|
+
fieldName: childName,
|
|
279
|
+
children: Object.keys(field.children).reverse(),
|
|
280
|
+
});
|
|
281
|
+
} else if (field.default != undefined) {
|
|
282
|
+
value[childName] = field.default;
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
return valueTree;
|
|
286
|
+
};
|
|
287
|
+
/**
|
|
288
|
+
* generates compatible TypeScript interface for this instance's schema
|
|
289
|
+
*
|
|
290
|
+
* @param {string} name the name of root type
|
|
291
|
+
* @return {string}
|
|
292
|
+
*/
|
|
293
|
+
generateTSTypes = (name) => {
|
|
294
|
+
const errorPreamble = (path) =>
|
|
295
|
+
`TypeScript type generation failed for "${path.join('.')}" field`;
|
|
296
|
+
const types = [];
|
|
297
|
+
const typeNames = new Map();
|
|
298
|
+
typeNames.set(name, 1n);
|
|
299
|
+
const stack = [
|
|
300
|
+
{
|
|
301
|
+
subSchema: this.schema,
|
|
302
|
+
children: Object.keys(this.schema),
|
|
303
|
+
parentPath: [],
|
|
304
|
+
typeName: name,
|
|
305
|
+
attributes: [],
|
|
306
|
+
},
|
|
307
|
+
];
|
|
308
|
+
// Traverses the schema object tree depth first
|
|
309
|
+
while (stack.length > 0) {
|
|
310
|
+
const { subSchema, children, parentPath, typeName, attributes } =
|
|
311
|
+
stack.at(-1);
|
|
312
|
+
const path = parentPath.concat([name]);
|
|
313
|
+
try {
|
|
314
|
+
// if a subtree's processing is finished go to the previous level
|
|
315
|
+
if (children.length == 0) {
|
|
316
|
+
types.push(this.genTSInterface(typeName, attributes));
|
|
317
|
+
stack.pop();
|
|
318
|
+
continue;
|
|
319
|
+
}
|
|
320
|
+
const childName = children.pop();
|
|
321
|
+
const field = subSchema[childName];
|
|
322
|
+
// if a node/field is of type object and thus is a subtree, add it to
|
|
323
|
+
// the stack to be traversed later. Otherwise it's a leaf and needs no
|
|
324
|
+
// traversal.
|
|
325
|
+
if (field.type === 'object') {
|
|
326
|
+
let childTypeName = `${childName[0].toUpperCase()}${childName.substring(
|
|
327
|
+
1
|
|
328
|
+
)}`;
|
|
329
|
+
const typeNameCount = typeNames.get(childTypeName);
|
|
330
|
+
typeNames.set(childTypeName, (typeNames.get(childName) || 0n) + 1n);
|
|
331
|
+
if (typeNameCount) {
|
|
332
|
+
childTypeName += typeNameCount.toString();
|
|
333
|
+
}
|
|
334
|
+
stack.push({
|
|
335
|
+
subSchema: field.children,
|
|
336
|
+
children: Object.keys(field.children).reverse(),
|
|
337
|
+
parentPath: path,
|
|
338
|
+
typeName: childTypeName,
|
|
339
|
+
attributes: [],
|
|
340
|
+
});
|
|
341
|
+
attributes.push([childName, childTypeName]);
|
|
342
|
+
} else {
|
|
343
|
+
attributes.push([childName, field.type]);
|
|
344
|
+
}
|
|
345
|
+
} catch (error) {
|
|
346
|
+
throw new Error(`${errorPreamble(path)}: ${error.message}`);
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
return types.reverse().join('\n\n') + '\n';
|
|
350
|
+
};
|
|
351
|
+
/**
|
|
352
|
+
* generates a TypeScript interface definition for passed name and attributes
|
|
353
|
+
*
|
|
354
|
+
* @param {string} name
|
|
355
|
+
* @param {Array<[string, string]>} attributes
|
|
356
|
+
* @return {string}
|
|
357
|
+
*/
|
|
358
|
+
genTSInterface = (name, attributes) => {
|
|
359
|
+
return `interface ${name} {
|
|
360
|
+
${attributes.map((attr) => `${attr[0]}: ${attr[1]};`).join('\n ')}
|
|
361
|
+
}`;
|
|
362
|
+
};
|
|
363
|
+
/**
|
|
364
|
+
* returns a characteristic object for values at a specific node config level
|
|
365
|
+
*
|
|
366
|
+
* @param {IConfig} config
|
|
367
|
+
* @param {string} level
|
|
368
|
+
* @return {Record<string, any>}
|
|
369
|
+
*/
|
|
370
|
+
getConfigForLevel(config, level) {
|
|
371
|
+
const confLevels = ConfigValidator.getNodeConfigLevels(config);
|
|
372
|
+
const levelIndex = confLevels.indexOf(level);
|
|
373
|
+
if (levelIndex === -1) {
|
|
374
|
+
throw new Error(
|
|
375
|
+
`The "${level}" level not found in the current system configuration levels`
|
|
376
|
+
);
|
|
377
|
+
}
|
|
378
|
+
const higherLevelSources = config.util
|
|
379
|
+
.getConfigSources()
|
|
380
|
+
.filter(
|
|
381
|
+
(source) => confLevels.indexOf(getSourceName(source)) > levelIndex
|
|
382
|
+
);
|
|
383
|
+
const currentLevelSource = config.util
|
|
384
|
+
.getConfigSources()
|
|
385
|
+
.filter((source) => getSourceName(source) === level)
|
|
386
|
+
.at(0);
|
|
387
|
+
const lowerLevelSources = config.util
|
|
388
|
+
.getConfigSources()
|
|
389
|
+
.filter(
|
|
390
|
+
(source) => confLevels.indexOf(getSourceName(source)) < levelIndex
|
|
391
|
+
);
|
|
392
|
+
// Traverses the schema object tree depth first
|
|
393
|
+
const valueTree = ConfigValidator.processConfigForLevelNode(
|
|
394
|
+
this.schema,
|
|
395
|
+
[],
|
|
396
|
+
higherLevelSources,
|
|
397
|
+
currentLevelSource,
|
|
398
|
+
lowerLevelSources
|
|
399
|
+
);
|
|
400
|
+
return valueTree;
|
|
401
|
+
}
|
|
402
|
+
/**
|
|
403
|
+
*traverses the config schema depth first to produce characteristic object
|
|
404
|
+
*
|
|
405
|
+
* @private
|
|
406
|
+
* @static
|
|
407
|
+
* @param {ConfigSchema} schema
|
|
408
|
+
* @param {string[]} path
|
|
409
|
+
* @param {IConfigSource[]} higherLevelSources
|
|
410
|
+
* @param {(IConfigSource | undefined)} currentLevelSource
|
|
411
|
+
* @param {IConfigSource[]} lowerLevelSources
|
|
412
|
+
* @return {Record<string, any>}
|
|
413
|
+
* @memberof ConfigValidator
|
|
414
|
+
*/
|
|
415
|
+
static processConfigForLevelNode(
|
|
416
|
+
schema,
|
|
417
|
+
path,
|
|
418
|
+
higherLevelSources,
|
|
419
|
+
currentLevelSource,
|
|
420
|
+
lowerLevelSources
|
|
421
|
+
) {
|
|
422
|
+
const value = Object.create(null);
|
|
423
|
+
for (const childName of Object.keys(schema).reverse()) {
|
|
424
|
+
const childPath = path.concat([childName]);
|
|
425
|
+
const field = schema[childName];
|
|
426
|
+
// if a field is of type object and thus is a subtree, recurse on it.
|
|
427
|
+
// Otherwise it's a leaf and needs no traversal.
|
|
428
|
+
value[childName] = Object.create(null);
|
|
429
|
+
if (field.type === 'object') {
|
|
430
|
+
value[childName] = ConfigValidator.processConfigForLevelNode(
|
|
431
|
+
field.children,
|
|
432
|
+
childPath,
|
|
433
|
+
higherLevelSources,
|
|
434
|
+
currentLevelSource,
|
|
435
|
+
lowerLevelSources
|
|
436
|
+
);
|
|
437
|
+
} else {
|
|
438
|
+
value[childName]['label'] =
|
|
439
|
+
field.label != undefined ? field.label : null;
|
|
440
|
+
value[childName]['description'] =
|
|
441
|
+
field.description != undefined ? field.description : null;
|
|
442
|
+
value[childName]['default'] = getValueFromConfigSources(
|
|
443
|
+
lowerLevelSources,
|
|
444
|
+
childPath
|
|
445
|
+
);
|
|
446
|
+
value[childName]['value'] = getValueFromConfigSources(
|
|
447
|
+
[...(currentLevelSource != undefined ? [currentLevelSource] : [])],
|
|
448
|
+
childPath
|
|
449
|
+
);
|
|
450
|
+
value[childName]['override'] = getValueFromConfigSources(
|
|
451
|
+
higherLevelSources,
|
|
452
|
+
childPath
|
|
453
|
+
);
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
return value;
|
|
457
|
+
}
|
|
458
|
+
/**
|
|
459
|
+
* returns a list of config sources used by node config package, ordered from
|
|
460
|
+
* the lowest to the highest priority
|
|
461
|
+
*
|
|
462
|
+
* @static
|
|
463
|
+
* @param {IConfig} config
|
|
464
|
+
* @return {string[]}
|
|
465
|
+
*/
|
|
466
|
+
static getNodeConfigLevels = (config) => {
|
|
467
|
+
const instance = config.util.getEnv('NODE_APP_INSTANCE');
|
|
468
|
+
let deployment = config.util.getEnv('NODE_ENV');
|
|
469
|
+
deployment = config.util.getEnv('NODE_CONFIG_ENV');
|
|
470
|
+
const fullHostname = config.util.getEnv('HOSTNAME');
|
|
471
|
+
const shortHostname =
|
|
472
|
+
fullHostname != undefined ? fullHostname.split('.')[0] : undefined;
|
|
473
|
+
const configLevels = [
|
|
474
|
+
'default',
|
|
475
|
+
...(instance != undefined ? [`default-${instance}`] : []),
|
|
476
|
+
...(deployment != undefined ? [`${deployment}`] : []),
|
|
477
|
+
...(instance != undefined && deployment != undefined
|
|
478
|
+
? [`${deployment}-${instance}`]
|
|
479
|
+
: []),
|
|
480
|
+
...(shortHostname != undefined ? [`${shortHostname}`] : []),
|
|
481
|
+
...(shortHostname != undefined && instance != undefined
|
|
482
|
+
? [`${shortHostname}-${instance}`]
|
|
483
|
+
: []),
|
|
484
|
+
...(shortHostname != undefined && deployment != undefined
|
|
485
|
+
? [`${shortHostname}-${deployment}`]
|
|
486
|
+
: []),
|
|
487
|
+
...(shortHostname != undefined &&
|
|
488
|
+
deployment != undefined &&
|
|
489
|
+
instance != undefined
|
|
490
|
+
? [`${shortHostname}-${deployment}-${instance}`]
|
|
491
|
+
: []),
|
|
492
|
+
...(fullHostname != undefined ? [`${fullHostname}`] : []),
|
|
493
|
+
...(fullHostname != undefined && instance != undefined
|
|
494
|
+
? [`${fullHostname}-${instance}`]
|
|
495
|
+
: []),
|
|
496
|
+
...(fullHostname != undefined && deployment != undefined
|
|
497
|
+
? [`${fullHostname}-${deployment}`]
|
|
498
|
+
: []),
|
|
499
|
+
...(fullHostname != undefined &&
|
|
500
|
+
deployment != undefined &&
|
|
501
|
+
instance != undefined
|
|
502
|
+
? [`${fullHostname}-${deployment}-${instance}`]
|
|
503
|
+
: []),
|
|
504
|
+
`local`,
|
|
505
|
+
...(instance != undefined ? [`local-${instance}`] : []),
|
|
506
|
+
...(deployment != undefined ? [`local-${deployment}`] : []),
|
|
507
|
+
...(deployment != undefined && instance != undefined
|
|
508
|
+
? [`local-${deployment}-${instance}`]
|
|
509
|
+
: []),
|
|
510
|
+
'$NODE_CONFIG',
|
|
511
|
+
'custom-environment-variables',
|
|
512
|
+
];
|
|
513
|
+
return configLevels;
|
|
514
|
+
};
|
|
515
|
+
/**
|
|
516
|
+
* validates a config object and writes it to the node-config file
|
|
517
|
+
* corresponding to the passed level
|
|
518
|
+
*
|
|
519
|
+
* @param {Record<string, any>} configObj
|
|
520
|
+
* @param {IConfig} config
|
|
521
|
+
* @param {string} level output node-config file level
|
|
522
|
+
* @param {string} format the format of the output file
|
|
523
|
+
*/
|
|
524
|
+
validateAndWriteConfig = (configObj, config, level, format) => {
|
|
525
|
+
const confLevels = ConfigValidator.getNodeConfigLevels(config).filter(
|
|
526
|
+
(l) => l !== 'custom-environment-variables'
|
|
527
|
+
);
|
|
528
|
+
const levelIndex = confLevels.indexOf(level);
|
|
529
|
+
if (levelIndex === -1) {
|
|
530
|
+
throw new Error(
|
|
531
|
+
`The [${level}] level not found in the current system's configuration levels`
|
|
532
|
+
);
|
|
533
|
+
}
|
|
534
|
+
const configDir =
|
|
535
|
+
process.env['NODE_CONFIG_DIR'] != undefined
|
|
536
|
+
? process.env['NODE_CONFIG_DIR']
|
|
537
|
+
: './config';
|
|
538
|
+
let output = '';
|
|
539
|
+
let ext = '';
|
|
540
|
+
switch (format) {
|
|
541
|
+
case 'json': {
|
|
542
|
+
const JsonBigInt = JsonBigIntFactory({
|
|
543
|
+
alwaysParseAsBig: false,
|
|
544
|
+
useNativeBigInt: true,
|
|
545
|
+
});
|
|
546
|
+
output = JsonBigInt.stringify(configObj);
|
|
547
|
+
ext = 'json';
|
|
548
|
+
break;
|
|
549
|
+
}
|
|
550
|
+
case 'yaml': {
|
|
551
|
+
output = yaml.dump(configObj);
|
|
552
|
+
ext = 'yaml';
|
|
553
|
+
break;
|
|
554
|
+
}
|
|
555
|
+
default:
|
|
556
|
+
throw Error(`Invalid format: ${format}`);
|
|
557
|
+
}
|
|
558
|
+
const outputPath = path.join(configDir, `${level}.${ext}`);
|
|
559
|
+
const backupPath = path.join(configDir, `${level}-backup.${ext}`);
|
|
560
|
+
const confFileExists = fs.existsSync(outputPath);
|
|
561
|
+
if (confFileExists) {
|
|
562
|
+
fs.renameSync(outputPath, backupPath);
|
|
563
|
+
}
|
|
564
|
+
fs.writeFileSync(outputPath, output);
|
|
565
|
+
const updatedConfObj = config.util.loadFileConfigs();
|
|
566
|
+
try {
|
|
567
|
+
this.validateConfig(updatedConfObj);
|
|
568
|
+
fs.unlinkSync(backupPath);
|
|
569
|
+
} catch (error) {
|
|
570
|
+
fs.unlinkSync(outputPath);
|
|
571
|
+
if (confFileExists) {
|
|
572
|
+
fs.renameSync(backupPath, outputPath);
|
|
573
|
+
}
|
|
574
|
+
throw error;
|
|
575
|
+
}
|
|
576
|
+
};
|
|
577
|
+
}
|
|
578
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../lib/config.ts"],"names":[],"mappings":"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","sourcesContent":["import { IConfig, IConfigSource } from 'config';\nimport * as fs from 'fs';\nimport * as yaml from 'js-yaml';\nimport JsonBigIntFactory from 'json-bigint';\nimport path from 'path';\nimport {\n  propertyValidators,\n  supportedTypes,\n} from './schema/Validators/fieldProperties';\nimport { ConfigField, ConfigSchema } from './schema/types/fields';\nimport { When } from './schema/types/validations';\nimport { getSourceName, getValueFromConfigSources } from './utils';\nimport { valueValidations, valueValidators } from './value/validators';\n\nexport class ConfigValidator {\n  constructor(private schema: ConfigSchema) {\n    this.validateSchema();\n  }\n\n  /**\n   * validates the passed config against the instance's schema\n   *\n   * @param {Record<string, any>} config\n   */\n  public validateConfig(config: Record<string, any>) {\n    const errorPreamble = (path: Array<string>) =>\n      `config validation failed for \"${path.join('.')}\" field`;\n\n    const stack: Array<{\n      subSchema: ConfigSchema;\n      subConfig: Record<string, any> | undefined;\n      parentPath: Array<string>;\n    }> = [\n      {\n        subSchema: this.schema,\n        subConfig: config,\n        parentPath: [],\n      },\n    ];\n\n    this.validateValue(\n      config,\n      { type: 'object', children: this.schema },\n      config\n    );\n\n    // Traverses the schema object tree depth first in coordination with config\n    // object tree and validate config using the schema\n    while (stack.length > 0) {\n      const { subSchema, subConfig, parentPath } = stack.pop()!;\n      // Process children of current field\n      for (const name of Object.keys(subSchema)) {\n        const path = parentPath.concat([name]);\n        try {\n          const field = subSchema[name];\n          let value = undefined;\n          if (subConfig != undefined && Object.hasOwn(subConfig, name)) {\n            value = subConfig[name];\n          }\n\n          this.validateValue(value, field, config);\n\n          // if a node/field is of type object and thus is a subtree, add it to\n          // the stack to be traversed later\n          if (field.type === 'object') {\n            stack.push({\n              subSchema: field.children,\n              subConfig: value,\n              parentPath: path,\n            });\n          }\n        } catch (error: any) {\n          throw new Error(`${errorPreamble(path)}: ${error.message}`);\n        }\n      }\n    }\n  }\n\n  /**\n   * validates a value in config object\n   *\n   * @private\n   * @param {*} value\n   * @param {ConfigField} field the field specification in schema\n   * @param {Record<string, any>} config the config object\n   */\n  private validateValue = (\n    value: any,\n    field: ConfigField,\n    config: Record<string, any>\n  ) => {\n    if (value != undefined) {\n      if (field.type === 'bigint') {\n        value = BigInt(value);\n      } else if (field.type === 'number' && value && !isNaN(value)) {\n        value = Number(value);\n      }\n      valueValidators[field.type](value, field);\n    }\n\n    if (field.type != 'object' && field.validations) {\n      for (const validation of field.validations) {\n        const name = Object.keys(validation).filter(\n          (key) => key !== 'when' && key !== 'error'\n        )[0];\n        if (Object.hasOwn(valueValidations[field.type], name)) {\n          try {\n            valueValidations[field.type][name](value, validation, config, this);\n          } catch (error: any) {\n            if (validation.error != undefined) {\n              throw new Error(validation.error);\n            }\n            throw error;\n          }\n        }\n      }\n    }\n  };\n\n  /**\n   * determines if a when clause in validations section of a schema field is\n   * satisfied\n   *\n   * @param {When} when\n   * @param {Record<string, any>} config\n   * @return {boolean}\n   */\n  public isWhenTrue = (when: When, config: Record<string, any>): boolean => {\n    const pathParts = when.path.split('.');\n    const value = ConfigValidator.valueAt(config, pathParts);\n    return value != undefined && value === when.value;\n  };\n\n  /**\n   * returns the value at specified path in config object\n   *\n   * @static\n   * @param {Record<string, any>} config\n   * @param {string[]} path\n   * @return {*}\n   */\n  static valueAt = (config: Record<string, any>, path: string[]) => {\n    let value: any = config;\n    for (const key of path) {\n      if (value != undefined && Object.hasOwn(value, key)) {\n        value = value[key];\n      } else {\n        return undefined;\n      }\n    }\n\n    return value;\n  };\n\n  /**\n   * validates this.schema and throws exception if any errors found\n   */\n  private validateSchema = () => {\n    const errorPreamble = (path: Array<string>) =>\n      `Schema validation failed for \"${path.join('.')}\" field`;\n\n    const stack: Array<{\n      subSchema: ConfigSchema;\n      parentPath: Array<string>;\n    }> = [\n      {\n        subSchema: this.schema,\n        parentPath: [],\n      },\n    ];\n\n    // Traverses the schema object tree depth first and validate fields\n    while (stack.length > 0) {\n      const { subSchema, parentPath } = stack.pop()!;\n\n      // process children of current object field\n      for (const name of Object.keys(subSchema).reverse()) {\n        const path = parentPath.concat([name]);\n        try {\n          this.validateConfigName(name);\n          const field = subSchema[name];\n\n          if (!Object.hasOwn(field, 'type') || typeof field.type !== 'string') {\n            throw new Error(\n              `every schema field must have a \"type\" property of type \"string\"`\n            );\n          }\n\n          if (!supportedTypes.includes(field.type)) {\n            throw new Error(`unsupported field type \"${field.type}\"`);\n          }\n\n          this.validateSchemaField(field);\n\n          // if the child is an object field itself add it to stack for\n          // processing\n          if (field.type === 'object') {\n            stack.push({\n              subSchema: field.children,\n              parentPath: path,\n            });\n          }\n        } catch (error: any) {\n          throw new Error(`${errorPreamble(path)}: ${error.message}`);\n        }\n      }\n    }\n  };\n\n  /**\n   * validates config key name\n   *\n   * @param {string} name\n   */\n  private validateConfigName = (name: string) => {\n    if (name.includes('.')) {\n      throw new Error(`config key name can not contain the '.' character`);\n    }\n  };\n\n  /**\n   * validates passed schema field structure\n   *\n   * @param {ConfigField} field\n   */\n  private validateSchemaField = (field: ConfigField) => {\n    for (const key of Object.keys(field)) {\n      if (\n        !Object.hasOwn(propertyValidators.all, key) &&\n        !(\n          field.type !== 'object' &&\n          Object.hasOwn(propertyValidators.primitive, key)\n        ) &&\n        !Object.hasOwn(propertyValidators[field.type], key)\n      ) {\n        throw new Error(`schema field has unknown property \"${key}\"`);\n      }\n    }\n\n    for (const validator of Object.values(propertyValidators.all)) {\n      validator(field, this);\n    }\n\n    if (field.type !== 'object') {\n      for (const validator of Object.values(propertyValidators.primitive)) {\n        validator(field, this);\n      }\n    }\n\n    for (const validator of Object.values(propertyValidators[field.type])) {\n      validator(field, this);\n    }\n  };\n\n  /**\n   * returns a field corresponding to a path in schema tree\n   *\n   * @param {string[]} path\n   * @return {(ConfigField | undefined)} returns undefined if field is not found\n   */\n  getSchemaField = (path: string[]): ConfigField | undefined => {\n    let subTree: ConfigSchema | undefined = this.schema;\n    let field: ConfigField | undefined = undefined;\n    for (const part of path) {\n      if (subTree != undefined && Object.hasOwn(subTree, part)) {\n        field = subTree[part];\n        subTree = 'children' in field ? field.children : undefined;\n      } else {\n        return undefined;\n      }\n    }\n    return field;\n  };\n\n  /**\n   * extracts default values from a schema\n   *\n   * @return {Record<string, any>} object of default values\n   */\n  generateDefault = (): Record<string, any> => {\n    const valueTree: Record<string, any> = Object.create(null);\n\n    const stack: {\n      schema: ConfigSchema;\n      parentValue: Record<string, any> | undefined;\n      fieldName: string;\n      children: string[];\n    }[] = [\n      {\n        schema: this.schema,\n        parentValue: undefined,\n        fieldName: '',\n        children: Object.keys(this.schema).reverse(),\n      },\n    ];\n\n    // Traverses the schema object tree depth first\n    while (stack.length > 0) {\n      const { schema, parentValue, fieldName, children } = stack.at(-1)!;\n\n      // if a subtree's processing is finished go to the previous level\n      if (children.length === 0) {\n        // if a subtree is empty (has no values) remove it from the result\n        if (\n          parentValue != undefined &&\n          Object.keys(parentValue[fieldName]).length === 0\n        ) {\n          delete parentValue[fieldName];\n        }\n        stack.pop();\n        continue;\n      }\n\n      const childName = children.pop()!;\n      const value =\n        parentValue != undefined ? parentValue[fieldName] : valueTree;\n      const field = schema[childName];\n      // if a node/field is of type object and thus is a subtree, add it both to\n      // value tree and to the stack to be traversed later. Otherwise it's a\n      // leaf and needs no traversal, so add it only to the value tree.\n      if (field.type === 'object') {\n        value[childName] = Object.create(null);\n        stack.push({\n          schema: field.children,\n          parentValue: value,\n          fieldName: childName,\n          children: Object.keys(field.children).reverse(),\n        });\n      } else if (field.default != undefined) {\n        value[childName] = field.default;\n      }\n    }\n\n    return valueTree;\n  };\n\n  /**\n   * generates compatible TypeScript interface for this instance's schema\n   *\n   * @param {string} name the name of root type\n   * @return {string}\n   */\n  generateTSTypes = (name: string): string => {\n    const errorPreamble = (path: Array<string>) =>\n      `TypeScript type generation failed for \"${path.join('.')}\" field`;\n\n    const types: Array<string> = [];\n\n    const typeNames: Map<string, bigint> = new Map<string, bigint>();\n    typeNames.set(name, 1n);\n\n    const stack: Array<{\n      subSchema: ConfigSchema;\n      children: string[];\n      parentPath: Array<string>;\n      typeName: string;\n      attributes: Array<[string, string]>;\n    }> = [\n      {\n        subSchema: this.schema,\n        children: Object.keys(this.schema),\n        parentPath: [],\n        typeName: name,\n        attributes: [],\n      },\n    ];\n\n    // Traverses the schema object tree depth first\n    while (stack.length > 0) {\n      const { subSchema, children, parentPath, typeName, attributes } =\n        stack.at(-1)!;\n      const path = parentPath.concat([name]);\n      try {\n        // if a subtree's processing is finished go to the previous level\n        if (children.length == 0) {\n          types.push(this.genTSInterface(typeName, attributes));\n          stack.pop();\n          continue;\n        }\n\n        const childName = children.pop()!;\n        const field = subSchema[childName];\n\n        // if a node/field is of type object and thus is a subtree, add it to\n        // the stack to be traversed later. Otherwise it's a leaf and needs no\n        // traversal.\n        if (field.type === 'object') {\n          let childTypeName = `${childName[0].toUpperCase()}${childName.substring(\n            1\n          )}`;\n          const typeNameCount = typeNames.get(childTypeName);\n          typeNames.set(childTypeName, (typeNames.get(childName) || 0n) + 1n);\n          if (typeNameCount) {\n            childTypeName += typeNameCount.toString();\n          }\n\n          stack.push({\n            subSchema: field.children,\n            children: Object.keys(field.children).reverse(),\n            parentPath: path,\n            typeName: childTypeName,\n            attributes: [],\n          });\n\n          attributes.push([childName, childTypeName]);\n        } else {\n          attributes.push([childName, field.type]);\n        }\n      } catch (error: any) {\n        throw new Error(`${errorPreamble(path)}: ${error.message}`);\n      }\n    }\n\n    return types.reverse().join('\\n\\n') + '\\n';\n  };\n\n  /**\n   * generates a TypeScript interface definition for passed name and attributes\n   *\n   * @param {string} name\n   * @param {Array<[string, string]>} attributes\n   * @return {string}\n   */\n  private genTSInterface = (\n    name: string,\n    attributes: Array<[string, string]>\n  ): string => {\n    return `interface ${name} {\n  ${attributes.map((attr) => `${attr[0]}: ${attr[1]};`).join('\\n  ')}\n}`;\n  };\n\n  /**\n   * returns a characteristic object for values at a specific node config level\n   *\n   * @param {IConfig} config\n   * @param {string} level\n   * @return {Record<string, any>}\n   */\n  getConfigForLevel(config: IConfig, level: string): Record<string, any> {\n    const confLevels = ConfigValidator.getNodeConfigLevels(config);\n    const levelIndex = confLevels.indexOf(level);\n    if (levelIndex === -1) {\n      throw new Error(\n        `The \"${level}\" level not found in the current system configuration levels`\n      );\n    }\n    const higherLevelSources = config.util\n      .getConfigSources()\n      .filter(\n        (source) => confLevels.indexOf(getSourceName(source)) > levelIndex\n      );\n    const currentLevelSource = config.util\n      .getConfigSources()\n      .filter((source) => getSourceName(source) === level)\n      .at(0);\n    const lowerLevelSources = config.util\n      .getConfigSources()\n      .filter(\n        (source) => confLevels.indexOf(getSourceName(source)) < levelIndex\n      );\n\n    // Traverses the schema object tree depth first\n    const valueTree = ConfigValidator.processConfigForLevelNode(\n      this.schema,\n      [],\n      higherLevelSources,\n      currentLevelSource,\n      lowerLevelSources\n    );\n\n    return valueTree;\n  }\n\n  /**\n   *traverses the config schema depth first to produce characteristic object\n   *\n   * @private\n   * @static\n   * @param {ConfigSchema} schema\n   * @param {string[]} path\n   * @param {IConfigSource[]} higherLevelSources\n   * @param {(IConfigSource | undefined)} currentLevelSource\n   * @param {IConfigSource[]} lowerLevelSources\n   * @return {Record<string, any>}\n   * @memberof ConfigValidator\n   */\n  private static processConfigForLevelNode(\n    schema: ConfigSchema,\n    path: string[],\n    higherLevelSources: IConfigSource[],\n    currentLevelSource: IConfigSource | undefined,\n    lowerLevelSources: IConfigSource[]\n  ): Record<string, any> {\n    const value = Object.create(null);\n    for (const childName of Object.keys(schema).reverse()) {\n      const childPath = path.concat([childName]);\n      const field = schema[childName];\n      // if a field is of type object and thus is a subtree, recurse on it.\n      // Otherwise it's a leaf and needs no traversal.\n      value[childName] = Object.create(null);\n      if (field.type === 'object') {\n        value[childName] = ConfigValidator.processConfigForLevelNode(\n          field.children,\n          childPath,\n          higherLevelSources,\n          currentLevelSource,\n          lowerLevelSources\n        );\n      } else {\n        value[childName]['label'] =\n          field.label != undefined ? field.label : null;\n        value[childName]['description'] =\n          field.description != undefined ? field.description : null;\n        value[childName]['default'] = getValueFromConfigSources(\n          lowerLevelSources,\n          childPath\n        );\n        value[childName]['value'] = getValueFromConfigSources(\n          [...(currentLevelSource != undefined ? [currentLevelSource] : [])],\n          childPath\n        );\n        value[childName]['override'] = getValueFromConfigSources(\n          higherLevelSources,\n          childPath\n        );\n      }\n    }\n\n    return value;\n  }\n\n  /**\n   * returns a list of config sources used by node config package, ordered from\n   * the lowest to the highest priority\n   *\n   * @static\n   * @param {IConfig} config\n   * @return  {string[]}\n   */\n  private static getNodeConfigLevels = (config: IConfig): string[] => {\n    const instance = config.util.getEnv('NODE_APP_INSTANCE');\n    let deployment = config.util.getEnv('NODE_ENV');\n    deployment = config.util.getEnv('NODE_CONFIG_ENV');\n    const fullHostname = config.util.getEnv('HOSTNAME');\n    const shortHostname =\n      fullHostname != undefined ? fullHostname.split('.')[0] : undefined;\n\n    const configLevels = [\n      'default',\n      ...(instance != undefined ? [`default-${instance}`] : []),\n      ...(deployment != undefined ? [`${deployment}`] : []),\n      ...(instance != undefined && deployment != undefined\n        ? [`${deployment}-${instance}`]\n        : []),\n      ...(shortHostname != undefined ? [`${shortHostname}`] : []),\n      ...(shortHostname != undefined && instance != undefined\n        ? [`${shortHostname}-${instance}`]\n        : []),\n      ...(shortHostname != undefined && deployment != undefined\n        ? [`${shortHostname}-${deployment}`]\n        : []),\n      ...(shortHostname != undefined &&\n      deployment != undefined &&\n      instance != undefined\n        ? [`${shortHostname}-${deployment}-${instance}`]\n        : []),\n      ...(fullHostname != undefined ? [`${fullHostname}`] : []),\n      ...(fullHostname != undefined && instance != undefined\n        ? [`${fullHostname}-${instance}`]\n        : []),\n      ...(fullHostname != undefined && deployment != undefined\n        ? [`${fullHostname}-${deployment}`]\n        : []),\n      ...(fullHostname != undefined &&\n      deployment != undefined &&\n      instance != undefined\n        ? [`${fullHostname}-${deployment}-${instance}`]\n        : []),\n      `local`,\n      ...(instance != undefined ? [`local-${instance}`] : []),\n      ...(deployment != undefined ? [`local-${deployment}`] : []),\n      ...(deployment != undefined && instance != undefined\n        ? [`local-${deployment}-${instance}`]\n        : []),\n      '$NODE_CONFIG',\n      'custom-environment-variables',\n    ];\n\n    return configLevels;\n  };\n\n  /**\n   * validates a config object and writes it to the node-config file\n   * corresponding to the passed level\n   *\n   * @param {Record<string, any>} configObj\n   * @param {IConfig} config\n   * @param {string} level output node-config file level\n   * @param {string} format the format of the output file\n   */\n  validateAndWriteConfig = (\n    configObj: Record<string, any>,\n    config: IConfig,\n    level: string,\n    format: string\n  ) => {\n    const confLevels = ConfigValidator.getNodeConfigLevels(config).filter(\n      (l) => l !== 'custom-environment-variables'\n    );\n    const levelIndex = confLevels.indexOf(level);\n    if (levelIndex === -1) {\n      throw new Error(\n        `The [${level}] level not found in the current system's configuration levels`\n      );\n    }\n\n    const configDir =\n      process.env['NODE_CONFIG_DIR'] != undefined\n        ? process.env['NODE_CONFIG_DIR']\n        : './config';\n    let output = '';\n    let ext = '';\n    switch (format) {\n      case 'json': {\n        const JsonBigInt = JsonBigIntFactory({\n          alwaysParseAsBig: false,\n          useNativeBigInt: true,\n        });\n        output = JsonBigInt.stringify(configObj);\n        ext = 'json';\n        break;\n      }\n      case 'yaml': {\n        output = yaml.dump(configObj);\n        ext = 'yaml';\n        break;\n      }\n      default:\n        throw Error(`Invalid format: ${format}`);\n    }\n\n    const outputPath = path.join(configDir, `${level}.${ext}`);\n    const backupPath = path.join(configDir, `${level}-backup.${ext}`);\n    const confFileExists = fs.existsSync(outputPath);\n    if (confFileExists) {\n      fs.renameSync(outputPath, backupPath);\n    }\n    fs.writeFileSync(outputPath, output);\n\n    const updatedConfObj = config.util.loadFileConfigs();\n\n    try {\n      this.validateConfig(updatedConfObj);\n      fs.unlinkSync(backupPath);\n    } catch (error) {\n      fs.unlinkSync(outputPath);\n      if (confFileExists) {\n        fs.renameSync(backupPath, outputPath);\n      }\n      throw error;\n    }\n  };\n}\n"]}
|
package/dist/index.d.ts
ADDED