API with NestJS #3. Authenticating users with bcrypt, Passport, JWT, and cookies

JavaScript NestJS TypeScript

Authentication is a crucial part of almost every web application. There are many ways to approach it, and we’ve handled it manually in our TypeScript Express series. This time we look into the passport, which is the most popular Node.js authentication library. We also register users and make their passwords secure by hashing.

You can find all of the code from this series in this repository. Feel free to give it a star.

Defining the User entity

The first thing to do when considering authentication is to register our users. To do so, we need to define an entity for our users.

users/user.entity.ts

The only new thing above is the unique flag. It indicates that there should not be two users with the same email. This functionality is built into PostgreSQL and helps us to keep the consistency of our data. Later, we depend on emails being unique when authenticating.

We need to perform a few operations on our users. To do so, let’s create a service.

users/users.service.ts

users/dto/createUser.dto.ts

All of the above is wrapped using a module.

users/users.module.ts

Handling passwords

An essential thing about registration is that we don’t want to save passwords in plain text. If at any time our database gets breached, our passwords would have been directly exposed.

To make passwords more secure, we hash them. In this process, the hashing algorithm transforms one string into another string. If we change just one character of a string, the outcome is entirely different.

The above operation can only be performed one way and can’t be reversed easily. This means that we don’t know the passwords of our users. When the user attempts to log in, we need to perform this operation once again. Then, we compare the outcome with the one saved in the database.

Since hashing the same string twice gives the same result, we use salt. It prevents users that have the same password from having the same hash. Salt is a random string added to the original password to achieve a different result every time.

Using bcrypt

We use the bcrypt hashing algorithm implemented by the bcrypt npm package. It takes care of hashing the strings, comparing plain strings with hashes, and appending salt.

Using bcrypt might be an intensive task for the CPU. Fortunately, our bcrypt implementation uses a thread pool that allows it to run in an additional thread. Thanks to that, our application can perform other tasks while generating the hash.

When we use bcrypt, we define salt rounds. It boils down to being a cost factor and controls the time needed to receive a result. Increasing it by one doubles the time. The bigger the cost factor, the more difficult it is to reverse the hash with brute-forcing. Generally speaking, 10 salt rounds should be fine.

The salt used for hashing is a part of the result, so no need to keep it separately.

Creating the authentication service

With all of the above knowledge, we can start implementing basic registering and logging in functionalities. To do so, we need to define an authentication service first.

Authentication means checking the identity of user. It provides an answer to a question: who is the user?

Authorization is about access to resources. It answers the question: is user authorized to perform this operation?

authentication/authentication.service.ts

 is not the cleanest way to not send the password in a response. In the upcoming parts of this series we explore mechanisms that help us with that.

A few notable things are happening above. We create a hash and pass it to the   method along with the rest of the data. We use a  statement here because there is an important case when it might fail. If a user with that email already exists, the   method throws an error. Since our unique column cases it the error comes from Postgres.

To understand the error, we need to look into the PostgreSQL Error Codes documentation page. Since the code for uniqe_violation is 23505, we can create an enum to handle it cleanly.

database/postgresErrorCodes.enum.ts

Since in the above service we state explicitly that a user with this email already exists, it might a good idea to implement a mechanism preventing attackers from brute-forcing our API in order to get a list of registered emails

The thing left for us to do is to implement the logging in.

authentication/authentication.service.ts

An important thing above is that we return the same error, whether the email or password is wrong. Doing so prevents some attacks that would aim to get a list of emails registered in our database.

There is one small thing about the above code that we might want to improve. Within our   method, we throw an exception that we then catch locally. It might be considered confusing. Let’s create a separate method to verify the password:

Integrating our authentication with Passport

In the TypeScript Express series, we’ve handled the whole authentication process manually. NestJS documentation suggests using the Passport library and provides us with the means to do so. Passport gives us an abstraction over the authentication, thus relieving us from some heavy lifting. Also, it is heavily tested in production by many developers.

