mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-11 21:07:28 -08:00
refactor(core): Separate listeners in scaling service (no-changelog) (#10487)
This commit is contained in:
parent
96e69ad5f2
commit
352aa2a9a4
|
@ -42,7 +42,7 @@ import type { WorkflowRepository } from '@db/repositories/workflow.repository';
|
||||||
import type { ExternalHooks } from './external-hooks';
|
import type { ExternalHooks } from './external-hooks';
|
||||||
import type { LICENSE_FEATURES, LICENSE_QUOTAS } from './constants';
|
import type { LICENSE_FEATURES, LICENSE_QUOTAS } from './constants';
|
||||||
import type { WorkflowWithSharingsAndCredentials } from './workflows/workflows.types';
|
import type { WorkflowWithSharingsAndCredentials } from './workflows/workflows.types';
|
||||||
import type { RunningJobSummary } from './scaling/types';
|
import type { RunningJobSummary } from './scaling/scaling.types';
|
||||||
import type { Scope } from '@n8n/permissions';
|
import type { Scope } from '@n8n/permissions';
|
||||||
|
|
||||||
export interface ICredentialsTypeData {
|
export interface ICredentialsTypeData {
|
||||||
|
|
|
@ -11,7 +11,7 @@ import type { WaitTracker } from '@/wait-tracker';
|
||||||
import type { ExecutionRepository } from '@/databases/repositories/execution.repository';
|
import type { ExecutionRepository } from '@/databases/repositories/execution.repository';
|
||||||
import type { ExecutionRequest } from '@/executions/execution.types';
|
import type { ExecutionRequest } from '@/executions/execution.types';
|
||||||
import type { ConcurrencyControlService } from '@/concurrency/concurrency-control.service';
|
import type { ConcurrencyControlService } from '@/concurrency/concurrency-control.service';
|
||||||
import type { Job } from '@/scaling/types';
|
import type { Job } from '@/scaling/scaling.types';
|
||||||
import { mockInstance } from '@test/mocking';
|
import { mockInstance } from '@test/mocking';
|
||||||
|
|
||||||
describe('ExecutionService', () => {
|
describe('ExecutionService', () => {
|
||||||
|
|
|
@ -3,7 +3,7 @@ import { ScalingService } from '../scaling.service';
|
||||||
import { JOB_TYPE_NAME, QUEUE_NAME } from '../constants';
|
import { JOB_TYPE_NAME, QUEUE_NAME } from '../constants';
|
||||||
import config from '@/config';
|
import config from '@/config';
|
||||||
import * as BullModule from 'bull';
|
import * as BullModule from 'bull';
|
||||||
import type { Job, JobData, JobOptions, JobQueue } from '../types';
|
import type { Job, JobData, JobOptions, JobQueue } from '../scaling.types';
|
||||||
import { ApplicationError } from 'n8n-workflow';
|
import { ApplicationError } from 'n8n-workflow';
|
||||||
import { mockInstance } from '@test/mocking';
|
import { mockInstance } from '@test/mocking';
|
||||||
import { GlobalConfig } from '@n8n/config';
|
import { GlobalConfig } from '@n8n/config';
|
||||||
|
@ -22,6 +22,8 @@ jest.mock('bull', () => ({
|
||||||
}));
|
}));
|
||||||
|
|
||||||
describe('ScalingService', () => {
|
describe('ScalingService', () => {
|
||||||
|
const Bull = jest.mocked(BullModule.default);
|
||||||
|
|
||||||
const globalConfig = mockInstance(GlobalConfig, {
|
const globalConfig = mockInstance(GlobalConfig, {
|
||||||
queue: {
|
queue: {
|
||||||
bull: {
|
bull: {
|
||||||
|
@ -40,11 +42,29 @@ describe('ScalingService', () => {
|
||||||
const instanceSettings = Container.get(InstanceSettings);
|
const instanceSettings = Container.get(InstanceSettings);
|
||||||
const orchestrationService = mock<OrchestrationService>({ isMultiMainSetupEnabled: false });
|
const orchestrationService = mock<OrchestrationService>({ isMultiMainSetupEnabled: false });
|
||||||
const jobProcessor = mock<JobProcessor>();
|
const jobProcessor = mock<JobProcessor>();
|
||||||
|
|
||||||
let scalingService: ScalingService;
|
let scalingService: ScalingService;
|
||||||
|
|
||||||
|
let registerMainListenersSpy: jest.SpyInstance;
|
||||||
|
let registerWorkerListenersSpy: jest.SpyInstance;
|
||||||
|
let scheduleQueueRecoverySpy: jest.SpyInstance;
|
||||||
|
let stopQueueRecoverySpy: jest.SpyInstance;
|
||||||
|
let getRunningJobsCountSpy: jest.SpyInstance;
|
||||||
|
|
||||||
|
const bullConstructorArgs = [
|
||||||
|
QUEUE_NAME,
|
||||||
|
{
|
||||||
|
prefix: globalConfig.queue.bull.prefix,
|
||||||
|
settings: globalConfig.queue.bull.settings,
|
||||||
|
createClient: expect.any(Function),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
jest.clearAllMocks();
|
jest.clearAllMocks();
|
||||||
config.set('generic.instanceType', 'main');
|
config.set('generic.instanceType', 'main');
|
||||||
|
instanceSettings.markAsLeader();
|
||||||
|
|
||||||
scalingService = new ScalingService(
|
scalingService = new ScalingService(
|
||||||
mock(),
|
mock(),
|
||||||
mock(),
|
mock(),
|
||||||
|
@ -54,98 +74,93 @@ describe('ScalingService', () => {
|
||||||
instanceSettings,
|
instanceSettings,
|
||||||
orchestrationService,
|
orchestrationService,
|
||||||
);
|
);
|
||||||
});
|
|
||||||
|
|
||||||
afterEach(() => {
|
getRunningJobsCountSpy = jest.spyOn(scalingService, 'getRunningJobsCount');
|
||||||
scalingService.stopQueueRecovery();
|
|
||||||
|
// @ts-expect-error Private method
|
||||||
|
ScalingService.prototype.scheduleQueueRecovery = jest.fn();
|
||||||
|
// @ts-expect-error Private method
|
||||||
|
registerMainListenersSpy = jest.spyOn(scalingService, 'registerMainListeners');
|
||||||
|
// @ts-expect-error Private method
|
||||||
|
registerWorkerListenersSpy = jest.spyOn(scalingService, 'registerWorkerListeners');
|
||||||
|
// @ts-expect-error Private method
|
||||||
|
scheduleQueueRecoverySpy = jest.spyOn(scalingService, 'scheduleQueueRecovery');
|
||||||
|
// @ts-expect-error Private method
|
||||||
|
stopQueueRecoverySpy = jest.spyOn(scalingService, 'stopQueueRecovery');
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('setupQueue', () => {
|
describe('setupQueue', () => {
|
||||||
it('should set up the queue', async () => {
|
describe('if leader main', () => {
|
||||||
/**
|
it('should set up queue + listeners + queue recovery', async () => {
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
const { prefix, settings } = globalConfig.queue.bull;
|
|
||||||
const Bull = jest.mocked(BullModule.default);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
|
|
||||||
/**
|
expect(Bull).toHaveBeenCalledWith(...bullConstructorArgs);
|
||||||
* Assert
|
expect(registerMainListenersSpy).toHaveBeenCalled();
|
||||||
*/
|
expect(registerWorkerListenersSpy).not.toHaveBeenCalled();
|
||||||
expect(Bull).toHaveBeenCalledWith(QUEUE_NAME, {
|
expect(scheduleQueueRecoverySpy).toHaveBeenCalled();
|
||||||
prefix,
|
});
|
||||||
settings,
|
});
|
||||||
createClient: expect.any(Function),
|
|
||||||
|
describe('if follower main', () => {
|
||||||
|
it('should set up queue + listeners', async () => {
|
||||||
|
instanceSettings.markAsFollower();
|
||||||
|
|
||||||
|
await scalingService.setupQueue();
|
||||||
|
|
||||||
|
expect(Bull).toHaveBeenCalledWith(...bullConstructorArgs);
|
||||||
|
expect(registerMainListenersSpy).toHaveBeenCalled();
|
||||||
|
expect(registerWorkerListenersSpy).not.toHaveBeenCalled();
|
||||||
|
expect(scheduleQueueRecoverySpy).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('if worker', () => {
|
||||||
|
it('should set up queue + listeners', async () => {
|
||||||
|
// @ts-expect-error Private field
|
||||||
|
scalingService.instanceType = 'worker';
|
||||||
|
|
||||||
|
await scalingService.setupQueue();
|
||||||
|
|
||||||
|
expect(Bull).toHaveBeenCalledWith(...bullConstructorArgs);
|
||||||
|
expect(registerWorkerListenersSpy).toHaveBeenCalled();
|
||||||
|
expect(registerMainListenersSpy).not.toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
expect(queue.on).toHaveBeenCalledWith('global:progress', expect.any(Function));
|
|
||||||
expect(queue.on).toHaveBeenCalledWith('error', expect.any(Function));
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('setupWorker', () => {
|
describe('setupWorker', () => {
|
||||||
it('should set up a worker with concurrency', async () => {
|
it('should set up a worker with concurrency', async () => {
|
||||||
/**
|
// @ts-expect-error Private field
|
||||||
* Arrange
|
scalingService.instanceType = 'worker';
|
||||||
*/
|
|
||||||
config.set('generic.instanceType', 'worker');
|
|
||||||
const scalingService = new ScalingService(
|
|
||||||
mock(),
|
|
||||||
mock(),
|
|
||||||
mock(),
|
|
||||||
globalConfig,
|
|
||||||
mock(),
|
|
||||||
instanceSettings,
|
|
||||||
orchestrationService,
|
|
||||||
);
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
const concurrency = 5;
|
const concurrency = 5;
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
scalingService.setupWorker(concurrency);
|
scalingService.setupWorker(concurrency);
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(queue.process).toHaveBeenCalledWith(JOB_TYPE_NAME, concurrency, expect.any(Function));
|
expect(queue.process).toHaveBeenCalledWith(JOB_TYPE_NAME, concurrency, expect.any(Function));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw if called on a non-worker instance', async () => {
|
it('should throw if called on a non-worker instance', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
|
|
||||||
/**
|
expect(() => scalingService.setupWorker(5)).toThrow();
|
||||||
* Act and Assert
|
});
|
||||||
*/
|
|
||||||
|
it('should throw if called before queue is ready', async () => {
|
||||||
|
// @ts-expect-error Private field
|
||||||
|
scalingService.instanceType = 'worker';
|
||||||
|
|
||||||
expect(() => scalingService.setupWorker(5)).toThrow();
|
expect(() => scalingService.setupWorker(5)).toThrow();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('stop', () => {
|
describe('stop', () => {
|
||||||
it('should pause the queue, check for running jobs, and stop queue recovery', async () => {
|
it('should pause queue, wait for running jobs, stop queue recovery', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
jobProcessor.getRunningJobIds.mockReturnValue([]);
|
jobProcessor.getRunningJobIds.mockReturnValue([]);
|
||||||
const stopQueueRecoverySpy = jest.spyOn(scalingService, 'stopQueueRecovery');
|
|
||||||
const getRunningJobsCountSpy = jest.spyOn(scalingService, 'getRunningJobsCount');
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
await scalingService.stop();
|
await scalingService.stop();
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(queue.pause).toHaveBeenCalledWith(true, true);
|
expect(queue.pause).toHaveBeenCalledWith(true, true);
|
||||||
expect(stopQueueRecoverySpy).toHaveBeenCalled();
|
expect(stopQueueRecoverySpy).toHaveBeenCalled();
|
||||||
expect(getRunningJobsCountSpy).toHaveBeenCalled();
|
expect(getRunningJobsCountSpy).toHaveBeenCalled();
|
||||||
|
@ -154,62 +169,35 @@ describe('ScalingService', () => {
|
||||||
|
|
||||||
describe('pingQueue', () => {
|
describe('pingQueue', () => {
|
||||||
it('should ping the queue', async () => {
|
it('should ping the queue', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
await scalingService.pingQueue();
|
await scalingService.pingQueue();
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(queue.client.ping).toHaveBeenCalled();
|
expect(queue.client.ping).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('addJob', () => {
|
describe('addJob', () => {
|
||||||
it('should add a job', async () => {
|
it('should add a job', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
queue.add.mockResolvedValue(mock<Job>({ id: '456' }));
|
queue.add.mockResolvedValue(mock<Job>({ id: '456' }));
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
const jobData = mock<JobData>({ executionId: '123' });
|
const jobData = mock<JobData>({ executionId: '123' });
|
||||||
const jobOptions = mock<JobOptions>();
|
const jobOptions = mock<JobOptions>();
|
||||||
await scalingService.addJob(jobData, jobOptions);
|
await scalingService.addJob(jobData, jobOptions);
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(queue.add).toHaveBeenCalledWith(JOB_TYPE_NAME, jobData, jobOptions);
|
expect(queue.add).toHaveBeenCalledWith(JOB_TYPE_NAME, jobData, jobOptions);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getJob', () => {
|
describe('getJob', () => {
|
||||||
it('should get a job', async () => {
|
it('should get a job', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
const jobId = '123';
|
const jobId = '123';
|
||||||
queue.getJob.mockResolvedValue(mock<Job>({ id: jobId }));
|
queue.getJob.mockResolvedValue(mock<Job>({ id: jobId }));
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
const job = await scalingService.getJob(jobId);
|
const job = await scalingService.getJob(jobId);
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(queue.getJob).toHaveBeenCalledWith(jobId);
|
expect(queue.getJob).toHaveBeenCalledWith(jobId);
|
||||||
expect(job?.id).toBe(jobId);
|
expect(job?.id).toBe(jobId);
|
||||||
});
|
});
|
||||||
|
@ -217,88 +205,49 @@ describe('ScalingService', () => {
|
||||||
|
|
||||||
describe('findJobsByStatus', () => {
|
describe('findJobsByStatus', () => {
|
||||||
it('should find jobs by status', async () => {
|
it('should find jobs by status', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
queue.getJobs.mockResolvedValue([mock<Job>({ id: '123' })]);
|
queue.getJobs.mockResolvedValue([mock<Job>({ id: '123' })]);
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
const jobs = await scalingService.findJobsByStatus(['active']);
|
const jobs = await scalingService.findJobsByStatus(['active']);
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(queue.getJobs).toHaveBeenCalledWith(['active']);
|
expect(queue.getJobs).toHaveBeenCalledWith(['active']);
|
||||||
expect(jobs).toHaveLength(1);
|
expect(jobs).toHaveLength(1);
|
||||||
expect(jobs.at(0)?.id).toBe('123');
|
expect(jobs.at(0)?.id).toBe('123');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should filter out `null` in Redis response', async () => {
|
it('should filter out `null` in Redis response', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
// @ts-expect-error - Untyped but possible Redis response
|
// @ts-expect-error - Untyped but possible Redis response
|
||||||
queue.getJobs.mockResolvedValue([mock<Job>(), null]);
|
queue.getJobs.mockResolvedValue([mock<Job>(), null]);
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
const jobs = await scalingService.findJobsByStatus(['waiting']);
|
const jobs = await scalingService.findJobsByStatus(['waiting']);
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(jobs).toHaveLength(1);
|
expect(jobs).toHaveLength(1);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('stopJob', () => {
|
describe('stopJob', () => {
|
||||||
it('should stop an active job', async () => {
|
it('should stop an active job', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
const job = mock<Job>({ isActive: jest.fn().mockResolvedValue(true) });
|
const job = mock<Job>({ isActive: jest.fn().mockResolvedValue(true) });
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
const result = await scalingService.stopJob(job);
|
const result = await scalingService.stopJob(job);
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(job.progress).toHaveBeenCalledWith({ kind: 'abort-job' });
|
expect(job.progress).toHaveBeenCalledWith({ kind: 'abort-job' });
|
||||||
expect(result).toBe(true);
|
expect(result).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should stop an inactive job', async () => {
|
it('should stop an inactive job', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
const job = mock<Job>({ isActive: jest.fn().mockResolvedValue(false) });
|
const job = mock<Job>({ isActive: jest.fn().mockResolvedValue(false) });
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
const result = await scalingService.stopJob(job);
|
const result = await scalingService.stopJob(job);
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(job.remove).toHaveBeenCalled();
|
expect(job.remove).toHaveBeenCalled();
|
||||||
expect(result).toBe(true);
|
expect(result).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should report failure to stop a job', async () => {
|
it('should report failure to stop a job', async () => {
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
await scalingService.setupQueue();
|
||||||
const job = mock<Job>({
|
const job = mock<Job>({
|
||||||
isActive: jest.fn().mockImplementation(() => {
|
isActive: jest.fn().mockImplementation(() => {
|
||||||
|
@ -306,53 +255,9 @@ describe('ScalingService', () => {
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
const result = await scalingService.stopJob(job);
|
const result = await scalingService.stopJob(job);
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(result).toBe(false);
|
expect(result).toBe(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('scheduleQueueRecovery', () => {
|
|
||||||
it('if leader, should schedule queue recovery', async () => {
|
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
const scheduleSpy = jest.spyOn(scalingService, 'scheduleQueueRecovery');
|
|
||||||
instanceSettings.markAsLeader();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(scheduleSpy).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('if follower, should not schedule queue recovery', async () => {
|
|
||||||
/**
|
|
||||||
* Arrange
|
|
||||||
*/
|
|
||||||
const scheduleSpy = jest.spyOn(scalingService, 'scheduleQueueRecovery');
|
|
||||||
instanceSettings.markAsFollower();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Act
|
|
||||||
*/
|
|
||||||
await scalingService.setupQueue();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Assert
|
|
||||||
*/
|
|
||||||
expect(scheduleSpy).not.toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -8,7 +8,7 @@ import { WorkflowRepository } from '@/databases/repositories/workflow.repository
|
||||||
import * as WorkflowExecuteAdditionalData from '@/workflow-execute-additional-data';
|
import * as WorkflowExecuteAdditionalData from '@/workflow-execute-additional-data';
|
||||||
import { NodeTypes } from '@/node-types';
|
import { NodeTypes } from '@/node-types';
|
||||||
import type { ExecutionStatus, IExecuteResponsePromiseData, IRun } from 'n8n-workflow';
|
import type { ExecutionStatus, IExecuteResponsePromiseData, IRun } from 'n8n-workflow';
|
||||||
import type { Job, JobId, JobResult, RunningJob, RunningJobSummary } from './types';
|
import type { Job, JobId, JobResult, RunningJob, RunningJobSummary } from './scaling.types';
|
||||||
import type PCancelable from 'p-cancelable';
|
import type PCancelable from 'p-cancelable';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -13,11 +13,11 @@ import type {
|
||||||
Job,
|
Job,
|
||||||
JobData,
|
JobData,
|
||||||
JobOptions,
|
JobOptions,
|
||||||
JobMessage,
|
|
||||||
JobStatus,
|
JobStatus,
|
||||||
JobId,
|
JobId,
|
||||||
QueueRecoveryContext,
|
QueueRecoveryContext,
|
||||||
} from './types';
|
PubSubMessage,
|
||||||
|
} from './scaling.types';
|
||||||
import type { IExecuteResponsePromiseData } from 'n8n-workflow';
|
import type { IExecuteResponsePromiseData } from 'n8n-workflow';
|
||||||
import { GlobalConfig } from '@n8n/config';
|
import { GlobalConfig } from '@n8n/config';
|
||||||
import { ExecutionRepository } from '@/databases/repositories/execution.repository';
|
import { ExecutionRepository } from '@/databases/repositories/execution.repository';
|
||||||
|
@ -71,6 +71,7 @@ export class ScalingService {
|
||||||
|
|
||||||
setupWorker(concurrency: number) {
|
setupWorker(concurrency: number) {
|
||||||
this.assertWorker();
|
this.assertWorker();
|
||||||
|
this.assertQueue();
|
||||||
|
|
||||||
void this.queue.process(
|
void this.queue.process(
|
||||||
JOB_TYPE_NAME,
|
JOB_TYPE_NAME,
|
||||||
|
@ -161,22 +162,6 @@ export class ScalingService {
|
||||||
// #region Listeners
|
// #region Listeners
|
||||||
|
|
||||||
private registerListeners() {
|
private registerListeners() {
|
||||||
this.queue.on('global:progress', (_jobId: JobId, msg: JobMessage) => {
|
|
||||||
if (msg.kind === 'respond-to-webhook') {
|
|
||||||
const { executionId, response } = msg;
|
|
||||||
this.activeExecutions.resolveResponsePromise(
|
|
||||||
executionId,
|
|
||||||
this.decodeWebhookResponse(response),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
this.queue.on('global:progress', (jobId: JobId, msg: JobMessage) => {
|
|
||||||
if (msg.kind === 'abort-job') {
|
|
||||||
this.jobProcessor.stopJob(jobId);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
let latestAttemptTs = 0;
|
let latestAttemptTs = 0;
|
||||||
let cumulativeTimeoutMs = 0;
|
let cumulativeTimeoutMs = 0;
|
||||||
|
|
||||||
|
@ -210,10 +195,28 @@ export class ScalingService {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
throw error;
|
||||||
this.instanceType === 'worker' &&
|
});
|
||||||
error.message.includes('job stalled more than maxStalledCount')
|
|
||||||
) {
|
if (this.instanceType === 'main') {
|
||||||
|
this.registerMainListeners();
|
||||||
|
} else if (this.instanceType === 'worker') {
|
||||||
|
this.registerWorkerListeners();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register listeners on a `worker` process for Bull queue events.
|
||||||
|
*/
|
||||||
|
private registerWorkerListeners() {
|
||||||
|
this.queue.on('global:progress', (jobId: JobId, msg: unknown) => {
|
||||||
|
if (!this.isPubSubMessage(msg)) return;
|
||||||
|
|
||||||
|
if (msg.kind === 'abort-job') this.jobProcessor.stopJob(jobId);
|
||||||
|
});
|
||||||
|
|
||||||
|
this.queue.on('error', (error: Error) => {
|
||||||
|
if (error.message.includes('job stalled more than maxStalledCount')) {
|
||||||
throw new MaxStalledCountError(error);
|
throw new MaxStalledCountError(error);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -221,10 +224,7 @@ export class ScalingService {
|
||||||
* Non-recoverable error on worker start with Redis unavailable.
|
* Non-recoverable error on worker start with Redis unavailable.
|
||||||
* Even if Redis recovers, worker will remain unable to process jobs.
|
* Even if Redis recovers, worker will remain unable to process jobs.
|
||||||
*/
|
*/
|
||||||
if (
|
if (error.message.includes('Error initializing Lua scripts')) {
|
||||||
this.instanceType === 'worker' &&
|
|
||||||
error.message.includes('Error initializing Lua scripts')
|
|
||||||
) {
|
|
||||||
this.logger.error('[ScalingService] Fatal error initializing worker', { error });
|
this.logger.error('[ScalingService] Fatal error initializing worker', { error });
|
||||||
this.logger.error('[ScalingService] Exiting process...');
|
this.logger.error('[ScalingService] Exiting process...');
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
|
@ -234,6 +234,24 @@ export class ScalingService {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register listeners on a `main` process for Bull queue events.
|
||||||
|
*/
|
||||||
|
private registerMainListeners() {
|
||||||
|
this.queue.on('global:progress', (_jobId: JobId, msg: unknown) => {
|
||||||
|
if (!this.isPubSubMessage(msg)) return;
|
||||||
|
|
||||||
|
if (msg.kind === 'respond-to-webhook') {
|
||||||
|
const decodedResponse = this.decodeWebhookResponse(msg.response);
|
||||||
|
this.activeExecutions.resolveResponsePromise(msg.executionId, decodedResponse);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private isPubSubMessage(candidate: unknown): candidate is PubSubMessage {
|
||||||
|
return typeof candidate === 'object' && candidate !== null && 'kind' in candidate;
|
||||||
|
}
|
||||||
|
|
||||||
// #endregion
|
// #endregion
|
||||||
|
|
||||||
private decodeWebhookResponse(
|
private decodeWebhookResponse(
|
||||||
|
@ -252,6 +270,12 @@ export class ScalingService {
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private assertQueue() {
|
||||||
|
if (this.queue) return;
|
||||||
|
|
||||||
|
throw new ApplicationError('This method must be called after `setupQueue`');
|
||||||
|
}
|
||||||
|
|
||||||
private assertWorker() {
|
private assertWorker() {
|
||||||
if (this.instanceType === 'worker') return;
|
if (this.instanceType === 'worker') return;
|
||||||
|
|
||||||
|
@ -265,7 +289,7 @@ export class ScalingService {
|
||||||
waitMs: config.getEnv('executions.queueRecovery.interval') * 60 * 1000,
|
waitMs: config.getEnv('executions.queueRecovery.interval') * 60 * 1000,
|
||||||
};
|
};
|
||||||
|
|
||||||
scheduleQueueRecovery(waitMs = this.queueRecoveryContext.waitMs) {
|
private scheduleQueueRecovery(waitMs = this.queueRecoveryContext.waitMs) {
|
||||||
this.queueRecoveryContext.timeout = setTimeout(async () => {
|
this.queueRecoveryContext.timeout = setTimeout(async () => {
|
||||||
try {
|
try {
|
||||||
const nextWaitMs = await this.recoverFromQueue();
|
const nextWaitMs = await this.recoverFromQueue();
|
||||||
|
@ -285,7 +309,7 @@ export class ScalingService {
|
||||||
this.logger.debug(`[ScalingService] Scheduled queue recovery check for next ${wait}`);
|
this.logger.debug(`[ScalingService] Scheduled queue recovery check for next ${wait}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
stopQueueRecovery() {
|
private stopQueueRecovery() {
|
||||||
clearTimeout(this.queueRecoveryContext.timeout);
|
clearTimeout(this.queueRecoveryContext.timeout);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -28,16 +28,19 @@ export type JobStatus = Bull.JobStatus;
|
||||||
|
|
||||||
export type JobOptions = Bull.JobOptions;
|
export type JobOptions = Bull.JobOptions;
|
||||||
|
|
||||||
/** Message sent by worker to queue or by queue to worker. */
|
export type PubSubMessage = MessageToMain | MessageToWorker;
|
||||||
export type JobMessage = RepondToWebhookMessage | AbortJobMessage;
|
|
||||||
|
|
||||||
export type RepondToWebhookMessage = {
|
type MessageToMain = RepondToWebhookMessage;
|
||||||
|
|
||||||
|
type MessageToWorker = AbortJobMessage;
|
||||||
|
|
||||||
|
type RepondToWebhookMessage = {
|
||||||
kind: 'respond-to-webhook';
|
kind: 'respond-to-webhook';
|
||||||
executionId: string;
|
executionId: string;
|
||||||
response: IExecuteResponsePromiseData;
|
response: IExecuteResponsePromiseData;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type AbortJobMessage = {
|
type AbortJobMessage = {
|
||||||
kind: 'abort-job';
|
kind: 'abort-job';
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import type { ExecutionStatus, WorkflowExecuteMode } from 'n8n-workflow';
|
import type { ExecutionStatus, WorkflowExecuteMode } from 'n8n-workflow';
|
||||||
import type { RedisServicePubSubPublisher } from '../../redis/RedisServicePubSubPublisher';
|
import type { RedisServicePubSubPublisher } from '../../redis/RedisServicePubSubPublisher';
|
||||||
import type { RunningJobSummary } from '@/scaling/types';
|
import type { RunningJobSummary } from '@/scaling/scaling.types';
|
||||||
|
|
||||||
export interface WorkerCommandReceivedHandlerOptions {
|
export interface WorkerCommandReceivedHandlerOptions {
|
||||||
queueModeId: string;
|
queueModeId: string;
|
||||||
|
|
|
@ -28,7 +28,7 @@ import { ExecutionRepository } from '@db/repositories/execution.repository';
|
||||||
import { ExternalHooks } from '@/external-hooks';
|
import { ExternalHooks } from '@/external-hooks';
|
||||||
import type { IExecutionResponse, IWorkflowExecutionDataProcess } from '@/Interfaces';
|
import type { IExecutionResponse, IWorkflowExecutionDataProcess } from '@/Interfaces';
|
||||||
import { NodeTypes } from '@/node-types';
|
import { NodeTypes } from '@/node-types';
|
||||||
import type { Job, JobData, JobResult } from '@/scaling/types';
|
import type { Job, JobData, JobResult } from '@/scaling/scaling.types';
|
||||||
import type { ScalingService } from '@/scaling/scaling.service';
|
import type { ScalingService } from '@/scaling/scaling.service';
|
||||||
import * as WorkflowHelpers from '@/workflow-helpers';
|
import * as WorkflowHelpers from '@/workflow-helpers';
|
||||||
import * as WorkflowExecuteAdditionalData from '@/workflow-execute-additional-data';
|
import * as WorkflowExecuteAdditionalData from '@/workflow-execute-additional-data';
|
||||||
|
|
|
@ -40,8 +40,15 @@ export class InstanceSettings {
|
||||||
|
|
||||||
readonly instanceId = this.generateInstanceId();
|
readonly instanceId = this.generateInstanceId();
|
||||||
|
|
||||||
/** Always `leader` in single-main setup. `leader` or `follower` in multi-main setup. */
|
/**
|
||||||
private instanceRole: InstanceRole = 'unset';
|
* A main is:
|
||||||
|
* - `unset` during bootup,
|
||||||
|
* - `leader` after bootup in single-main setup,
|
||||||
|
* - `leader` or `follower` after bootup in multi-main setup.
|
||||||
|
*
|
||||||
|
* A non-main instance type (e.g. `worker`) is always `unset`.
|
||||||
|
*/
|
||||||
|
instanceRole: InstanceRole = 'unset';
|
||||||
|
|
||||||
get isLeader() {
|
get isLeader() {
|
||||||
return this.instanceRole === 'leader';
|
return this.instanceRole === 'leader';
|
||||||
|
|
Loading…
Reference in a new issue