@wcaservices/config 1.0.1 → 1.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/package.json +1 -1
- package/src/environment/EnvironmentVariables.ts +103 -0
- package/src/environment/Errors.ts +10 -0
- package/src/environment/Types.ts +50 -0
- package/src/{defineEnvironmentVariables.test-d.ts → environment/defineEnvironmentVariables.test-d.ts} +65 -2
- package/src/environment/defineEnvironmentVariables.test.ts +50 -0
- package/src/environment/defineEnvironmentVariables.ts +14 -0
- package/src/environment/index.ts +1 -0
- package/src/index.ts +1 -1
- package/src/defineEnvironmentVariables.ts +0 -92
package/package.json
CHANGED
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
import { envFlag } from '@wcaservices/util/lib';
|
|
2
|
+
import { EnvironmentVariableError, RequiredEnvironmentVariableError } from './Errors.ts';
|
|
3
|
+
import type { VariableDefinitionMap, VariableOutputType } from './Types.ts';
|
|
4
|
+
|
|
5
|
+
export class EnvironmentVariables<TVariables extends VariableDefinitionMap> {
|
|
6
|
+
constructor(protected readonly variables: TVariables) {
|
|
7
|
+
return new Proxy(this, {
|
|
8
|
+
get(target, prop, receiver) {
|
|
9
|
+
if (prop in target) {
|
|
10
|
+
return Reflect.get(target, prop, receiver);
|
|
11
|
+
}
|
|
12
|
+
return target.getVariable(prop as string);
|
|
13
|
+
},
|
|
14
|
+
set(target, prop, value, receiver) {
|
|
15
|
+
return target.setVariable(prop as string, value);
|
|
16
|
+
}
|
|
17
|
+
})
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
public validate(): void {
|
|
21
|
+
this.validateRequiredVariables();
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
protected validateRequiredVariables(): void {
|
|
25
|
+
Object.entries(this.variables).forEach(([key, config]) => {
|
|
26
|
+
if (!config.required) {
|
|
27
|
+
return;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const value = this.getVariable(key);
|
|
31
|
+
|
|
32
|
+
if (typeof value === 'undefined') {
|
|
33
|
+
throw new RequiredEnvironmentVariableError(Object.assign({ name: key }, config));
|
|
34
|
+
}
|
|
35
|
+
})
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
protected getVariable(key: string): unknown {
|
|
39
|
+
const config = this.variables[key];
|
|
40
|
+
|
|
41
|
+
if (!config) {
|
|
42
|
+
throw new EnvironmentVariableError(`No configuration found for environment variable '${key}'`);
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
const originalValue = process.env[key] ?? config.default;
|
|
46
|
+
let value: unknown = originalValue;
|
|
47
|
+
const cast = config.type ?? 'default';
|
|
48
|
+
|
|
49
|
+
if (config.required && typeof value === 'undefined') {
|
|
50
|
+
throw new RequiredEnvironmentVariableError(Object.assign({ name: key }, config));
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
switch (cast) {
|
|
54
|
+
case 'BOOLEAN':
|
|
55
|
+
value = envFlag(key);
|
|
56
|
+
break;
|
|
57
|
+
case 'NUMBER':
|
|
58
|
+
value = Number(value);
|
|
59
|
+
if (Number.isNaN(value)) {
|
|
60
|
+
throw new EnvironmentVariableError(`Invalid number type for '${key}': ${originalValue}`);
|
|
61
|
+
}
|
|
62
|
+
break;
|
|
63
|
+
case 'JSON':
|
|
64
|
+
JSON.parse(String(value));
|
|
65
|
+
break;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
if (config.transform) {
|
|
69
|
+
value = config.transform(value);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
return value;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
protected setVariable(key: string, value: unknown): boolean {
|
|
76
|
+
const config = this.variables[key];
|
|
77
|
+
|
|
78
|
+
if (!config) {
|
|
79
|
+
throw new EnvironmentVariableError(`No configuration found for environment variable '${key}'`);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
if (config.transform) {
|
|
83
|
+
throw new EnvironmentVariableError('Setting environment variables with transform functions is not yet supported.');
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
switch (config.type) {
|
|
87
|
+
case 'BOOLEAN':
|
|
88
|
+
case 'NUMBER':
|
|
89
|
+
case 'default':
|
|
90
|
+
process.env[key] = String(value);
|
|
91
|
+
return true;
|
|
92
|
+
case 'JSON':
|
|
93
|
+
process.env[key] = JSON.stringify(value);
|
|
94
|
+
return true;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
throw new EnvironmentVariableError(`Invalid type '${config.type}' for environment variable '${key}'`);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
export type EnvironmentVariablesInstance<TVariables extends VariableDefinitionMap> = EnvironmentVariables<TVariables> & {
|
|
102
|
+
[key in keyof TVariables]: VariableOutputType<TVariables[key]>;
|
|
103
|
+
}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import type { EnvironmentVariableConfig } from './Types.ts';
|
|
2
|
+
|
|
3
|
+
export class EnvironmentVariableError extends Error {
|
|
4
|
+
}
|
|
5
|
+
|
|
6
|
+
export class RequiredEnvironmentVariableError extends EnvironmentVariableError {
|
|
7
|
+
constructor(public readonly config: EnvironmentVariableConfig & { name: string }) {
|
|
8
|
+
super(`Required environment variable "${config.name}" is missing. Make sure one is set before starting the process!`);
|
|
9
|
+
}
|
|
10
|
+
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
export interface EnvironmentVariableConfig<
|
|
2
|
+
TRequired extends boolean = boolean,
|
|
3
|
+
TTypeCast extends keyof VariableTypeCast = keyof VariableTypeCast,
|
|
4
|
+
TTransform = unknown
|
|
5
|
+
> {
|
|
6
|
+
required?: TRequired;
|
|
7
|
+
type?: TTypeCast;
|
|
8
|
+
default?: VariableTypeCast[TTypeCast];
|
|
9
|
+
transform?: TransformFn<any, TTransform>;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export type VariableDefinitionMap = {
|
|
13
|
+
[key in keyof NodeJS.ProcessEnv]: EnvironmentVariableConfig;
|
|
14
|
+
}
|
|
15
|
+
export type VariableOutputType<
|
|
16
|
+
TConfig extends EnvironmentVariableConfig,
|
|
17
|
+
TTypeCast extends keyof VariableTypeCast = TConfig['type'] extends keyof VariableTypeCast
|
|
18
|
+
? TConfig['type']
|
|
19
|
+
: 'default',
|
|
20
|
+
TType = VariableTypeCast[TTypeCast],
|
|
21
|
+
TCastOutputType = TConfig['required'] extends true
|
|
22
|
+
? TType
|
|
23
|
+
: TType | undefined,
|
|
24
|
+
> = TConfig['transform'] extends TransformFn<TCastOutputType, infer TTransformOutput>
|
|
25
|
+
? TTransformOutput
|
|
26
|
+
: TCastOutputType;
|
|
27
|
+
|
|
28
|
+
type TransformFn<TInput, TOutput> = (value: TInput) => TOutput;
|
|
29
|
+
|
|
30
|
+
interface VariableTypeCast {
|
|
31
|
+
/**
|
|
32
|
+
* Default type for environment variables.
|
|
33
|
+
*/
|
|
34
|
+
default: string;
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Boolean flag (1, 0, true, false, yes, no)
|
|
38
|
+
*/
|
|
39
|
+
BOOLEAN: boolean;
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Cast inputs to number.
|
|
43
|
+
*/
|
|
44
|
+
NUMBER: number;
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Parse input as JSON
|
|
48
|
+
*/
|
|
49
|
+
JSON: unknown; // todo
|
|
50
|
+
}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import { defineEnvironmentVariables } from './defineEnvironmentVariables';
|
|
1
|
+
import { describe, expectTypeOf, it } from 'vitest';
|
|
2
|
+
import { defineEnvironmentVariables } from './defineEnvironmentVariables.ts';
|
|
3
3
|
|
|
4
4
|
describe('Type inference for string types', () => {
|
|
5
5
|
const env = defineEnvironmentVariables({
|
|
@@ -97,3 +97,66 @@ describe('Type inference for BOOLEAN types', () => {
|
|
|
97
97
|
expectTypeOf(env.REQUIRED_ENV_VAR).not.toEqualTypeOf<undefined>();
|
|
98
98
|
});
|
|
99
99
|
});
|
|
100
|
+
|
|
101
|
+
describe('Type inference with transformer functions', () => {
|
|
102
|
+
const env = defineEnvironmentVariables({
|
|
103
|
+
EXPLICIT_RETURN_TYPE: {
|
|
104
|
+
transform: (value): 'yes' | 'no' => value === 'yes' ? 'yes' : 'no',
|
|
105
|
+
},
|
|
106
|
+
IMPLICIT_RETURN_TYPE: {
|
|
107
|
+
transform: (value) => value === 'yes' ? 'yes' as const : 'no' as const,
|
|
108
|
+
},
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
it('Can infer types with explicit return types', () => {
|
|
112
|
+
expectTypeOf(env.EXPLICIT_RETURN_TYPE).toEqualTypeOf<'yes' | 'no'>();
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
it('Can infer types with implicit return types', () => {
|
|
116
|
+
expectTypeOf(env.IMPLICIT_RETURN_TYPE).toEqualTypeOf<'yes' | 'no'>();
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it.todo('Will infer input types for required variables', () => {
|
|
120
|
+
const env = defineEnvironmentVariables({
|
|
121
|
+
IMPLICIT_REQUIRED_ARGUMENT_TYPE: {
|
|
122
|
+
required: true,
|
|
123
|
+
transform: (value) => {
|
|
124
|
+
// @ts-expect-error TODO: fix transform input type inference
|
|
125
|
+
expectTypeOf(value).toEqualTypeOf<string>()
|
|
126
|
+
return 'ok' as const;
|
|
127
|
+
}
|
|
128
|
+
},
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
expectTypeOf(env.IMPLICIT_REQUIRED_ARGUMENT_TYPE).toEqualTypeOf<'ok'>();
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
it.todo('Will infer input types for optional variables', () => {
|
|
135
|
+
const env = defineEnvironmentVariables({
|
|
136
|
+
IMPLICIT_OPTIONAL_ARGUMENT_TYPE: {
|
|
137
|
+
transform: (value) => {
|
|
138
|
+
// @ts-expect-error TODO: fix transform input type inference
|
|
139
|
+
expectTypeOf(value).toEqualTypeOf<string | undefined>()
|
|
140
|
+
return 'ok' as const;
|
|
141
|
+
}
|
|
142
|
+
},
|
|
143
|
+
});
|
|
144
|
+
|
|
145
|
+
expectTypeOf(env.IMPLICIT_OPTIONAL_ARGUMENT_TYPE).toEqualTypeOf<'ok'>();
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
it('Will allow explicit argument types', () => {
|
|
149
|
+
const env = defineEnvironmentVariables({
|
|
150
|
+
EXPLICIT_REQUIRED_ARGUMENT_TYPE: {
|
|
151
|
+
required: true,
|
|
152
|
+
transform: (value: string) => 'ok' as const,
|
|
153
|
+
},
|
|
154
|
+
EXPLICIT_OPTIONAL_ARGUMENT_TYPE: {
|
|
155
|
+
transform: (value: string | undefined) => 'ok' as const,
|
|
156
|
+
},
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
expectTypeOf(env.EXPLICIT_REQUIRED_ARGUMENT_TYPE).toEqualTypeOf<'ok'>();
|
|
160
|
+
expectTypeOf(env.EXPLICIT_OPTIONAL_ARGUMENT_TYPE).toEqualTypeOf<'ok'>();
|
|
161
|
+
})
|
|
162
|
+
})
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { beforeEach, describe, expect, it } from 'vitest';
|
|
2
|
+
import { defineEnvironmentVariables } from './defineEnvironmentVariables.ts';
|
|
3
|
+
import { RequiredEnvironmentVariableError } from './Errors.ts';
|
|
4
|
+
|
|
5
|
+
const ORIGINAL_ENV = Object.freeze({
|
|
6
|
+
...process.env,
|
|
7
|
+
})
|
|
8
|
+
|
|
9
|
+
describe('Environment variable parsing', () => {
|
|
10
|
+
beforeEach(() => {
|
|
11
|
+
process.env = {
|
|
12
|
+
...ORIGINAL_ENV,
|
|
13
|
+
TEST_STRING_HELLO: 'world',
|
|
14
|
+
TEST_NUMBER_123: '123',
|
|
15
|
+
TEST_BOOLEAN_TRUE: 'true',
|
|
16
|
+
TEST_BOOLEAN_FALSE: 'false',
|
|
17
|
+
};
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it('should enforce required environment variables', () => {
|
|
21
|
+
expect(() => {
|
|
22
|
+
const env = defineEnvironmentVariables({
|
|
23
|
+
REQUIRED_ENV_VALUE: {
|
|
24
|
+
required: true,
|
|
25
|
+
}
|
|
26
|
+
})
|
|
27
|
+
}).toThrow(RequiredEnvironmentVariableError);
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it('should apply transform functions', () => {
|
|
31
|
+
const transform = (value: boolean) => value ? 'yes' : 'no';
|
|
32
|
+
|
|
33
|
+
const env = defineEnvironmentVariables({
|
|
34
|
+
TEST_BOOLEAN_TRUE: {
|
|
35
|
+
type: 'BOOLEAN',
|
|
36
|
+
required: true,
|
|
37
|
+
transform,
|
|
38
|
+
},
|
|
39
|
+
TEST_BOOLEAN_FALSE: {
|
|
40
|
+
type: 'BOOLEAN',
|
|
41
|
+
required: true,
|
|
42
|
+
transform,
|
|
43
|
+
}
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
expect(env.TEST_BOOLEAN_TRUE).toEqual('yes');
|
|
47
|
+
expect(env.TEST_BOOLEAN_FALSE).toEqual('no');
|
|
48
|
+
})
|
|
49
|
+
})
|
|
50
|
+
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import { EnvironmentVariables, type EnvironmentVariablesInstance } from './EnvironmentVariables.ts';
|
|
2
|
+
import type { VariableDefinitionMap } from './Types.ts';
|
|
3
|
+
|
|
4
|
+
export function defineEnvironmentVariables<
|
|
5
|
+
TVariables extends VariableDefinitionMap
|
|
6
|
+
>(variables: TVariables): EnvironmentVariablesInstance<TVariables> {
|
|
7
|
+
const instance = new EnvironmentVariables(variables);
|
|
8
|
+
|
|
9
|
+
instance.validate();
|
|
10
|
+
|
|
11
|
+
// @ts-expect-error Type mismatch between result and explicit return type
|
|
12
|
+
return instance;
|
|
13
|
+
}
|
|
14
|
+
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './defineEnvironmentVariables';
|
package/src/index.ts
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
export * from './
|
|
1
|
+
export * from './environment';
|
|
@@ -1,92 +0,0 @@
|
|
|
1
|
-
import { envFlag } from '@wcaservices/util/lib';
|
|
2
|
-
|
|
3
|
-
export function defineEnvironmentVariables<
|
|
4
|
-
TVariables extends VariableDefinitionMap
|
|
5
|
-
>(variables: TVariables): {
|
|
6
|
-
[key in keyof TVariables]: VariableOutputType<TVariables[key]>;
|
|
7
|
-
} {
|
|
8
|
-
const result = {};
|
|
9
|
-
|
|
10
|
-
for (const [key, config] of Object.entries(variables)) {
|
|
11
|
-
const originalValue = process.env[key] ?? config.default;
|
|
12
|
-
let value: unknown = originalValue;
|
|
13
|
-
const cast = config.type ?? 'default';
|
|
14
|
-
|
|
15
|
-
if (config.required && typeof value === 'undefined') {
|
|
16
|
-
throw new RequiredEnvironmentVariableError(Object.assign({ name: key }, config));
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
switch (cast) {
|
|
20
|
-
case 'BOOLEAN':
|
|
21
|
-
value = envFlag(key);
|
|
22
|
-
break;
|
|
23
|
-
case 'NUMBER':
|
|
24
|
-
value = Number(value);
|
|
25
|
-
if (Number.isNaN(value)) {
|
|
26
|
-
throw new EnvironmentVariableError(`Invalid number type for '${key}': ${originalValue}`);
|
|
27
|
-
}
|
|
28
|
-
break;
|
|
29
|
-
case 'JSON':
|
|
30
|
-
JSON.parse(String(value));
|
|
31
|
-
break;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
Object.assign(result, { [key]: value });
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
// @ts-expect-error Type mismatch between result and explicit return type
|
|
38
|
-
return result;
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
interface EnvironmentVariableConfig<
|
|
42
|
-
TRequired extends boolean = boolean,
|
|
43
|
-
TTypeCast extends keyof VariableTypeCast = keyof VariableTypeCast,
|
|
44
|
-
> {
|
|
45
|
-
required?: TRequired;
|
|
46
|
-
type?: TTypeCast;
|
|
47
|
-
default?: VariableTypeCast[TTypeCast];
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
type VariableDefinitionMap = {
|
|
51
|
-
[key in keyof NodeJS.ProcessEnv]: EnvironmentVariableConfig;
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
type VariableOutputType<
|
|
55
|
-
TConfig extends EnvironmentVariableConfig,
|
|
56
|
-
TTypeCast extends keyof VariableTypeCast = TConfig['type'] extends keyof VariableTypeCast
|
|
57
|
-
? TConfig['type']
|
|
58
|
-
: 'default',
|
|
59
|
-
TType = VariableTypeCast[TTypeCast]
|
|
60
|
-
> = TConfig['required'] extends true
|
|
61
|
-
? TType
|
|
62
|
-
: TType | undefined;
|
|
63
|
-
|
|
64
|
-
class EnvironmentVariableError extends Error {}
|
|
65
|
-
|
|
66
|
-
class RequiredEnvironmentVariableError extends EnvironmentVariableError {
|
|
67
|
-
constructor(public readonly config: EnvironmentVariableConfig & { name: string }) {
|
|
68
|
-
super(`Required environment variable "${config.name}" is missing. Make sure one is set before starting the process!`);
|
|
69
|
-
}
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
interface VariableTypeCast {
|
|
73
|
-
/**
|
|
74
|
-
* Default type for environment variables.
|
|
75
|
-
*/
|
|
76
|
-
default: string;
|
|
77
|
-
|
|
78
|
-
/**
|
|
79
|
-
* Boolean flag (1, 0, true, false, yes, no)
|
|
80
|
-
*/
|
|
81
|
-
BOOLEAN: boolean;
|
|
82
|
-
|
|
83
|
-
/**
|
|
84
|
-
* Cast inputs to number.
|
|
85
|
-
*/
|
|
86
|
-
NUMBER: number;
|
|
87
|
-
|
|
88
|
-
/**
|
|
89
|
-
* Parse input as JSON
|
|
90
|
-
*/
|
|
91
|
-
JSON: unknown; // todo
|
|
92
|
-
}
|