Awesome
NestJS v10 Webpack Boilerplate
š„ Features
-
š Graceful Production Deployment
-
š§ Unified API Response Structure
-
ā” Extreme Performance Optimize
-
š Fully Integrated Coding Quality Tools
ā ļø Attention
Although there are advantages to use Webpack bundling your code (especially for serverless applications), there are some constraints, and details here (UPDATE: Both examples stated by the NestJS creator have already unblinded the native driver and are good to use now). Therefore, please make sure your application does not contain native bindings library, then you can enjoy the benefits.
Extra Configuration for Dependency Packages
Bull
You should install copy-webpack-plugin
and copy bull default commands to the output directory when building the code.
pnpm install -D copy-webpack-plugin
// webpack.config.js;
const CopyWebpackPlugin = require('copy-webpack-plugin');
// ...
module.exports = {
plugins: [
// ...
new CopyWebpackPlugin({
patterns: [
{
context: 'node_modules/bull/lib/commands',
from: '**/*.lua',
},
],
}),
];
}
Pino Pretty
By default we assume the application will run in production
mode after building the app, so if you still using development
mode you will get the error since you enable pino-pretty and it does not include in the production bundle. Therefore, if you want to use pino-pretty after bundling for any reason, you should install the pino-webpack-plugin
.
pnpm install -D pino-webpack-plugin
// webpack.config.js
const { PinoWebpackPlugin } = require('pino-webpack-plugin');
// ...
module.exports = {
// ...
plugins: [
// ...
new PinoWebpackPlugin({ transports: ['pino-pretty'] }),
],
};
š Commands
Commands Description
# build the app
$ pnpm build
# format the code
$ pnpm lint
# start the app
$ pnpm start
# run in development mode
$ pnpm start:dev || pnpm dev
# build the app and run it in production mode
$ pnpm start:prod || pnpm prod
# generate Swagger JSON schema
$ pnpm swagger
# test both unit test and e2e test
$ pnpm test
# test all the e2e test
$ pnpm test:e2e
# test all the unit test
$ pnpm test:unit
Running Application for Development
$ git clone <repo>
$ pnpm install
# Fill in require information in .env file
$ cp .env.example .env
# Linux / Mac users may require (allow git hook script executable)
$ chmod +x .husky -R
$ pnpm dev
š Boilerplate Structure
āāā ci
ā āāā docker-compose.yaml
ā āāā Dockerfile
āāā .husky
ā āāā _
ā ā āāā .gitignore
ā ā āāā husky.sh
ā āāā commit-msg
ā āāā pre-commit
ā āāā pre-push
āāā src
ā āāā exception
ā ā āāā index.ts
ā ā āāā normal.exception.ts
ā āāā filter
ā ā āāā all-exception.filter.ts
ā ā āāā index.ts
ā ā āāā normal-exception.filter.ts
ā ā āāā validator-exception.filter.ts
ā āāā interceptor
ā ā āāā response.interceptor.ts
ā āāā modules
ā ā āāā app
ā ā ā āāā dto
ā ā ā ā āāā response
ā ā ā ā ā āāā index.ts
ā ā ā ā ā āāā version.dto.ts
ā ā ā ā āāā index.ts
ā ā ā āāā app.config.ts
ā ā ā āāā app.controller.ts
ā ā ā āāā app.module.ts
ā ā ā āāā app.service.spec.ts
ā ā ā āāā app.service.ts
ā ā ā āāā index.ts
ā ā āāā http
ā ā āāā http.module.ts
ā ā āāā http.service.ts
ā āāā shared
ā ā āāā enums
ā ā ā āāā index.ts
ā ā ā āāā log-level.ts
ā ā ā āāā node-env.ts
ā ā āāā interfaces
ā ā ā āāā index.ts
ā ā ā āāā response.ts
ā ā āāā constants.ts
ā āāā utils
ā ā āāā clustering.ts
ā ā āāā helper.ts
ā ā āāā swagger.ts
ā āāā env.d.ts
ā āāā main.ts
āāā test
ā āāā app.e2e-spec.ts
ā āāā common.ts
ā āāā jest.e2e.config.ts
āāā .vscode
ā āāā extensions.json
ā āāā settings.json
āāā .commitlintrc.js
āāā .dockerignore
āāā .editorconfig
āāā .env.example
āāā .eslintignore
āāā .eslintrc.js
āāā .gitattributes
āāā .gitignore
āāā jest.config.ts
āāā .lintstagedrc.js
āāā nest-cli.json
āāā .npmrc
āāā package.json
āāā pnpm-lock.yaml
āāā .prettierrc.js
āāā README.md
āāā tsconfig.json
āāā webpack.config.js
ā Coding Quality Tools Details Description
ESLint
It statically analyzes your code to help you detect formatting issues and find code inconsistencies, here we extend the ESLint TypeScript recommend rules, the most popular JavaScript style Airbnb, auto import sorting and shaking plugins.
# Config File
āāā .eslintignore
āāā .eslintrc.js
Prettier
Similar to ESLint, but mainly focus on auto-formatting, not the code quality. Actually, ESLint can do all the jobs that Prettier can do, but for the formatting part, Prettier does better, so we import both and achieve each of the advantages. About the conflict of the formatting part, we can import plugin:prettier/recommended
to solve this, but keep in mind that this plugin should extend at the last.
# Config File
āāā .prettierrc.js
Editorconfig
It defines a standard code formatting style guide among all the IDEs and editors used within a team of developers. Basically, all the rules in the Editorconfig should sync with Prettier, Editorconfig focus on newly created files, ESLint and Prettier focus on existing files.
# Config File
āāā .editorconfig
Husky + Commitlint + Lint-staged
These tools are the wrapper of Git Hook. Lint-staged enforces you to format your code (run pnpm lint
) before committing, but the tools will cache the file that is already formatted to improve performance. Commitlint enforces your commit message to fit a specific format, here we extend Conventional Commits (officially recommend setting).
# Type: build, chore, ci, docs, feat, fix, perf, refactor, revert, style, test
# Commitlint Format:
<type>[optional scope]: <description>
[optional body]
[optional footer(s)]
# Config File for Lint-staged
āāā .lintstagedrc.js
# Config File for Commitlint
āāā .commitlintrc.js
# Config File for Husky
āāā .husky
| āāā commit-msg # call Commitlint to check the commit message
| āāā pre-commit # call Eslint to lint the coding issue
| āāā pre-push # call Jest to do the unit + e2e test
Git Attributes
To synchronize the end-of-line of the git repository.
# Config File
āāā .gitattributes
āļø Other Configuration
SWC
SWC (stands for Speedy Web Compiler) is a super-fast TypeScript / JavaScript compiler written in Rust. NestJS v10 should be officially support it now, so we replace ts-loader
to swc-loader
for better building performance.
Pnpm
We use preinstall script forcing Pnpm as default package manager because it is a fast and disk space efficient manager compare with Npm and Yarn.
Webpack
We overwrite the default webpack.config.js
so that the production build can bundle all required libraries in main.ts
. For the configuration, we ignored a list of the nestjs-buildin library so that we could build it without error. If you need these libraries for your development, you can comment it in the lazy imports list.
Alias Path
Using an alias path can prevent dirty relative paths (e.g. ../../../), also it is easier to import files in the deep directory (e.g. src/assets/img/testing/...).
# Config File
āāā tsconfig.json
API Response
Success Response
{
"data": {
"...": "..."
}
}
Error Response
{
"error": {
"code": 400,
"message": "..."
}
}
We use Google JSON guide to be the response format implemented by filtering + interceptor, which is the built-in feature of NestJS, to sync with the response format. All exceptions will be caught by filtering, and all normal returns will be transformed by the interceptor.
# Related Directory
āāā src
| āāā exception
| āāā filter
| āāā interceptor
Environment Variables Validation
We use Joi library for the validation, which is recommended by NestJS.
# Config File
āāā src
| āāā app.config.ts
HTTP Request
Since @nestjs/axios default return Observable, it does not fit the common use case (Promise based), so we use a custom module to implement secondary encapsulation of the native Axios library, also extract .data from the response to prevent .data.data.data... chaining.
Reference:
Pino Logger
We used nestjs-pino to auto-log every request metadata and response time. We also centralized Pino config in app.config.ts
for main.ts
to reuse it.
Swagger
@nestjs/swagger allows you to auto-generate the API document, but here we decouple the document and the service. You can run pnpm swagger
to generate the schema and put it into Swagger UI to host your API document as a static page. We have two examples in app.controller.ts
to show you how to integrate the Google JSON response format. We also have a GitHub Action example to auto-update the schema and host it to the GitHub Pages. If you do not want this setup, you can just follow NestJS official guideline to host your document inside the service.
Attention:
You do not need to wrap the data object to your DTO for every response, you only have to name your DTO end with 'Res', swagger.ts
script will auto-wrap for you and display correctly in the Swagger UI.
Docker Containerization
We also set up the Dockerfile
with multi-stage builds to optimize your image size and building time. For the docker-compose config, we also included health checking.
# Config File
āāā ci
| āāā docker-compose.yaml
| āāā Dockerfile
Clustering
We also configured the clustering feature for the service to improve performance. All you need to do is just config the environment variable CLUSTERING=true
.
āļø Naming Convention
JS variable / function:
lower camel case [e.g. twoWords]
JS global const + enum's attributes:
upper case [e.g. TWO_WORDS]
JS class / interface / type / enum:
pascal case [e.g. TwoWords]
Asset name (e.g. image):
kebab case [e.g. two-words]
š Performance Optimization
By default, we used Fastify instead of Express to achieve twice of performance, below are the benchmarks tested by NestJS:
Express.js
Stat | 1% | 2.5% | 50% | 97.5% | Avg | Stdev | Min |
---|---|---|---|---|---|---|---|
Req/Sec | 14183 | 14183 | 15767 | 15991 | 15640 | 501.13 | 14182 |
Bytes/Sec | 3.06 MB | 3.06 MB | 3.41 MB | 3.45 MB | 3.38 MB | 108 kB | 3.06 MB |
Fastify
Stat | 1% | 2.5% | 50% | 97.5% | Avg | Stdev | Min |
---|---|---|---|---|---|---|---|
Req/Sec | 19935 | 19935 | 33247 | 34111 | 32030.4 | 4103.84 | 19931 |
Bytes/Sec | 3.03 MB | 3.03 MB | 5.05 MB | 5.19 MB | 4.87 MB | 624 kB | 3.03 MB |
Reference:
License
This project is licensed under the MIT License, Copyright Ā© 2022. See LICENSE for more information.