import { NestFactory, Reflector } from '@nestjs/core'; import { AppModule } from './app.module'; import { FastifyAdapter, NestFastifyApplication, } from '@nestjs/platform-fastify'; import { Logger, NotFoundException, ValidationPipe } from '@nestjs/common'; import { Logger as PinoLogger } from 'nestjs-pino'; import { TransformHttpResponseInterceptor } from './common/interceptors/http-response.interceptor'; import { WsRedisIoAdapter } from './ws/adapter/ws-redis.adapter'; import fastifyMultipart from '@fastify/multipart'; import fastifyCookie from '@fastify/cookie'; import fastifyIp from 'fastify-ip'; import { InternalLogFilter } from './common/logger/internal-log-filter'; async function bootstrap() { const app = await NestFactory.create( AppModule, new FastifyAdapter({ trustProxy: true, routerOptions: { maxParamLength: 1000, ignoreTrailingSlash: true, ignoreDuplicateSlashes: true, }, }), { rawBody: true, // captures NestJS internal errors logger: new InternalLogFilter(), // bufferLogs must be false else pino will fail // to log OnApplicationBootstrap logs bufferLogs: false, }, ); app.useLogger(app.get(PinoLogger)); app.setGlobalPrefix('api', { exclude: ['robots.txt', 'share/:shareId/p/:pageSlug', 'mcp'], }); const reflector = app.get(Reflector); const redisIoAdapter = new WsRedisIoAdapter(app); await redisIoAdapter.connectToRedis(); app.useWebSocketAdapter(redisIoAdapter); await app.register(fastifyIp); await app.register(fastifyMultipart); await app.register(fastifyCookie); app .getHttpAdapter() .getInstance() .addContentTypeParser( 'application/scim+json', { parseAs: 'string' }, (_, body, done) => { try { const json = JSON.parse(body.toString()); done(null, json); } catch (err: any) { done(err); } }, ); app .getHttpAdapter() .getInstance() .decorateReply('setHeader', function (name: string, value: unknown) { this.header(name, value); }) .decorateReply('end', function () { this.send(''); }) .addHook('preHandler', function (req, reply, done) { // don't require workspaceId for the following paths const excludedPaths = [ '/api/auth/setup', '/api/health', '/api/billing/stripe/webhook', '/api/workspace/check-hostname', '/api/sso/google', '/api/workspace/create', '/api/workspace/joined', '/api/workspace/find-by-email', ]; if ( req.originalUrl.startsWith('/api') && !excludedPaths.some((path) => req.originalUrl.startsWith(path)) ) { if (!req.raw?.['workspaceId'] && req.originalUrl !== '/api') { throw new NotFoundException('Workspace not found'); } done(); } else { done(); } }); app.useGlobalPipes( new ValidationPipe({ whitelist: true, stopAtFirstError: true, transform: true, }), ); app.enableCors(); app.useGlobalInterceptors(new TransformHttpResponseInterceptor(reflector)); app.enableShutdownHooks(); const logger = new Logger('NestApplication'); process.on('unhandledRejection', (reason, promise) => { logger.error(`UnhandledRejection, reason: ${reason}`, promise); }); process.on('uncaughtException', (error) => { logger.error('UncaughtException:', error); }); const port = process.env.PORT || 3000; const host = process.env.HOST || '0.0.0.0'; await app.listen(port, host, () => { logger.log( `Listening on http://127.0.0.1:${port} / ${process.env.APP_URL}`, ); }); } bootstrap();