2024-09-12 09:07:18 -07:00
|
|
|
import { GlobalConfig } from '@n8n/config';
|
|
|
|
import compression from 'compression';
|
2023-01-04 02:38:48 -08:00
|
|
|
import express from 'express';
|
2024-04-29 01:55:45 -07:00
|
|
|
import { engine as expressHandlebars } from 'express-handlebars';
|
2024-09-12 09:07:18 -07:00
|
|
|
import { readFile } from 'fs/promises';
|
|
|
|
import type { Server } from 'http';
|
2023-09-01 04:54:35 -07:00
|
|
|
import isbot from 'isbot';
|
2024-09-12 09:07:18 -07:00
|
|
|
import { Container, Service } from 'typedi';
|
2023-09-01 04:54:35 -07:00
|
|
|
|
2023-01-04 02:38:48 -08:00
|
|
|
import config from '@/config';
|
2024-04-29 01:55:45 -07:00
|
|
|
import { N8N_VERSION, TEMPLATES_DIR, inDevelopment, inTest } from '@/constants';
|
2024-08-28 08:57:46 -07:00
|
|
|
import * as Db from '@/db';
|
2024-09-12 09:07:18 -07:00
|
|
|
import { OnShutdown } from '@/decorators/on-shutdown';
|
2024-08-22 02:10:37 -07:00
|
|
|
import { ExternalHooks } from '@/external-hooks';
|
2024-09-12 09:07:18 -07:00
|
|
|
import { N8nInstanceType } from '@/interfaces';
|
|
|
|
import { Logger } from '@/logger';
|
2023-08-11 00:18:33 -07:00
|
|
|
import { rawBodyReader, bodyParser, corsMiddleware } from '@/middlewares';
|
2024-09-12 09:07:18 -07:00
|
|
|
import { send, sendErrorResponse } from '@/response-helper';
|
2024-08-22 02:10:37 -07:00
|
|
|
import { WaitingForms } from '@/waiting-forms';
|
2024-09-12 09:07:18 -07:00
|
|
|
import { LiveWebhooks } from '@/webhooks/live-webhooks';
|
2024-08-22 02:10:37 -07:00
|
|
|
import { TestWebhooks } from '@/webhooks/test-webhooks';
|
|
|
|
import { WaitingWebhooks } from '@/webhooks/waiting-webhooks';
|
|
|
|
import { createWebhookHandlerFor } from '@/webhooks/webhook-request-handler';
|
2024-09-12 09:07:18 -07:00
|
|
|
|
2023-08-07 08:03:21 -07:00
|
|
|
import { generateHostInstanceId } from './databases/utils/generators';
|
2023-11-28 01:19:27 -08:00
|
|
|
import { ServiceUnavailableError } from './errors/response-errors/service-unavailable.error';
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2023-12-22 02:39:58 -08:00
|
|
|
@Service()
|
2023-01-04 02:38:48 -08:00
|
|
|
export abstract class AbstractServer {
|
2023-10-25 07:35:22 -07:00
|
|
|
protected logger: Logger;
|
|
|
|
|
2023-02-10 06:02:47 -08:00
|
|
|
protected server: Server;
|
|
|
|
|
2023-08-01 08:32:30 -07:00
|
|
|
readonly app: express.Application;
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2023-12-27 02:50:43 -08:00
|
|
|
protected externalHooks: ExternalHooks;
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2024-07-31 08:45:11 -07:00
|
|
|
protected globalConfig = Container.get(GlobalConfig);
|
2023-01-04 02:38:48 -08:00
|
|
|
|
|
|
|
protected sslKey: string;
|
|
|
|
|
|
|
|
protected sslCert: string;
|
|
|
|
|
|
|
|
protected restEndpoint: string;
|
|
|
|
|
2023-12-13 07:00:51 -08:00
|
|
|
protected endpointForm: string;
|
|
|
|
|
|
|
|
protected endpointFormTest: string;
|
|
|
|
|
|
|
|
protected endpointFormWaiting: string;
|
|
|
|
|
2023-01-04 02:38:48 -08:00
|
|
|
protected endpointWebhook: string;
|
|
|
|
|
|
|
|
protected endpointWebhookTest: string;
|
|
|
|
|
|
|
|
protected endpointWebhookWaiting: string;
|
|
|
|
|
2023-08-01 08:32:30 -07:00
|
|
|
protected webhooksEnabled = true;
|
|
|
|
|
|
|
|
protected testWebhooksEnabled = false;
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2023-08-07 08:03:21 -07:00
|
|
|
readonly uniqueInstanceId: string;
|
|
|
|
|
|
|
|
constructor(instanceType: N8nInstanceType = 'main') {
|
2023-01-04 02:38:48 -08:00
|
|
|
this.app = express();
|
|
|
|
this.app.disable('x-powered-by');
|
|
|
|
|
2024-04-29 01:55:45 -07:00
|
|
|
this.app.engine('handlebars', expressHandlebars({ defaultLayout: false }));
|
|
|
|
this.app.set('view engine', 'handlebars');
|
|
|
|
this.app.set('views', TEMPLATES_DIR);
|
|
|
|
|
2023-11-03 10:44:12 -07:00
|
|
|
const proxyHops = config.getEnv('proxy_hops');
|
|
|
|
if (proxyHops > 0) this.app.set('trust proxy', proxyHops);
|
|
|
|
|
2023-01-04 02:38:48 -08:00
|
|
|
this.sslKey = config.getEnv('ssl_key');
|
|
|
|
this.sslCert = config.getEnv('ssl_cert');
|
|
|
|
|
2024-07-31 08:45:11 -07:00
|
|
|
this.restEndpoint = this.globalConfig.endpoints.rest;
|
2023-12-13 07:00:51 -08:00
|
|
|
|
2024-07-31 08:45:11 -07:00
|
|
|
this.endpointForm = this.globalConfig.endpoints.form;
|
|
|
|
this.endpointFormTest = this.globalConfig.endpoints.formTest;
|
|
|
|
this.endpointFormWaiting = this.globalConfig.endpoints.formWaiting;
|
2023-12-13 07:00:51 -08:00
|
|
|
|
2024-07-31 08:45:11 -07:00
|
|
|
this.endpointWebhook = this.globalConfig.endpoints.webhook;
|
|
|
|
this.endpointWebhookTest = this.globalConfig.endpoints.webhookTest;
|
|
|
|
this.endpointWebhookWaiting = this.globalConfig.endpoints.webhookWaiting;
|
2023-08-07 08:03:21 -07:00
|
|
|
|
|
|
|
this.uniqueInstanceId = generateHostInstanceId(instanceType);
|
2023-10-25 07:35:22 -07:00
|
|
|
|
|
|
|
this.logger = Container.get(Logger);
|
2023-01-04 02:38:48 -08:00
|
|
|
}
|
|
|
|
|
2023-08-01 08:32:30 -07:00
|
|
|
async configure(): Promise<void> {
|
|
|
|
// Additional configuration in derived classes
|
|
|
|
}
|
|
|
|
|
2023-02-10 06:02:47 -08:00
|
|
|
private async setupErrorHandlers() {
|
2023-01-04 02:38:48 -08:00
|
|
|
const { app } = this;
|
|
|
|
|
|
|
|
// Augment errors sent to Sentry
|
|
|
|
const {
|
|
|
|
Handlers: { requestHandler, errorHandler },
|
|
|
|
} = await import('@sentry/node');
|
|
|
|
app.use(requestHandler());
|
|
|
|
app.use(errorHandler());
|
2023-02-10 06:02:47 -08:00
|
|
|
}
|
|
|
|
|
2023-08-01 08:32:30 -07:00
|
|
|
private setupCommonMiddlewares() {
|
2023-01-04 02:38:48 -08:00
|
|
|
// Compress the response data
|
2023-08-01 08:32:30 -07:00
|
|
|
this.app.use(compression());
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2023-08-01 08:32:30 -07:00
|
|
|
// Read incoming data into `rawBody`
|
2023-08-11 00:18:33 -07:00
|
|
|
this.app.use(rawBodyReader);
|
2023-01-04 02:38:48 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
private setupDevMiddlewares() {
|
|
|
|
this.app.use(corsMiddleware);
|
|
|
|
}
|
|
|
|
|
2023-02-10 06:02:47 -08:00
|
|
|
protected setupPushServer() {}
|
|
|
|
|
2023-01-04 02:38:48 -08:00
|
|
|
private async setupHealthCheck() {
|
2024-09-05 02:04:48 -07:00
|
|
|
// main health check should not care about DB connections
|
2024-03-28 01:46:39 -07:00
|
|
|
this.app.get('/healthz', async (_req, res) => {
|
2023-05-10 01:27:04 -07:00
|
|
|
res.send({ status: 'ok' });
|
|
|
|
});
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2024-09-05 02:04:48 -07:00
|
|
|
this.app.get('/healthz/readiness', async (_req, res) => {
|
|
|
|
return Db.connectionState.connected && Db.connectionState.migrated
|
|
|
|
? res.status(200).send({ status: 'ok' })
|
|
|
|
: res.status(503).send({ status: 'error' });
|
|
|
|
});
|
|
|
|
|
2023-05-10 01:27:04 -07:00
|
|
|
const { connectionState } = Db;
|
2024-03-28 01:46:39 -07:00
|
|
|
this.app.use((_req, res, next) => {
|
2023-05-10 01:27:04 -07:00
|
|
|
if (connectionState.connected) {
|
|
|
|
if (connectionState.migrated) next();
|
|
|
|
else res.send('n8n is starting up. Please wait');
|
|
|
|
} else sendErrorResponse(res, new ServiceUnavailableError('Database is not ready!'));
|
2023-01-04 02:38:48 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-05-10 01:27:04 -07:00
|
|
|
async init(): Promise<void> {
|
2024-07-31 08:45:11 -07:00
|
|
|
const { app, sslKey, sslCert } = this;
|
|
|
|
const { protocol } = this.globalConfig;
|
2023-01-04 02:38:48 -08:00
|
|
|
|
|
|
|
if (protocol === 'https' && sslKey && sslCert) {
|
|
|
|
const https = await import('https');
|
2023-02-10 06:02:47 -08:00
|
|
|
this.server = https.createServer(
|
2023-01-04 02:38:48 -08:00
|
|
|
{
|
|
|
|
key: await readFile(this.sslKey, 'utf8'),
|
|
|
|
cert: await readFile(this.sslCert, 'utf8'),
|
|
|
|
},
|
|
|
|
app,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
const http = await import('http');
|
2023-02-10 06:02:47 -08:00
|
|
|
this.server = http.createServer(app);
|
2023-01-04 02:38:48 -08:00
|
|
|
}
|
|
|
|
|
2024-07-29 05:32:20 -07:00
|
|
|
const { port, listen_address: address } = Container.get(GlobalConfig);
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2023-02-10 06:02:47 -08:00
|
|
|
this.server.on('error', (error: Error & { code: string }) => {
|
2023-01-04 02:38:48 -08:00
|
|
|
if (error.code === 'EADDRINUSE') {
|
2024-05-03 06:24:27 -07:00
|
|
|
this.logger.info(
|
2024-07-29 05:32:20 -07:00
|
|
|
`n8n's port ${port} is already in use. Do you have another instance of n8n running already?`,
|
2023-01-04 02:38:48 -08:00
|
|
|
);
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2024-07-29 05:32:20 -07:00
|
|
|
await new Promise<void>((resolve) => this.server.listen(port, address, () => resolve()));
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2023-07-25 09:17:34 -07:00
|
|
|
this.externalHooks = Container.get(ExternalHooks);
|
|
|
|
|
2023-05-10 01:27:04 -07:00
|
|
|
await this.setupHealthCheck();
|
|
|
|
|
2024-07-29 05:32:20 -07:00
|
|
|
this.logger.info(`n8n ready on ${address}, port ${port}`);
|
2023-05-10 01:27:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
async start(): Promise<void> {
|
2023-08-01 08:32:30 -07:00
|
|
|
if (!inTest) {
|
|
|
|
await this.setupErrorHandlers();
|
|
|
|
this.setupPushServer();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.setupCommonMiddlewares();
|
|
|
|
|
|
|
|
// Setup webhook handlers before bodyParser, to let the Webhook node handle binary data in requests
|
|
|
|
if (this.webhooksEnabled) {
|
2024-08-12 23:49:24 -07:00
|
|
|
const liveWebhooksRequestHandler = createWebhookHandlerFor(Container.get(LiveWebhooks));
|
|
|
|
// Register a handler for live forms
|
|
|
|
this.app.all(`/${this.endpointForm}/:path(*)`, liveWebhooksRequestHandler);
|
2023-12-13 07:00:51 -08:00
|
|
|
|
2024-08-12 23:49:24 -07:00
|
|
|
// Register a handler for live webhooks
|
|
|
|
this.app.all(`/${this.endpointWebhook}/:path(*)`, liveWebhooksRequestHandler);
|
2023-12-13 07:00:51 -08:00
|
|
|
|
|
|
|
// Register a handler for waiting forms
|
|
|
|
this.app.all(
|
|
|
|
`/${this.endpointFormWaiting}/:path/:suffix?`,
|
2024-08-09 07:08:15 -07:00
|
|
|
createWebhookHandlerFor(Container.get(WaitingForms)),
|
2023-08-01 08:32:30 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
// Register a handler for waiting webhooks
|
|
|
|
this.app.all(
|
|
|
|
`/${this.endpointWebhookWaiting}/:path/:suffix?`,
|
2024-08-09 07:08:15 -07:00
|
|
|
createWebhookHandlerFor(Container.get(WaitingWebhooks)),
|
2023-08-01 08:32:30 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.testWebhooksEnabled) {
|
2024-08-09 07:08:15 -07:00
|
|
|
const testWebhooksRequestHandler = createWebhookHandlerFor(Container.get(TestWebhooks));
|
2023-08-01 08:32:30 -07:00
|
|
|
|
2023-12-13 07:00:51 -08:00
|
|
|
// Register a handler
|
2024-08-09 07:08:15 -07:00
|
|
|
this.app.all(`/${this.endpointFormTest}/:path(*)`, testWebhooksRequestHandler);
|
|
|
|
this.app.all(`/${this.endpointWebhookTest}/:path(*)`, testWebhooksRequestHandler);
|
2023-08-01 08:32:30 -07:00
|
|
|
}
|
|
|
|
|
2023-09-01 04:54:35 -07:00
|
|
|
// Block bots from scanning the application
|
|
|
|
const checkIfBot = isbot.spawn(['bot']);
|
|
|
|
this.app.use((req, res, next) => {
|
|
|
|
const userAgent = req.headers['user-agent'];
|
2023-09-12 10:57:25 -07:00
|
|
|
if (userAgent && checkIfBot(userAgent)) {
|
2023-10-25 07:35:22 -07:00
|
|
|
this.logger.info(`Blocked ${req.method} ${req.url} for "${userAgent}"`);
|
2023-09-01 04:54:35 -07:00
|
|
|
res.status(204).end();
|
|
|
|
} else next();
|
|
|
|
});
|
|
|
|
|
2023-01-04 02:38:48 -08:00
|
|
|
if (inDevelopment) {
|
|
|
|
this.setupDevMiddlewares();
|
|
|
|
}
|
|
|
|
|
2024-05-17 08:55:29 -07:00
|
|
|
if (this.testWebhooksEnabled) {
|
|
|
|
const testWebhooks = Container.get(TestWebhooks);
|
|
|
|
// Removes a test webhook
|
|
|
|
// TODO UM: check if this needs validation with user management.
|
|
|
|
this.app.delete(
|
|
|
|
`/${this.restEndpoint}/test-webhook/:id`,
|
|
|
|
send(async (req) => await testWebhooks.cancelWebhook(req.params.id)),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-08-11 00:18:33 -07:00
|
|
|
// Setup body parsing middleware after the webhook handlers are setup
|
|
|
|
this.app.use(bodyParser);
|
2023-08-01 08:32:30 -07:00
|
|
|
|
2023-01-04 02:38:48 -08:00
|
|
|
await this.configure();
|
|
|
|
|
2023-08-01 08:32:30 -07:00
|
|
|
if (!inTest) {
|
2024-05-03 06:24:27 -07:00
|
|
|
this.logger.info(`Version: ${N8N_VERSION}`);
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2023-08-01 08:32:30 -07:00
|
|
|
const defaultLocale = config.getEnv('defaultLocale');
|
|
|
|
if (defaultLocale !== 'en') {
|
2024-05-03 06:24:27 -07:00
|
|
|
this.logger.info(`Locale: ${defaultLocale}`);
|
2023-08-01 08:32:30 -07:00
|
|
|
}
|
2023-01-04 02:38:48 -08:00
|
|
|
|
2023-08-01 08:32:30 -07:00
|
|
|
await this.externalHooks.run('n8n.ready', [this, config]);
|
|
|
|
}
|
2023-01-04 02:38:48 -08:00
|
|
|
}
|
2023-12-22 02:39:58 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Stops the HTTP(S) server from accepting new connections. Gives all
|
|
|
|
* connections configured amount of time to finish their work and
|
|
|
|
* then closes them forcefully.
|
|
|
|
*/
|
|
|
|
@OnShutdown()
|
|
|
|
async onShutdown(): Promise<void> {
|
|
|
|
if (!this.server) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-07-31 08:45:11 -07:00
|
|
|
const { protocol } = this.globalConfig;
|
|
|
|
|
|
|
|
this.logger.debug(`Shutting down ${protocol} server`);
|
2023-12-22 02:39:58 -08:00
|
|
|
|
|
|
|
this.server.close((error) => {
|
|
|
|
if (error) {
|
2024-07-31 08:45:11 -07:00
|
|
|
this.logger.error(`Error while shutting down ${protocol} server`, { error });
|
2023-12-22 02:39:58 -08:00
|
|
|
}
|
|
|
|
|
2024-07-31 08:45:11 -07:00
|
|
|
this.logger.debug(`${protocol} server shut down`);
|
2023-12-22 02:39:58 -08:00
|
|
|
});
|
|
|
|
}
|
2023-01-04 02:38:48 -08:00
|
|
|
}
|