n8n/packages/cli/src/controller.registry.ts
Ricardo Espinoza cd167ac6db
feat(core): Add OAuth to MCP server (#21469)
Co-authored-by: Milorad FIlipović <milorad@n8n.io>
2025-11-05 19:26:31 -05:00

207 lines
6.6 KiB
TypeScript

import { RESPONSE_ERROR_MESSAGES } from '@/constants';
import { inProduction } from '@n8n/backend-common';
import { GlobalConfig } from '@n8n/config';
import { type BooleanLicenseFeature } from '@n8n/constants';
import type { AuthenticatedRequest } from '@n8n/db';
import { ControllerRegistryMetadata } from '@n8n/decorators';
import type { AccessScope, Controller, RateLimit, StaticRouterMetadata } from '@n8n/decorators';
import { Container, Service } from '@n8n/di';
import { Router } from 'express';
import type { Application, Request, Response, RequestHandler } from 'express';
import { rateLimit as expressRateLimit } from 'express-rate-limit';
import { UnexpectedError } from 'n8n-workflow';
import type { ZodClass } from 'zod-class';
import { NotFoundError } from './errors/response-errors/not-found.error';
import { LastActiveAtService } from './services/last-active-at.service';
import { AuthService } from '@/auth/auth.service';
import { UnauthenticatedError } from '@/errors/response-errors/unauthenticated.error';
import { License } from '@/license';
import { userHasScopes } from '@/permissions.ee/check-access';
import { send } from '@/response-helper';
@Service()
export class ControllerRegistry {
constructor(
private readonly license: License,
private readonly authService: AuthService,
private readonly globalConfig: GlobalConfig,
private readonly metadata: ControllerRegistryMetadata,
private readonly lastActiveAtService: LastActiveAtService,
) {}
activate(app: Application) {
for (const controllerClass of this.metadata.controllerClasses) {
this.activateController(app, controllerClass);
}
}
private activateController(app: Application, controllerClass: Controller) {
const metadata = this.metadata.getControllerMetadata(controllerClass);
const router = Router({ mergeParams: true });
const basePath = metadata.registerOnRootPath
? metadata.basePath
: `/${this.globalConfig.endpoints.rest}/${metadata.basePath}`;
const prefix = basePath.replace(/\/+/g, '/').replace(/\/$/, '');
app.use(prefix === '' ? '/' : prefix, router);
const controller = Container.get(controllerClass) as Controller;
const controllerMiddlewares = metadata.middlewares.map(
(handlerName) => controller[handlerName].bind(controller) as RequestHandler,
);
const staticRouters = (controllerClass as any).routers as StaticRouterMetadata[] | undefined;
if (staticRouters) {
for (const routerConfig of staticRouters) {
if (!routerConfig.router) {
throw new UnexpectedError(
`Router is undefined for path "${routerConfig.path}" in controller "${controllerClass.name}"`,
);
}
const middlewares = this.buildMiddlewares(routerConfig, controllerMiddlewares);
router.use(routerConfig.path, ...middlewares, routerConfig.router);
}
}
// Register regular routes
for (const [handlerName, route] of metadata.routes) {
// Original handler logic for non-router routes
const argTypes = Reflect.getMetadata(
'design:paramtypes',
controller,
handlerName,
) as unknown[];
const handler = async (req: Request, res: Response) => {
const args: unknown[] = [req, res];
for (let index = 0; index < route.args.length; index++) {
const arg = route.args[index];
if (!arg) continue;
if (arg.type === 'param') args.push(req.params[arg.key]);
else if (['body', 'query'].includes(arg.type)) {
const paramType = argTypes[index] as ZodClass;
if (paramType && 'safeParse' in paramType) {
const output = paramType.safeParse(req[arg.type]);
if (output.success) args.push(output.data);
else {
return res.status(400).json(output.error.errors[0]);
}
}
} else throw new UnexpectedError('Unknown arg type: ' + arg.type);
}
return await controller[handlerName](...args);
};
const middlewares = this.buildMiddlewares(route, controllerMiddlewares);
const finalHandler = route.usesTemplates
? async (req: Request, res: Response) => {
await handler(req, res);
}
: send(handler);
router[route.method](route.path, ...middlewares, finalHandler);
}
}
/**
* Builds middleware array based on route configuration.
* Used for both static routers and inline router definitions.
*/
private buildMiddlewares(
route: {
skipAuth?: boolean;
allowSkipMFA?: boolean;
allowSkipPreviewAuth?: boolean;
rateLimit?: boolean | RateLimit;
licenseFeature?: BooleanLicenseFeature;
accessScope?: AccessScope;
middlewares?: RequestHandler[];
},
controllerMiddlewares: RequestHandler[],
): RequestHandler[] {
const middlewares: RequestHandler[] = [];
if (inProduction && route.rateLimit) {
middlewares.push(this.createRateLimitMiddleware(route.rateLimit));
}
if (!route.skipAuth) {
middlewares.push(
this.authService.createAuthMiddleware({
allowSkipMFA: route.allowSkipMFA ?? false,
allowSkipPreviewAuth: route.allowSkipPreviewAuth ?? false,
}),
this.lastActiveAtService.middleware.bind(this.lastActiveAtService) as RequestHandler,
);
}
if (route.licenseFeature) {
middlewares.push(this.createLicenseMiddleware(route.licenseFeature));
}
if (route.accessScope) {
middlewares.push(this.createScopedMiddleware(route.accessScope));
}
middlewares.push(...controllerMiddlewares);
if (route.middlewares) {
middlewares.push(...route.middlewares);
}
return middlewares;
}
private createRateLimitMiddleware(rateLimit: true | RateLimit): RequestHandler {
if (typeof rateLimit === 'boolean') rateLimit = {};
return expressRateLimit({
windowMs: rateLimit.windowMs,
limit: rateLimit.limit,
message: { message: 'Too many requests' },
});
}
private createLicenseMiddleware(feature: BooleanLicenseFeature): RequestHandler {
return (_req, res, next) => {
if (!this.license.isLicensed(feature)) {
res.status(403).json({ status: 'error', message: 'Plan lacks license for this feature' });
return;
}
next();
};
}
private createScopedMiddleware(accessScope: AccessScope): RequestHandler {
return async (
req: AuthenticatedRequest<{ credentialId?: string; workflowId?: string; projectId?: string }>,
res,
next,
) => {
if (!req.user) throw new UnauthenticatedError();
const { scope, globalOnly } = accessScope;
try {
if (!(await userHasScopes(req.user, [scope], globalOnly, req.params))) {
res.status(403).json({
status: 'error',
message: RESPONSE_ERROR_MESSAGES.MISSING_SCOPE,
});
return;
}
} catch (error) {
if (error instanceof NotFoundError) {
res.status(404).json({ status: 'error', message: error.message });
return;
}
throw error;
}
next();
};
}
}