Configuring NestJS applications
Mon Feb 05 2024
1. The challenges around configuration
When working on a complex project, being able to easily configure an application to run in multiple environments becomes a necessity.
Depending on the environment (local, staging or production) the requirements around environment variables might change and cause some headaches. Luckily, we will see that this is not such a big problem for applications that use NestJS, as it can help us drastically reduce the possibility of human errors.
The nest documentation already provides a comprehensive guide on configuration that you can view here. Using that as a starting point, I will explain some of the basic concepts and showcase the setup that I prefer using.
2. Prerequisites
- Have Node.js installed
- A NestJS project with
@nestjs/config
installed. Basic knowledge of the framework and Node.js is required as I will only be focusing on configuration.
The easiest way to get everything set up is to run:
$ npm i -g @nestjs/cli
$ nest new my-project
$ cd my-project
$ npm i --save @nestjs/config
3. Setting up the config module
Setting up the config module, as shown in the documentation, is as easy as importing it in the app module:
// app.module.ts
import { Module } from '@nestjs/common';
import { ConfigModule } from '@nestjs/config';
@Module({
imports: [ConfigModule.forRoot()],
})
export class AppModule {}
You could also specify custom/multiple env file paths if you wish:
ConfigModule.forRoot({
envFilePath: ['.env.development.local', '.env.development'],
});
Under the hood, this will parse the environment variables using the dotenv package and assign them to process.env.
Defining it in the app module works just fine, but I generally opt for a slightly different approach:
3.1. Generate a core module
$ nest g module core
The idea of creating a core module as opposed to keeping everything at the top level and importing it in the app module is to define the cross-cutting functionalities of the application and export the ones that will be needed by other modules. For example:
// core.module.ts
import { Global, Module } from '@nestjs/common';
import { CqrsModule } from '@nestjs/cqrs';
import { ConfigModule } from './config/config.module';
import { DatabaseModule } from './database/database.module';
import { HealthModule } from './health/health.module';
import { AuthModule } from './auth/auth.module';
@Global()
@Module({
imports: [
CqrsModule.forRoot(),
ConfigModule,
DatabaseModule,
HealthModule,
AuthModule,
],
exports: [ConfigModule, DatabaseModule, AuthModule],
})
export class CoreModule {}
This is the core module implementation from a project I am currently working on that showcases my point. I would also set up basic integrations with external services such as Stripe here as well.
An important topic to keep in mind as well is to only export the things you need! There would be no point in exporting the health module as it will not be used by other modules in app.
3.2. Generate a config module inside the core module
To generate the config module, run:
$ cd ./src/core
$ nest g module config
After that, we can set up the nest config module.
// config.module.ts
import { Module, Provider } from '@nestjs/common';
import { ConfigModule as NestConfigModule } from '@nestjs/config';
@Module({
imports: [
NestConfigModule.forRoot(),
],
})
export class ConfigModule {}
4. The issues of the basic setup
After following these steps, you should easily be able to inject the ConfigService
which is exported from @nestjs/config
and access your environment variables.
const dbUser = this.configService.get<string>('POSTGRES_USER');
// you can also provide a fallback in case the value is not found
const dbUser = this.configService.get<string>('POSTGRES_USER', 'postgres');
As you can see, you can even pass a generic parameter to specify a type for the result. The problem is that, by default, all properties from the environment are parsed as strings, and to handle other value types such as numbers or comma-separated values that you might want to parse into an array would probably be done through custom getters such as the one below:
import { ConfigService } from '@nestjs/config';
@Injectable()
export class ConfigService {
constructor(private readonly configService: ConfigService) {}
get port(): number {
return Number(this.configService.get('POSTGRES_PORT'));
}
}
You can even provide a type for the config service to have autocompletion:
export class DbConfigService {
constructor(private configService: ConfigService<{ POSTGRES_PORT: string }>) {}
// ...
}
In my opinion, there are two main issues with this setup:
- There is no explicit validation of the provided variables.
- For example, to validate that the input for
POSTGRES_PORT
is actually a valid port number.
- For example, to validate that the input for
- All of our variables are grouped together in one place. This might not be an issue at first, but as applications grow and have more and more third-party integrations separation becomes desirable.
5. My preferred setup
To avoid the issues described above, I employ two primary techniques: namespacing and runtime validation.
5.1 Using namespaces
// postgres.configuration.ts
import { registerAs } from '@nestjs/config';
export const POSTGRES_CONFIG_KEY = 'postgres';
export const postgresConfig = registerAs(POSTGRES_CONFIG_KEY, () => {
return {
host: process.env.POSTGRES_HOST,
port: process.env.POSTGRES_PORT,
user: process.env.POSTGRES_USER,
password: process.env.POSTGRES_PASSWORD,
databaseName: process.env.POSTGRES_DB_NAME,
logging: process.env.POSTGRES_LOGS_ENABLED,
};
});
// config.module.ts
@Module({
imports: [
NestConfigModule.forRoot({
load: [postgresConfig],
}),
],
})
export class ConfigModule {}
Using namespaces allows you to break up the configuration of your app into multiple slices that can be used separately.
5.2 Environment validation at runtime
class-validator and class-transformer integrate really nicely with Nest and they are my go-to packages for request validation, so I see no point in adding another validation library for this as these two work really well.
You can install them by running:
$ npm i class-validator class-transformer
Before creating the validation schemas, let’s create a function that applies our validation logic.
// validate.ts
import { plainToInstance } from 'class-transformer';
import { ValidationError, validateSync } from 'class-validator';
import { Type } from '@nestjs/common';
export function validate<TDto extends Type>(
config: Record<string, unknown>,
dto: TDto,
): TDto {
const validatedConfig = plainToInstance(dto, config);
const errors = validateSync(validatedConfig, {
skipMissingProperties: false,
});
if (errors.length > 0) {
throw new Error(getPrettyErrorMessage(errors));
}
return validatedConfig;
}
// Formatting the error is a thing of preference, but this is how I like to do it
function getPrettyErrorMessage(errors: ValidationError[]): string {
let errorToPrint = '';
errors.forEach((err) => {
if (!err.constraints) return;
const violatedConstraints = Object.values(err.constraints);
errorToPrint += violatedConstraints.join('\n') + '\n';
});
return errorToPrint;
}
The validate
function does the following:
- Takes the raw config and validation class as inputs.
- Tries to validate the raw config according to the specifications in the validation class. (we will look at an example in a moment)
- Throws an error that prints the validation constraints that were not respected if the validation is not successful.
- Returns the validated configuration if every check passed.
5.3 Updated namespace registration + example schema
// postgres.configuration.ts
import { registerAs } from '@nestjs/config';
import { validate } from '../validate';
import { PostgresSchema } from './postgres.schema';
export const POSTGRES_CONFIG_KEY = 'postgres';
export const postgresConfig = registerAs(POSTGRES_CONFIG_KEY, () => {
const rawConfig = {
host: process.env.POSTGRES_HOST,
port: process.env.POSTGRES_PORT,
user: process.env.POSTGRES_USER,
password: process.env.POSTGRES_PASSWORD,
databaseName: process.env.POSTGRES_DB_NAME,
logging: process.env.POSTGRES_LOGS_ENABLED,
};
return validate(rawConfig, PostgresSchema);
});
// postgres.schema.ts
import { Transform } from 'class-transformer';
import { IsBoolean, IsPort, IsString } from 'class-validator';
export class PostgresSchema {
@IsString()
host: string;
@IsPort()
port: string;
@IsString()
user: string;
@IsString()
password: string;
@IsString()
databaseName: string;
@IsBoolean()
@Transform(({ value }) => value === 'true')
logging: boolean;
}
A note on implicit conversion:
// implementation of "validate" found on nest docs
export function validate(config: Record<string, unknown>) {
const validatedConfig = plainToInstance(
EnvironmentVariables,
config,
{ enableImplicitConversion: true },
);
const errors = validateSync(validatedConfig, { skipMissingProperties: false });
if (errors.length > 0) {
throw new Error(errors.toString());
}
return validatedConfig;
}
A key difference between our implementation and the one found in the documentation is that we did not set { enableImplicitConversion: true }
.
This is a matter of preference, but there some things that must be accounted for. Let’s look at an example:
If implicit conversion is enabled and the POSTGRES_LOGS_ENABLED
variable is equal to the string false
, since that is a truthy value it will be coerced to the boolean true
when processed by plainToInstance
.
In consequence, to achieve our desired result of disabling logging when that variable is set to false we would have to do some processing when defining our configuration namespace.
{
// ...
logging: process.env.POSTGRES_LOGS_ENABLED === 'true',
// ...
}
If we don’t allow implicit conversion we have to define the transformation logic in our validation class, as showcased in our schema definition.
I personally avoid implicit conversion as I find it easier to treat everything that our validation schema receives as strings and perform the necessary processing there.
5.4 Consuming the configuration
One of the biggest benefits of configuration namespaces is that you can inject the configuration object directly and benefit from strong typings. (especially since we also know that the contents are validated when our application spins up)
// postgres.service.ts
import { ConfigType } from '@nestjs/config';
import { postgresConfig } from '../config/postgres/postgres.configuration';
export class PostgresService {
constructor(
@Inject(postgresConfig.KEY)
private pgConfig: ConfigType<typeof postgresConfig>,
) {}
}
The KEY
property is a unique string returned by registerAs
that can be used as the injection token to gain access to the configuration object.
To avoid the typeof
gymnastics, I like to do the following:
// config.module.ts
import { Module, Provider } from '@nestjs/common';
import { ConfigType, ConfigModule as NestConfigModule } from '@nestjs/config';
import { postgresConfig } from './postgres/postgres.configuration';
import { PostgresSchema } from './postgres/postgres.schema';
const PostgresSchemaProvider: Provider = {
provide: PostgresSchema,
inject: [postgresConfig.KEY],
useFactory: (config: ConfigType<typeof postgresConfig>) => config,
};
@Module({
imports: [
NestConfigModule.forRoot({
load: [postgresConfig],
}),
],
providers: [PostgresSchemaProvider],
exports: [PostgresSchemaProvider],
})
export class ConfigModule {}
The code above registers a custom provider so that we can gain access to the configuration using the PostgresSchema
class both as the injection token and as the type. I like this approach the most.
The updated usage example:
// postgres.service.ts
import { PostgresSchema } from '../config/postgres/postgres.schema';
export class PostgresService {
constructor(
@Inject(PostgresSchema)
private pgConfig: PostgresSchema,
) {}
}
6. Conclusion
Environment configuration errors can be some of the most frustrating, especially when the person deploying the code is not the same as the one who wrote it.
I strongly believe that validating your application’s environment is one of the most important things to think about when scaffolding a new project. One thing that we have not looked at in this post is how to apply different validation criteria depending on the deployment environment (staging, production), but I am planning to get into that soon and I will make a separate post on mixins, as they are especially useful in this scenario.
Typos can also be extremely annoying, but that can also be solved really easily when using typescript. A good video by Matt Pocock covers pretty much everything on the topic.
I hope you have found the content useful, and I am open to discuss anything regarding this post or Nest in general on twitter or linkedin. Thank you for reading!