Skip to content
/ p-env Public

TypeScript tools for parsing process.env type-safely

License

Notifications You must be signed in to change notification settings

carnesen/p-env

Repository files navigation

build status badge

An isomorphic TypeScript library for parsing process environment variables type-safely

Usage

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
// }

default and optional

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:

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.

API

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".

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 configs is provided to p.env and to the class constructor, the class constructor one takes precedence.

Environment variable factories

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.

boolean

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.

number

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.

string

p.string: Factory for string-valued environment variables. The string parser returns the environment value as-is.

stringArray

p.stringArray: Factory for string[]-valued environment variables. The stringArray parser splits the environment value on ,.

stringOneOf

p.stringOneOf: Factory for environment variables that must be one of the allowed values provided.

Custom types

To create your own custom environment variable type, extend PEnvVar using the built-in classes (PEnvBoolean etc.) as your guide.

More information

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.

Related

  • @carnesen/cli: A library for building command-line interfaces in Node.js and the browser

License

MIT © Chris Arnesen