NestJS boilerplate with TypeORM, GraphQL and PostgreSQL
Totally open for any Pull Request, please feel free to contribute in any ways. There can be errors related with type or something. It would be very helpful to me for you to fix these errors.
Base NestJS, We like it
We use Nestjs/TypeORM
In this template, We've been trying not to use Pure SQL
to make the most of TypeORM.
We use postgresQL for backend database, The default database taht will be used is named 'postgres' You have to have postgresql Database server before getting started. You can use Docker postgresQL to have server easily
packages: graphql, apollo-server-express and @nestjs/graphql, graphqlUpload ...
We use GraphQL in a Code First approach (our code will create the GraphQL Schemas).
We don't use swagger now, but you can use this if you want to. You can see playground
We use apollographql as playground. but if you want to use default playground, you can do like below.
// setting.service.ts
GraphQLModule.forRootAsync <
ApolloDriverConfig >
{
...
useFactory: (configService: ConfigService) => ({
...
playground: true,
...
}),
...
};
Some of the GraphQL queries are protected by a NestJS Guard (GraphqlPassportAuthGuard
) and requires you to be authenticated (and some also requires to have the Admin role).
You can solve them with Sending JWT token in Http Header
with the Authorization
.
# Http Header
{
"Authorization": "Bearer TOKEN"
}
- getMe (must be authenticated)
- All methods generated by the generator (must be authenticated and must be admin)
-
Automatically maps GraphQL queries to optimized SELECT and JOIN clauses in TypeORM.
-
Ensures that only the requested fields and necessary relations are retrieved, reducing over-fetching and improving performance.
-
With using interceptor (name:
UseRepositoryInterceptor
) and paramDecorator (name:GraphQLQueryToOption
)
- You can find example code in /src/user/user.resolver.ts
The permission guard is used to block access to specific fields in client requests.
-
In GraphQL, clients can request any field, which could expose sensitive information. This guard ensures that sensitive fields are protected.
-
It allows controlling access to specific fields based on the server's permissions.
@Query(()=>Some)
@UseQueryPermissionGuard(Some, { something: true })
async getManySomeList(){
return this.someService.getMany()
}
With this API, if the client request includes the field "something," a Forbidden
error will be triggered.
There might be duplicate code when using this guard alongside other interceptors
(name: UseRepositoryInterceptor
) in this boilerplate. In such cases, you may need to adjust the code to ensure compatibility.
MIT
To make most of GraphQL's advantage, We created its own api, such as GetMany or GetOne. We tried to make it as comfortable as possible, but if you find any mistakes or improvements, please point them out or promote them.
You can see detail in folder /src/common/graphql files
// query
query($input:GetManyInput) {
getManyPlaces(input:$input){
data{
id
logitude
count
}
}
}
// variables
{
input: {
pagination: {
size: 10,
page: 0, // Started from 0
},
order: { id: 'DESC' },
dataType: 'data', //all or count or data - default: all
where: {
id: 3,
},
},
};
You can see detail here.
There is CRUD Generator in NestJS. In this repository, It has its own generator with plopjs. You can use like below.
$ yarn g
This project provides a custom decorator that makes it easy to implement method caching in NestJS applications.
- Caching Functionality: Utilizes
DiscoveryService
andMetadataScanner
to handle method caching automatically at runtime. - Usage: Designed for use with any provider.
- GraphQL Resolvers: Resolvers are also part of providers, but due to GraphQL's internal logic, method overrides do not work. Therefore, the functionality has been replaced with an interceptor.
You can use like below
@Injectable()
export class ExampleService {
@Cache(...)
async exampleMethod(...args: unknown) {
...
}
}
You can find related codes here
Before you start, make sure you have a recent version of NodeJS environment >=14.0 with NPM 6 or Yarn.
The first thing you will need is to install NestJS CLI.
$ yarn -g @nestjs/cli
And do install the dependencies
$ yarn install # or npm install
for development
$ yarn dev # or npm run dev
for production
$ yarn build # or npm run build
$ yarn start # or npm run start
or run with docker following below
Download docker from Official website
Before running Docker, you need to create an env file named .production.env
.
The content should be modified based on .example.env
.
The crucial point is that DB_HOST must be set to 'postgres'.
Open terminal and navigate to project directory and run the following command.
# Only for prduction
$ docker compose --env-file ./.production.env up
If you want to use docker, you have to set DB_HOST in .production.env to be postgres
.
The default set is postgres
You can just create postgresql by below code, sync with .development.env
$ docker run -p 5432:5432 --name postgres -e POSTGRES_PASSWORD=1q2w3e4r -d postgres
@nestjs/testing
= supertest
+ jest
Before starting the test, you need to set at least jwt-related environment variables in an env file named .test.env
.
Unit test(with jest mock) for services & resolvers (*.service.spec.ts & *.resolver.spec.ts)
$ yarn test:unit
Integration test(with pg-mem) for modules (*.module.spec.ts)
$ yarn test:integration
E2E Test(with docker container)
$ yarn test:e2e:docker
To ensure github actions execution, please set the 'ENV' variable within your github actions secrets as your .test.env configuration.
Note: Github Actions does not recognize newline characters. Therefore, you must remove any newline characters from each environment variable value in your .env
file, ensuring that the entire content is on a single line when setting the Secret. If you need to use an environment variable value that includes newline characters, encode the value using Base64 and store it in the Github Secret, then decode it within the workflow.
ex)
JWT_PRIVATE_KEY= -----BEGIN RSA PRIVATE KEY-----...MIIEogIBAAKCAQBZ...-----END RSA PRIVATE KEY-----
$ yarn prepare
Before commit, The pre-commit hooks is executed.
Lint checks have been automated to run before a commit is made.
If you want to add test before commit actions, you can add follow line in pre-commit file.
...
yarn test
...
The pre-push hooks is executed before the push action.
The default rule set in the pre-push hook is to prevent direct pushes to the main branch.
If you want to enable this action, you should uncomment the lines in the pre push file.
SWC (Speedy Web Compiler) is an extensible Rust-based platform that can be used for both compilation and bundling. Using SWC with Nest CLI is a great and simple way to significantly speed up your development process.
After applying SWC
, the following error was displayed in jest using an in-memory database (pg-mem
):
QueryFailedError: ERROR: function obj_description(regclass,text) does not exist
HINT: π¨ Please note that pg-mem implements very few native functions.
π You can specify the functions you would like to use via "db.public.registerFunction(...)"
π This seems to be an execution error, which means that your request syntax seems okay,
but the resulting statement cannot be executed β Probably not a pg-mem error.
*οΈβ£ Failed SQL statement: SELECT "table_schema", "table_name", obj_description(('"' || "table_schema" || '"."' || "table_name" || '"')::regclass, 'pg_class') AS table_comment FROM "information_schema"."tables" WHERE ("table_schema" = 'public' AND "table_name" = 'user');
π You can file an issue at https://github.com/oguimbal/pg-mem along with a way to reproduce this error (if you can), and the stacktrace:
pg-mem
is a library designed to emulate PostgreSQL
, however, it does not support all features, which is why the above error occurred.
This error can be resolved by implementing or overriding existing functions. Below is the function implementation for the resolution. Related issues can be checked here.
db.public.registerFunction({
name: 'obj_description',
args: [DataType.text, DataType.text],
returns: DataType.text,
implementation: () => 'test',
});
-
TDD
- Unit Test (Use mock)
- Integration Test (Use in-memory DB)
- End To End Test (Use docker)
-
CI
- Github actions
- husky
-
GraphQL Upload
-
Healthcheck
-
Divide usefactory
-
SWC Compiler
-
Refresh Token
-
Redis
-
ElasticSearch
-
Caching
-
Graphql Subscription
-
Remove lodash