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