From a573db2ef78024a4254ff0f468dc47b12148aa28 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E0=A4=95=E0=A4=BE=E0=A4=B0=E0=A4=A4=E0=A5=8B=E0=A4=AB?= =?UTF-8?q?=E0=A5=8D=E0=A4=AB=E0=A5=87=E0=A4=B2=E0=A4=B8=E0=A5=8D=E0=A4=95?= =?UTF-8?q?=E0=A5=8D=E0=A4=B0=E0=A4=BF=E0=A4=AA=E0=A5=8D=E0=A4=9F=E2=84=A2?= Date: Wed, 11 Jan 2023 18:28:35 +0100 Subject: [PATCH] fix: Report app startup and DB migration errors to Sentry (#5127) --- packages/cli/src/commands/start.ts | 48 +++++++++--------- packages/cli/src/commands/webhook.ts | 49 +++++++++---------- packages/cli/src/commands/worker.ts | 54 +++++++++++---------- packages/workflow/src/ErrorReporterProxy.ts | 2 +- 4 files changed, 74 insertions(+), 79 deletions(-) diff --git a/packages/cli/src/commands/start.ts b/packages/cli/src/commands/start.ts index bb12929429f33..32d66158dcc54 100644 --- a/packages/cli/src/commands/start.ts +++ b/packages/cli/src/commands/start.ts @@ -1,9 +1,5 @@ -/* eslint-disable @typescript-eslint/no-non-null-assertion */ /* eslint-disable @typescript-eslint/await-thenable */ -/* eslint-disable @typescript-eslint/no-unsafe-assignment */ -/* eslint-disable @typescript-eslint/explicit-module-boundary-types */ /* eslint-disable @typescript-eslint/unbound-method */ -/* eslint-disable no-console */ /* eslint-disable @typescript-eslint/no-unsafe-call */ /* eslint-disable @typescript-eslint/no-unsafe-member-access */ import path from 'path'; @@ -17,7 +13,7 @@ import replaceStream from 'replacestream'; import { promisify } from 'util'; import glob from 'fast-glob'; -import { LoggerProxy, sleep } from 'n8n-workflow'; +import { LoggerProxy, ErrorReporterProxy as ErrorReporter, sleep } from 'n8n-workflow'; import { createHash } from 'crypto'; import config from '@/config'; @@ -49,7 +45,20 @@ const open = require('open'); const pipeline = promisify(stream.pipeline); let activeWorkflowRunner: ActiveWorkflowRunner.ActiveWorkflowRunner | undefined; -let processExitCode = 0; + +const exitWithCrash = async (message: string, error: unknown) => { + ErrorReporter.error(new Error(message, { cause: error }), { level: 'fatal' }); + await sleep(2000); + process.exit(1); +}; + +const exitSuccessFully = async () => { + try { + await CrashJournal.cleanup(); + } finally { + process.exit(); + } +}; export class Start extends Command { static description = 'Starts n8n. Makes Web-UI available and starts active workflows'; @@ -101,12 +110,6 @@ export class Start extends Command { static async stopProcess() { getLogger().info('\nStopping n8n...'); - const exit = () => { - CrashJournal.cleanup().finally(() => { - process.exit(processExitCode); - }); - }; - try { // Stop with trying to activate workflows that could not be activated activeWorkflowRunner?.removeAllQueuedWorkflowActivations(); @@ -114,11 +117,11 @@ export class Start extends Command { const externalHooks = ExternalHooks(); await externalHooks.run('n8n.stop', []); - setTimeout(() => { + setTimeout(async () => { // In case that something goes wrong with shutdown we // kill after max. 30 seconds no matter what console.log('process exited after 30s'); - exit(); + await exitSuccessFully(); }, 30000); await InternalHooksManager.getInstance().onN8nStop(); @@ -159,10 +162,10 @@ export class Start extends Command { //finally shut down Event Bus await eventBus.close(); } catch (error) { - console.error('There was an error shutting down n8n.', error); + await exitWithCrash('There was an error shutting down n8n.', error); } - exit(); + await exitSuccessFully(); } static async generateStaticAssets() { @@ -236,12 +239,9 @@ export class Start extends Command { try { // Start directly with the init of the database to improve startup time - const startDbInitPromise = Db.init().catch((error: Error) => { - logger.error(`There was an error initializing DB: "${error.message}"`); - - processExitCode = 1; - process.emit('exit', processExitCode); - }); + const startDbInitPromise = Db.init().catch(async (error: Error) => + exitWithCrash('There was an error initializing DB', error), + ); // Make sure the settings exist const userSettings = await UserSettings.prepareUserSettings(); @@ -456,9 +456,7 @@ export class Start extends Command { }); } } catch (error) { - console.error('There was an error', error); - processExitCode = 1; - process.emit('exit', processExitCode); + await exitWithCrash('There was an error', error); } } } diff --git a/packages/cli/src/commands/webhook.ts b/packages/cli/src/commands/webhook.ts index 6085560fe895d..2caccd5942949 100644 --- a/packages/cli/src/commands/webhook.ts +++ b/packages/cli/src/commands/webhook.ts @@ -1,13 +1,8 @@ -/* eslint-disable @typescript-eslint/no-unsafe-argument */ -/* eslint-disable no-console */ -/* eslint-disable @typescript-eslint/no-unsafe-call */ -/* eslint-disable @typescript-eslint/no-unsafe-member-access */ -/* eslint-disable @typescript-eslint/no-unsafe-assignment */ /* eslint-disable @typescript-eslint/unbound-method */ import { BinaryDataManager, UserSettings } from 'n8n-core'; import { Command, flags } from '@oclif/command'; -import { LoggerProxy, sleep } from 'n8n-workflow'; +import { LoggerProxy, ErrorReporterProxy as ErrorReporter, sleep } from 'n8n-workflow'; import config from '@/config'; import * as ActiveExecutions from '@/ActiveExecutions'; import { CredentialsOverwrites } from '@/CredentialsOverwrites'; @@ -22,7 +17,19 @@ import { getLogger } from '@/Logger'; import { initErrorHandling } from '@/ErrorReporting'; import * as CrashJournal from '@/CrashJournal'; -let processExitCode = 0; +const exitWithCrash = async (message: string, error: unknown) => { + ErrorReporter.error(new Error(message, { cause: error }), { level: 'fatal' }); + await sleep(2000); + process.exit(1); +}; + +const exitSuccessFully = async () => { + try { + await CrashJournal.cleanup(); + } finally { + process.exit(); + } +}; export class Webhook extends Command { static description = 'Starts n8n webhook process. Intercepts only production URLs.'; @@ -42,20 +49,14 @@ export class Webhook extends Command { static async stopProcess() { LoggerProxy.info('\nStopping n8n...'); - const exit = () => { - CrashJournal.cleanup().finally(() => { - process.exit(processExitCode); - }); - }; - try { const externalHooks = ExternalHooks(); await externalHooks.run('n8n.stop', []); - setTimeout(() => { + setTimeout(async () => { // In case that something goes wrong with shutdown we // kill after max. 30 seconds no matter what - exit(); + await exitSuccessFully(); }, 30000); // Wait for active workflow executions to finish @@ -74,10 +75,10 @@ export class Webhook extends Command { executingWorkflows = activeExecutionsInstance.getActiveExecutions(); } } catch (error) { - LoggerProxy.error('There was an error shutting down n8n.', error); + await exitWithCrash('There was an error shutting down n8n.', error); } - exit(); + await exitSuccessFully(); } // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types @@ -110,13 +111,9 @@ export class Webhook extends Command { try { // Start directly with the init of the database to improve startup time - const startDbInitPromise = Db.init().catch((error) => { - // eslint-disable-next-line @typescript-eslint/restrict-template-expressions, @typescript-eslint/no-unsafe-member-access - logger.error(`There was an error initializing DB: "${error.message}"`); - - processExitCode = 1; - process.emit('exit', processExitCode); - }); + const startDbInitPromise = Db.init().catch(async (error: Error) => + exitWithCrash('There was an error initializing DB', error), + ); // Make sure the settings exist // eslint-disable-next-line @typescript-eslint/no-unused-vars @@ -151,9 +148,7 @@ export class Webhook extends Command { console.info('Webhook listener waiting for requests.'); } catch (error) { - console.error('Exiting due to error.', error); - processExitCode = 1; - process.emit('exit', processExitCode); + await exitWithCrash('Exiting due to error.', error); } } } diff --git a/packages/cli/src/commands/worker.ts b/packages/cli/src/commands/worker.ts index 08dca8667e4da..67d024f25421c 100644 --- a/packages/cli/src/commands/worker.ts +++ b/packages/cli/src/commands/worker.ts @@ -1,12 +1,8 @@ /* eslint-disable @typescript-eslint/no-unsafe-argument */ -/* eslint-disable no-console */ /* eslint-disable @typescript-eslint/no-shadow */ -/* eslint-disable @typescript-eslint/explicit-module-boundary-types */ /* eslint-disable @typescript-eslint/unbound-method */ /* eslint-disable @typescript-eslint/no-unsafe-member-access */ -/* eslint-disable @typescript-eslint/no-non-null-assertion */ /* eslint-disable @typescript-eslint/restrict-template-expressions */ -/* eslint-disable @typescript-eslint/no-unused-vars */ import express from 'express'; import http from 'http'; import PCancelable from 'p-cancelable'; @@ -20,6 +16,7 @@ import { IRun, Workflow, LoggerProxy, + ErrorReporterProxy as ErrorReporter, sleep, } from 'n8n-workflow'; @@ -29,7 +26,6 @@ import { CredentialsOverwrites } from '@/CredentialsOverwrites'; import { CredentialTypes } from '@/CredentialTypes'; import * as Db from '@/Db'; import { ExternalHooks } from '@/ExternalHooks'; -import * as GenericHelpers from '@/GenericHelpers'; import { NodeTypes } from '@/NodeTypes'; import * as ResponseHelper from '@/ResponseHelper'; import * as WebhookHelpers from '@/WebhookHelpers'; @@ -41,9 +37,25 @@ import { PermissionChecker } from '@/UserManagement/PermissionChecker'; import config from '@/config'; import * as Queue from '@/Queue'; +import * as CrashJournal from '@/CrashJournal'; import { getWorkflowOwner } from '@/UserManagement/UserManagementHelper'; import { generateFailedExecutionFromError } from '@/WorkflowHelpers'; import { N8N_VERSION } from '@/constants'; +import { initErrorHandling } from '@/ErrorReporting'; + +const exitWithCrash = async (message: string, error: unknown) => { + ErrorReporter.error(new Error(message, { cause: error }), { level: 'fatal' }); + await sleep(2000); + process.exit(1); +}; + +const exitSuccessFully = async () => { + try { + await CrashJournal.cleanup(); + } finally { + process.exit(); + } +}; export class Worker extends Command { static description = '\nStarts a n8n worker'; @@ -64,9 +76,6 @@ export class Worker extends Command { static jobQueue: Queue.JobQueue; - static processExitCode = 0; - // static activeExecutions = ActiveExecutions.getInstance(); - /** * Stop n8n in a graceful way. * Make for example sure that all the webhooks from third party services @@ -87,10 +96,10 @@ export class Worker extends Command { const stopTime = new Date().getTime() + maxStopTime; - setTimeout(() => { + setTimeout(async () => { // In case that something goes wrong with shutdown we // kill after max. 30 seconds no matter what - process.exit(Worker.processExitCode); + await exitSuccessFully(); }, maxStopTime); // Wait for active workflow executions to finish @@ -108,10 +117,10 @@ export class Worker extends Command { await sleep(500); } } catch (error) { - LoggerProxy.error('There was an error shutting down n8n.', error); + await exitWithCrash('There was an error shutting down n8n.', error); } - process.exit(Worker.processExitCode); + await exitSuccessFully(); } async runJob(job: Queue.Job, nodeTypes: INodeTypes): Promise { @@ -261,20 +270,18 @@ export class Worker extends Command { process.once('SIGTERM', Worker.stopProcess); process.once('SIGINT', Worker.stopProcess); + await initErrorHandling(); + await CrashJournal.init(); + // Wrap that the process does not close but we can still use async await (async () => { try { const { flags } = this.parse(Worker); // Start directly with the init of the database to improve startup time - const startDbInitPromise = Db.init().catch((error) => { - logger.error(`There was an error initializing DB: "${error.message}"`); - - Worker.processExitCode = 1; - // @ts-ignore - process.emit('SIGINT'); - process.exit(1); - }); + const startDbInitPromise = Db.init().catch(async (error: Error) => + exitWithCrash('There was an error initializing DB', error), + ); // Make sure the settings exist await UserSettings.prepareUserSettings(); @@ -428,12 +435,7 @@ export class Worker extends Command { }); } } catch (error) { - logger.error(`Worker process cannot continue. "${error.message}"`); - - Worker.processExitCode = 1; - // @ts-ignore - process.emit('SIGINT'); - process.exit(1); + await exitWithCrash('Worker process cannot continue.', error); } })(); } diff --git a/packages/workflow/src/ErrorReporterProxy.ts b/packages/workflow/src/ErrorReporterProxy.ts index 2dd3dfce6ab20..46ca6c56e977e 100644 --- a/packages/workflow/src/ErrorReporterProxy.ts +++ b/packages/workflow/src/ErrorReporterProxy.ts @@ -2,7 +2,7 @@ import type { Primitives } from './utils'; import * as Logger from './LoggerProxy'; export interface ReportingOptions { - level?: 'warning' | 'error'; + level?: 'warning' | 'error' | 'fatal'; tags?: Record; extra?: Record; }