Developing a Node.js Authentication API with TypeScript: Comprehensive Guide
Introduction
Type safety and code quality are becoming increasingly important when developing modern web applications. In this tutorial, we will learn TypeScript’s powerful features through a real project. While developing a Todo application with JWT and Google OAuth authentication, we will implement TypeScript’s core concepts and best practices. You can access the complete project at this GitHub repository.
This project will provide you with:
- Practice in writing secure code with TypeScript
- Experience in designing a modern REST API
- Authentication and Authorization implementation
- Using TypeScript with MongoDB
- Applying Clean Architecture principles
TypeScript Features and Project Structure
We’ll develop a secure and scalable API using TypeScript’s core features. Let’s explain each feature with examples from our actual project code:
1. TypeScript Basics (Basics)
TypeScript’s basic building blocks are used in our project like this:
// src/config/env.ts'de Tip Tanımlamaları
const PORT: number = Number(process.env.PORT) || 3000;
const JWT_EXPIRES_IN: string = '1d';
// src/middleware/auth.middleware.ts'de Type Assertion
const decoded = jwt.verify(token, JWT_SECRET) as IJwtPayload;
// Burada JWT'den gelen veriyi IJwtPayload tipine dönüştürüyoruz
// src/interfaces/user.interface.ts'de Literal Types
type UserRole = 'user' | 'admin';
// User modelinde kullanıcı rollerini sadece bu iki değerle sınırlıyoruz
Projedeki Kullanım Örnekleri:
PORT
tanımısrc/index.ts
‘de server başlatırken kullanılıyor- Type assertion
auth.middleware.ts
‘de JWT doğrulamasında kullanılıyor - UserRole tipi
IUser
interface’inde kullanıcı rolünü kısıtlamak için kullanılıyor
2. Functions
TypeScript’te functions are typed like this:
// src/services/auth.service.ts'de Method Signatures
interface IAuthService {
login(credentials: IUserLogin): Promise<{ user: IUser; token: string }>;
register(userData: IUserRegistration): Promise<IUser>;
}
// src/controllers/auth.controller.ts'de Implementation
public async login(req: Request, res: Response): Promise<void> {
const { email, password } = req.body;
const result = await this.authService.login({ email, password });
// ...
}
Projedeki Kullanım Örnekleri:
IAuthService
interface’iauth.service.ts
‘de servis implementasyonunu tanımlıyor- Controller’lardaki tüm handler functions use Request and Response types
- All async functions are typed with Promise return type
3. Object Types
We model complex data structures with object types in our project:
// src/config/database.ts'de Configuration Types
type DatabaseConfig = {
uri: string;
options: {
useNewUrlParser: boolean;
useUnifiedTopology: boolean;
};
};
// src/controllers/todo.controller.ts'de Request Types
interface ITodoCreate {
title: string;
description?: string; // Optional property example
}
Projedeki Kullanım Örnekleri:
DatabaseConfig
type defines MongoDB connection settingsITodoCreate
interface is used for request body validation in todo creation endpoint- Optional properties allow partial updates in todo updates
4. Interfaces
Interfaces are used in our project both for type definition and for contracts:
// src/interfaces/base.interface.ts'de Base Interface
interface IBaseEntity {
_id: string;
createdAt: Date;
updatedAt: Date;
}
// src/interfaces/user.interface.ts'de Interface Extension
interface IUser extends IBaseEntity {
email: string;
password?: string;
name: string;
role: UserRole;
}
Projedeki Kullanım Örnekleri:
IBaseEntity
defines common fields for all MongoDB modelsIUser
interface defines User model schema and methods- Interfaces ensure type safety in mongoose model definitions
5. TypeScript Compiler
We configure TypeScript compiler specifically for our project:
{
"compilerOptions": {
"target": "es2016",
"module": "commonjs",
"strict": true,
"esModuleInterop": true,
"skipLibCheck": true,
"forceConsistentCasingInFileNames": true,
"outDir": "./dist",
"rootDir": "./src"
}
}
Compiler Settings’s Importance:
strict
: Enables strict type checkingtarget
: Allows us to use modern JavaScript featuresmodule
: Uses Node.js compatible module system
6. Classes
We implement OOP principles with TypeScript classes:
// src/services/base.service.ts'de Abstract Base Class
abstract class BaseService<T extends IBaseEntity> {
constructor(protected model: Model<T>) {}
abstract create(data: Partial<T>): Promise<T>;
async findById(id: string): Promise<T | null> {
return this.model.findById(id);
}
}
// src/services/todo.service.ts'de Class Implementation
class TodoService extends BaseService<ITodo> {
async create(data: ICreateTodo): Promise<ITodo> {
return this.model.create(data);
}
async markAsCompleted(id: string): Promise<ITodo | null> {
return this.model.findByIdAndUpdate(id, { completed: true });
}
}
Why This Feature?
- Abstract classes enforce common behaviors
- Inheritance reduces code repetition
- Organizes service layer
Projedeki Kullanım Örnekleri:
BaseService
defines basic CRUD operations for all servicesTodoService
andAuthService
extend this base class to add their own specific methods- Abstract methods ensure each service must implement its own create method
7. Generics
We use generics in our project like this:
// src/services/base.service.ts'de Generic Service
class CrudService<T extends IBaseEntity> {
async findOne(filter: FilterQuery<T>): Promise<T | null> {
return this.model.findOne(filter);
}
}
// src/utils/response.ts'de Generic Response Handler
function createResponse<T>(success: boolean, message: string, data?: T): IApiResponse<T> {
return { success, message, data };
}
// src/utils/error.ts'de Generic Error Handler
class ApiError<T = unknown> extends Error {
constructor(public statusCode: number, message: string, public data?: T) {
super(message);
}
}
Why This Feature?
- Keeps type safety while writing reusable code
- Creates functions that work with different data types
- Creates flexible structures with type parameters
Projedeki Kullanım Örnekleri:
CrudService
works with different model types (User, Todo, etc.)createResponse
creates consistent API responses for all data structuresApiError
provides customizable error handling for different error types
8. Type Narrowing
We safely perform runtime type checking and narrowing in TypeScript:
// src/utils/error.ts'de Type Guards
function isError(error: unknown): error is Error {
return error instanceof Error;
}
// src/middleware/error.middleware.ts'de Error Handling
function handleError(error: unknown): IApiResponse<null> {
if (isError(error)) {
return createResponse(false, error.message);
}
if (typeof error === 'string') {
return createResponse(false, error);
}
return createResponse(false, 'Unknown error occurred');
}
// src/types/error.types.ts'de Discriminated Unions
type ValidationError = {
type: 'validation';
fields: { [key: string]: string };
};
type AuthError = {
type: 'auth';
message: string;
};
type AppError = ValidationError | AuthError;
// src/utils/error-handler.ts'de Error Type Handling
function handleAppError(error: AppError) {
switch (error.type) {
case 'validation':
return error.fields;
case 'auth':
return error.message;
}
}
Why This Feature?
- Keeps runtime type safety
- Improves error handling
- Works with Union types correctly
Projedeki Kullanım Örnekleri:
isError
type guard in middleware detects error type correctly- Error handling middleware distinguishes different error types
- Discriminated unions allow handling validation and auth errors separately
Project Summary
Our API will include the following features:
User Management
- Registration and Login
- JWT Authentication
- Google OAuth Integration
- Role-based authorization
Todo Operations
- Create, read, update, delete todos
- User-specific todos
- Todo status changes
Security and Validation
- Input validation
- Route protection
- Error handling
Project Structure
Our project is organized as follows:
src/
├── config/ # Configuration files
├── controllers/ # HTTP request handlers
├── interfaces/ # TypeScript interfaces
├── middleware/ # Express middleware
├── models/ # Mongoose models
├── routes/ # API routes
├── services/ # Business logic
├── utils/ # Helper functions
└── index.ts # Application entry point
Developing the Project with TypeScript
1. Project Setup and TypeScript Configuration
First step is to integrate TypeScript into our project:
mkdir nodejs-typescript-auth
cd nodejs-typescript-auth
npm init -y
npm install typescript ts-node @types/node --save-dev
Dependencies
Let’s install necessary packages for our project:
# Main dependencies
npm install express mongoose dotenv jsonwebtoken bcrypt passport passport-google-oauth20 passport-jwt cors
# Type definitions
npm install @types/express @types/mongoose @types/jsonwebtoken @types/bcrypt @types/passport @types/passport-google-oauth20 @types/passport-jwt @types/cors --save-dev
TypeScript Configuration
{
"compilerOptions": {
"target": "es2016",
"module": "commonjs",
"outDir": "./dist",
"rootDir": "./src",
"strict": true,
"esModuleInterop": true,
"skipLibCheck": true,
"forceConsistentCasingInFileNames": true
}
}
2. Defining Data Models
User Model
Let’s define our User model with TypeScript interfaces:
// src/interfaces/user.interface.ts
// Base interface - basic user properties
interface IBaseUser {
email: string;
name: string;
}
// Main user interface - includes all properties
interface IUser extends IBaseUser {
password?: string; // Optional: Google OAuth users may not have a password
googleId?: string; // Optional: Only for users who sign in with Google
role: 'user' | 'admin'; // Union type to limit roles
comparePassword(candidatePassword: string): Promise<boolean>;
}
// Required fields for registration
interface IUserRegistration {
email: string;
password: string;
name: string;
}
// Required fields for login
interface IUserLogin {
email: string;
password: string;
}
Todo Model
Let’s define the interfaces needed for todo operations:
// src/interfaces/todo.interface.ts
interface ITodo {
title: string;
description?: string;
completed: boolean;
user: string; // Reference: User ID
createdAt: Date;
updatedAt: Date;
}
// Required fields for todo creation
interface ICreateTodo {
title: string;
description?: string;
}
// Optional fields for todo updates
interface IUpdateTodo {
title?: string;
description?: string;
completed?: boolean;
}
3. Service Layer Implementation
Base Service
Let’s create a generic base service to reduce code repetition:
// src/services/base.service.ts
abstract class BaseService<T> {
constructor(protected model: Model<T>) {}
async findById(id: string): Promise<T | null> {
return this.model.findById(id);
}
async findOne(filter: FilterQuery<T>): Promise<T | null> {
return this.model.findOne(filter);
}
async find(filter: FilterQuery<T>): Promise<T[]> {
return this.model.find(filter);
}
}
Auth Service
Service to handle authentication operations:
// src/services/auth.service.ts
class AuthService extends BaseService<IUser> {
public async register(userData: IUserRegistration): Promise<IUser> {
const existingUser = await this.findOne({ email: userData.email });
if (existingUser) {
throw new Error('This email is already in use');
}
const user = await this.model.create(userData);
return user;
}
public async login(loginData: IUserLogin): Promise<{ user: IUser; token: string }> {
const user = await this.findOne({ email: loginData.email });
if (!user || !(await user.comparePassword(loginData.password))) {
throw new Error('Invalid credentials');
}
return {
user,
token: this.generateToken(user),
};
}
private generateToken(user: IUser): string {
return jwt.sign({ id: user._id, email: user.email, role: user.role }, process.env.JWT_SECRET!, { expiresIn: '1d' });
}
}
Todo Service
Service to handle todo operations:
// src/services/todo.service.ts
class TodoService extends BaseService<ITodo> {
public async getAllTodos(userId: string): Promise<ITodo[]> {
return this.find({ user: userId });
}
public async createTodo(todoData: ICreateTodo, userId: string): Promise<ITodo> {
return this.model.create({
...todoData,
user: userId,
completed: false,
});
}
public async updateTodo(todoId: string, todoData: Partial<ITodo>, userId: string): Promise<ITodo | null> {
return this.model.findOneAndUpdate({ _id: todoId, user: userId }, todoData, { new: true });
}
}
4. Middleware Implementation
TypeScript’s safe middleware writing:
// src/middleware/auth.middleware.ts
// Request type extension
declare global {
namespace Express {
interface Request {
user?: IUser;
}
}
}
export const isAuthenticated = async (req: Request, res: Response, next: NextFunction): Promise<void> => {
try {
const token = req.headers.authorization?.split(' ')[1];
if (!token) {
throw new Error('Token not found');
}
const decoded = jwt.verify(token, process.env.JWT_SECRET!) as IJwtPayload;
const user = await UserModel.findById(decoded.id);
if (!user) {
throw new Error('User not found');
}
req.user = user;
next();
} catch (error) {
res.status(401).json({
success: false,
message: 'Authorization error',
error: error instanceof Error ? error.message : 'Unknown error',
});
}
};
5. Controller Layer
TypeScript’s safe controller:
// src/controllers/todo.controller.ts
class TodoController {
constructor(private todoService: TodoService) {}
public async getAllTodos(req: Request, res: Response): Promise<void> {
try {
const todos = await this.todoService.getAllTodos(req.user!._id);
res.status(200).json({
success: true,
message: 'Todos fetched successfully',
data: todos,
});
} catch (error) {
res.status(500).json({
success: false,
message: 'Error fetching todos',
error: error instanceof Error ? error.message : 'Unknown error',
});
}
}
}
API Endpoints
Auth Endpoints
POST /api/auth/register
- Request Body: { email: string, password: string, name: string }
- Response: { success: boolean, message: string, data: { user: IUser, token: string } }
POST /api/auth/login
- Request Body: { email: string, password: string }
- Response: { success: boolean, message: string, data: { user: IUser, token: string } }
GET /api/auth/google
- Google OAuth initiation endpoint
GET /api/auth/google/callback
- Google OAuth callback endpoint
Todo Endpoints
GET /api/todos
- Headers: { Authorization: "Bearer ${token}" }
- Response: { success: boolean, message: string, data: ITodo[] }
POST /api/todos
- Headers: { Authorization: "Bearer ${token}" }
- Request Body: { title: string, description?: string }
- Response: { success: boolean, message: string, data: ITodo }
PUT /api/todos/:id
- Headers: { Authorization: "Bearer ${token}" }
- Request Body: { title?: string, description?: string, completed?: boolean }
- Response: { success: boolean, message: string, data: ITodo }
DELETE /api/todos/:id
- Headers: { Authorization: "Bearer ${token}" }
- Response: { success: boolean, message: string }
Best Practices
Type Safety
- Always use specific types
- Avoid the
any
type - Limit value sets with Union types
- Write reusable code with Generic types
Code Organization
- Use separate folders for each layer
- Keep interfaces in relevant domain folders
- Generalize service layer with abstract classes
Error Handling
- Create custom error classes
- Use global error handler
- Standardize error messages
Security
- Store sensitive information in environment variables
- Implement input validation
- Apply rate limiting
- Configure CORS policies correctly
Conclusion
In this project, we learned:
- Writing secure code with TypeScript’s type system
- Implementing OOP principles with TypeScript
- Designing a modern API architecture
- Implementing authentication and authorization
TypeScript provided our project with important advantages such as:
- Compile-time error detection
- Better IDE support
- Self-documenting code
- Maintainability