n8n
Version:
n8n Workflow Automation Tool
334 lines • 18.4 kB
JavaScript
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
var __metadata = (this && this.__metadata) || function (k, v) {
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.Server = void 0;
const typedi_1 = require("typedi");
const child_process_1 = require("child_process");
const promises_1 = require("fs/promises");
const util_1 = require("util");
const cookie_parser_1 = __importDefault(require("cookie-parser"));
const express_1 = __importDefault(require("express"));
const helmet_1 = __importDefault(require("helmet"));
const n8n_core_1 = require("n8n-core");
const google_timezones_json_1 = __importDefault(require("google-timezones-json"));
const config_1 = __importDefault(require("./config"));
const Queue_1 = require("./Queue");
const workflows_controller_1 = require("./workflows/workflows.controller");
const constants_1 = require("./constants");
const credentials_controller_1 = require("./credentials/credentials.controller");
const decorators_1 = require("./decorators");
const auth_controller_1 = require("./controllers/auth.controller");
const binaryData_controller_1 = require("./controllers/binaryData.controller");
const curl_controller_1 = require("./controllers/curl.controller");
const dynamicNodeParameters_controller_1 = require("./controllers/dynamicNodeParameters.controller");
const me_controller_1 = require("./controllers/me.controller");
const mfa_controller_1 = require("./controllers/mfa.controller");
const nodeTypes_controller_1 = require("./controllers/nodeTypes.controller");
const oAuth1Credential_controller_1 = require("./controllers/oauth/oAuth1Credential.controller");
const oAuth2Credential_controller_1 = require("./controllers/oauth/oAuth2Credential.controller");
const owner_controller_1 = require("./controllers/owner.controller");
const passwordReset_controller_1 = require("./controllers/passwordReset.controller");
const tags_controller_1 = require("./controllers/tags.controller");
const translation_controller_1 = require("./controllers/translation.controller");
const users_controller_1 = require("./controllers/users.controller");
const workflowStatistics_controller_1 = require("./controllers/workflowStatistics.controller");
const ExternalSecrets_controller_ee_1 = require("./ExternalSecrets/ExternalSecrets.controller.ee");
const executions_controller_1 = require("./executions/executions.controller");
const PublicApi_1 = require("./PublicApi");
const CredentialsOverwrites_1 = require("./CredentialsOverwrites");
const LoadNodesAndCredentials_1 = require("./LoadNodesAndCredentials");
const ResponseHelper = __importStar(require("./ResponseHelper"));
const eventBus_controller_1 = require("./eventbus/eventBus.controller");
const eventBus_controller_ee_1 = require("./eventbus/eventBus.controller.ee");
const license_controller_1 = require("./license/license.controller");
const push_1 = require("./push");
const helpers_1 = require("./Ldap/helpers");
const AbstractServer_1 = require("./AbstractServer");
const posthog_1 = require("./posthog");
const MessageEventBus_1 = require("./eventbus/MessageEventBus/MessageEventBus");
const InternalHooks_1 = require("./InternalHooks");
const saml_controller_ee_1 = require("./sso/saml/routes/saml.controller.ee");
const saml_service_ee_1 = require("./sso/saml/saml.service.ee");
const variables_controller_ee_1 = require("./environments/variables/variables.controller.ee");
const sourceControl_service_ee_1 = require("./environments/sourceControl/sourceControl.service.ee");
const sourceControl_controller_ee_1 = require("./environments/sourceControl/sourceControl.controller.ee");
const ai_controller_1 = require("./controllers/ai.controller");
const helpers_2 = require("./Mfa/helpers");
const activeWorkflows_controller_1 = require("./controllers/activeWorkflows.controller");
const orchestration_controller_1 = require("./controllers/orchestration.controller");
const workflowHistory_controller_ee_1 = require("./workflows/workflowHistory/workflowHistory.controller.ee");
const invitation_controller_1 = require("./controllers/invitation.controller");
const orchestration_service_1 = require("./services/orchestration.service");
const project_controller_1 = require("./controllers/project.controller");
const role_controller_1 = require("./controllers/role.controller");
const exec = (0, util_1.promisify)(child_process_1.exec);
let Server = class Server extends AbstractServer_1.AbstractServer {
constructor() {
super('main');
this.testWebhooksEnabled = true;
this.webhooksEnabled = !config_1.default.getEnv('endpoints.disableProductionWebhooksOnMainProcess');
}
async start() {
this.loadNodesAndCredentials = typedi_1.Container.get(LoadNodesAndCredentials_1.LoadNodesAndCredentials);
if (!config_1.default.getEnv('endpoints.disableUi')) {
this.frontendService = typedi_1.Container.get(require('./services/frontend.service').FrontendService);
}
this.presetCredentialsLoaded = false;
this.endpointPresetCredentials = config_1.default.getEnv('credentials.overwrite.endpoint');
await super.start();
this.logger.debug(`Server ID: ${this.uniqueInstanceId}`);
if (constants_1.inDevelopment && process.env.N8N_DEV_RELOAD === 'true') {
void this.loadNodesAndCredentials.setupHotReload();
}
void typedi_1.Container.get(InternalHooks_1.InternalHooks).onServerStarted();
}
async registerControllers() {
const { app } = this;
const controllers = [
eventBus_controller_1.EventBusController,
eventBus_controller_ee_1.EventBusControllerEE,
auth_controller_1.AuthController,
license_controller_1.LicenseController,
oAuth1Credential_controller_1.OAuth1CredentialController,
oAuth2Credential_controller_1.OAuth2CredentialController,
owner_controller_1.OwnerController,
me_controller_1.MeController,
dynamicNodeParameters_controller_1.DynamicNodeParametersController,
nodeTypes_controller_1.NodeTypesController,
passwordReset_controller_1.PasswordResetController,
tags_controller_1.TagsController,
translation_controller_1.TranslationController,
users_controller_1.UsersController,
saml_controller_ee_1.SamlController,
sourceControl_controller_ee_1.SourceControlController,
workflowStatistics_controller_1.WorkflowStatisticsController,
ExternalSecrets_controller_ee_1.ExternalSecretsController,
orchestration_controller_1.OrchestrationController,
workflowHistory_controller_ee_1.WorkflowHistoryController,
binaryData_controller_1.BinaryDataController,
variables_controller_ee_1.VariablesController,
invitation_controller_1.InvitationController,
variables_controller_ee_1.VariablesController,
activeWorkflows_controller_1.ActiveWorkflowsController,
workflows_controller_1.WorkflowsController,
executions_controller_1.ExecutionsController,
credentials_controller_1.CredentialsController,
ai_controller_1.AIController,
project_controller_1.ProjectController,
role_controller_1.RoleController,
curl_controller_1.CurlController,
];
if (process.env.NODE_ENV !== 'production' &&
typedi_1.Container.get(orchestration_service_1.OrchestrationService).isMultiMainSetupEnabled) {
const { DebugController } = await Promise.resolve().then(() => __importStar(require('./controllers/debug.controller')));
controllers.push(DebugController);
}
if ((0, helpers_1.isLdapEnabled)()) {
const { LdapService } = await Promise.resolve().then(() => __importStar(require('./Ldap/ldap.service')));
const { LdapController } = await require('./Ldap/ldap.controller');
await typedi_1.Container.get(LdapService).init();
controllers.push(LdapController);
}
if (config_1.default.getEnv('nodes.communityPackages.enabled')) {
const { CommunityPackagesController } = await Promise.resolve().then(() => __importStar(require('./controllers/communityPackages.controller')));
controllers.push(CommunityPackagesController);
}
if (constants_1.inE2ETests) {
const { E2EController } = await Promise.resolve().then(() => __importStar(require('./controllers/e2e.controller')));
controllers.push(E2EController);
}
if ((0, helpers_2.isMfaFeatureEnabled)()) {
controllers.push(mfa_controller_1.MFAController);
}
if (!config_1.default.getEnv('endpoints.disableUi')) {
const { CtaController } = await Promise.resolve().then(() => __importStar(require('./controllers/cta.controller')));
controllers.push(CtaController);
}
controllers.forEach((controller) => (0, decorators_1.registerController)(app, controller));
}
async configure() {
if (config_1.default.getEnv('endpoints.metrics.enable')) {
const { MetricsService } = await Promise.resolve().then(() => __importStar(require('./services/metrics.service')));
await typedi_1.Container.get(MetricsService).configureMetrics(this.app);
}
const { frontendService } = this;
if (frontendService) {
frontendService.addToSettings({
isNpmAvailable: await exec('npm --version')
.then(() => true)
.catch(() => false),
versionCli: constants_1.N8N_VERSION,
});
await this.externalHooks.run('frontend.settings', [frontendService.getSettings()]);
}
await typedi_1.Container.get(posthog_1.PostHogClient).init();
const publicApiEndpoint = config_1.default.getEnv('publicApi.path');
if ((0, PublicApi_1.isApiEnabled)()) {
const { apiRouters, apiLatestVersion } = await (0, PublicApi_1.loadPublicApiVersions)(publicApiEndpoint);
this.app.use(...apiRouters);
if (frontendService) {
frontendService.settings.publicApi.latestVersion = apiLatestVersion;
}
}
this.app.use((req, _, next) => {
req.browserId = req.headers['browser-id'];
next();
});
this.app.use((0, cookie_parser_1.default)());
const { restEndpoint, app } = this;
(0, push_1.setupPushHandler)(restEndpoint, app);
if (config_1.default.getEnv('executions.mode') === 'queue') {
await typedi_1.Container.get(Queue_1.Queue).init();
}
await (0, helpers_2.handleMfaDisable)();
await this.registerControllers();
try {
await typedi_1.Container.get(saml_service_ee_1.SamlService).init();
}
catch (error) {
this.logger.warn(`SAML initialization failed: ${error.message}`);
}
try {
await typedi_1.Container.get(sourceControl_service_ee_1.SourceControlService).init();
}
catch (error) {
this.logger.warn(`Source Control initialization failed: ${error.message}`);
}
this.app.get(`/${this.restEndpoint}/options/timezones`, ResponseHelper.send(async () => google_timezones_json_1.default));
if (frontendService) {
this.app.get(`/${this.restEndpoint}/settings`, ResponseHelper.send(async (req) => frontendService.getSettings(req.headers['push-ref'])));
}
const eventBus = typedi_1.Container.get(MessageEventBus_1.MessageEventBus);
await eventBus.initialize();
if (this.endpointPresetCredentials !== '') {
this.app.post(`/${this.endpointPresetCredentials}`, async (req, res) => {
if (!this.presetCredentialsLoaded) {
const body = req.body;
if (req.contentType !== 'application/json') {
ResponseHelper.sendErrorResponse(res, new Error('Body must be a valid JSON, make sure the content-type is application/json'));
return;
}
typedi_1.Container.get(CredentialsOverwrites_1.CredentialsOverwrites).setData(body);
await (frontendService === null || frontendService === void 0 ? void 0 : frontendService.generateTypes());
this.presetCredentialsLoaded = true;
ResponseHelper.sendSuccessResponse(res, { success: true }, true, 200);
}
else {
ResponseHelper.sendErrorResponse(res, new Error('Preset credentials can be set once'));
}
});
}
const maxAge = constants_1.Time.days.toMilliseconds;
const cacheOptions = constants_1.inE2ETests || constants_1.inDevelopment ? {} : { maxAge };
const { staticCacheDir } = typedi_1.Container.get(n8n_core_1.InstanceSettings);
if (frontendService) {
const serveIcons = async (req, res) => {
let { scope, packageName } = req.params;
if (scope)
packageName = `@${scope}/${packageName}`;
const filePath = this.loadNodesAndCredentials.resolveIcon(packageName, req.originalUrl);
if (filePath) {
try {
await (0, promises_1.access)(filePath);
return res.sendFile(filePath, cacheOptions);
}
catch { }
}
res.sendStatus(404);
};
this.app.use('/icons/@:scope/:packageName/*/*.(svg|png)', serveIcons);
this.app.use('/icons/:packageName/*/*.(svg|png)', serveIcons);
const isTLSEnabled = this.protocol === 'https' && !!(this.sslKey && this.sslCert);
const isPreviewMode = process.env.N8N_PREVIEW_MODE === 'true';
const securityHeadersMiddleware = (0, helmet_1.default)({
contentSecurityPolicy: false,
xFrameOptions: isPreviewMode || constants_1.inE2ETests || constants_1.inDevelopment ? false : { action: 'sameorigin' },
dnsPrefetchControl: false,
ieNoOpen: false,
xPoweredBy: false,
strictTransportSecurity: isTLSEnabled
? {
maxAge: 180 * constants_1.Time.days.toSeconds,
includeSubDomains: false,
preload: false,
}
: false,
});
const nonUIRoutes = [
'assets',
'types',
'healthz',
'metrics',
'e2e',
this.restEndpoint,
this.endpointPresetCredentials,
(0, PublicApi_1.isApiEnabled)() ? '' : publicApiEndpoint,
...config_1.default.getEnv('endpoints.additionalNonUIRoutes').split(':'),
].filter((u) => !!u);
const nonUIRoutesRegex = new RegExp(`^/(${nonUIRoutes.join('|')})/?.*$`);
const historyApiHandler = (req, res, next) => {
const { method, headers: { accept }, } = req;
if (method === 'GET' &&
accept &&
(accept.includes('text/html') || accept.includes('*/*')) &&
!nonUIRoutesRegex.test(req.path)) {
res.setHeader('Cache-Control', 'no-cache, no-store, must-revalidate');
securityHeadersMiddleware(req, res, () => {
res.sendFile('index.html', { root: staticCacheDir, maxAge, lastModified: true });
});
}
else {
next();
}
};
this.app.use('/', express_1.default.static(staticCacheDir, cacheOptions), express_1.default.static(constants_1.EDITOR_UI_DIST_DIR, cacheOptions), historyApiHandler);
}
else {
this.app.use('/', express_1.default.static(staticCacheDir, cacheOptions));
}
}
setupPushServer() {
const { restEndpoint, server, app } = this;
(0, push_1.setupPushServer)(restEndpoint, server, app);
}
};
exports.Server = Server;
exports.Server = Server = __decorate([
(0, typedi_1.Service)(),
__metadata("design:paramtypes", [])
], Server);
//# sourceMappingURL=Server.js.map
;