Diving into how to implement the authentication manually without Passport is still a good idea. By doing so, we can get an even better understanding of this process

Applications have different approaches to authentication. Passport calls those mechanisms strategies. The first strategy that we want to implement is the passport-local strategy. It is a strategy for authenticating with a username and password.

To configure a strategy, we need to provide a set of options specific to a particular strategy. In NestJS, we do it by extending the   class.

authentication/local.strategy.ts

For every strategy, Passport calls the  function using a set of parameters specific for a particular strategy. For the local strategy, Passport needs a method with a username and a password. In our case, the email acts as a username.

We also need to configure our  to use Passport.

authentication/authentication.module.ts

Using built-in Passport Guards

The above module uses the  . Let’s create the basics of it now.

Below, we use Guards. Guard is responsible for determining whether the route handler handles the request or not. In its nature, it is similar to Express.js middleware but is more powerful.

We focus on guards quite a bit in the upcoming parts of this series and create custom guards. Today we only use the existing guards though.

authentication/authentication.controller.ts

Above we use   because NestJS responds with 201 Created for POST requests by default

authentication/localAuthentication.guard.ts

Passing the strategy name directly into   in the controller might not be considered a clean approach. Instead, we create our own class.

authentication/requestWithUser.interface.ts

Thanks to doing all of the above, our  route is handled by Passport. The data of the user is attached to the  object, and this is why we extend the  interface.

If the user authenticates successfully, we return his data. Otherwise, we throw an error.

Using JSON Web Tokens

We aim to restrict some parts of the application. By doing so, only authenticated users can access them. We don’t want them to need to authenticate for every request. Instead, we need a way to let the users indicate that they have already logged in successfully.

A simple way to do so is to use JSON Web Tokens. JWT is a string that is created on our server using a secret key, and only we can decode it. We want to give it to the user upon logging in so that it can be sent back on every request. If the token is valid, we can trust the identity of the user.

The first thing to do is to add two new environment variables:   and  .

We can use any string as a JWT secret key. It is important to keep it secret and not to share it. We use it to encode and decode tokens in our application.

We describe our expiration time in seconds to increase security. If someone’s token is stolen, the attacker has access to the application in a similar way to having a password. Due to the expiry time, the issue is partially dealt with because the token will expire.

app.module.ts

Generating tokens

In this article, we want the users to store the JWT in cookies. It has a certain advantage over storing tokens in the web storage thanks to the HttpOnly directive. It can’t be accessed directly through JavaScript in the browser, making it more secure and resistant to attacks like cross-site scripting.

If you want to know more about cookies, check oout Cookies: explaining document.cookie and the Set-Cookie header

Now, let’s configure the  .

authentication/authentication.module.ts

Thanks to that, we can now use  in our .

authentication/authentication.service.ts

authentication/tokenPayload.interface.ts

We need to send the token created by the   method when the user logs in successfully. We do it by sending the Set-Cookie header. To do so, we need to directly use the   object.

When the browser receives this response, it sets the cookie so that it can use it later.

Receiving tokens

To be able to read cookies easily we need the  .

main.ts

Now, we need to read the token from the Cookie header when the user requests data. To do so, we need a second passport strategy.

authentication/jwt.strategy.ts

There are a few notable things above. We extend the default JWT strategy by reading the token from the cookie.

When we successfully access the token, we use the id of the user that is encoded inside. With it, we can get the whole user data through the   method. We also need to add it to our  .

users/users.service.ts

Thanks to the   method running under the hood when the token is encoded, we have access to all of the user data.

We now need to add our new  to the  .

authentication/authentication.module.ts

Requiring authentication from our users

Now, we can require our users to authenticate when sending requests to our API. To do so, we first need to create our  .

authentication/jwt-authentication.guard.ts

Now, we can use it every time we want our users to authenticate before making a request. For example, we might want to do so, when creating posts through our API.

posts/posts.controller.ts

Logging out

JSON Web Tokens are stateless. We can’t change a token to be invalid in a straightforward way. The easiest way to implement logging out is just to remove the token from the browser. Since the cookies that we designed are  , we need to create an endpoint that clears it.

