UNPKG

n8n

Version:

n8n Workflow Automation Tool

373 lines 19.1 kB
"use strict"; 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 () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __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 config_1 = require("@n8n/config"); const di_1 = require("@n8n/di"); const cookie_parser_1 = __importDefault(require("cookie-parser")); const express_1 = __importDefault(require("express")); const promises_1 = require("fs/promises"); const helmet_1 = __importDefault(require("helmet")); const lodash_1 = require("lodash"); const n8n_core_1 = require("n8n-core"); const n8n_workflow_1 = require("n8n-workflow"); const path_1 = require("path"); const abstract_server_1 = require("./abstract-server"); const config_2 = __importDefault(require("./config")); const constants_1 = require("./constants"); const credentials_overwrites_1 = require("./credentials-overwrites"); const decorators_1 = require("./decorators"); const message_event_bus_1 = require("./eventbus/message-event-bus/message-event-bus"); const event_service_1 = require("./events/event.service"); const log_streaming_event_relay_1 = require("./events/relays/log-streaming.event-relay"); const helpers_ee_1 = require("./ldap.ee/helpers.ee"); const load_nodes_and_credentials_1 = require("./load-nodes-and-credentials"); const helpers_1 = require("./mfa/helpers"); const posthog_1 = require("./posthog"); const public_api_1 = require("./public-api"); const push_1 = require("./push"); const ResponseHelper = __importStar(require("./response-helper")); require("./controllers/active-workflows.controller"); require("./controllers/annotation-tags.controller.ee"); require("./controllers/auth.controller"); require("./controllers/binary-data.controller"); require("./controllers/ai.controller"); require("./controllers/dynamic-node-parameters.controller"); require("./controllers/invitation.controller"); require("./controllers/me.controller"); require("./controllers/node-types.controller"); require("./controllers/oauth/oauth1-credential.controller"); require("./controllers/oauth/oauth2-credential.controller"); require("./controllers/orchestration.controller"); require("./controllers/owner.controller"); require("./controllers/password-reset.controller"); require("./controllers/project.controller"); require("./controllers/role.controller"); require("./controllers/tags.controller"); require("./controllers/translation.controller"); require("./controllers/folder.controller"); require("./controllers/users.controller"); require("./controllers/user-settings.controller"); require("./controllers/workflow-statistics.controller"); require("./controllers/api-keys.controller"); require("./credentials/credentials.controller"); require("./eventbus/event-bus.controller"); require("./events/events.controller"); require("./executions/executions.controller"); require("./external-secrets.ee/external-secrets.controller.ee"); require("./license/license.controller"); require("./evaluation.ee/test-definitions.controller.ee"); require("./evaluation.ee/test-runs.controller.ee"); require("./workflows/workflow-history.ee/workflow-history.controller.ee"); require("./workflows/workflows.controller"); let Server = class Server extends abstract_server_1.AbstractServer { constructor(loadNodesAndCredentials, postHogClient, eventService, instanceSettings) { super(); this.loadNodesAndCredentials = loadNodesAndCredentials; this.postHogClient = postHogClient; this.eventService = eventService; this.instanceSettings = instanceSettings; this.testWebhooksEnabled = true; this.webhooksEnabled = !this.globalConfig.endpoints.disableProductionWebhooksOnMainProcess; } async start() { if (!this.globalConfig.endpoints.disableUi) { const { FrontendService } = await Promise.resolve().then(() => __importStar(require('./services/frontend.service'))); this.frontendService = di_1.Container.get(FrontendService); } this.presetCredentialsLoaded = false; this.endpointPresetCredentials = this.globalConfig.credentials.overwrite.endpoint; await super.start(); this.logger.debug(`Server ID: ${this.instanceSettings.hostId}`); if (constants_1.inDevelopment && process.env.N8N_DEV_RELOAD === 'true') { void this.loadNodesAndCredentials.setupHotReload(); } this.eventService.emit('server-started'); } async registerAdditionalControllers() { if (!constants_1.inProduction && this.instanceSettings.isMultiMain) { await Promise.resolve().then(() => __importStar(require('./controllers/debug.controller'))); } if ((0, helpers_ee_1.isLdapEnabled)()) { const { LdapService } = await Promise.resolve().then(() => __importStar(require('./ldap.ee/ldap.service.ee'))); await Promise.resolve().then(() => __importStar(require('./ldap.ee/ldap.controller.ee'))); await di_1.Container.get(LdapService).init(); } if (this.globalConfig.nodes.communityPackages.enabled) { await Promise.resolve().then(() => __importStar(require('./controllers/community-packages.controller'))); } if (constants_1.inE2ETests) { await Promise.resolve().then(() => __importStar(require('./controllers/e2e.controller'))); } if ((0, helpers_1.isMfaFeatureEnabled)()) { await Promise.resolve().then(() => __importStar(require('./controllers/mfa.controller'))); } if (!this.globalConfig.endpoints.disableUi) { await Promise.resolve().then(() => __importStar(require('./controllers/cta.controller'))); } if (!this.globalConfig.tags.disabled) { await Promise.resolve().then(() => __importStar(require('./controllers/tags.controller'))); } try { const { SamlService } = await Promise.resolve().then(() => __importStar(require('./sso.ee/saml/saml.service.ee'))); await di_1.Container.get(SamlService).init(); await Promise.resolve().then(() => __importStar(require('./sso.ee/saml/routes/saml.controller.ee'))); } catch (error) { this.logger.warn(`SAML initialization failed: ${error.message}`); } try { const { SourceControlService } = await Promise.resolve().then(() => __importStar(require('./environments.ee/source-control/source-control.service.ee'))); await di_1.Container.get(SourceControlService).init(); await Promise.resolve().then(() => __importStar(require('./environments.ee/source-control/source-control.controller.ee'))); await Promise.resolve().then(() => __importStar(require('./environments.ee/variables/variables.controller.ee'))); } catch (error) { this.logger.warn(`Source Control initialization failed: ${error.message}`); } } async configure() { if (this.globalConfig.endpoints.metrics.enable) { const { PrometheusMetricsService } = await Promise.resolve().then(() => __importStar(require('./metrics/prometheus-metrics.service'))); await di_1.Container.get(PrometheusMetricsService).init(this.app); } const { frontendService } = this; if (frontendService) { await this.externalHooks.run('frontend.settings', [frontendService.getSettings()]); } await this.postHogClient.init(); const publicApiEndpoint = this.globalConfig.publicApi.path; if ((0, public_api_1.isApiEnabled)()) { const { apiRouters, apiLatestVersion } = await (0, public_api_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; const push = di_1.Container.get(push_1.Push); push.setupPushHandler(restEndpoint, app); if (push.isBidirectional) { const { CollaborationService } = await Promise.resolve().then(() => __importStar(require('./collaboration/collaboration.service'))); const collaborationService = di_1.Container.get(CollaborationService); collaborationService.init(); } else { this.logger.warn('Collaboration features are disabled because push is configured unidirectional. Use N8N_PUSH_BACKEND=websocket environment variable to enable them.'); } if (config_2.default.getEnv('executions.mode') === 'queue') { const { ScalingService } = await Promise.resolve().then(() => __importStar(require('./scaling/scaling.service'))); await di_1.Container.get(ScalingService).setupQueue(); } await (0, helpers_1.handleMfaDisable)(); await this.registerAdditionalControllers(); di_1.Container.get(decorators_1.ControllerRegistry).activate(app); const tzDataFile = (0, path_1.resolve)(constants_1.CLI_DIR, 'dist/timezones.json'); this.app.get(`/${this.restEndpoint}/options/timezones`, (_, res) => res.sendFile(tzDataFile)); if (frontendService) { this.app.get(`/${this.restEndpoint}/settings`, ResponseHelper.send(async () => frontendService.getSettings())); this.app.get(`/${this.restEndpoint}/sentry.js`, (_, res) => { res.type('js'); res.write('window.sentry='); res.write(JSON.stringify({ dsn: this.globalConfig.sentry.frontendDsn, environment: process.env.ENVIRONMENT || 'development', serverName: process.env.DEPLOYMENT_NAME, release: `n8n@${constants_1.N8N_VERSION}`, })); res.end(); }); } const eventBus = di_1.Container.get(message_event_bus_1.MessageEventBus); await eventBus.initialize(); di_1.Container.get(log_streaming_event_relay_1.LogStreamingEventRelay).init(); 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; } di_1.Container.get(credentials_overwrites_1.CredentialsOverwrites).setData(body); await 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 } = di_1.Container.get(n8n_core_1.InstanceSettings); if (frontendService) { this.app.use([ '/icons/{@:scope/}:packageName/*path/*file.svg', '/icons/{@:scope/}:packageName/*path/*file.png', ], 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, { maxAge, dotfiles: 'allow' }); } catch { } } res.sendStatus(404); }); const serveSchemas = async (req, res) => { const { node, version, resource, operation } = req.params; const filePath = this.loadNodesAndCredentials.resolveSchema({ node, resource, operation, version, }); if (filePath) { try { await (0, promises_1.access)(filePath); return res.sendFile(filePath, cacheOptions); } catch { } } res.sendStatus(404); }; this.app.use('/schemas/:node/:version{/:resource}{/:operation}.json', serveSchemas); const isTLSEnabled = this.globalConfig.protocol === 'https' && !!(this.sslKey && this.sslCert); const isPreviewMode = process.env.N8N_PREVIEW_MODE === 'true'; const cspDirectives = (0, n8n_workflow_1.jsonParse)(di_1.Container.get(config_1.SecurityConfig).contentSecurityPolicy, { errorMessage: 'The contentSecurityPolicy is not valid JSON.', }); const securityHeadersMiddleware = (0, helmet_1.default)({ contentSecurityPolicy: (0, lodash_1.isEmpty)(cspDirectives) ? false : { useDefaults: false, directives: { ...cspDirectives, }, }, 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 = [ 'favicon.ico', 'assets', 'static', 'types', 'healthz', 'metrics', 'e2e', this.restEndpoint, this.endpointPresetCredentials, (0, public_api_1.isApiEnabled)() ? '' : publicApiEndpoint, ...this.globalConfig.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('*/*')) && !req.path.endsWith('.wasm') && !nonUIRoutesRegex.test(req.path)) { res.setHeader('Cache-Control', 'no-cache, no-store, must-revalidate, proxy-revalidate'); securityHeadersMiddleware(req, res, () => { res.sendFile('index.html', { root: staticCacheDir, maxAge: 0, lastModified: false }); }); } else { next(); } }; const setCustomCacheHeader = (res) => { if (/^\/types\/(nodes|credentials).json$/.test(res.req.url)) { res.setHeader('Cache-Control', 'no-cache, must-revalidate'); } }; this.app.use('/', historyApiHandler, express_1.default.static(staticCacheDir, { ...cacheOptions, setHeaders: setCustomCacheHeader, }), express_1.default.static(constants_1.EDITOR_UI_DIST_DIR, cacheOptions)); } else { this.app.use('/', express_1.default.static(staticCacheDir, cacheOptions)); } } setupPushServer() { const { restEndpoint, server, app } = this; di_1.Container.get(push_1.Push).setupPushServer(restEndpoint, server, app); } }; exports.Server = Server; exports.Server = Server = __decorate([ (0, di_1.Service)(), __metadata("design:paramtypes", [load_nodes_and_credentials_1.LoadNodesAndCredentials, posthog_1.PostHogClient, event_service_1.EventService, n8n_core_1.InstanceSettings]) ], Server); //# sourceMappingURL=server.js.map