2025-08-20 18:02:50 +09:00
|
|
|
import { NestFactory } from '@nestjs/core';
|
|
|
|
|
import { ValidationPipe } from '@nestjs/common';
|
|
|
|
|
import { SwaggerModule, DocumentBuilder } from '@nestjs/swagger';
|
|
|
|
|
import { ConfigService } from '@nestjs/config';
|
2025-08-21 15:24:40 +09:00
|
|
|
import { Logger } from 'nestjs-pino';
|
2025-08-20 18:02:50 +09:00
|
|
|
import helmet from 'helmet';
|
|
|
|
|
import cookieParser from 'cookie-parser';
|
|
|
|
|
import { AppModule } from './app.module';
|
2025-08-21 15:24:40 +09:00
|
|
|
import { GlobalExceptionFilter } from './common/filters/http-exception.filter';
|
2025-08-20 18:02:50 +09:00
|
|
|
|
|
|
|
|
async function bootstrap() {
|
2025-08-21 15:24:40 +09:00
|
|
|
const app = await NestFactory.create(AppModule, { bufferLogs: true });
|
|
|
|
|
|
|
|
|
|
// Set Pino as the logger
|
|
|
|
|
app.useLogger(app.get(Logger));
|
|
|
|
|
|
2025-08-20 18:02:50 +09:00
|
|
|
const configService = app.get(ConfigService);
|
2025-08-21 15:24:40 +09:00
|
|
|
const logger = app.get(Logger);
|
2025-08-20 18:02:50 +09:00
|
|
|
|
|
|
|
|
// Security
|
|
|
|
|
app.use(helmet());
|
2025-08-21 15:24:40 +09:00
|
|
|
app.getHttpAdapter().getInstance().disable('x-powered-by');
|
2025-08-20 18:02:50 +09:00
|
|
|
app.use(cookieParser());
|
2025-08-21 15:24:40 +09:00
|
|
|
// Behind reverse proxies (e.g., Nginx) to ensure correct IPs and secure cookies
|
|
|
|
|
if (configService.get('TRUST_PROXY', 'false') === 'true') {
|
|
|
|
|
const httpAdapter = app.getHttpAdapter();
|
|
|
|
|
const instance: any = httpAdapter.getInstance();
|
|
|
|
|
if (instance?.set) {
|
|
|
|
|
instance.set('trust proxy', 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-08-20 18:02:50 +09:00
|
|
|
|
|
|
|
|
// CORS
|
|
|
|
|
app.enableCors({
|
|
|
|
|
origin: configService.get('CORS_ORIGIN', 'http://localhost:3000'),
|
|
|
|
|
credentials: true,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Global validation pipe
|
|
|
|
|
app.useGlobalPipes(
|
|
|
|
|
new ValidationPipe({
|
|
|
|
|
transform: true,
|
|
|
|
|
whitelist: true,
|
|
|
|
|
forbidNonWhitelisted: true,
|
|
|
|
|
}),
|
|
|
|
|
);
|
|
|
|
|
|
2025-08-21 15:24:40 +09:00
|
|
|
// Global exception filter
|
|
|
|
|
app.useGlobalFilters(new GlobalExceptionFilter());
|
|
|
|
|
|
|
|
|
|
// Ensure proper shutdown for Prisma/Redis and logger flush
|
|
|
|
|
app.enableShutdownHooks();
|
|
|
|
|
const shutdown = async (signal: string) => {
|
|
|
|
|
try {
|
|
|
|
|
logger.warn(`Received ${signal}. Shutting down gracefully...`);
|
|
|
|
|
await app.close();
|
|
|
|
|
if (typeof (logger as any).flush === 'function') {
|
|
|
|
|
await (logger as any).flush();
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
// eslint-disable-next-line no-console
|
|
|
|
|
console.error('Error during shutdown', err);
|
|
|
|
|
} finally {
|
|
|
|
|
process.exit(0);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
process.on('SIGINT', () => shutdown('SIGINT'));
|
|
|
|
|
process.on('SIGTERM', () => shutdown('SIGTERM'));
|
|
|
|
|
|
2025-08-20 18:02:50 +09:00
|
|
|
// Global prefix
|
|
|
|
|
app.setGlobalPrefix('api');
|
|
|
|
|
|
|
|
|
|
// Swagger documentation
|
|
|
|
|
if (configService.get('NODE_ENV') !== 'production') {
|
|
|
|
|
const config = new DocumentBuilder()
|
|
|
|
|
.setTitle('Customer Portal API')
|
|
|
|
|
.setDescription('Backend for Frontend API for customer portal')
|
|
|
|
|
.setVersion('1.0')
|
|
|
|
|
.addBearerAuth()
|
|
|
|
|
.addCookieAuth('auth-cookie')
|
|
|
|
|
.build();
|
|
|
|
|
|
|
|
|
|
const document = SwaggerModule.createDocument(app, config);
|
|
|
|
|
SwaggerModule.setup('api/docs', app, document);
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-21 15:24:40 +09:00
|
|
|
const port = Number(configService.get('BFF_PORT', 4000));
|
|
|
|
|
|
2025-08-20 18:02:50 +09:00
|
|
|
await app.listen(port);
|
|
|
|
|
|
2025-08-21 15:24:40 +09:00
|
|
|
logger.log(`🚀 BFF API running on: http://localhost:${port}/api`);
|
2025-08-20 18:02:50 +09:00
|
|
|
if (configService.get('NODE_ENV') !== 'production') {
|
2025-08-21 15:24:40 +09:00
|
|
|
logger.log(`📚 API Documentation: http://localhost:${port}/api/docs`);
|
2025-08-20 18:02:50 +09:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bootstrap();
|