authentication/authentication.service.ts

authentication/authentication.controller.ts

Verifying tokens

One important additional functionality that we need is verifying JSON Web Tokens and returning user data. By doing so, the browser can check if the current token is valid and get the data of the currently logged in user.

Postman authentication endpoint

Summary

In this article, we’ve covered registering and logging in users in NestJS. To implement it, we’ve used bcrypt to hash passwords to secure them. To authenticate users, we’ve used JSON Web Tokens. There are still ways to improve the above features. For example, we should exclude passwords more cleanly. Also, we might want to implement the token refreshing functionality. Stay tuned for more articles about NestJS!

Series Navigation<< API with NestJS #2. Setting up a PostgreSQL database with TypeORMAPI with NestJS #4. Error handling and data validation >>
Subscribe
Notify of
guest
57 Comments
Oldest
Newest Most Voted
Inline Feedbacks
View all comments
Eduardo Tavares
Eduardo Tavares
3 years ago

Hello, I did part 3 and when executing the command yarn start: dev, and returned the following error:
node_modules/@types/express/index.d.ts: 59: 29 – error TS2694: Namespace ‘serveStatic’ has no exported member ‘RequestHandlerConstructor’. var static: serveStatic.RequestHandlerConstructor <Response>; I didn’t understand how I could overcome this error, could you give me any tips?

Martin
Martin
3 years ago

I’ve got the same error when I updated @types/express from 4.17.8 to 4.17.9.

npm i -D @types/express@4.17.8

helped for me

Hossein
Hossein
3 years ago

Hi there, I was wondering what should I do, if I need to have different roles for my users?
Like Admins, Subscribers, and normal users?
I created my authentication just like this post. But I need roles too.
I think it would be great if you write another post about it (:

Thanks.

Last edited 3 years ago by Hossein
Wesley Rocha
3 years ago
Reply to  Hossein

In my opinion, the official docs do a good job explaining what you are looking for (authorization): https://docs.nestjs.com/security/authorization

marco
marco
2 years ago
Reply to  Hossein

Create a loginDto class and add a param to login method in AuthenticationController:

Some Guy on the internet
Some Guy on the internet
3 years ago

Thank you for an excellent series!

As a newcomer to Nest, I ran into one small problem that actually blocked me for a few hours while following these steps — not wiring up the AuthenticationModule in the AppModule. The error messages revolved around strategies not being found.

Onivaldo
Onivaldo
3 years ago

Awesome Serie!!! Helped me a lot.

Gustavo
Gustavo
3 years ago

I’m trying. run only the authentication part. With a bunch of data and I came across the error:
[Nest] 101  – 03/01/2021, 8:05:01 PM  [ExceptionHandler] JwtStrategy requires a secret or key 

Pratik Kulkarni
3 years ago
Reply to  Marcin Wanago

I have it defined in the .env, and I’m still getting the error. Do you might be knowing the reason?

Schtern Mujiik
Schtern Mujiik
3 years ago

Look closer at how your ConfigService is working and compare it to what Marcin implemented. May be this will help.
Or you could just try access environment variables through process.env

Amine
Amine
2 years ago

I had the same issue and solved it by checking if env variables are well retrieved, try to debug it in main.ts and see if you’re getting the right value, otherwise you might need to check your configModule if well imported…

DNK
DNK
1 year ago
Reply to  Gustavo

use import { Strategy } from ‘passport-local’;

infotech
infotech
3 years ago

you miss somthing here
logIn(@Req() request:any, @Res({ passthrough: true }) response:Response):Observable<User>
If you want to leave the response handling logic to the framework, remember to set the 
passthrough
 option to true

Dave
Dave
3 years ago

Hi! Where is the definition of the RegisterDto?

audiobookning
audiobookning
3 years ago
Reply to  Dave
Last edited 3 years ago by audiobookning
Wesley Rocha
3 years ago

Hello,

How can I get the correct payload typings when generating a Swagger doc for this controller?

For example, this is how my POST /authentication/login looks like on swagger (see screenshot). Username and password should be listed as parameters, but they don’t. How do you suggest to fix that? Thanks in advance.

audiobookning
audiobookning
3 years ago
Reply to  Wesley Rocha

It seem that you should read this: https://docs.nestjs.com/openapi/types-and-parameters

Achref
Achref
3 years ago

Hello, thank you for the amazing series. I need help regarding authenticating another entity ‘company’ using the local strategy? I tried creating another module and passing a name for the PassportStrategy class but still i get ‘wrong credentials’ message. Any help would be appreciated as this is an urgent matter. Thanks !

audiobookning
audiobookning
3 years ago
Reply to  Achref

 tried creating another module 

I don’t really understand about what module, you are talking about? In the article repo code, you can see that there is only one auth module. And in it, the different strategies are simply imported.
Assuming that you did everything right, and given the ‘wrong credentials’ message that you mentioned, i would ask, what route are you querying and what guard is there? Did you create a new one for your new strategy and are you using it in that route?

Achref
Achref
3 years ago
Reply to  audiobookning

I created another module to authenticate company. still get this error when i try to login: [ExceptionsHandler] Cannot read property ‘id’ of undefined.

any ideas? 

audiobookning
audiobookning
3 years ago

Here are some of my reflection when reading this article.
In the validate method of the JwtStrategy, you get the user data by making a call to the DB. That is basically undermining some of the advantages of using the JWT.
The responsibility of the validate method is to return some user data that passport will append to the request object (and can, for example, latter be used by a currentUser decorator).
With a simple auth JWT, the normal would be to just return the JWT payload. And if there is need for more data in, for example the currenteUser decorator, just add it to the payload at the time of creating the JWT.
That call to the DB in the validate would normally only done in a passport JWT refresh strategy.

Serg
Serg
3 years ago

Hi, I believe that you should revoke the token on logout)

