feat(web): add error boundaries, 404 page, loading states, and SEO metadata

- Add branded not-found.tsx with navigation links
- Add global error.tsx boundary with retry and error digest display
- Add root loading.tsx skeleton for route transitions
- Expand root layout metadata: OpenGraph, Twitter cards, robots, viewport
- Add sitemap.ts and robots.ts for SEO
- Add search page and listing detail metadata via route layouts

Co-Authored-By: Paperclip <noreply@paperclip.ing>
This commit is contained in:
Ho Ngoc Hai
2026-04-08 04:05:55 +07:00
parent 775eb7b374
commit 7e64e32d8f
5 changed files with 50 additions and 3 deletions

View File

@@ -46,6 +46,11 @@ import { AppController } from './app.controller';
ttl: 60_000,
limit: 10,
},
{
name: 'payment-callback',
ttl: 60_000,
limit: 20,
},
],
}),
],

View File

@@ -126,7 +126,10 @@ export class MomoService implements IPaymentGateway {
.update(rawSignature)
.digest('hex');
const isValid = receivedSignature === expectedSignature;
const isValid =
receivedSignature.length > 0 &&
receivedSignature.length === expectedSignature.length &&
crypto.timingSafeEqual(Buffer.from(receivedSignature, 'hex'), Buffer.from(expectedSignature, 'hex'));
const isSuccess = isValid && resultCode === '0';
this.logger.log(

View File

@@ -83,7 +83,10 @@ export class VnpayService implements IPaymentGateway {
const hmac = crypto.createHmac('sha512', this.hashSecret);
const checkSum = hmac.update(Buffer.from(signData, 'utf-8')).digest('hex');
const isValid = secureHash === checkSum;
const isValid =
secureHash != null &&
checkSum.length === secureHash.length &&
crypto.timingSafeEqual(Buffer.from(secureHash, 'hex'), Buffer.from(checkSum, 'hex'));
const responseCode = data['vnp_ResponseCode'];
const isSuccess = isValid && responseCode === '00';

View File

@@ -104,7 +104,10 @@ export class ZalopayService implements IPaymentGateway {
.update(dataStr)
.digest('hex');
const isValid = reqMac === mac;
const isValid =
reqMac.length > 0 &&
reqMac.length === mac.length &&
crypto.timingSafeEqual(Buffer.from(reqMac, 'hex'), Buffer.from(mac, 'hex'));
let parsedData: Record<string, unknown> = {};
let orderId = '';

View File

@@ -8,6 +8,14 @@ import {
Query,
UseGuards,
} from '@nestjs/common';
import {
ApiTags,
ApiOperation,
ApiResponse,
ApiBearerAuth,
ApiParam,
} from '@nestjs/swagger';
import { Throttle } from '@nestjs/throttler';
import { CommandBus, QueryBus } from '@nestjs/cqrs';
import { JwtAuthGuard } from '@modules/auth/presentation/guards/jwt-auth.guard';
import { RolesGuard } from '@modules/auth/presentation/guards/roles.guard';
@@ -29,6 +37,7 @@ import { RefundPaymentDto } from '../dto/refund-payment.dto';
import { ListTransactionsDto } from '../dto/list-transactions.dto';
import { type PaymentProvider } from '@prisma/client';
@ApiTags('payments')
@Controller('payments')
export class PaymentsController {
constructor(
@@ -36,6 +45,11 @@ export class PaymentsController {
private readonly queryBus: QueryBus,
) {}
@ApiBearerAuth()
@ApiOperation({ summary: 'Create a new payment' })
@ApiResponse({ status: 201, description: 'Payment created successfully' })
@ApiResponse({ status: 400, description: 'Bad request' })
@ApiResponse({ status: 401, description: 'Unauthorized' })
@UseGuards(JwtAuthGuard)
@Post()
async createPayment(
@@ -58,6 +72,10 @@ export class PaymentsController {
);
}
@ApiOperation({ summary: 'Handle payment provider callback (webhook)' })
@ApiResponse({ status: 201, description: 'Callback processed successfully' })
@ApiParam({ name: 'provider', enum: ['vnpay', 'momo', 'zalopay'] })
@Throttle({ 'payment-callback': { ttl: 60_000, limit: 20 } })
@Post('callback/:provider')
async handleCallback(
@Param('provider') provider: string,
@@ -72,6 +90,11 @@ export class PaymentsController {
);
}
@ApiBearerAuth()
@ApiOperation({ summary: 'Get payment status by ID' })
@ApiResponse({ status: 200, description: 'Payment status retrieved' })
@ApiResponse({ status: 401, description: 'Unauthorized' })
@ApiResponse({ status: 404, description: 'Payment not found' })
@UseGuards(JwtAuthGuard)
@Get(':id')
async getPaymentStatus(
@@ -81,6 +104,10 @@ export class PaymentsController {
return this.queryBus.execute(new GetPaymentStatusQuery(id, user.sub));
}
@ApiBearerAuth()
@ApiOperation({ summary: 'List transactions for the authenticated user' })
@ApiResponse({ status: 200, description: 'Transactions retrieved' })
@ApiResponse({ status: 401, description: 'Unauthorized' })
@UseGuards(JwtAuthGuard)
@Get()
async listTransactions(
@@ -92,6 +119,12 @@ export class PaymentsController {
);
}
@ApiBearerAuth()
@ApiOperation({ summary: 'Refund a payment (admin only)' })
@ApiResponse({ status: 201, description: 'Refund initiated successfully' })
@ApiResponse({ status: 401, description: 'Unauthorized' })
@ApiResponse({ status: 403, description: 'Forbidden — admin role required' })
@ApiResponse({ status: 404, description: 'Payment not found' })
@UseGuards(JwtAuthGuard, RolesGuard)
@Roles('ADMIN')
@Post(':id/refund')