fix(core): Disconnect Redis after pausing queue during worker shutdown (#9928)

Co-authored-by: कारतोफ्फेलस्क्रिप्ट™ <aditya@netroy.in>
This commit is contained in:
Iván Ovejero 2024-07-04 18:07:47 +02:00 committed by GitHub
parent e5c324753f
commit c82579bf76
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 15 additions and 15 deletions

View file

@ -9,6 +9,7 @@ import {
} from 'n8n-workflow'; } from 'n8n-workflow';
import { ActiveExecutions } from '@/ActiveExecutions'; import { ActiveExecutions } from '@/ActiveExecutions';
import config from '@/config'; import config from '@/config';
import { HIGHEST_PRIORITY, OnShutdown } from './decorators/OnShutdown';
export type JobId = Bull.JobId; export type JobId = Bull.JobId;
export type Job = Bull.Job<JobData>; export type Job = Bull.Job<JobData>;
@ -108,11 +109,10 @@ export class Queue {
return await this.jobQueue.client.ping(); return await this.jobQueue.client.ping();
} }
async pause({ @OnShutdown(HIGHEST_PRIORITY)
isLocal, // Stop accepting new jobs, `doNotWaitActive` allows reporting progress
doNotWaitActive, async pause(): Promise<void> {
}: { isLocal?: boolean; doNotWaitActive?: boolean } = {}): Promise<void> { return await this.jobQueue?.pause(true, true);
return await this.jobQueue.pause(isLocal, doNotWaitActive);
} }
getBullObjectInstance(): JobQueue { getBullObjectInstance(): JobQueue {

View file

@ -64,9 +64,6 @@ export class Worker extends BaseCommand {
async stopProcess() { async stopProcess() {
this.logger.info('Stopping n8n...'); this.logger.info('Stopping n8n...');
// Stop accepting new jobs, `doNotWaitActive` allows reporting progress
await Worker.jobQueue.pause({ isLocal: true, doNotWaitActive: true });
try { try {
await this.externalHooks?.run('n8n.stop', []); await this.externalHooks?.run('n8n.stop', []);

View file

@ -2,6 +2,10 @@ import { Container } from 'typedi';
import { ApplicationError } from 'n8n-workflow'; import { ApplicationError } from 'n8n-workflow';
import { type ServiceClass, ShutdownService } from '@/shutdown/Shutdown.service'; import { type ServiceClass, ShutdownService } from '@/shutdown/Shutdown.service';
export const LOWEST_PRIORITY = 0;
export const DEFAULT_PRIORITY = 100;
export const HIGHEST_PRIORITY = 200;
/** /**
* Decorator that registers a method as a shutdown hook. The method will * Decorator that registers a method as a shutdown hook. The method will
* be called when the application is shutting down. * be called when the application is shutting down.
@ -22,7 +26,7 @@ import { type ServiceClass, ShutdownService } from '@/shutdown/Shutdown.service'
* ``` * ```
*/ */
export const OnShutdown = export const OnShutdown =
(priority = 100): MethodDecorator => (priority = DEFAULT_PRIORITY): MethodDecorator =>
(prototype, propertyKey, descriptor) => { (prototype, propertyKey, descriptor) => {
const serviceClass = prototype.constructor as ServiceClass; const serviceClass = prototype.constructor as ServiceClass;
const methodName = String(propertyKey); const methodName = String(propertyKey);

View file

@ -40,10 +40,6 @@ class RedisServiceBase {
} }
this.redisClient = this.redisClientService.createClient({ type }); this.redisClient = this.redisClientService.createClient({ type });
this.redisClient.on('close', () => {
this.logger.warn('Redis unavailable - trying to reconnect...');
});
this.redisClient.on('error', (error) => { this.redisClient.on('error', (error) => {
if (!String(error).includes('ECONNREFUSED')) { if (!String(error).includes('ECONNREFUSED')) {
this.logger.warn('Error with Redis: ', error); this.logger.warn('Error with Redis: ', error);

View file

@ -4,7 +4,7 @@ import { Logger } from '@/Logger';
import ioRedis from 'ioredis'; import ioRedis from 'ioredis';
import type { Cluster, RedisOptions } from 'ioredis'; import type { Cluster, RedisOptions } from 'ioredis';
import type { RedisClientType } from './RedisServiceBaseClasses'; import type { RedisClientType } from './RedisServiceBaseClasses';
import { OnShutdown } from '@/decorators/OnShutdown'; import { LOWEST_PRIORITY, OnShutdown } from '@/decorators/OnShutdown';
@Service() @Service()
export class RedisClientService { export class RedisClientService {
@ -23,7 +23,7 @@ export class RedisClientService {
return client; return client;
} }
@OnShutdown() @OnShutdown(LOWEST_PRIORITY)
disconnectClients() { disconnectClients() {
for (const client of this.clients) { for (const client of this.clients) {
client.disconnect(); client.disconnect();
@ -144,6 +144,8 @@ export class RedisClientService {
} }
} }
this.logger.warn('Redis unavailable - trying to reconnect...');
return RETRY_INTERVAL; return RETRY_INTERVAL;
}; };
} }

View file

@ -11,6 +11,7 @@ export const setupTestCommand = <T extends BaseCommand>(Command: Class<T>) => {
// mock SIGINT/SIGTERM registration // mock SIGINT/SIGTERM registration
process.once = jest.fn(); process.once = jest.fn();
process.exit = jest.fn() as never;
beforeAll(async () => { beforeAll(async () => {
await testDb.init(); await testDb.init();