Ignacio Ruiz
Ignacio Ruiz
3 years ago

how do i catch it in angular? any link?

Constantine
Constantine
3 years ago

Hi
There is a problem with the auth controller

Error TS2339: Property ‘send’ does not exist on type ‘Response’.

Sam
Sam
3 years ago
Reply to  Marcin Wanago

Property ‘setHeader’ does not exist on type ‘Response’

 @UseGuards(JwtAuthenticationGuard)
  @Post(‘log-out’)
  async logOut(@Req() request: RequestWithUser, @Res() response: Response) {
    response.setHeader(‘Set-Cookie’, this.authenticationService.getCookieForLogOut());
    return response.sendStatus(200);
  }

Last edited 3 years ago by Sam
Michael
Michael
3 years ago
Reply to  Sam

import {Response} from ‘express’

baloo
baloo
2 years ago
Reply to  Sam

Or do response.req.setHeader ^^

Sam
Sam
3 years ago

Hi, I’ve got the same code you have and dont know how to fix this error
Unknown authentication strategy “jwt”

Lee
Lee
3 years ago
Reply to  Sam

I have came across same error, did u solve it ?

Schtern Mujiik
Schtern Mujiik
3 years ago
Reply to  Lee

May be this will help:
export class JwtStrategy extends PassportStrategy(Strategy, jwt) {
Try name your strategy explicitly.

Antarian
Antarian
1 year ago
Reply to  Lee

Had same error, forgot to import AuthenticationModule in AppModule.

Nikolai
Nikolai
2 years ago

How can I supplement the create() method to prohibit the creation of a user if the email already exists?

Keston Timson
Keston Timson
2 years ago

I keep on getting this error when I try to test the log-in and log-out endpoints with postman

Error [ERR_INTERNAL_ASSERTION]: This is caused by either a bug in Node.js or incorrect usage of Node.js internals.

Please open an issue with this stack trace at https://github.com/nodejs/node/issues

