safe-env-getter 0.1.9 → 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/README.md +40 -6
- package/lib/index.d.ts +83 -3
- package/lib/index.js +126 -11
- package/package.json +21 -21
package/README.md
CHANGED
|
@@ -10,7 +10,7 @@ Type-safe environment variable getter for Node.js. Reads and parses `process.env
|
|
|
10
10
|
|
|
11
11
|
- **Typed specs**: `string`, `number`, `boolean`, and `enum` with TypeScript inference
|
|
12
12
|
- **Optional defaults**: Fallback values when the variable is unset or empty
|
|
13
|
-
- **Strict validation**: Throws with
|
|
13
|
+
- **Strict validation**: Throws `SafeEnvGetterValidationError` with structured errors
|
|
14
14
|
- **Boolean parsing**: Accepts `1`, `true`, `yes`, `on` (case-insensitive) as `true`
|
|
15
15
|
- **Enum constraint**: Restricts values to a fixed set of choices
|
|
16
16
|
|
|
@@ -31,7 +31,7 @@ yarn add safe-env-getter
|
|
|
31
31
|
## Usage
|
|
32
32
|
|
|
33
33
|
```ts
|
|
34
|
-
import { SafeEnvGetter, SafeEnvType } from 'safe-env-getter';
|
|
34
|
+
import { SafeEnvGetter, SafeEnvGetterValidationError, SafeEnvType } from 'safe-env-getter';
|
|
35
35
|
|
|
36
36
|
// String (spec omitted → defaults to SafeEnvType.String; throws if missing)
|
|
37
37
|
const nodeEnv = SafeEnvGetter.getEnv('NODE_ENV');
|
|
@@ -46,14 +46,36 @@ const port = SafeEnvGetter.getEnv('PORT', SafeEnvType.Number, { default: 3000 })
|
|
|
46
46
|
const debug = SafeEnvGetter.getEnv('DEBUG', SafeEnvType.Boolean, { default: false });
|
|
47
47
|
|
|
48
48
|
// Enum (required)
|
|
49
|
-
const mode = SafeEnvGetter.getEnv('MODE', SafeEnvType.Enum(['read', 'write']));
|
|
49
|
+
const mode = SafeEnvGetter.getEnv('MODE', SafeEnvType.Enum(['read', 'write'] as const));
|
|
50
50
|
// Enum with default
|
|
51
|
-
const modeWithDefault = SafeEnvGetter.getEnv('MODE', SafeEnvType.Enum(['read', 'write']), { default: 'read' });
|
|
51
|
+
const modeWithDefault = SafeEnvGetter.getEnv('MODE', SafeEnvType.Enum(['read', 'write'] as const), { default: 'read' });
|
|
52
|
+
|
|
53
|
+
// Read multiple envs at once (collects all missing/invalid errors and throws once)
|
|
54
|
+
const envs = SafeEnvGetter.getEnvs({
|
|
55
|
+
PORT: [SafeEnvType.Number, { default: 3000 }],
|
|
56
|
+
DEBUG: [SafeEnvType.Boolean, { default: false }],
|
|
57
|
+
MODE: SafeEnvType.Enum(['read', 'write'] as const),
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
// Access structured validation errors
|
|
61
|
+
try {
|
|
62
|
+
SafeEnvGetter.getEnvs({
|
|
63
|
+
PORT: SafeEnvType.Number,
|
|
64
|
+
MODE: SafeEnvType.Enum(['read', 'write'] as const),
|
|
65
|
+
});
|
|
66
|
+
} catch (e) {
|
|
67
|
+
if (e instanceof SafeEnvGetterValidationError) {
|
|
68
|
+
// e.errors: [{ key, message, raw?, kind }, ...]
|
|
69
|
+
// e.keys: ['PORT', 'MODE', ...]
|
|
70
|
+
console.error(e.errors);
|
|
71
|
+
}
|
|
72
|
+
throw e;
|
|
73
|
+
}
|
|
52
74
|
```
|
|
53
75
|
|
|
54
76
|
## Options
|
|
55
77
|
|
|
56
|
-
`SafeEnvGetter.getEnv(key, spec?, options?)`
|
|
78
|
+
### `SafeEnvGetter.getEnv(key, spec?, options?)`
|
|
57
79
|
|
|
58
80
|
| Argument | Required | Description |
|
|
59
81
|
|----------|----------|-------------|
|
|
@@ -61,6 +83,13 @@ const modeWithDefault = SafeEnvGetter.getEnv('MODE', SafeEnvType.Enum(['read', '
|
|
|
61
83
|
| **spec** | No | Type spec; defaults to `SafeEnvType.String` when omitted. Use `SafeEnvType.String`, `SafeEnvType.Number`, `SafeEnvType.Boolean`, or `SafeEnvType.Enum(choices)`. |
|
|
62
84
|
| **options** | No | Optional object. Use `{ default: value }` to provide a fallback when the variable is missing or empty. |
|
|
63
85
|
|
|
86
|
+
### `SafeEnvGetter.getEnvs(schema)`
|
|
87
|
+
|
|
88
|
+
`schema` is an object where each key is an env var name and each value is either:
|
|
89
|
+
|
|
90
|
+
- A spec (e.g. `SafeEnvType.Number`)
|
|
91
|
+
- A tuple of `[spec, { default }]` (e.g. `[SafeEnvType.Number, { default: 3000 }]`)
|
|
92
|
+
|
|
64
93
|
**Spec types:**
|
|
65
94
|
|
|
66
95
|
| Spec | Shape | Description |
|
|
@@ -72,10 +101,15 @@ const modeWithDefault = SafeEnvGetter.getEnv('MODE', SafeEnvType.Enum(['read', '
|
|
|
72
101
|
|
|
73
102
|
**Errors:**
|
|
74
103
|
|
|
75
|
-
-
|
|
104
|
+
- Missing/empty without a default: `Missing required environment variable: <key>`.
|
|
76
105
|
- For `number`, invalid values: `Env <key>: expected number, got "<raw>"`.
|
|
77
106
|
- For `enum`, invalid values: `Env <key>: must be one of [choice1, choice2, ...]`.
|
|
78
107
|
|
|
108
|
+
Both `getEnv()` and `getEnvs()` throw `SafeEnvGetterValidationError`, which exposes structured data:
|
|
109
|
+
|
|
110
|
+
- `errors`: `[{ key, message, raw?, kind }, ...]`
|
|
111
|
+
- `keys`: `['KEY1', 'KEY2', ...]`
|
|
112
|
+
|
|
79
113
|
## Requirements
|
|
80
114
|
|
|
81
115
|
- **Node.js** >= 20.0.0
|
package/lib/index.d.ts
CHANGED
|
@@ -38,6 +38,62 @@ export type SafeEnvTypeEnum<T extends string = string> = {
|
|
|
38
38
|
};
|
|
39
39
|
/** Union of all environment variable spec types. */
|
|
40
40
|
export type SafeEnvSpec = SafeEnvTypeString | SafeEnvTypeNumber | SafeEnvTypeBoolean | SafeEnvTypeEnum;
|
|
41
|
+
/**
|
|
42
|
+
* Discriminated union error kinds emitted during env parsing/validation.
|
|
43
|
+
*/
|
|
44
|
+
export type SafeEnvErrorKind = 'missing' | 'invalid_number' | 'invalid_enum';
|
|
45
|
+
/**
|
|
46
|
+
* A structured validation error entry for a single env var.
|
|
47
|
+
*
|
|
48
|
+
* @template K - Environment variable key type.
|
|
49
|
+
*/
|
|
50
|
+
export type SafeEnvError<K extends string = string> = {
|
|
51
|
+
key: K;
|
|
52
|
+
message: string;
|
|
53
|
+
raw?: string;
|
|
54
|
+
kind: SafeEnvErrorKind;
|
|
55
|
+
};
|
|
56
|
+
/**
|
|
57
|
+
* Base error class for this package.
|
|
58
|
+
*
|
|
59
|
+
* @example
|
|
60
|
+
* ```ts
|
|
61
|
+
* try {
|
|
62
|
+
* SafeEnvGetter.getEnv('PORT', SafeEnvType.Number);
|
|
63
|
+
* } catch (e) {
|
|
64
|
+
* if (e instanceof SafeEnvGetterError) {
|
|
65
|
+
* // Handle all safe-env-getter errors
|
|
66
|
+
* }
|
|
67
|
+
* }
|
|
68
|
+
* ```
|
|
69
|
+
*/
|
|
70
|
+
export declare abstract class SafeEnvGetterError extends Error {
|
|
71
|
+
protected constructor(message: string);
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Validation error that carries one or more environment variable issues.
|
|
75
|
+
*
|
|
76
|
+
* It is thrown by `SafeEnvGetter.getEnv()` (single-entry `errors`) and
|
|
77
|
+
* `SafeEnvGetter.getEnvs()` (multi-entry `errors`).
|
|
78
|
+
*/
|
|
79
|
+
export declare class SafeEnvGetterValidationError<K extends string = string> extends SafeEnvGetterError {
|
|
80
|
+
/**
|
|
81
|
+
* Formats validation errors into a human-readable error message.
|
|
82
|
+
*/
|
|
83
|
+
static format<K extends string>(errors: readonly SafeEnvError<K>[]): string;
|
|
84
|
+
/**
|
|
85
|
+
* Structured list of validation errors.
|
|
86
|
+
*/
|
|
87
|
+
readonly errors: readonly SafeEnvError<K>[];
|
|
88
|
+
/**
|
|
89
|
+
* Convenience list of keys included in `errors`.
|
|
90
|
+
*/
|
|
91
|
+
readonly keys: readonly K[];
|
|
92
|
+
/**
|
|
93
|
+
* Creates a new validation error from one or more `SafeEnvError` entries.
|
|
94
|
+
*/
|
|
95
|
+
constructor(errors: readonly SafeEnvError<K>[]);
|
|
96
|
+
}
|
|
41
97
|
/**
|
|
42
98
|
* Predefined spec constants for use as the second argument to `getEnv`.
|
|
43
99
|
* Use the third argument `options: { default: value }` to provide a fallback when the variable is missing.
|
|
@@ -67,6 +123,31 @@ export declare const SafeEnvType: {
|
|
|
67
123
|
* @template S - A `SafeEnvSpec` variant.
|
|
68
124
|
*/
|
|
69
125
|
export type SafeEnvSpecToType<S> = S extends SafeEnvTypeString ? string : S extends SafeEnvTypeNumber ? number : S extends SafeEnvTypeBoolean ? boolean : S extends SafeEnvTypeEnum<infer T> ? T : never;
|
|
126
|
+
/**
|
|
127
|
+
* Options for reading an env var with an optional default.
|
|
128
|
+
*/
|
|
129
|
+
export type SafeGetEnvOptions<S extends SafeEnvSpec> = {
|
|
130
|
+
default?: SafeEnvSpecToType<S>;
|
|
131
|
+
};
|
|
132
|
+
/**
|
|
133
|
+
* Schema entry for a single env var.
|
|
134
|
+
*
|
|
135
|
+
* Either provide a spec directly, or a tuple of `[spec, options]` to attach a default.
|
|
136
|
+
*/
|
|
137
|
+
export type SafeEnvSchemaEntry<S extends SafeEnvSpec = SafeEnvSpec> = S | readonly [S, SafeGetEnvOptions<S>];
|
|
138
|
+
/**
|
|
139
|
+
* Schema object used by `getEnvs()`.
|
|
140
|
+
*
|
|
141
|
+
* Keys are env var names, values are specs (optionally with defaults).
|
|
142
|
+
*/
|
|
143
|
+
export type SafeEnvSchema = Record<string, SafeEnvSchemaEntry>;
|
|
144
|
+
type SafeEnvSchemaEntryToSpec<E> = E extends readonly [infer S, unknown] ? S : E;
|
|
145
|
+
/**
|
|
146
|
+
* Maps a schema object to the resulting parsed env object type.
|
|
147
|
+
*/
|
|
148
|
+
export type SafeEnvSchemaToType<TSchema extends SafeEnvSchema> = {
|
|
149
|
+
[K in keyof TSchema]: SafeEnvSpecToType<SafeEnvSchemaEntryToSpec<TSchema[K]> & SafeEnvSpec>;
|
|
150
|
+
};
|
|
70
151
|
/**
|
|
71
152
|
* Reads and parses an environment variable according to the given spec.
|
|
72
153
|
* If the variable is missing or empty and no default is provided in `options`, throws an error.
|
|
@@ -75,9 +156,7 @@ export type SafeEnvSpecToType<S> = S extends SafeEnvTypeString ? string : S exte
|
|
|
75
156
|
* @param spec - Type spec; defaults to `SafeEnvType.String` when omitted.
|
|
76
157
|
* @param options - Optional. Use `{ default: value }` to provide a fallback when the variable is missing.
|
|
77
158
|
* @returns Parsed value with type inferred from `spec`.
|
|
78
|
-
* @throws {
|
|
79
|
-
* @throws {Error} When `spec.type` is `"number"` and the value is not a valid number.
|
|
80
|
-
* @throws {Error} When `spec.type` is `"enum"` and the value is not in `spec.choices`.
|
|
159
|
+
* @throws {SafeEnvGetterValidationError} When the variable is missing/invalid and `options.default` is not set (or not applicable).
|
|
81
160
|
*/
|
|
82
161
|
declare function getEnv<K extends string, S extends SafeEnvSpec = SafeEnvTypeString>(key: K, spec?: S, options?: {
|
|
83
162
|
default?: SafeEnvSpecToType<S>;
|
|
@@ -88,5 +167,6 @@ declare function getEnv<K extends string, S extends SafeEnvSpec = SafeEnvTypeStr
|
|
|
88
167
|
*/
|
|
89
168
|
export declare const SafeEnvGetter: {
|
|
90
169
|
readonly getEnv: typeof getEnv;
|
|
170
|
+
readonly getEnvs: <TSchema extends SafeEnvSchema>(schema: TSchema) => SafeEnvSchemaToType<TSchema>;
|
|
91
171
|
};
|
|
92
172
|
export {};
|
package/lib/index.js
CHANGED
|
@@ -1,6 +1,53 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.SafeEnvGetter = exports.SafeEnvType = void 0;
|
|
3
|
+
exports.SafeEnvGetter = exports.SafeEnvType = exports.SafeEnvGetterValidationError = exports.SafeEnvGetterError = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* Base error class for this package.
|
|
6
|
+
*
|
|
7
|
+
* @example
|
|
8
|
+
* ```ts
|
|
9
|
+
* try {
|
|
10
|
+
* SafeEnvGetter.getEnv('PORT', SafeEnvType.Number);
|
|
11
|
+
* } catch (e) {
|
|
12
|
+
* if (e instanceof SafeEnvGetterError) {
|
|
13
|
+
* // Handle all safe-env-getter errors
|
|
14
|
+
* }
|
|
15
|
+
* }
|
|
16
|
+
* ```
|
|
17
|
+
*/
|
|
18
|
+
class SafeEnvGetterError extends Error {
|
|
19
|
+
constructor(message) {
|
|
20
|
+
super(message);
|
|
21
|
+
this.name = 'SafeEnvGetterError';
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
exports.SafeEnvGetterError = SafeEnvGetterError;
|
|
25
|
+
/**
|
|
26
|
+
* Validation error that carries one or more environment variable issues.
|
|
27
|
+
*
|
|
28
|
+
* It is thrown by `SafeEnvGetter.getEnv()` (single-entry `errors`) and
|
|
29
|
+
* `SafeEnvGetter.getEnvs()` (multi-entry `errors`).
|
|
30
|
+
*/
|
|
31
|
+
class SafeEnvGetterValidationError extends SafeEnvGetterError {
|
|
32
|
+
/**
|
|
33
|
+
* Formats validation errors into a human-readable error message.
|
|
34
|
+
*/
|
|
35
|
+
static format(errors) {
|
|
36
|
+
const lines = errors.map((e) => `- ${e.key}: ${e.message}${e.raw == null ? '' : ` (raw="${e.raw}")`}`);
|
|
37
|
+
return `Invalid environment variables (${errors.length}):\n${lines.join('\n')}`;
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Creates a new validation error from one or more `SafeEnvError` entries.
|
|
41
|
+
*/
|
|
42
|
+
constructor(errors) {
|
|
43
|
+
const msg = SafeEnvGetterValidationError.format(errors);
|
|
44
|
+
super(msg);
|
|
45
|
+
this.name = 'SafeEnvGetterValidationError';
|
|
46
|
+
this.errors = errors;
|
|
47
|
+
this.keys = errors.map((e) => e.key);
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
exports.SafeEnvGetterValidationError = SafeEnvGetterValidationError;
|
|
4
51
|
/**
|
|
5
52
|
* Predefined spec constants for use as the second argument to `getEnv`.
|
|
6
53
|
* Use the third argument `options: { default: value }` to provide a fallback when the variable is missing.
|
|
@@ -27,41 +74,109 @@ exports.SafeEnvType = {
|
|
|
27
74
|
* @param spec - Type spec; defaults to `SafeEnvType.String` when omitted.
|
|
28
75
|
* @param options - Optional. Use `{ default: value }` to provide a fallback when the variable is missing.
|
|
29
76
|
* @returns Parsed value with type inferred from `spec`.
|
|
30
|
-
* @throws {
|
|
31
|
-
* @throws {Error} When `spec.type` is `"number"` and the value is not a valid number.
|
|
32
|
-
* @throws {Error} When `spec.type` is `"enum"` and the value is not in `spec.choices`.
|
|
77
|
+
* @throws {SafeEnvGetterValidationError} When the variable is missing/invalid and `options.default` is not set (or not applicable).
|
|
33
78
|
*/
|
|
34
79
|
function getEnv(key, spec = exports.SafeEnvType.String, options) {
|
|
35
80
|
const raw = process.env[key];
|
|
36
81
|
const defaultValue = options?.default;
|
|
37
82
|
const hasDefault = defaultValue !== undefined;
|
|
38
83
|
if (raw == null || raw === '') {
|
|
39
|
-
if (!hasDefault)
|
|
40
|
-
throw new
|
|
84
|
+
if (!hasDefault) {
|
|
85
|
+
throw new SafeEnvGetterValidationError([
|
|
86
|
+
{ key, message: `Missing required environment variable: ${key}`, raw, kind: 'missing' },
|
|
87
|
+
]);
|
|
88
|
+
}
|
|
41
89
|
return defaultValue;
|
|
42
90
|
}
|
|
43
91
|
switch (spec.type) {
|
|
44
92
|
case 'number': {
|
|
45
93
|
const n = Number(raw);
|
|
46
|
-
if (Number.isNaN(n))
|
|
47
|
-
throw new
|
|
94
|
+
if (Number.isNaN(n)) {
|
|
95
|
+
throw new SafeEnvGetterValidationError([
|
|
96
|
+
{ key, message: `Env ${key}: expected number, got "${raw}"`, raw, kind: 'invalid_number' },
|
|
97
|
+
]);
|
|
98
|
+
}
|
|
48
99
|
return n;
|
|
49
100
|
}
|
|
50
101
|
case 'boolean':
|
|
51
102
|
return (/^(1|true|yes|on)$/i.test(raw) ? true : false);
|
|
52
103
|
case 'enum':
|
|
53
|
-
if (!spec.choices.includes(raw))
|
|
54
|
-
throw new
|
|
104
|
+
if (!spec.choices.includes(raw)) {
|
|
105
|
+
throw new SafeEnvGetterValidationError([
|
|
106
|
+
{ key, message: `Env ${key}: must be one of [${spec.choices.join(', ')}]`, raw, kind: 'invalid_enum' },
|
|
107
|
+
]);
|
|
108
|
+
}
|
|
55
109
|
return raw;
|
|
56
110
|
default:
|
|
57
111
|
return raw;
|
|
58
112
|
}
|
|
59
113
|
}
|
|
114
|
+
/**
|
|
115
|
+
* Reads and parses multiple environment variables according to the given schema.
|
|
116
|
+
*
|
|
117
|
+
* This function always evaluates every key in `schema`. If any missing/invalid values
|
|
118
|
+
* are found, it throws once with a `SafeEnvGetterValidationError` that contains all issues.
|
|
119
|
+
*
|
|
120
|
+
* @param schema - Map of env var names to specs, optionally with per-key defaults via `[spec, { default }]`.
|
|
121
|
+
* @returns An object of parsed envs with types inferred from the schema.
|
|
122
|
+
* @throws {SafeEnvGetterValidationError} When one or more required env vars are missing or invalid.
|
|
123
|
+
*/
|
|
124
|
+
const getEnvs = (schema) => {
|
|
125
|
+
const envs = {};
|
|
126
|
+
const errors = [];
|
|
127
|
+
for (const key of Object.keys(schema)) {
|
|
128
|
+
const entry = schema[key];
|
|
129
|
+
const spec = (Array.isArray(entry) ? entry[0] : entry);
|
|
130
|
+
const options = (Array.isArray(entry) ? entry[1] : undefined);
|
|
131
|
+
const raw = process.env[key];
|
|
132
|
+
const defaultValue = options?.default;
|
|
133
|
+
const hasDefault = defaultValue !== undefined;
|
|
134
|
+
if (raw == null || raw === '') {
|
|
135
|
+
if (!hasDefault) {
|
|
136
|
+
errors.push({ key, message: `Missing required environment variable: ${key}`, raw, kind: 'missing' });
|
|
137
|
+
continue;
|
|
138
|
+
}
|
|
139
|
+
envs[key] = defaultValue;
|
|
140
|
+
continue;
|
|
141
|
+
}
|
|
142
|
+
if (spec.type === 'number') {
|
|
143
|
+
const n = Number(raw);
|
|
144
|
+
if (Number.isNaN(n)) {
|
|
145
|
+
errors.push({ key, message: `Env ${key}: expected number, got "${raw}"`, raw, kind: 'invalid_number' });
|
|
146
|
+
continue;
|
|
147
|
+
}
|
|
148
|
+
envs[key] = n;
|
|
149
|
+
continue;
|
|
150
|
+
}
|
|
151
|
+
if (spec.type === 'boolean') {
|
|
152
|
+
envs[key] = (/^(1|true|yes|on)$/i.test(raw) ? true : false);
|
|
153
|
+
continue;
|
|
154
|
+
}
|
|
155
|
+
if (spec.type === 'enum') {
|
|
156
|
+
if (!spec.choices.includes(raw)) {
|
|
157
|
+
errors.push({
|
|
158
|
+
key,
|
|
159
|
+
message: `Env ${key}: must be one of [${spec.choices.join(', ')}]`,
|
|
160
|
+
raw,
|
|
161
|
+
kind: 'invalid_enum',
|
|
162
|
+
});
|
|
163
|
+
continue;
|
|
164
|
+
}
|
|
165
|
+
envs[key] = raw;
|
|
166
|
+
continue;
|
|
167
|
+
}
|
|
168
|
+
envs[key] = raw;
|
|
169
|
+
}
|
|
170
|
+
if (errors.length > 0)
|
|
171
|
+
throw new SafeEnvGetterValidationError(errors);
|
|
172
|
+
return envs;
|
|
173
|
+
};
|
|
60
174
|
/**
|
|
61
175
|
* Safe environment variable getter.
|
|
62
176
|
* Use `SafeEnvGetter.getEnv(key, spec?, options?)` to read and parse environment variables with type safety.
|
|
63
177
|
*/
|
|
64
178
|
exports.SafeEnvGetter = {
|
|
65
179
|
getEnv,
|
|
180
|
+
getEnvs,
|
|
66
181
|
};
|
|
67
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAoCA;;;GAGG;AACU,QAAA,WAAW,GAAG;IACzB,+BAA+B;IAC/B,MAAM,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAuC;IAC/D,gCAAgC;IAChC,MAAM,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAuC;IAC/D,uDAAuD;IACvD,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAwC;IAClE;;;;OAIG;IACH,IAAI,EAAE,CAAmB,OAAqB,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,CAAuB;CAC5F,CAAC;AAaX;;;;;;;;;;;GAWG;AACH,SAAS,MAAM,CACb,GAAM,EACN,OAAU,mBAAW,CAAC,MAAW,EACjC,OAA4C;IAE5C,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IAC7B,MAAM,YAAY,GAAG,OAAO,EAAE,OAAO,CAAC;IACtC,MAAM,UAAU,GAAG,YAAY,KAAK,SAAS,CAAC;IAE9C,IAAI,GAAG,IAAI,IAAI,IAAI,GAAG,KAAK,EAAE,EAAE,CAAC;QAC9B,IAAI,CAAC,UAAU;YAAE,MAAM,IAAI,KAAK,CAAC,0CAA0C,GAAG,EAAE,CAAC,CAAC;QAClF,OAAO,YAAoC,CAAC;IAC9C,CAAC;IAED,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;QAClB,KAAK,QAAQ,CAAC,CAAC,CAAC;YACd,MAAM,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;YACtB,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;gBAAE,MAAM,IAAI,KAAK,CAAC,OAAO,GAAG,2BAA2B,GAAG,GAAG,CAAC,CAAC;YAClF,OAAO,CAAyB,CAAC;QACnC,CAAC;QACD,KAAK,SAAS;YACZ,OAAO,CAAC,oBAAoB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAyB,CAAC;QACjF,KAAK,MAAM;YACT,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC;gBAAE,MAAM,IAAI,KAAK,CAAC,OAAO,GAAG,qBAAqB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAC5G,OAAO,GAA2B,CAAC;QACrC;YACE,OAAO,GAA2B,CAAC;IACvC,CAAC;AACH,CAAC;AAED;;;GAGG;AACU,QAAA,aAAa,GAAG;IAC3B,MAAM;CACE,CAAC","sourcesContent":["/**\n * Spec for a string environment variable.\n * The default value is passed via the third argument of `getEnv`, not in the spec.\n */\nexport type SafeEnvTypeString = { type: 'string'; default?: string };\n\n/**\n * Spec for a numeric environment variable.\n * Values are parsed with `Number()`; invalid values throw.\n * The default value is passed via the third argument of `getEnv`, not in the spec.\n */\nexport type SafeEnvTypeNumber = { type: 'number'; default?: number };\n\n/**\n * Spec for a boolean environment variable.\n * Parses `1`, `true`, `yes`, `on` (case-insensitive) as `true`; anything else as `false`.\n * The default value is passed via the third argument of `getEnv`, not in the spec.\n */\nexport type SafeEnvTypeBoolean = { type: 'boolean'; default?: boolean };\n\n/**\n * Spec for an enum environment variable with a fixed set of choices.\n * The value must be one of `choices`; otherwise an error is thrown.\n * The default value is passed via the third argument of `getEnv`, not in the spec.\n *\n * @template T - Literal string union of allowed values.\n */\nexport type SafeEnvTypeEnum<T extends string = string> = { type: 'enum'; choices: readonly T[]; default?: T };\n\n/** Union of all environment variable spec types. */\nexport type SafeEnvSpec =\n  | SafeEnvTypeString\n  | SafeEnvTypeNumber\n  | SafeEnvTypeBoolean\n  | SafeEnvTypeEnum;\n\n/**\n * Predefined spec constants for use as the second argument to `getEnv`.\n * Use the third argument `options: { default: value }` to provide a fallback when the variable is missing.\n */\nexport const SafeEnvType = {\n  /** Spec for a string value. */\n  String: { type: 'string' } as const satisfies SafeEnvTypeString,\n  /** Spec for a numeric value. */\n  Number: { type: 'number' } as const satisfies SafeEnvTypeNumber,\n  /** Spec for a boolean value (1/true/yes/on → true). */\n  Boolean: { type: 'boolean' } as const satisfies SafeEnvTypeBoolean,\n  /**\n   * Returns a spec that restricts the value to one of the given choices.\n   * @param choices - Allowed string literals.\n   * @returns Enum spec for use with `getEnv`.\n   */\n  Enum: <T extends string>(choices: readonly T[]) => ({ type: 'enum', choices }) as SafeEnvTypeEnum<T>,\n} as const;\n\n/**\n * Infers the return type from the given spec.\n * @template S - A `SafeEnvSpec` variant.\n */\nexport type SafeEnvSpecToType<S> =\n  S extends SafeEnvTypeString ? string\n    : S extends SafeEnvTypeNumber ? number\n      : S extends SafeEnvTypeBoolean ? boolean\n        : S extends SafeEnvTypeEnum<infer T> ? T\n          : never;\n\n/**\n * Reads and parses an environment variable according to the given spec.\n * If the variable is missing or empty and no default is provided in `options`, throws an error.\n *\n * @param key - Environment variable name (e.g. `\"PORT\"`, `\"NODE_ENV\"`).\n * @param spec - Type spec; defaults to `SafeEnvType.String` when omitted.\n * @param options - Optional. Use `{ default: value }` to provide a fallback when the variable is missing.\n * @returns Parsed value with type inferred from `spec`.\n * @throws {Error} When the variable is missing and `options.default` is not set.\n * @throws {Error} When `spec.type` is `\"number\"` and the value is not a valid number.\n * @throws {Error} When `spec.type` is `\"enum\"` and the value is not in `spec.choices`.\n */\nfunction getEnv<K extends string, S extends SafeEnvSpec = SafeEnvTypeString>(\n  key: K,\n  spec: S = SafeEnvType.String as S,\n  options?: { default?: SafeEnvSpecToType<S> },\n): SafeEnvSpecToType<S> {\n  const raw = process.env[key];\n  const defaultValue = options?.default;\n  const hasDefault = defaultValue !== undefined;\n\n  if (raw == null || raw === '') {\n    if (!hasDefault) throw new Error(`Missing required environment variable: ${key}`);\n    return defaultValue as SafeEnvSpecToType<S>;\n  }\n\n  switch (spec.type) {\n    case 'number': {\n      const n = Number(raw);\n      if (Number.isNaN(n)) throw new Error(`Env ${key}: expected number, got \"${raw}\"`);\n      return n as SafeEnvSpecToType<S>;\n    }\n    case 'boolean':\n      return (/^(1|true|yes|on)$/i.test(raw) ? true : false) as SafeEnvSpecToType<S>;\n    case 'enum':\n      if (!spec.choices.includes(raw)) throw new Error(`Env ${key}: must be one of [${spec.choices.join(', ')}]`);\n      return raw as SafeEnvSpecToType<S>;\n    default:\n      return raw as SafeEnvSpecToType<S>;\n  }\n}\n\n/**\n * Safe environment variable getter.\n * Use `SafeEnvGetter.getEnv(key, spec?, options?)` to read and parse environment variables with type safety.\n */\nexport const SafeEnvGetter = {\n  getEnv,\n} as const;"]}
|
|
182
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAqDA;;;;;;;;;;;;;GAaG;AACH,MAAsB,kBAAmB,SAAQ,KAAK;IACpD,YAAsB,OAAe;QACnC,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,IAAI,GAAG,oBAAoB,CAAC;IACnC,CAAC;CACF;AALD,gDAKC;AAED;;;;;GAKG;AACH,MAAa,4BAAwD,SAAQ,kBAAkB;IAC7F;;OAEG;IACI,MAAM,CAAC,MAAM,CAAmB,MAAkC;QACvE,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,KAAK,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC,GAAG,IAAI,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,GAAG,IAAI,EAAE,CAAC,CAAC;QACvG,OAAO,kCAAkC,MAAM,CAAC,MAAM,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;IAClF,CAAC;IAWD;;OAEG;IACH,YAAmB,MAAkC;QACnD,MAAM,GAAG,GAAG,4BAA4B,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACxD,KAAK,CAAC,GAAG,CAAC,CAAC;QACX,IAAI,CAAC,IAAI,GAAG,8BAA8B,CAAC;QAC3C,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,IAAI,CAAC,IAAI,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;IACvC,CAAC;CACF;AA5BD,oEA4BC;AAED;;;GAGG;AACU,QAAA,WAAW,GAAG;IACzB,+BAA+B;IAC/B,MAAM,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAuC;IAC/D,gCAAgC;IAChC,MAAM,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAuC;IAC/D,uDAAuD;IACvD,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAwC;IAClE;;;;OAIG;IACH,IAAI,EAAE,CAAmB,OAAqB,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,CAAuB;CAC5F,CAAC;AAyCX;;;;;;;;;GASG;AACH,SAAS,MAAM,CACb,GAAM,EACN,OAAU,mBAAW,CAAC,MAAW,EACjC,OAA4C;IAE5C,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IAC7B,MAAM,YAAY,GAAG,OAAO,EAAE,OAAO,CAAC;IACtC,MAAM,UAAU,GAAG,YAAY,KAAK,SAAS,CAAC;IAE9C,IAAI,GAAG,IAAI,IAAI,IAAI,GAAG,KAAK,EAAE,EAAE,CAAC;QAC9B,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,IAAI,4BAA4B,CAAC;gBACrC,EAAE,GAAG,EAAE,OAAO,EAAE,0CAA0C,GAAG,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE;aACxF,CAAC,CAAC;QACL,CAAC;QACD,OAAO,YAAoC,CAAC;IAC9C,CAAC;IAED,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;QAClB,KAAK,QAAQ,CAAC,CAAC,CAAC;YACd,MAAM,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;YACtB,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC;gBACpB,MAAM,IAAI,4BAA4B,CAAC;oBACrC,EAAE,GAAG,EAAE,OAAO,EAAE,OAAO,GAAG,2BAA2B,GAAG,GAAG,EAAE,GAAG,EAAE,IAAI,EAAE,gBAAgB,EAAE;iBAC3F,CAAC,CAAC;YACL,CAAC;YACD,OAAO,CAAyB,CAAC;QACnC,CAAC;QACD,KAAK,SAAS;YACZ,OAAO,CAAC,oBAAoB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAyB,CAAC;QACjF,KAAK,MAAM;YACT,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;gBAChC,MAAM,IAAI,4BAA4B,CAAC;oBACrC,EAAE,GAAG,EAAE,OAAO,EAAE,OAAO,GAAG,qBAAqB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,EAAE,IAAI,EAAE,cAAc,EAAE;iBACvG,CAAC,CAAC;YACL,CAAC;YACD,OAAO,GAA2B,CAAC;QACrC;YACE,OAAO,GAA2B,CAAC;IACvC,CAAC;AACH,CAAC;AAED;;;;;;;;;GASG;AACH,MAAM,OAAO,GAAG,CAAgC,MAAe,EAAgC,EAAE;IAC/F,MAAM,IAAI,GAA0C,EAAE,CAAC;IACvD,MAAM,MAAM,GAAmD,EAAE,CAAC;IAElE,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAA0C,EAAE,CAAC;QAC/E,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;QAC1B,MAAM,IAAI,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAgB,CAAC;QACtE,MAAM,OAAO,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAA+C,CAAC;QAE5G,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAC7B,MAAM,YAAY,GAAG,OAAO,EAAE,OAAO,CAAC;QACtC,MAAM,UAAU,GAAG,YAAY,KAAK,SAAS,CAAC;QAE9C,IAAI,GAAG,IAAI,IAAI,IAAI,GAAG,KAAK,EAAE,EAAE,CAAC;YAC9B,IAAI,CAAC,UAAU,EAAE,CAAC;gBAChB,MAAM,CAAC,IAAI,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,0CAA0C,GAAG,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC,CAAC;gBACrG,SAAS;YACX,CAAC;YACD,IAAI,CAAC,GAAG,CAAC,GAAG,YAAwD,CAAC;YACrE,SAAS;QACX,CAAC;QAED,IAAI,IAAI,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC3B,MAAM,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;YACtB,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC;gBACpB,MAAM,CAAC,IAAI,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,OAAO,GAAG,2BAA2B,GAAG,GAAG,EAAE,GAAG,EAAE,IAAI,EAAE,gBAAgB,EAAE,CAAC,CAAC;gBACxG,SAAS;YACX,CAAC;YACD,IAAI,CAAC,GAAG,CAAC,GAAG,CAA6C,CAAC;YAC1D,SAAS;QACX,CAAC;QAED,IAAI,IAAI,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;YAC5B,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,oBAAoB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAA6C,CAAC;YACxG,SAAS;QACX,CAAC;QAED,IAAI,IAAI,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;YACzB,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;gBAChC,MAAM,CAAC,IAAI,CAAC;oBACV,GAAG;oBACH,OAAO,EAAE,OAAO,GAAG,qBAAqB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG;oBAClE,GAAG;oBACH,IAAI,EAAE,cAAc;iBACrB,CAAC,CAAC;gBACH,SAAS;YACX,CAAC;YACD,IAAI,CAAC,GAAG,CAAC,GAAG,GAA+C,CAAC;YAC5D,SAAS;QACX,CAAC;QAED,IAAI,CAAC,GAAG,CAAC,GAAG,GAA+C,CAAC;IAC9D,CAAC;IAED,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC;QAAE,MAAM,IAAI,4BAA4B,CAAC,MAAM,CAAC,CAAC;IACtE,OAAO,IAAoC,CAAC;AAC9C,CAAC,CAAC;AAEF;;;GAGG;AACU,QAAA,aAAa,GAAG;IAC3B,MAAM;IACN,OAAO;CACC,CAAC","sourcesContent":["/**\n * Spec for a string environment variable.\n * The default value is passed via the third argument of `getEnv`, not in the spec.\n */\nexport type SafeEnvTypeString = { type: 'string'; default?: string };\n\n/**\n * Spec for a numeric environment variable.\n * Values are parsed with `Number()`; invalid values throw.\n * The default value is passed via the third argument of `getEnv`, not in the spec.\n */\nexport type SafeEnvTypeNumber = { type: 'number'; default?: number };\n\n/**\n * Spec for a boolean environment variable.\n * Parses `1`, `true`, `yes`, `on` (case-insensitive) as `true`; anything else as `false`.\n * The default value is passed via the third argument of `getEnv`, not in the spec.\n */\nexport type SafeEnvTypeBoolean = { type: 'boolean'; default?: boolean };\n\n/**\n * Spec for an enum environment variable with a fixed set of choices.\n * The value must be one of `choices`; otherwise an error is thrown.\n * The default value is passed via the third argument of `getEnv`, not in the spec.\n *\n * @template T - Literal string union of allowed values.\n */\nexport type SafeEnvTypeEnum<T extends string = string> = { type: 'enum'; choices: readonly T[]; default?: T };\n\n/** Union of all environment variable spec types. */\nexport type SafeEnvSpec =\n  | SafeEnvTypeString\n  | SafeEnvTypeNumber\n  | SafeEnvTypeBoolean\n  | SafeEnvTypeEnum;\n\n/**\n * Discriminated union error kinds emitted during env parsing/validation.\n */\nexport type SafeEnvErrorKind = 'missing' | 'invalid_number' | 'invalid_enum';\n\n/**\n * A structured validation error entry for a single env var.\n *\n * @template K - Environment variable key type.\n */\nexport type SafeEnvError<K extends string = string> = {\n  key: K;\n  message: string;\n  raw?: string;\n  kind: SafeEnvErrorKind;\n};\n\n/**\n * Base error class for this package.\n *\n * @example\n * ```ts\n * try {\n *   SafeEnvGetter.getEnv('PORT', SafeEnvType.Number);\n * } catch (e) {\n *   if (e instanceof SafeEnvGetterError) {\n *     // Handle all safe-env-getter errors\n *   }\n * }\n * ```\n */\nexport abstract class SafeEnvGetterError extends Error {\n  protected constructor(message: string) {\n    super(message);\n    this.name = 'SafeEnvGetterError';\n  }\n}\n\n/**\n * Validation error that carries one or more environment variable issues.\n *\n * It is thrown by `SafeEnvGetter.getEnv()` (single-entry `errors`) and\n * `SafeEnvGetter.getEnvs()` (multi-entry `errors`).\n */\nexport class SafeEnvGetterValidationError<K extends string = string> extends SafeEnvGetterError {\n  /**\n   * Formats validation errors into a human-readable error message.\n   */\n  public static format<K extends string>(errors: readonly SafeEnvError<K>[]): string {\n    const lines = errors.map((e) => `- ${e.key}: ${e.message}${e.raw == null ? '' : ` (raw=\"${e.raw}\")`}`);\n    return `Invalid environment variables (${errors.length}):\\n${lines.join('\\n')}`;\n  }\n\n  /**\n   * Structured list of validation errors.\n   */\n  public readonly errors: readonly SafeEnvError<K>[];\n  /**\n   * Convenience list of keys included in `errors`.\n   */\n  public readonly keys: readonly K[];\n\n  /**\n   * Creates a new validation error from one or more `SafeEnvError` entries.\n   */\n  public constructor(errors: readonly SafeEnvError<K>[]) {\n    const msg = SafeEnvGetterValidationError.format(errors);\n    super(msg);\n    this.name = 'SafeEnvGetterValidationError';\n    this.errors = errors;\n    this.keys = errors.map((e) => e.key);\n  }\n}\n\n/**\n * Predefined spec constants for use as the second argument to `getEnv`.\n * Use the third argument `options: { default: value }` to provide a fallback when the variable is missing.\n */\nexport const SafeEnvType = {\n  /** Spec for a string value. */\n  String: { type: 'string' } as const satisfies SafeEnvTypeString,\n  /** Spec for a numeric value. */\n  Number: { type: 'number' } as const satisfies SafeEnvTypeNumber,\n  /** Spec for a boolean value (1/true/yes/on → true). */\n  Boolean: { type: 'boolean' } as const satisfies SafeEnvTypeBoolean,\n  /**\n   * Returns a spec that restricts the value to one of the given choices.\n   * @param choices - Allowed string literals.\n   * @returns Enum spec for use with `getEnv`.\n   */\n  Enum: <T extends string>(choices: readonly T[]) => ({ type: 'enum', choices }) as SafeEnvTypeEnum<T>,\n} as const;\n\n/**\n * Infers the return type from the given spec.\n * @template S - A `SafeEnvSpec` variant.\n */\nexport type SafeEnvSpecToType<S> =\n  S extends SafeEnvTypeString ? string\n    : S extends SafeEnvTypeNumber ? number\n      : S extends SafeEnvTypeBoolean ? boolean\n        : S extends SafeEnvTypeEnum<infer T> ? T\n          : never;\n\n/**\n * Options for reading an env var with an optional default.\n */\nexport type SafeGetEnvOptions<S extends SafeEnvSpec> = { default?: SafeEnvSpecToType<S> };\n\n/**\n * Schema entry for a single env var.\n *\n * Either provide a spec directly, or a tuple of `[spec, options]` to attach a default.\n */\nexport type SafeEnvSchemaEntry<S extends SafeEnvSpec = SafeEnvSpec> = S | readonly [S, SafeGetEnvOptions<S>];\n\n/**\n * Schema object used by `getEnvs()`.\n *\n * Keys are env var names, values are specs (optionally with defaults).\n */\nexport type SafeEnvSchema = Record<string, SafeEnvSchemaEntry>;\n\ntype SafeEnvSchemaEntryToSpec<E> = E extends readonly [infer S, unknown] ? S : E;\n\n/**\n * Maps a schema object to the resulting parsed env object type.\n */\nexport type SafeEnvSchemaToType<TSchema extends SafeEnvSchema> = {\n  [K in keyof TSchema]: SafeEnvSpecToType<SafeEnvSchemaEntryToSpec<TSchema[K]> & SafeEnvSpec>;\n};\n\n/**\n * Reads and parses an environment variable according to the given spec.\n * If the variable is missing or empty and no default is provided in `options`, throws an error.\n *\n * @param key - Environment variable name (e.g. `\"PORT\"`, `\"NODE_ENV\"`).\n * @param spec - Type spec; defaults to `SafeEnvType.String` when omitted.\n * @param options - Optional. Use `{ default: value }` to provide a fallback when the variable is missing.\n * @returns Parsed value with type inferred from `spec`.\n * @throws {SafeEnvGetterValidationError} When the variable is missing/invalid and `options.default` is not set (or not applicable).\n */\nfunction getEnv<K extends string, S extends SafeEnvSpec = SafeEnvTypeString>(\n  key: K,\n  spec: S = SafeEnvType.String as S,\n  options?: { default?: SafeEnvSpecToType<S> },\n): SafeEnvSpecToType<S> {\n  const raw = process.env[key];\n  const defaultValue = options?.default;\n  const hasDefault = defaultValue !== undefined;\n\n  if (raw == null || raw === '') {\n    if (!hasDefault) {\n      throw new SafeEnvGetterValidationError([\n        { key, message: `Missing required environment variable: ${key}`, raw, kind: 'missing' },\n      ]);\n    }\n    return defaultValue as SafeEnvSpecToType<S>;\n  }\n\n  switch (spec.type) {\n    case 'number': {\n      const n = Number(raw);\n      if (Number.isNaN(n)) {\n        throw new SafeEnvGetterValidationError([\n          { key, message: `Env ${key}: expected number, got \"${raw}\"`, raw, kind: 'invalid_number' },\n        ]);\n      }\n      return n as SafeEnvSpecToType<S>;\n    }\n    case 'boolean':\n      return (/^(1|true|yes|on)$/i.test(raw) ? true : false) as SafeEnvSpecToType<S>;\n    case 'enum':\n      if (!spec.choices.includes(raw)) {\n        throw new SafeEnvGetterValidationError([\n          { key, message: `Env ${key}: must be one of [${spec.choices.join(', ')}]`, raw, kind: 'invalid_enum' },\n        ]);\n      }\n      return raw as SafeEnvSpecToType<S>;\n    default:\n      return raw as SafeEnvSpecToType<S>;\n  }\n}\n\n/**\n * Reads and parses multiple environment variables according to the given schema.\n *\n * This function always evaluates every key in `schema`. If any missing/invalid values\n * are found, it throws once with a `SafeEnvGetterValidationError` that contains all issues.\n *\n * @param schema - Map of env var names to specs, optionally with per-key defaults via `[spec, { default }]`.\n * @returns An object of parsed envs with types inferred from the schema.\n * @throws {SafeEnvGetterValidationError} When one or more required env vars are missing or invalid.\n */\nconst getEnvs = <TSchema extends SafeEnvSchema>(schema: TSchema): SafeEnvSchemaToType<TSchema> => {\n  const envs: Partial<SafeEnvSchemaToType<TSchema>> = {};\n  const errors: SafeEnvError<Extract<keyof TSchema, string>>[] = [];\n\n  for (const key of Object.keys(schema) as Array<Extract<keyof TSchema, string>>) {\n    const entry = schema[key];\n    const spec = (Array.isArray(entry) ? entry[0] : entry) as SafeEnvSpec;\n    const options = (Array.isArray(entry) ? entry[1] : undefined) as SafeGetEnvOptions<SafeEnvSpec> | undefined;\n\n    const raw = process.env[key];\n    const defaultValue = options?.default;\n    const hasDefault = defaultValue !== undefined;\n\n    if (raw == null || raw === '') {\n      if (!hasDefault) {\n        errors.push({ key, message: `Missing required environment variable: ${key}`, raw, kind: 'missing' });\n        continue;\n      }\n      envs[key] = defaultValue as SafeEnvSchemaToType<TSchema>[typeof key];\n      continue;\n    }\n\n    if (spec.type === 'number') {\n      const n = Number(raw);\n      if (Number.isNaN(n)) {\n        errors.push({ key, message: `Env ${key}: expected number, got \"${raw}\"`, raw, kind: 'invalid_number' });\n        continue;\n      }\n      envs[key] = n as SafeEnvSchemaToType<TSchema>[typeof key];\n      continue;\n    }\n\n    if (spec.type === 'boolean') {\n      envs[key] = (/^(1|true|yes|on)$/i.test(raw) ? true : false) as SafeEnvSchemaToType<TSchema>[typeof key];\n      continue;\n    }\n\n    if (spec.type === 'enum') {\n      if (!spec.choices.includes(raw)) {\n        errors.push({\n          key,\n          message: `Env ${key}: must be one of [${spec.choices.join(', ')}]`,\n          raw,\n          kind: 'invalid_enum',\n        });\n        continue;\n      }\n      envs[key] = raw as SafeEnvSchemaToType<TSchema>[typeof key];\n      continue;\n    }\n\n    envs[key] = raw as SafeEnvSchemaToType<TSchema>[typeof key];\n  }\n\n  if (errors.length > 0) throw new SafeEnvGetterValidationError(errors);\n  return envs as SafeEnvSchemaToType<TSchema>;\n};\n\n/**\n * Safe environment variable getter.\n * Use `SafeEnvGetter.getEnv(key, spec?, options?)` to read and parse environment variables with type safety.\n */\nexport const SafeEnvGetter = {\n  getEnv,\n  getEnvs,\n} as const;"]}
|
package/package.json
CHANGED
|
@@ -5,24 +5,24 @@
|
|
|
5
5
|
"url": "https://github.com/gammarers-labs/safe-env-getter.git"
|
|
6
6
|
},
|
|
7
7
|
"scripts": {
|
|
8
|
-
"build": "
|
|
9
|
-
"bump": "
|
|
10
|
-
"clobber": "
|
|
11
|
-
"compile": "
|
|
12
|
-
"default": "
|
|
13
|
-
"eject": "
|
|
14
|
-
"eslint": "
|
|
15
|
-
"package": "
|
|
16
|
-
"post-compile": "
|
|
17
|
-
"post-upgrade": "
|
|
18
|
-
"pre-compile": "
|
|
19
|
-
"release": "
|
|
20
|
-
"test": "
|
|
21
|
-
"test:watch": "
|
|
22
|
-
"unbump": "
|
|
23
|
-
"upgrade": "
|
|
24
|
-
"watch": "
|
|
25
|
-
"projen": "
|
|
8
|
+
"build": "projen build",
|
|
9
|
+
"bump": "projen bump",
|
|
10
|
+
"clobber": "projen clobber",
|
|
11
|
+
"compile": "projen compile",
|
|
12
|
+
"default": "projen default",
|
|
13
|
+
"eject": "projen eject",
|
|
14
|
+
"eslint": "projen eslint",
|
|
15
|
+
"package": "projen package",
|
|
16
|
+
"post-compile": "projen post-compile",
|
|
17
|
+
"post-upgrade": "projen post-upgrade",
|
|
18
|
+
"pre-compile": "projen pre-compile",
|
|
19
|
+
"release": "projen release",
|
|
20
|
+
"test": "projen test",
|
|
21
|
+
"test:watch": "projen test:watch",
|
|
22
|
+
"unbump": "projen unbump",
|
|
23
|
+
"upgrade": "projen upgrade",
|
|
24
|
+
"watch": "projen watch",
|
|
25
|
+
"projen": "projen"
|
|
26
26
|
},
|
|
27
27
|
"author": {
|
|
28
28
|
"name": "yicr",
|
|
@@ -42,8 +42,8 @@
|
|
|
42
42
|
"eslint-plugin-import": "^2.32.0",
|
|
43
43
|
"jest": "^30.3.0",
|
|
44
44
|
"jest-junit": "^16",
|
|
45
|
-
"projen": "^0.99.
|
|
46
|
-
"ts-jest": "^29.4.
|
|
45
|
+
"projen": "^0.99.42",
|
|
46
|
+
"ts-jest": "^29.4.9",
|
|
47
47
|
"ts-node": "^10.9.2",
|
|
48
48
|
"typescript": "5.9.x"
|
|
49
49
|
},
|
|
@@ -55,7 +55,7 @@
|
|
|
55
55
|
"publishConfig": {
|
|
56
56
|
"access": "public"
|
|
57
57
|
},
|
|
58
|
-
"version": "0.
|
|
58
|
+
"version": "0.2.0",
|
|
59
59
|
"jest": {
|
|
60
60
|
"coverageProvider": "v8",
|
|
61
61
|
"testMatch": [
|