code examples
code examples
Build NestJS SMS OTP/2FA with MessageBird Verify API 2025
Build production-ready SMS OTP and two-factor authentication in NestJS using MessageBird Verify API. Step-by-step TypeScript tutorial with code examples, error handling, and security best practices.
Build SMS OTP/2FA Authentication with MessageBird Verify API in NestJS
Integrate MessageBird's Verify API into your NestJS application to implement robust SMS-based Two-Factor Authentication (2FA) or phone number verification using One-Time Passwords (OTPs). This guide shows you how to build a secure, scalable backend API that sends OTPs via SMS and verifies user-submitted tokens.
Project Overview and Goals
What You'll Build:
- NestJS application with dedicated endpoints for initiating OTP requests and verifying submitted tokens
- Integration with the MessageBird Verify API via their official Node.js SDK
- Secure API key handling using NestJS configuration modules
- Input validation for phone numbers and tokens
- Robust error handling for common scenarios
Problem This Solves:
Verify user phone numbers and add a 2FA layer to your application. This prevents fraudulent account creation, secures accounts against unauthorized access, and verifies transactions by confirming control over a registered phone number. Common use cases include e-commerce checkout verification, financial transaction confirmation, and account recovery flows.
Technologies You'll Use:
- Node.js: JavaScript runtime environment
- NestJS: Progressive Node.js framework for building efficient, reliable, and scalable server-side applications with modular architecture and dependency injection
- MessageBird: Third-party service provider for sending SMS OTPs via their Verify API
- MessageBird Node.js SDK: Simplifies interaction with the MessageBird API
- TypeScript: Provides static typing for enhanced code quality and maintainability
- (Optional) Prisma: Modern ORM for database interaction (not required for basic OTP flow)
System Architecture:
graph LR
A[User Client (Web/Mobile)] -- 1. Request OTP (POST /otp/send) --> B(NestJS API);
B -- 2. Call OtpService.sendOtp --> C(OtpService);
C -- 3. Call MessageBird SDK (verify.create) --> D(MessageBird Verify API);
D -- 4. Send SMS OTP --> E[User's Phone];
D -- 5. Return Verification ID --> C;
C -- 6. Return Verification ID --> B;
B -- 7. Send Verification ID back --> A;
A -- 8. Submit Token & ID (POST /otp/verify) --> B;
B -- 9. Call OtpService.verifyOtp --> C;
C -- 10. Call MessageBird SDK (verify.verify) --> D;
D -- 11. Return Verification Status --> C;
C -- 12. Return Status --> B;
B -- 13. Send Verification Result back --> A;
style B fill:#f9f,stroke:#333,stroke-width:2px
style C fill:#ccf,stroke:#333,stroke-width:2px
style D fill:#ff9,stroke:#333,stroke-width:2pxPrerequisites:
Install these requirements before starting:
- Node.js: Install LTS version v22 (Active LTS) or v20 (Maintenance LTS) recommended as of 2025. Node.js 18 reaches end-of-life April 2025.
- MessageBird Account: Sign up and obtain API credentials (both test and live keys)
- Knowledge: Basic understanding of TypeScript, NestJS concepts (modules, controllers, services), REST APIs, and authentication flows
- NestJS CLI: Install globally with
npm install -g @nestjs/cli(NestJS v10+ requires Node.js v16 or higher) - (Optional) Docker: For database setup if using Prisma
- (Optional) PostgreSQL: Local or cloud-based database instance
Final Outcome:
Build a functional NestJS API that:
- Accepts a phone number via POST request
- Sends an SMS OTP to that number using MessageBird
- Returns a unique verification ID to the client
- Accepts the verification ID and user-submitted OTP via another POST request
- Verifies the token against the ID using MessageBird
- Returns success or failure status to the client
After successful verification, integrate this into your full authentication flow by issuing JWT tokens or creating user sessions.
1. Setting Up the Project
Initialize a new NestJS project and install the necessary dependencies.
-
Create Your NestJS Project: Run the NestJS CLI command to create a new project named
nestjs-messagebird-otp.bashnest new nestjs-messagebird-otpChoose your preferred package manager (npm or yarn) when prompted.
-
Navigate to Your Project Directory:
bashcd nestjs-messagebird-otp -
Install Required Dependencies: Install these essential packages:
@nestjs/config: Manages environment variablesmessagebird: Official MessageBird Node.js SDK (v4.0.1, last updated 2022 — monitor for alternatives if maintenance becomes a concern)class-validator,class-transformer: Validate request data using DTOs
bash# Using npm npm install @nestjs/config messagebird class-validator class-transformer # Using yarn yarn add @nestjs/config messagebird class-validator class-transformerPackage Versions (as of 2025):
messagebird: v4.0.1 (last published 3 years ago — check for security vulnerabilities before production deployment)@nestjs/config: Compatible with NestJS v10+ and v11- NestJS v11.1.6 is the latest stable version
Security Note: The MessageBird SDK hasn't been updated since 2022. Before production deployment, check for known CVEs and consider alternatives like direct REST API integration or maintained community SDKs.
-
(Optional) Install Prisma Dependencies: If you plan to integrate with a database (e.g., to link verified numbers to users), install Prisma. This section is optional for the core OTP functionality.
bash# Using npm npm install prisma @prisma/client --save-dev # Using yarn yarn add prisma @prisma/client --dev -
Project Structure: Key directories:
src/: Application source code (main.ts,app.module.ts, etc.)src/otp/: Module for OTP logic (you'll create this).env: Environment variables (you'll create this)(Optional) prisma/: Prisma schema and migrations
2. Environment Configuration
Securely manage API keys using the @nestjs/config module and a .env file. For production, use secret managers like AWS Secrets Manager, Azure Key Vault, or HashiCorp Vault instead of plain .env files.
-
Create
.envfile: Create a file named.envin the project's root directory. -
Add MessageBird API Key: Obtain your API keys from the MessageBird Dashboard:
- Log in to your MessageBird account.
- Navigate to the Developers section in the left sidebar.
- Go to the API access (REST) tab.
- You'll find both Live and Test API keys.
- Test Key: Use during development and testing. Simulates API calls without sending actual SMS messages or incurring costs. Test against specific MessageBird test numbers.
- Live Key: Use in production to send real SMS messages to users.
- Add the key you intend to use (start with the test key) to your
.envfile:
dotenv# .env # Use your TEST key for development, switch to LIVE key for production MESSAGEBIRD_API_KEY=YOUR_TEST_API_KEY_HEREReplace
YOUR_TEST_API_KEY_HEREwith your actual test key initially. Switch to the live key in your production environment.Purpose: This variable holds the secret key required to authenticate requests to the MessageBird API. Using test keys avoids costs and real messages during development.
-
(Optional) Add Database URL: If using Prisma (optional feature), add your database connection string:
dotenv# .env (Optional - only if using Prisma) # Example for PostgreSQL: postgresql://USER:PASSWORD@HOST:PORT/DATABASE DATABASE_URL="postgresql://postgres:password@localhost:5432/otp_app?schema=public"Adjust the URL according to your database credentials and provider.
Purpose: This variable tells Prisma how to connect to your database.
-
Load
ConfigModule: Import and configureConfigModulein your main application module (src/app.module.ts) to make environment variables available throughout the app viaConfigService. Make it global and load.envvariables.typescript// src/app.module.ts import { Module } from '@nestjs/common'; import { ConfigModule } from '@nestjs/config'; // Import ConfigModule import { AppController } from './app.controller'; import { AppService } from './app.service'; import { OtpModule } from './otp/otp.module'; // We'll create this soon // import { PrismaModule } from './prisma.module'; // Import if using Prisma @Module({ imports: [ ConfigModule.forRoot({ // Load .env variables globally isGlobal: true, }), OtpModule, // Import our feature module // PrismaModule, // Add PrismaModule if using database (ensure it's defined) ], controllers: [AppController], providers: [AppService], }) export class AppModule {}Why
isGlobal: true? This makes theConfigServiceavailable in any module without needing to importConfigModulerepeatedly.
3. Implementing Core Functionality (OTP Service)
Create a dedicated module and service to handle the logic of interacting with the MessageBird API.
-
Generate OTP Module and Service: Use the NestJS CLI to generate the
otpmodule and service.bashnest generate module otp nest generate service otpThis creates
src/otp/otp.module.tsandsrc/otp/otp.service.ts(and spec file). NestJS automatically updatessrc/app.module.tsto importOtpModule. -
Implement
OtpService: Opensrc/otp/otp.service.ts. InjectConfigServiceto access the API key and instantiate the MessageBird client.typescript// src/otp/otp.service.ts import { Injectable, InternalServerErrorException, BadRequestException, Logger } from '@nestjs/common'; import { ConfigService } from '@nestjs/config'; import * as MessageBird from 'messagebird'; // Import MessageBird SDK @Injectable() export class OtpService { private readonly logger = new Logger(OtpService.name); private messagebird: MessageBird.MessageBird; constructor(private configService: ConfigService) { const apiKey = this.configService.get<string>('MESSAGEBIRD_API_KEY'); if (!apiKey) { throw new Error('MessageBird API Key not found in environment variables.'); } // Initialize MessageBird client // Type assertion might be needed due to MessageBird SDK typings; this could change with SDK updates. this.messagebird = MessageBird(apiKey) as unknown as MessageBird.MessageBird; } /** * Sends an OTP code to the specified phone number via MessageBird Verify API. * @param phoneNumber The recipient's phone number in E.164 format (e.g., +14155552671) * @returns The verification ID from MessageBird. * @throws InternalServerErrorException if the API call fails unexpectedly. * @throws BadRequestException if MessageBird returns a specific user-related error (e.g., invalid number format - code 21). */ async sendOtp(phoneNumber: string): Promise<string> { // Basic normalization example: remove spaces, dashes, parentheses // For production, use libphonenumber-js for comprehensive E.164 format validation and normalization const normalizedPhoneNumber = phoneNumber.replace(/[\s\-()]/g, ''); this.logger.log(`Sending OTP to normalized number: ${normalizedPhoneNumber}`); const params: MessageBird.VerifyCreateParams = { originator: 'VerifyApp', // Or your App Name (max 11 chars alphanumeric) or verified phone number template: 'Your verification code is %token.', type: 'sms', // tokenLength: 6, // Default is 6, range: 6-10 (per MessageBird Verify API docs) // timeout: 30, // Default is 30 seconds, range: 30-172801 seconds (up to 2 days) // maxAttempts: 1, // Default is 1, range: 1-10 attempts }; return new Promise((resolve, reject) => { this.messagebird.verify.create(normalizedPhoneNumber, params, (err, response) => { if (err) { this.logger.error('MessageBird verify.create error:', JSON.stringify(err)); // Log the full error structure // Check for specific, actionable MessageBird error codes if (err.errors && err.errors.length > 0) { const mbError = err.errors[0]; this.logger.error(`MessageBird Error Code: ${mbError.code}, Description: ${mbError.description}`); // Example: Code 21 indicates an invalid phone number if (mbError.code === 21) { return reject(new BadRequestException(`Invalid phone number: ${mbError.description}`)); } // Add checks for other relevant codes if needed (e.g., originator issues) // Fallback for other specific MessageBird errors return reject(new BadRequestException(`Failed to send OTP: ${mbError.description}`)); } // Generic internal error if no specific code matched or error structure is unexpected return reject(new InternalServerErrorException('Failed to initiate OTP verification due to an unexpected error.')); } this.logger.log(`OTP request successful for ${normalizedPhoneNumber}. ID: ${response?.id}`); // Ensure response.id is treated as a string if (response && typeof response.id === 'string') { resolve(response.id); } else { this.logger.error('MessageBird verify.create response missing or invalid ID:', response); reject(new InternalServerErrorException('Failed to get a valid verification ID from MessageBird.')); } }); }); } /** * Verifies the OTP token submitted by the user against the verification ID. * @param id The verification ID received from sendOtp. * @param token The 6-digit token entered by the user. * @returns An object indicating successful verification. * @throws BadRequestException if the token is invalid or expired (e.g., MessageBird error code 10). * @throws InternalServerErrorException on other API errors. */ async verifyOtp(id: string, token: string): Promise<{ status: string }> { this.logger.log(`Verifying OTP for ID: ${id}`); return new Promise((resolve, reject) => { this.messagebird.verify.verify(id, token, (err, response) => { if (err) { this.logger.error(`MessageBird verify.verify error for ID ${id}:`, JSON.stringify(err)); // Handle specific errors, e.g., invalid token (code 10) if (err.errors && err.errors.length > 0) { const mbError = err.errors[0]; this.logger.error(`MessageBird Verify Error Code: ${mbError.code}, Description: ${mbError.description}`); if (mbError.code === 10) { // Code 10 typically means invalid token return reject(new BadRequestException('Invalid or expired OTP token.')); } // Fallback for other specific MessageBird errors during verify return reject(new BadRequestException(`Failed to verify OTP: ${mbError.description}`)); } // Generic internal error return reject(new InternalServerErrorException('Failed to verify OTP due to an unexpected error.')); } // Check the status if available if (response && response.status === 'verified') { this.logger.log(`OTP verification successful for ID: ${id}`); resolve({ status: 'verified' }); } else { // Log the actual status received if not 'verified' (e.g., 'expired', 'failed') this.logger.warn(`OTP verification for ID ${id} returned non-verified status: ${response?.status ?? 'unknown'}`); // Treat any non-verified status as a failure from the client's perspective reject(new BadRequestException('Invalid or expired OTP token.')); } }); }); } }Key Implementation Details:
- Why
new Promise? The MessageBird SDK uses callbacks. Wrap the callback logic in a Promise to work seamlessly with NestJS's async/await syntax. - Error Handling: Catch errors from the SDK. Throw
BadRequestExceptionfor known user-input issues or specific MessageBird error codes (like invalid number code 21, invalid token code 10). UseInternalServerErrorExceptionfor unexpected API failures or missing IDs. Logging the full error (JSON.stringify(err)) and specific codes helps diagnose issues. - Configuration:
originatorshould ideally be a purchased virtual number from MessageBird or a short alphanumeric code (max 11 characters, check country-specific restrictions).templatemust include the%tokenplaceholder (verified from MessageBird Verify API documentation at https://developers.messagebird.com/api/verify/). - Normalization: The basic regex
phoneNumber.replace(/[\s\-()]/g, '')strips common characters before sending to MessageBird. For production, use a robust phone number library likelibphonenumber-jsfor comprehensive E.164 format validation and normalization, especially for international formats and country-specific validation. - API Parameters: Based on MessageBird Verify API official documentation (https://developers.messagebird.com/api/verify/):
timeout: Range 30–172,801 seconds (default: 30s)tokenLength: Range 6–10 digits (default: 6)maxAttempts: Range 1–10 attempts (default: 1)type: Options aresms,flash,tts(text-to-speech), oremail
Common MessageBird Error Codes:
- Code 10: Invalid or expired OTP token
- Code 21: Invalid phone number format
- Code 9: Missing required parameter
- Why
-
Register
ConfigServiceinOtpModule: SinceConfigModuleis global andConfigServiceis injected intoOtpService(which is part ofOtpModule), no explicit import ofConfigModuleor registration ofConfigServiceis needed withinOtpModuleitself.typescript// src/otp/otp.module.ts import { Module } from '@nestjs/common'; import { OtpService } from './otp.service'; import { OtpController } from './otp.controller'; // We'll create this next @Module({ controllers: [OtpController], // Add controller here providers: [OtpService], exports: [OtpService], // Export if other modules need it }) export class OtpModule {}
4. Building the API Layer (OTP Controller)
Create the controller with endpoints that clients can call. Protect these endpoints with rate limiting and authentication in production.
-
Generate OTP Controller:
bashnest generate controller otpThis creates
src/otp/otp.controller.ts(and spec file) and adds it toOtpModule. -
Define Data Transfer Objects (DTOs): Create DTOs to define the expected request body structure and apply validation rules using
class-validator.-
Create
src/otp/dto/send-otp.dto.ts:typescript// src/otp/dto/send-otp.dto.ts import { IsNotEmpty, IsPhoneNumber, IsString } from 'class-validator'; export class SendOtpDto { @IsNotEmpty() @IsString() @IsPhoneNumber(null) // Use null for generic E.164 format validation // Input might still contain formatting chars; normalization happens in the service. phoneNumber: string; }About @IsPhoneNumber: This validator accepts E.164 format phone numbers (e.g., +14155552671) but has limitations with formatting variations. Service-level normalization remains necessary to handle spaces, dashes, and parentheses that users might include.
-
Create
src/otp/dto/verify-otp.dto.ts:typescript// src/otp/dto/verify-otp.dto.ts import { IsNotEmpty, IsString, Length } from 'class-validator'; export class VerifyOtpDto { @IsNotEmpty() @IsString() // Add validation if MessageBird always returns IDs of a specific format/length id: string; @IsNotEmpty() @IsString() @Length(6, 6) // Assuming default 6-digit token token: string; }
-
-
Implement
OtpController: Opensrc/otp/otp.controller.ts. InjectOtpServiceand define the endpoints.typescript// src/otp/otp.controller.ts import { Controller, Post, Body, HttpCode, HttpStatus, ValidationPipe, UsePipes, Logger } from '@nestjs/common'; import { OtpService } from './otp.service'; import { SendOtpDto } from './dto/send-otp.dto'; import { VerifyOtpDto } from './dto/verify-otp.dto'; @Controller('otp') // Route prefix: /otp export class OtpController { private readonly logger = new Logger(OtpController.name); constructor(private readonly otpService: OtpService) {} @Post('send') // Endpoint: POST /otp/send @HttpCode(HttpStatus.OK) // Return 200 OK on success // If ValidationPipe is global, @UsePipes is not needed here // @UsePipes(new ValidationPipe({ transform: true, whitelist: true })) async sendOtp(@Body() sendOtpDto: SendOtpDto): Promise<{ id: string }> { this.logger.log(`Received request to send OTP to ${sendOtpDto.phoneNumber}`); const verificationId = await this.otpService.sendOtp(sendOtpDto.phoneNumber); return { id: verificationId }; } @Post('verify') // Endpoint: POST /otp/verify @HttpCode(HttpStatus.OK) // Return 200 OK on successful verification // If ValidationPipe is global, @UsePipes is not needed here // @UsePipes(new ValidationPipe({ transform: true, whitelist: true })) async verifyOtp(@Body() verifyOtpDto: VerifyOtpDto): Promise<{ status: string }> { this.logger.log(`Received request to verify OTP for ID: ${verifyOtpDto.id}`); return this.otpService.verifyOtp(verifyOtpDto.id, verifyOtpDto.token); } }Endpoint Details:
@Controller('otp'): Sets the base route for all methods in this controller to/otp@Post('send'),@Post('verify'): Define POST endpoints at/otp/sendand/otp/verify@Body(): Injects the request bodyValidationPipe: (Applied globally below) Automatically validates the incoming request body against the DTO@HttpCode(HttpStatus.OK): Ensures a200 OKstatus is returned on success
Production Security:
- Implement rate limiting to prevent abuse (use
@nestjs/throttlerpackage) - Add authentication to protect endpoints (JWT tokens, API keys)
- Track OTP request attempts per IP and phone number
- Implement cooldown periods between OTP requests
-
Enable
ValidationPipeGlobally (Recommended): Enable the validation pipe globally insrc/main.tsfor cleaner controllers.typescript// src/main.ts import { NestFactory } from '@nestjs/core'; import { AppModule } from './app.module'; import { ValidationPipe, Logger } from '@nestjs/common'; // Import Logger & ValidationPipe async function bootstrap() { const app = await NestFactory.create(AppModule); const logger = new Logger('Bootstrap'); // Enable global validation pipe app.useGlobalPipes(new ValidationPipe({ whitelist: true, // Strip properties not in DTO transform: true, // Transform payload to DTO instances forbidNonWhitelisted: true, // Throw error if non-whitelisted properties are sent transformOptions: { enableImplicitConversion: true, // Allow auto-conversion of types }, })); // Enable CORS - By default allows all origins. // **IMPORTANT**: Restrict origins in production for security! app.enableCors(); // Example for production: // app.enableCors({ origin: 'https://your-frontend-app.com' }); logger.log('CORS enabled. Remember to restrict origins in production.'); const port = process.env.PORT || 3000; await app.listen(port); logger.log(`Application listening on port ${port}`); } bootstrap();With the global pipe enabled, remove the
@UsePipes(...)decorators from the controller methods (as shown in the controller example above).Production Deployment Checklist:
- Install
helmetfor security headers - Enable compression middleware
- Configure structured logging (Pino recommended)
- Add health check endpoints
- Implement graceful shutdown
- Use environment-specific configurations
- Set up centralized logging (CloudWatch, Datadog, etc.)
- Install
-
Testing Endpoints with
curl:-
Start the application:
npm run start:dev -
Send OTP: Replace
+12345678900with a real phone number (use a test number if using your test API key).bashcurl -X POST http://localhost:3000/otp/send \ -H "Content-Type: application/json" \ -d '{ "phoneNumber": "+12345678900" }'Expected Response (Success):
json{ "id": "some-verification-id-from-messagebird" }Expected Response (Validation Error – e.g., invalid number format):
json{ "statusCode": 400, "message": [ "phoneNumber must be a valid phone number" ], "error": "Bad Request" } -
Verify OTP: Replace
some-verification-id-from-messagebirdwith the ID you received, and123456with the code sent to your phone (or the test code provided by MessageBird if using test keys).bashcurl -X POST http://localhost:3000/otp/verify \ -H "Content-Type: application/json" \ -d '{ "id": "some-verification-id-from-messagebird", "token": "123456" }'Expected Response (Success):
json{ "status": "verified" }Expected Response (Incorrect Token):
json{ "statusCode": 400, "message": "Invalid or expired OTP token.", "error": "Bad Request" }
-
5. Integrating with Third-Party Services (MessageBird)
Core integration points:
- Configuration: Store API key securely in
.env(use Test key for dev, Live key for prod) and load via@nestjs/config. Ensure the.envfile is never committed to version control (add it to.gitignore). - SDK Initialization: Initialize the
messagebirdSDK in theOtpServiceconstructor using the API key fromConfigService. The type assertionas unknown as MessageBird.MessageBirdis included due to potential SDK typing nuances. - API Calls: Use
verify.createandverify.verifymethods withinOtpServiceto interact with MessageBird. - Dashboard Navigation for API Key: MessageBird Dashboard → Developers → API access (REST) → Copy/Create Test and Live API Keys.
- Fallback Mechanisms: The current implementation relies directly on MessageBird. For production robustness:
- Retries: Implement retry logic (see Section 6) within
OtpServicearound the MessageBird calls, especially for transient network errors (distinguish from user errors like invalid tokens). - Alternative Providers: Consider integrating a secondary SMS provider as a fallback if MessageBird experiences prolonged outages (adds complexity).
- Voice Fallback: MessageBird Verify supports
type: 'tts'(Text-to-Speech). ModifysendOtpto offer this if SMS fails or as a preference.
- Retries: Implement retry logic (see Section 6) within
MessageBird Pricing Considerations:
- SMS costs vary by destination country
- Test API keys don't incur charges
- Monitor usage to stay within budget
- Consider volume pricing tiers for scaling
Geographic Coverage: MessageBird supports SMS delivery to 200+ countries. Check country-specific restrictions and compliance requirements (e.g., sender ID registration, content filtering) before deploying internationally.
6. Implementing Error Handling and Logging
NestJS provides robust mechanisms for handling errors and logging.
-
Error Handling Strategy:
- Validation Errors: The global
ValidationPipehandles these automatically, returning400 Bad Request. - Service-Level Errors:
OtpServicethrows specificHttpExceptionsubclasses (BadRequestException,InternalServerErrorException) based on MessageBird API responses or internal issues. Specific MessageBird error codes (e.g., 10, 21) map toBadRequestException. - Unhandled Errors: NestJS catches unhandled exceptions and returns
500 Internal Server Error.
Custom Exception Filters: Create custom exception filters to format error responses consistently or integrate with error tracking services like Sentry. Example:
typescriptimport { ExceptionFilter, Catch, ArgumentsHost, HttpException } from '@nestjs/common'; import { Response } from 'express'; @Catch(HttpException) export class HttpExceptionFilter implements ExceptionFilter { catch(exception: HttpException, host: ArgumentsHost) { const ctx = host.switchToHttp(); const response = ctx.getResponse<Response>(); const status = exception.getStatus(); response.status(status).json({ statusCode: status, timestamp: new Date().toISOString(), message: exception.message, }); } } - Validation Errors: The global
-
Logging:
- Use the built-in
Logger(@nestjs/common) inOtpService,OtpController, andmain.ts. - NestJS logs basic request/response info and exceptions.
OtpServicelogs specific MessageBird errors. - Production Best Practices:
- Use structured logging (
pino,nestjs-pino) for easier parsing by log aggregation tools - Include correlation IDs for request tracing
- Redact PII (Personal Identifiable Information) like phone numbers in logs
- Configure appropriate log levels (debug, info, warn, error)
- Implement log rotation
- Use centralized logging solutions (CloudWatch, Datadog, Splunk)
- Use structured logging (
- Use the built-in
-
Retry Mechanisms (Conceptual Example): Add retries to
OtpServicecalls to improve resilience against transient network issues.typescript// Simplified conceptual retry logic for OtpService methods // Consider using a library like 'nestjs-retry' or 'async-retry' for production // Define logger at a scope accessible by isTransientError if needed, or pass it const logger = new Logger('RetryLogic'); async function callWithRetry<T>(action: () => Promise<T>, maxAttempts: number, logger: Logger): Promise<T> { let attempts = 0; while (attempts < maxAttempts) { attempts++; try { return await action(); // Attempt the action } catch (error) { logger.error(`Attempt ${attempts} failed: ${error.message}`, error.stack); // Check if the error is potentially transient AND if more attempts remain if (isTransientError(error) && attempts < maxAttempts) { const delay = 1000 * Math.pow(2, attempts - 1); // Exponential backoff logger.warn(`Retrying after ${delay}ms…`); await new Promise(resolve => setTimeout(resolve, delay)); continue; // Go to the next iteration } else { // If not transient or max attempts reached, re-throw the error throw error; } } } // This line should theoretically not be reached if maxAttempts > 0 throw new InternalServerErrorException('Action failed after maximum retry attempts.'); } // **Placeholder Function - Requires Real Implementation** // This function MUST be implemented to identify errors that justify a retry // (e.g., network timeouts, specific 5xx errors from MessageBird if identifiable). // Do NOT retry on errors like 'invalid phone number' or 'invalid token'. function isTransientError(error: any): boolean { logger.warn('`isTransientError` check is currently a placeholder and always returns false. Implement real transient error detection logic.'); // Example checks (modify based on actual error structure and codes): // if (error?.status === 503) return true; // Service Unavailable // if (error?.code === 'ETIMEDOUT' || error?.code === 'ENETUNREACH') return true; // Network issues // if (error?.errors?.[0]?.code === SOME_MESSAGEBIRD_TRANSIENT_CODE) return true; return false; // Default: Do not retry } // Usage within OtpService (Example for sendOtp): // Assume OtpService class context for 'this.logger', 'this.messagebird' etc. /* async sendOtp(phoneNumber: string): Promise<string> { const normalizedPhoneNumber = phoneNumber.replace(/[\s\-()]/g, ''); this.logger.log(`Sending OTP to normalized number: ${normalizedPhoneNumber}`); const action = () => new Promise<string>((resolve, reject) => { const params: MessageBird.VerifyCreateParams = { originator: 'VerifyApp', template: 'Your verification code is %token.', type: 'sms', }; this.messagebird.verify.create(normalizedPhoneNumber, params, (err, response) => { if (err) return reject(err); // Reject promise on error to be caught by retry logic if (response && typeof response.id === 'string') { resolve(response.id); } else { reject(new InternalServerErrorException('Failed to get a valid verification ID from MessageBird.')); } }); }); try { // Wrap the promise-based SDK call within the retry function // Important: The retry function handles the actual SDK error mapping (BadRequest vs InternalServer) return await callWithRetry(action, 3, this.logger); } catch (error) { // Map final error after retries (if any occurred) this.logger.error('Final error after retries (or on first attempt if non-transient):', JSON.stringify(error)); if (error instanceof HttpException) { throw error; // Re-throw exceptions already mapped (like BadRequest) } // Map specific MessageBird errors after retries failed if (error.errors && error.errors.length > 0) { const mbError = error.errors[0]; if (mbError.code === 21) { throw new BadRequestException(`Invalid phone number: ${mbError.description}`); } throw new BadRequestException(`Failed to send OTP: ${mbError.description}`); } // Fallback generic error throw new InternalServerErrorException('Failed to initiate OTP verification after multiple attempts.'); } } // Apply similar retry logic structure to verifyOtp if needed. */Caveat: The
isTransientErrorfunction is critical and provided as a non-working placeholder. You must implement logic to correctly identify which errors (based on codes, status, or types) should trigger a retry. Retrying on user errors (like invalid number) is incorrect. The usage example withinsendOtpis commented out to avoid making the code block non-functional due to the placeholderisTransientErrorand context assumptions.Circuit Breaker Pattern: Protect your application from cascading failures when MessageBird is down. Use the circuit breaker pattern (libraries like
opossumornestjs-circuit-breaker) to temporarily stop sending requests to a failing service and provide fallback responses.
7. (Optional) Creating a Database Schema and Data Layer (Prisma Example)
This section is optional and only relevant if you need to store user data or link verified numbers.
-
Initialize Prisma: If not done in Section 1, install Prisma dependencies and initialize:
bash# Install if needed: npm install prisma @prisma/client --save-dev npx prisma init --datasource-provider postgresqlEnsure your
.envhas the correctDATABASE_URL. -
Create User and Verification Schema:
prisma// prisma/schema.prisma model User { id String @id @default(cuid()) phoneNumber String @unique isVerified Boolean @default(false) createdAt DateTime @default(now()) updatedAt DateTime @updatedAt verifications Verification[] } model Verification { id String @id @default(cuid()) userId String? user User? @relation(fields: [userId], references: [id]) phoneNumber String verificationId String @unique status String // 'pending', 'verified', 'failed', 'expired' attempts Int @default(0) createdAt DateTime @default(now()) expiresAt DateTime verifiedAt DateTime? } -
Run Migrations:
bashnpx prisma migrate dev --name init npx prisma generate -
Integration Example: Track verification attempts for audit trails, implement cooldown periods between OTP requests, and link verified phone numbers to user accounts:
typescript// In OtpService after successful verification: await this.prisma.verification.update({ where: { verificationId: id }, data: { status: 'verified', verifiedAt: new Date(), }, }); await this.prisma.user.update({ where: { phoneNumber }, data: { isVerified: true }, });
Source Citations:
- Node.js LTS versions: https://nodejs.org/en/about/previous-releases (verified January 2025)
- NestJS version compatibility: NestJS v10+ requires Node.js v16+, v11.1.6 latest (https://github.com/nestjs/nest/releases)
- MessageBird Node.js SDK: v4.0.1 on npm (https://www.npmjs.com/package/messagebird)
- MessageBird Verify API documentation: https://developers.messagebird.com/api/verify/