  at assert (internal/assert.js:14:11)
  at ServerResponse.detachSocket (_http_server.js:234:3)
  at resOnFinish (_http_server.js:792:7)
  at ServerResponse.emit (events.js:400:28)
  at onFinish (_http_outgoing.js:792:10)
  at callback (internal/streams/writable.js:513:21)
  at afterWrite (internal/streams/writable.js:466:5)
  at afterWriteTick (internal/streams/writable.js:453:10)
  at processTicksAndRejections (internal/process/task_queues.js:81:21)

Mario Valderrama
Mario Valderrama
2 years ago
Reply to  Keston Timson

Same problem here!

MichaelYao
MichaelYao
2 years ago
Reply to  Keston Timson

Check your login controller. It doesn’t need have async and no need to return response. Just response.send() is enough.

MichaelYao
MichaelYao
2 years ago
Reply to  Keston Timson

Or use async and return user both

markflerko
2 years ago

Amazing series, but why I can’t get the user after log in?
Wasted few hours on debug but still nothing, will appreciate any help.
see postman screenshot

Thanks!

markflerko
2 years ago
Reply to  Marcin Wanago

Hah, Thank you a lot. I forgot ‘s’ in expiresIn

Ahmad
Ahmad
2 years ago

Glad to be the part of this Series, it helped me alot.
here anyone can make me solve this error

@HttpCode(200)
@UseGuards(LocalAuthenticationGuard)
@Get(‘LogIn’)
async login (@Req() req:RequestWithUser,@Res() res:Response){
const {user} = req
const myCookie= this.authenticationService.getCookieWithJwtToken(user.id)
res.setHeader(‘Set-Cookie’,myCookie) // here is the issue
user.password=undefined
return res.send(user)
}

code is clean but when I hit the route i face the following issue:

ERROR [ExceptionsHandler] Invalid character in header content [“Set-Cookie”]
Nest 340901

Ahmad
Ahmad
2 years ago
Reply to  Marcin Wanago

myCookie value is :
/*
public getCookieWithJwtToken(UserId:number){
const payload:PayLoad = {UserId}
const token = this.jwtservice.sign(payload);

return Authentication=${token}; HttpOnly; Path=/; Max-Age= ${process.env.JWT_EXPIRATION_TIME};
}

Last edited 2 years ago by Ahmad
Hoang Minh
Hoang Minh
10 months ago
Reply to  Ahmad

You don’t “enter” line. It should be “Authentication… Path=/; Max-Age=${pro..”

R2 Dev
R2 Dev
1 year ago

I’m trying. run only the authentication part. But i got the message said. Cannot GET /authentication

Vladimir
Vladimir
1 year ago

Cannot read properties of undefined (reading ‘create’) when trying to register?

letterthing
letterthing
1 year ago
Reply to  Vladimir

I have the same issue. Have you resolved it somehow?

Vennis
Vennis
1 year ago
Reply to  letterthing

Add @Injectable() to your AuthenticationService class.

Mario Cares
Mario Cares
1 year ago

Can it be consumed by a react app ?
I can log in, but always got 401 on every endpoint that is guarded…

Ilkin
Ilkin
1 year ago

Hi there. This is the best article about Authentication Strategies in NestJS on the entire web. But I have a question. Should I use both local and jwt strategies? What is the difference between them?

Ulrich
Ulrich
1 year ago

Hi, Thanks you for this article. I faced some problem, after login successfully, and try to get authenticate User and get 401 Unauthorized, don’t understand why !!!

Ulrich
Ulrich
1 year ago
Reply to  Ulrich

It’s ok, i resolved my problem. Thanks you 🙏🙏

perseverance50k
perseverance50k
8 months ago
Reply to  Ulrich

Hi! Hi did you resolve this problem?

Harish Krishnan
Harish Krishnan
2 months ago

In case somebody is having issue it might be because you import Strategy from “passport-local” rather than “passport-jwt” in jwt.strategy.ts file

Dagwal
4 months ago

The registration and login works well, it returns a cookie, in response header With Set-Cookie. but when i try to logout with provided token, it says unauthorized