Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/notiz-dev/nestjs-prisma-starter
Starter template for NestJS 😻 includes GraphQL with Prisma Client, Passport-JWT authentication, Swagger Api and Docker
https://github.com/notiz-dev/nestjs-prisma-starter
bcrypt code-first graphql nestjs passport-jwt passportjs prisma
Last synced: about 24 hours ago
JSON representation
Starter template for NestJS 😻 includes GraphQL with Prisma Client, Passport-JWT authentication, Swagger Api and Docker
- Host: GitHub
- URL: https://github.com/notiz-dev/nestjs-prisma-starter
- Owner: notiz-dev
- License: mit
- Created: 2018-12-02T16:35:28.000Z (about 6 years ago)
- Default Branch: main
- Last Pushed: 2024-07-08T07:07:30.000Z (5 months ago)
- Last Synced: 2024-10-29T15:35:08.312Z (about 1 month ago)
- Topics: bcrypt, code-first, graphql, nestjs, passport-jwt, passportjs, prisma
- Language: TypeScript
- Homepage:
- Size: 3.19 MB
- Stars: 2,347
- Watchers: 37
- Forks: 337
- Open Issues: 13
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-nestjs - NestJS Prisma Starter - Starter project for NestJS includes Graphql with Prisma Client, Passport-JWT authentication, Swagger Api and Docker. (Resources)
README
# Instructions
Starter template for 😻 [NestJS](https://nestjs.com/) and [Prisma](https://www.prisma.io/).
> Checkout [NestJS Prisma Schematics](https://github.com/marcjulian/nestjs-prisma) to automatically add Prisma support to your Nest application.
## Version
| Branch | Nest | Prisma | Graphql |
| ------------------------------------------------------------------------------------------------------------ | ----- | ---------------------------------------------------- | --------------------------------------------------------------------- |
| main | v9 | [v4](https://github.com/prisma/prisma) | [Code-first](https://docs.nestjs.com/graphql/quick-start#code-first) |
| [nest-8-prisma-3](https://github.com/fivethree-team/nestjs-prisma-starter/tree/nest-8-prisma-3) | v8 | [v3](https://github.com/prisma/prisma) | [Code-first](https://docs.nestjs.com/graphql/quick-start#code-first) |
| [nest-7](https://github.com/fivethree-team/nestjs-prisma-starter/tree/nest-7) | v7 | [v2](https://github.com/prisma/prisma2) | [Code-first](https://docs.nestjs.com/graphql/quick-start#code-first) |
| [nest-6-prisma2-code-first](https://github.com/fivethree-team/nestjs-prisma-starter/tree/nest-6-prisma2-code-first) | v6 | [v2-preview](https://github.com/prisma/prisma2) | [Code-first](https://github.com/19majkel94/type-graphql) |
| [nest-6-code-first](https://github.com/fivethree-team/nestjs-prisma-starter/tree/nest-6-code-first) | v6 | [v1](https://github.com/prisma/prisma) | [Code-first](https://github.com/19majkel94/type-graphql) |
| [nest-6-sdl-first](https://github.com/fivethree-team/nestjs-prisma-starter/tree/nest-6-sdl-first) | v6 | [v1](https://github.com/prisma/prisma) | [SDL First](https://docs.nestjs.com/graphql/quick-start#schema-first) |
| [nest-5](https://github.com/fivethree-team/nestjs-prisma-starter/tree/nest-5) | v5 | [v1](https://github.com/prisma/prisma) | [SDL First](https://docs.nestjs.com/graphql/quick-start#schema-first) |## Features
- GraphQL w/ [playground](https://github.com/prisma/graphql-playground)
- Code-First w/ [decorators](https://docs.nestjs.com/graphql/quick-start#code-first)
- [Prisma](https://www.prisma.io/) for database modelling, migration and type-safe access (Postgres, MySQL & MongoDB)
- 🔐 JWT authentication w/ [passport-jwt](https://github.com/mikenicholson/passport-jwt)
- REST API docs w/ [Swagger](https://swagger.io/)## Overview
- [Instructions](#instructions)
- [Features](#features)
- [Overview](#overview)
- [Prisma Setup](#prisma-setup)
- [1. Install Dependencies](#1-install-dependencies)
- [2. PostgreSQL with Docker](#2-PostgreSQL-with-docker)
- [3. Prisma: Prisma Migrate](#3-prisma-prisma-migrate)
- [4. Prisma: Prisma Client JS](#4-prisma-client-js)
- [5. Seed the database data with this script](#5-seed-the-database-data-with-this-script)
- [6. Start NestJS Server](#6-start-nestjs-server)
- [GraphQL Playground](#graphql-playground)
- [Rest Api](#rest-api)
- [Docker](#docker)
- [Schema Development](#schema-development)
- [NestJS - Api Schema](#nestjs---api-schema)
- [Resolver](#resolver)
- [GraphQL Client](#graphql-client)
- [Angular](#angular)
- [Setup](#setup)
- [Queries](#queries)
- [Mutations](#mutations)
- [Subscriptions](#subscriptions)
- [Authentication](#authentication)## Prisma Setup
### 1. Install Dependencies
Install [Nestjs CLI](https://docs.nestjs.com/cli/usages) to start and [generate CRUD resources](https://trilon.io/blog/introducing-cli-generators-crud-api-in-1-minute)
```bash
# npm
npm i -g @nestjs/cli
# yarn
yarn add -g @nestjs/cli
```Install the dependencies for the Nest application:
```bash
# npm
npm install
# yarn
yarn install
```### 2. PostgreSQL with Docker
Setup a development PostgreSQL with Docker. Copy [.env.example](./.env.example) and rename to `.env` - `cp .env.example .env` - which sets the required environments for PostgreSQL such as `POSTGRES_USER`, `POSTGRES_PASSWORD` and `POSTGRES_DB`. Update the variables as you wish and select a strong password.
Start the PostgreSQL database
```bash
docker-compose -f docker-compose.db.yml up -d
# or
npm run docker:db
```### 3. Prisma Migrate
[Prisma Migrate](https://github.com/prisma/prisma2/tree/master/docs/prisma-migrate) is used to manage the schema and migration of the database. Prisma datasource requires an environment variable `DATABASE_URL` for the connection to the PostgreSQL database. Prisma reads the `DATABASE_URL` from the root [.env](./.env) file.
Use Prisma Migrate in your [development environment](https://www.prisma.io/blog/prisma-migrate-preview-b5eno5g08d0b#evolving-the-schema-in-development) to
1. Creates `migration.sql` file
2. Updates Database Schema
3. Generates Prisma Client```bash
npx prisma migrate dev
# or
npm run migrate:dev
```If you like to customize your `migration.sql` file run the following command. After making your customizations run `npx prisma migrate dev` to apply it.
```bash
npx prisma migrate dev --create-only
# or
npm run migrate:dev:create
```If you are happy with your database changes you want to deploy those changes to your [production database](https://www.prisma.io/blog/prisma-migrate-preview-b5eno5g08d0b#applying-migrations-in-production-and-other-environments). Use `prisma migrate deploy` to apply all pending migrations, can also be used in CI/CD pipelines as it works without prompts.
```bash
npx prisma migrate deploy
# or
npm run migrate:deploy
```### 4. Prisma: Prisma Client JS
[Prisma Client JS](https://www.prisma.io/docs/reference/tools-and-interfaces/prisma-client/api) is a type-safe database client auto-generated based on the data model.
Generate Prisma Client JS by running
> **Note**: Every time you update [schema.prisma](prisma/schema.prisma) re-generate Prisma Client JS
```bash
npx prisma generate
# or
npm run prisma:generate
```### 5. Seed the database data with this script
Execute the script with this command:
```bash
npm run seed
```### 6. Start NestJS Server
Run Nest Server in Development mode:
```bash
npm run start# watch mode
npm run start:dev
```Run Nest Server in Production mode:
```bash
npm run start:prod
```GraphQL Playground for the NestJS Server is available here: http://localhost:3000/graphql
**[⬆ back to top](#overview)**
## GraphQL Playground
Open up the [example GraphQL queries](graphql/auth.graphql) and copy them to the GraphQL Playground. Some queries and mutations are secured by an auth guard. You have to acquire a JWT token from `signup` or `login`. Add the `accessToken`as followed to **HTTP HEADERS** in the playground and replace `YOURTOKEN` here:
```json
{
"Authorization": "Bearer YOURTOKEN"
}
```## Rest Api
[RESTful API](http://localhost:3000/api) documentation available with Swagger.
## Docker
Nest server is a Node.js application and it is easily [dockerized](https://nodejs.org/de/docs/guides/nodejs-docker-webapp/).
See the [Dockerfile](./Dockerfile) on how to build a Docker image of your Nest server.
Now to build a Docker image of your own Nest server simply run:
```bash
# give your docker image a name
docker build -t /nest-prisma-server .
# for example
docker build -t nest-prisma-server .
```After Docker build your docker image you are ready to start up a docker container running the nest server:
```bash
docker run -d -t -p 3000:3000 --env-file .env nest-prisma-server
```Now open up [localhost:3000](http://localhost:3000) to verify that your nest server is running.
When you run your NestJS application in a Docker container update your [.env](.env) file
```diff
- DB_HOST=localhost
# replace with name of the database container
+ DB_HOST=postgres# Prisma database connection
+ DATABASE_URL=postgresql://${POSTGRES_USER}:${POSTGRES_PASSWORD}@${DB_HOST}:${DB_PORT}/${POSTGRES_DB}?schema=${DB_SCHEMA}&sslmode=prefer
```If `DATABASE_URL` is missing in the root `.env` file, which is loaded into the Docker container, the NestJS application will exit with the following error:
```bash
(node:19) UnhandledPromiseRejectionWarning: Error: error: Environment variable not found: DATABASE_URL.
--> schema.prisma:3
|
2 | provider = "postgresql"
3 | url = env("DATABASE_URL")
```### Docker Compose
You can also setup a the database and Nest application with the docker-compose
```bash
# building new NestJS docker image
docker-compose build
# or
npm run docker:build# start docker-compose
docker-compose up -d
# or
npm run docker
```## Schema Development
Update the Prisma schema `prisma/schema.prisma` and after that run the following two commands:
```bash
npx prisma generate
# or in watch mode
npx prisma generate --watch
# or
npm run prisma:generate
npm run prisma:generate:watch
```**[⬆ back to top](#overview)**
## NestJS - Api Schema
The [schema.graphql](./src/schema.graphql) is generated with [code first approach](https://docs.nestjs.com/graphql/quick-start#code-first) from the models, resolvers and input classes.
You can use [class-validator](https://docs.nestjs.com/techniques/validation) to validate your inputs and arguments.
### Resolver
To implement the new query, a new resolver function needs to be added to `users.resolver.ts`.
```ts
@Query(returns => User)
async getUser(@Args() args): Promise {
return await this.prisma.client.user(args);
}
```Restart the NestJS server and this time the Query to fetch a `user` should work.
**[⬆ back to top](#overview)**
## GraphQL Client
A GraphQL client is necessary to consume the GraphQL api provided by the NestJS Server.
Checkout [Apollo](https://www.apollographql.com/) a popular GraphQL client which offers several clients for React, Angular, Vue.js, Native iOS, Native Android and more.
### Angular
#### Setup
To start using [Apollo Angular](https://www.apollographql.com/docs/angular/basics/setup.html) simply run in an Angular and Ionic project:
```bash
ng add apollo-angular
````HttpLink` from apollo-angular requires the `HttpClient`. Therefore, you need to add the `HttpClientModule` to the `AppModule`:
```ts
imports: [BrowserModule,
HttpClientModule,
...,
GraphQLModule],
```You can also add the `GraphQLModule` in the `AppModule` to make `Apollo` available in your Angular App.
You need to set the URL to the NestJS GraphQL Api. Open the file `src/app/graphql.module.ts` and update `uri`:
```ts
const uri = 'http://localhost:3000/graphql';
```To use Apollo-Angular you can inject `private apollo: Apollo` into the constructor of a page, component or service.
**[⬆ back to top](#overview)**
#### Queries
To execute a query you can use:
```ts
this.apollo.query({query: YOUR_QUERY});# or
this.apollo.watchQuery({
query: YOUR_QUERY
}).valueChanges;
```Here is an example how to fetch your profile from the NestJS GraphQL Api:
```ts
const CurrentUserProfile = gql`
query CurrentUserProfile {
me {
id
name
}
}
`;@Component({
selector: 'app-home',
templateUrl: 'home.page.html',
styleUrls: ['home.page.scss'],
})
export class HomePage implements OnInit {
data: Observable;constructor(private apollo: Apollo) {}
ngOnInit() {
this.data = this.apollo.watchQuery({
query: CurrentUserProfile,
}).valueChanges;
}
}
```Use the `AsyncPipe` and [SelectPipe](https://www.apollographql.com/docs/angular/basics/queries.html#select-pipe) to unwrap the data Observable in the template:
```html
Me id: {{me.id}}
Me email: {{me.email}}
Me name: {{me.name}}
```Or unwrap the data using [RxJs](https://www.apollographql.com/docs/angular/basics/queries.html#rxjs).
This will end up in an `GraphQL error` because `Me` is protected by an `@UseGuards(GqlAuthGuard)` and requires an `Bearer TOKEN`.
Please refer to the [Authentication](#authentication) section.**[⬆ back to top](#overview)**
#### Mutations
To execute a mutation you can use:
```ts
this.apollo.mutate({
mutation: YOUR_MUTATION,
});
```Here is an example how to login into your profile using the `login` Mutation:
```ts
const Login = gql`
mutation Login {
login(email: "[email protected]", password: "pizzaHawaii") {
token
user {
id
name
}
}
}
`;@Component({
selector: 'app-home',
templateUrl: 'home.page.html',
styleUrls: ['home.page.scss'],
})
export class HomePage implements OnInit {
data: Observable;constructor(private apollo: Apollo) {}
ngOnInit() {
this.data = this.apollo.mutate({
mutation: Login,
});
}
}
```**[⬆ back to top](#overview)**
#### Subscriptions
To execute a subscription you can use:
```ts
this.apollo.subscribe({
query: YOUR_SUBSCRIPTION_QUERY,
});
```**[⬆ back to top](#overview)**
#### Authentication
To authenticate your requests you have to add your `TOKEN` you receive on `signup` and `login` [mutation](#mutations) to each request which is protected by the `@UseGuards(GqlAuthGuard)`.
Because the apollo client is using `HttpClient` under the hood you are able to simply use an `Interceptor` to add your token to the requests.
Create the following class:
```ts
import { Injectable } from '@angular/core';
import {
HttpEvent,
HttpInterceptor,
HttpHandler,
HttpRequest,
} from '@angular/common/http';
import { Observable } from 'rxjs';@Injectable()
export class TokenInterceptor implements HttpInterceptor {
constructor() {}intercept(
req: HttpRequest,
next: HttpHandler
): Observable> {
const token = 'YOUR_TOKEN'; // get from local storage
if (token !== undefined) {
req = req.clone({
setHeaders: {
Authorization: `Bearer ${token}`,
},
});
}return next.handle(req);
}
}
```Add the Interceptor to the `AppModule` providers like this:
```ts
providers: [
...
{ provide: HTTP_INTERCEPTORS, useClass: TokenInterceptor, multi: true },
...
]
```After you configured the Interceptor and retrieved the `TOKEN` from storage your request will succeed on resolvers with `@UseGuards(GqlAuthGuard)`.
**[⬆ back to top](#overview)**