Create Authentication system with NodeJS, ExpressJS, TypeScript and Jest E2E testing -- PART 2.

·

10 min read


JWT Middleware

Let's create the middleware to handle the authentications. Create this file src/middlewares/jwt.middleware.ts.

This middleware takes the Authorization header (Bearer [token]) and get the token. jsonwebtoken packages handles the process of validating the token behind the scene.

If the token doesn't exist, the access is denied (401 Error). Else, we validated the token, the package handles the token and return an error if it's expired or invalid (in case not expired, then it's invalid). Otherwise, it's valid, and we move on to the next handler (The route handler probably).

import { Response, NextFunction } from 'express';
import jwt, { TokenExpiredError } from 'jsonwebtoken';
import { ResponseHandler } from '@/utils/responseHandler';
import appConfig from '@/config/app.config';

export function authenticateJWT(req: IRequest, res: Response, next: NextFunction) {
  const authHeader = req.headers.authorization;

  if (authHeader) {
    const token = authHeader.split(' ')[1];
    jwt.verify(token, appConfig.jwt.secret, (err, user) => {
      if (err || !user) {
        if (err instanceof TokenExpiredError) {
          const resBody = ResponseHandler.Unauthorized('Unauthenticated');
          res.status(resBody.error!.code).json(resBody);
        } else {
          const resBody = ResponseHandler.Forbidden('Access forbidden: Invalid token');
          res.status(resBody.error!.code).json(resBody);
        }
        return;
      }
      req.user = user;
      next();
    });
  } else {
    const resBody = ResponseHandler.Unauthorized('Access denied: No token provided');
    res.status(resBody.error!.code).json(resBody);
  }
}

The IRequest interface is a globally declared interface, it extends default Express Request, and add user payload (userId). We can use this userId in other route handler to get the logged in user. Create src/index.d.ts

// Refrence files, to import the types from other definition files. We will be importing them here instead of importing the types in each file.

/// <reference path="./schemas/auth/auth.schema.d.ts" />
/// <reference path="./types/auth.d.ts" />

import { Request as ExpressRequest } from 'express';

declare global {
  interface IRequest extends ExpressRequest {
    user?: any;
  }
}

Auth Router

Router

As I mentioned in the Architecture section, the code will be separated, the router will only handle routing and listening. The logic will be in repositories files. Let's create the file src/routes/v1/auth.route.ts

import { NextFunction, Request, Response, Router } from 'express';
import { validate } from '@/middlewares/validateRequest.middleware';
import { AuthRepository } from '@/repositories/auth.repo';
import { authenticateJWT } from '@/middlewares/jwt.middleware';
import HttpStatusCode from '@/utils/HTTPStatusCodes';
import { AuthZODSchema } from '@/schemas/auth.schema';

const AuthRoutes = Router();

AuthRoutes.post(
  '/login',
  validate(AuthZODSchema.authSchema),
  async (req: Request, res: Response, next: NextFunction) => {
    try {
      const body: TAuthSchema = req.body;
      const resBody = await AuthRepository.loginUser(body);
      res.status(resBody.error ? resBody.error.code : HttpStatusCode.OK).json(resBody);
      next();
    } catch (err) {
      next(err);
    }
  }
);

AuthRoutes.post(
  '/refresh-token',
  validate(AuthZODSchema.refreshTokenSchema),
  async (req: Request, res: Response, next: NextFunction) => {
    try {
      const body: TRefreshTokenSchema = req.body;
      const resBody = await AuthRepository.refreshToken(body);
      res.status(resBody.error ? resBody.error.code : HttpStatusCode.OK).json(resBody);
      next();
    } catch (err) {
      next(err);
    }
  }
);

AuthRoutes.post(
  '/register',
  validate(AuthZODSchema.registerSchema),
  async (req: Request, res: Response, next: NextFunction) => {
    try {
      const body: TRegisterSchema = req.body;
      const resBody = await AuthRepository.createUser(body);
      res.status(resBody.error ? resBody.error.code : HttpStatusCode.OK).json(resBody);
      next();
    } catch (err) {
      next(err);
    }
  }
);

AuthRoutes.post(
  '/forget-password',
  validate(AuthZODSchema.forgetPasswordSchema),
  async (req: Request, res: Response, next: NextFunction) => {
    try {
      const body: TForgetPasswordSchema = req.body;
      const resBody = await AuthRepository.forgotPassword(body);
      res.status(resBody.error ? resBody.error.code : HttpStatusCode.OK).json(resBody);
      next();
    } catch (err) {
      next(err);
    }
  }
);

AuthRoutes.post(
  '/reset-password',
  validate(AuthZODSchema.resetPasswordSchema),
  async (req: Request, res: Response, next: NextFunction) => {
    try {
      const body: TResetPasswordSchema = req.body;
      const resBody = await AuthRepository.resetPassword(body);
      res.status(resBody.error ? resBody.error.code : HttpStatusCode.OK).json(resBody);
      next();
    } catch (err) {
      next(err);
    }
  }
);

AuthRoutes.post(
  '/update-password',
  authenticateJWT,
  validate(AuthZODSchema.updatePasswordSchema),
  async (req: IRequest, res: Response, next: NextFunction) => {
    try {
      const body: TUpdatePasswordSchema = req.body;
      // We get the userId from the request. It's passed by "authenticateJWT"
      const resBody = await AuthRepository.updatePassword(body, req.user.userId);
      res.status(resBody.error ? resBody.error.code : HttpStatusCode.OK).json(resBody);
      next();
    } catch (err) {
      next(err);
    }
  }
);

AuthRoutes.post(
  '/confirm-update-password',
  validate(AuthZODSchema.validateUserSchema),
  async (req: Request, res: Response, next: NextFunction) => {
    try {
      const body: TValidateUserSchema = req.body;
      const resBody = await AuthRepository.confirmUpdatePassword(body);
      res.status(resBody.error ? resBody.error.code : HttpStatusCode.OK).json(resBody);
      next();
    } catch (err) {
      next(err);
    }
  }
);

AuthRoutes.post(
  '/verify-user',
  validate(AuthZODSchema.validateUserSchema),
  async (req: Request, res: Response, next: NextFunction) => {
    try {
      const body: TValidateUserSchema = req.body;
      const resBody = await AuthRepository.verifyUser(body);
      res.status(resBody.error ? resBody.error.code : HttpStatusCode.OK).json(resBody);
      next();
    } catch (err) {
      next(err);
    }
  }
);

export default AuthRoutes;

This router depends on 3 other files; The types and ZOD schema, the ZOD validator middleware, and the Auth Repo.

ZOD Validator Middleware.

All middlewares are injected in src/middlewares directory. So, let's create the middleware. Create validateRequest.middleware.ts file.

import { Request, Response, NextFunction } from 'express';
import { ZodError, ZodSchema } from 'zod';
import { ResponseHandler } from '@/utils/responseHandler';

function parseZodErrors(errors: ZodError) {
  return errors.errors.map((err) => `${err.path.join(', ')}: ${err.message}`);
}

export function validate(schema: ZodSchema) {
  return (req: Request, res: Response, next: NextFunction) => {
    try {
      schema.parse(req.body);
      next();
    } catch (error: any) {
      const resBody = ResponseHandler.InvalidBody({
        message: 'Validation Error',
        errors: parseZodErrors(error),
      });
      res.status(resBody.error!.code).json(resBody);
    }
  };
}

This middleware is simple. ZOD validates the schema passed through the middleware in Router handler. If it's valid, move to next handler. Otherwise, return a 422 response with the ZOD error to know which property isn't valid and what's the error. That's the role of parseZodErrors. It parses the ZodErrors to the ApiResponseBody message property.

ZOD Schemas

I used a class with static ZOD schema object to have a clean import section. And used .d.ts to declare the types. So, we got two files; the schema objects file, and type definitions file. Create a file src/schemas/auth/auth.schema.ts

import { z } from 'zod';

export class AuthZODSchema {
  static readonly authSchema = z.object({
    email: z.string().email(),
    password: z.string().min(8),
    type: z.enum(['DOCTOR', 'PATIENT', 'ADMIN']),
  });
  static readonly registerSchema = z.object({
    name: z.string().min(2),
    phone: z.string().refine((phone) => /^\+\d{10,15}$/.test(phone), 'Invalid phone number'),
    email: z.string().email(),
    password: z.string().min(8),
    type: z.enum(['DOCTOR', 'PATIENT']),
  });
  static readonly refreshTokenSchema = z.object({
    refreshToken: z.string().uuid(),
  });
  static readonly forgetPasswordSchema = z.object({
    email: z.string().email(),
    type: z.enum(['DOCTOR', 'PATIENT', 'ADMIN']),
  });
  static readonly resetPasswordSchema = z.object({
    newPassword: z.string().min(8),
    token: z.string().uuid(),
  });
  static readonly validateUserSchema = z.object({
    token: z.string().uuid(),
  });
  static readonly updatePasswordSchema = z.object({
    oldPassword: z.string().min(8),
    newPassword: z.string().min(8),
    type: z.enum(['DOCTOR', 'PATIENT', 'ADMIN']),
  });
}

Now create the definition file src/schemas/auth/auth.schema.d.ts

import { AuthZODSchema } from './auth.schema';
import { z } from 'zod';

declare global {
  type TAuthSchema = z.infer<typeof AuthZODSchema.authSchema>;
  type TRegisterSchema = z.infer<typeof AuthZODSchema.registerSchema>;
  type TForgetPasswordSchema = z.infer<typeof AuthZODSchema.forgetPasswordSchema>;
  type TResetPasswordSchema = z.infer<typeof AuthZODSchema.resetPasswordSchema>;
  type TUpdatePasswordSchema = z.infer<typeof AuthZODSchema.updatePasswordSchema>;
  type TValidateUserSchema = z.infer<typeof AuthZODSchema.validateUserSchema>;
  type TRefreshTokenSchema = z.infer<typeof AuthZODSchema.refreshTokenSchema>;
}

This definition is already declared in index.d.ts. We won't need to import each type in each file we use.

Setting up prisma

Before we start with the repository, we need to create the prisma models and migrate them. Create the schema in prisma/schema.prisma

generator client {
  provider = "prisma-client-js"
}

datasource db {
  provider = "postgresql"
  url      = env("DATABASE_URL")
}

enum UserType {
  DOCTOR
  PATIENT
  ADMIN
}

model User {
  id                  String               @id @unique @default(uuid())
  name                String
  email               String               @unique
  phone               String               @unique
  password            String
  verifiedEmail       Boolean              @default(false)
  userType            UserType
  refreshToken        RefreshToken[]
  resetPasswordToken  ResetPasswordToken?
  updatePasswordToken UpdatePasswordToken?
  verifyEmailToken    VerifyEmailToken?
  createdAt           DateTime             @default(now())
  updatedAt           DateTime             @updatedAt

  @@map("users")
}

model ResetPasswordToken {
  id        String   @id @unique @default(uuid())
  token     String   @unique
  expiresAt DateTime
  userId    String   @unique
  user      User     @relation(fields: [userId], references: [id], onDelete: Cascade)
  createdAt DateTime @default(now())
  updatedAt DateTime @updatedAt

  @@map("reset_password_tokens")
}

model UpdatePasswordToken {
  id          String   @id @unique @default(uuid())
  token       String   @unique
  newPassword String
  expiresAt   DateTime
  userId      String   @unique
  user        User     @relation(fields: [userId], references: [id], onDelete: Cascade)
  createdAt   DateTime @default(now())
  updatedAt   DateTime @updatedAt

  @@map("update_password_tokens")
}

model VerifyEmailToken {
  id        String   @id @unique @default(uuid())
  token     String   @unique
  expiresAt DateTime
  userId    String   @unique
  user      User     @relation(fields: [userId], references: [id], onDelete: Cascade)
  createdAt DateTime @default(now())
  updatedAt DateTime @updatedAt

  @@map("verify_email_tokens")
}

model RefreshToken {
  id        String   @id @default(uuid())
  token     String   @unique
  userId    String
  user      User     @relation(fields: [userId], references: [id], onDelete: Cascade)
  expiresAt DateTime
  createdAt DateTime @default(now())

  @@map("refresh_tokens")
}

Now we migrate our schema. Make sure you created the DB and added the URL in .env

npx prisma migrate dev --name init

And because we will handle also the TEST DB, I created a script to migrate the migrations in that DB. Create the file in prisma/scripts/migrate.ts.

import { execSync } from 'child_process';
import { config } from 'dotenv';
config()

async function runMigrations() {
  try {
    console.log('Running migrations...');
    execSync(`cross-env DATABASE_URL=${process.env.TEST_DATABASE_URL} npx prisma migrate deploy`, {
      stdio: 'inherit',
    });
    console.log('Migrations completed successfully.');
  } catch (error) {
    console.error('Error running migrations:', error);
    process.exit(1);
  }
}

runMigrations()

Install cross-env before, and make sure you installed ts-node like mentioned before.

npm i -D cross-env

After that, you can run the script

npx ts-node prisma/scripts/migrate.ts