refactor(core): Separate listeners in scaling service (no-changelog) (#10487)

This commit is contained in:
Iván Ovejero 2024-08-26 12:35:39 +02:00 committed by GitHub
parent 96e69ad5f2
commit 352aa2a9a4
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
9 changed files with 149 additions and 210 deletions

View file

@ -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 {

View file

@ -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', () => {

View file

@ -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();
});
});
}); });

View file

@ -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';
/** /**

View file

@ -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);
} }

View file

@ -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';
}; };

View file

@ -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;

View file

@ -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';

View file

@ -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';