An isomorphic TypeScript library for parsing process environment variables type-safely
Install this package as a dependency in your library or application:
npm install @carnesen/p-env
Here's an example of how to use it in your code:
import { p } from '@carnesen/p-env';
class AppEnv extends p.env({
APP: p.string({ default: 'my-app', optional: true }),
DATE: p.date({ default: new Date() }),
JSON: p.json({ default: [] }),
MODE: p.stringOneOf({
default: 'dev',
values: ['dev', 'qa', 'prod'] as const,
}),
PORT: p.port({ default: 8080 }),
SECRET: p.string({ default: 'abc123', secret: true }),
STRICT: p.boolean({ default: false, optional: true }),
}) {}
// The `AppEnv` constructor parses `process.env` and assigns the
// parsed values to the new instance. Suppose STRICT is
// set to "1" (or "yes" or "true") in the process environment.
const appEnv = new AppEnv({ logger: console });
// APP="my-app" (default)
// DATE="2023-02-12T16:45:24.607Z"
// JSON=[] (default)
// MODE="dev" (default)
// PORT=8080 (default)
// SECRET=<redacted> (default)
// STRICT=true ("1")
// ^^ Parsed values for fields with `secret: true` are redacted in logs and
// error messages
// ^^ A log with (default) means a default value was used. Otherwise the logged
// line has ("<raw value>") where <raw value> is the string value from the
// process environment.
console.log(appEnv);
// AppEnv {
// APP: 'my-app',
// DATE: 2023-02-12T16:45:24.607Z,
// JSON: [],
// MODE: 'dev'
// PORT: 8080,
// SECRET: 'abc123',
// STRICT: true
// }
Every environment variable in the schema must have a default
value. A environment value always takes precedence over the default
value. If an environment value is not provided, two factors determine whether the default
value is used:
- Is Node.js is running in development or production mode?
- Does the field have
optional
set totrue
?
optional | mode | use default? |
---|---|---|
true | any | yes |
any | development | yes |
false | production | no |
When NODE_ENV
is "production"
and no environment value is provided for a non-optional field, the parser throws a PEnvError
.
This primary documentation for this package's API is its rich, strict types along with their associated TypeDoc strings viewable in your IDE by hovering over a symbol. All exports are named. The primary export is the p
namespace object import { p } from "@carnesen/p-env"
.
p.env
: Takes a single argument defining your environment object's schema and returns an anonymous class whose constructor parses process.env
and assigns the parsed values to the new instance. We recommend extending the anonymous class as a named one e.g. class MyEnv extends p.env({...}) {}
, but you're welcome to use the anonymous one directly as e.g. const MyClass = p.env({...})
. The class constructor takes an optional config
object argument new MyEnv({ logger, loader }).
If logger.log
is provided, the parsed values are logged. If logger.error
is provided, parse/validation errors are logged. Use the loader
property to define a custom process.env
loader. This is mostly useful for unit testing. config
can also/instead be passed as the second argument of p.env
. If a config
s is provided to p.env
and to the class constructor, the class constructor one takes precedence.
A p-env schema declares a name (e.g. PORT) and a type for each environment variable. The environment variable factories p.boolean
, p.string
, etc. take a config
object with a mandatory property default
and optional properties optional
and secret
. The meaning of optional
is described above. If secret
is true, the value shows as "" in logs and error messages. Some of the environment variable types have other optional properties too.
p.boolean
: Factory for boolean
-valued environment variables. "1", "true", "yes" (and their upper-case/white-spaced variations) parse to true
. "0", "false", and "no" and their upper-case/white-spaced variations) parse to false
.
There are three factories for number
valued environment variables
p.number
: Parses the environment value as a number
. NaN
is not allowed value. Has optional configuration properties maximum
(default +Infinity
) and minimum
(default -Infinity
) defining the allowed range for the number
value.
p.integer
: Same as p.number
but the parsed number
must be an integer. Equivalent to p.number
with { integer: true }
.
p.port
: Same as p.integer
with minimum: 0
and maximum: 65535
. Optional config properties maximum
and minimum
further restrict the allowed range.
p.string
: Factory for string
-valued environment variables. The string
parser returns the environment value as-is.
p.stringArray
: Factory for string[]
-valued environment variables. The stringArray
parser splits the environment value on ,
.
p.stringOneOf
: Factory for environment variables that must be one of the allowed values provided.
To create your own custom environment variable type, extend PEnvVar
using the built-in classes (PEnvBoolean
etc.) as your guide.
If you encounter any bugs or have any questions or feature requests, please don't hesitate to file an issue or submit a pull request on this project's repository on GitHub.
- @carnesen/cli: A library for building command-line interfaces in Node.js and the browser
MIT © Chris Arnesen