mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-16 09:36:44 +00:00
173 lines
5.0 KiB
TypeScript
173 lines
5.0 KiB
TypeScript
import { mockLogger } from '@n8n/backend-test-utils';
|
|
import type { GlobalConfig } from '@n8n/config';
|
|
import type { DbConnection } from '@n8n/db';
|
|
import type express from 'express';
|
|
import { mock } from 'jest-mock-extended';
|
|
import type { InstanceSettings } from 'n8n-core';
|
|
import { AssertionError } from 'node:assert';
|
|
import * as http from 'node:http';
|
|
|
|
import type { ExternalHooks } from '@/external-hooks';
|
|
import type { PrometheusMetricsService } from '@/metrics/prometheus-metrics.service';
|
|
import { bodyParser, rawBodyReader } from '@/middlewares';
|
|
|
|
import { WorkerServer } from '../worker-server';
|
|
|
|
const app = mock<express.Application>();
|
|
|
|
jest.mock('node:http');
|
|
jest.mock('express', () => ({ __esModule: true, default: () => app }));
|
|
|
|
const addressInUseError = () => {
|
|
const error: NodeJS.ErrnoException = new Error('Port already in use');
|
|
error.code = 'EADDRINUSE';
|
|
|
|
return error;
|
|
};
|
|
|
|
describe('WorkerServer', () => {
|
|
let globalConfig: GlobalConfig;
|
|
|
|
const externalHooks = mock<ExternalHooks>();
|
|
const instanceSettings = mock<InstanceSettings>({ instanceType: 'worker' });
|
|
const prometheusMetricsService = mock<PrometheusMetricsService>();
|
|
const dbConnection = mock<DbConnection>();
|
|
|
|
const newWorkerServer = () =>
|
|
new WorkerServer(
|
|
globalConfig,
|
|
mockLogger(),
|
|
dbConnection,
|
|
mock(),
|
|
externalHooks,
|
|
instanceSettings,
|
|
prometheusMetricsService,
|
|
mock(),
|
|
);
|
|
|
|
beforeEach(() => {
|
|
globalConfig = mock<GlobalConfig>({
|
|
queue: {
|
|
health: { active: true, port: 5678, address: '::' },
|
|
},
|
|
credentials: {
|
|
overwrite: { endpoint: '' },
|
|
},
|
|
});
|
|
jest.restoreAllMocks();
|
|
});
|
|
|
|
describe('constructor', () => {
|
|
it('should throw if non-worker instance type', () => {
|
|
expect(
|
|
() =>
|
|
new WorkerServer(
|
|
globalConfig,
|
|
mockLogger(),
|
|
dbConnection,
|
|
mock(),
|
|
externalHooks,
|
|
mock<InstanceSettings>({ instanceType: 'webhook' }),
|
|
prometheusMetricsService,
|
|
mock(),
|
|
),
|
|
).toThrowError(AssertionError);
|
|
});
|
|
|
|
it('should exit if port taken', async () => {
|
|
const server = mock<http.Server>();
|
|
const processExitSpy = jest
|
|
.spyOn(process, 'exit')
|
|
.mockImplementation(() => undefined as never);
|
|
|
|
jest.spyOn(http, 'createServer').mockReturnValue(server);
|
|
|
|
server.on.mockImplementation((event: string, callback: (arg?: unknown) => void) => {
|
|
if (event === 'error') callback(addressInUseError());
|
|
return server;
|
|
});
|
|
|
|
newWorkerServer();
|
|
|
|
expect(processExitSpy).toHaveBeenCalledWith(1);
|
|
|
|
processExitSpy.mockRestore();
|
|
});
|
|
});
|
|
|
|
describe('init', () => {
|
|
it('should mount all endpoints when all are enabled', async () => {
|
|
const server = mock<http.Server>();
|
|
jest.spyOn(http, 'createServer').mockReturnValue(server);
|
|
|
|
server.listen.mockImplementation((...args: unknown[]) => {
|
|
const callback = args.find((arg) => typeof arg === 'function');
|
|
if (callback) callback();
|
|
return server;
|
|
});
|
|
|
|
const workerServer = newWorkerServer();
|
|
|
|
const CREDENTIALS_OVERWRITE_ENDPOINT = 'credentials/overwrites';
|
|
globalConfig.credentials.overwrite.endpoint = CREDENTIALS_OVERWRITE_ENDPOINT;
|
|
|
|
await workerServer.init({ health: true, overwrites: true, metrics: true });
|
|
|
|
expect(app.get).toHaveBeenCalledWith('/healthz', expect.any(Function));
|
|
expect(app.post).toHaveBeenCalledWith(
|
|
`/${CREDENTIALS_OVERWRITE_ENDPOINT}`,
|
|
rawBodyReader,
|
|
bodyParser,
|
|
expect.any(Function),
|
|
);
|
|
expect(prometheusMetricsService.init).toHaveBeenCalledWith(app);
|
|
});
|
|
|
|
it('should mount only health and overwrites endpoints if only those are enabled', async () => {
|
|
const server = mock<http.Server>();
|
|
jest.spyOn(http, 'createServer').mockReturnValue(server);
|
|
|
|
server.listen.mockImplementation((...args: unknown[]) => {
|
|
const callback = args.find((arg) => typeof arg === 'function');
|
|
if (callback) callback();
|
|
return server;
|
|
});
|
|
|
|
const workerServer = newWorkerServer();
|
|
|
|
await workerServer.init({ health: true, overwrites: false, metrics: true });
|
|
|
|
expect(app.get).toHaveBeenCalledWith('/healthz', expect.any(Function));
|
|
expect(app.post).not.toHaveBeenCalled();
|
|
expect(prometheusMetricsService.init).toHaveBeenCalledWith(app);
|
|
});
|
|
|
|
it('should throw if no endpoints are enabled', async () => {
|
|
const server = mock<http.Server>();
|
|
jest.spyOn(http, 'createServer').mockReturnValue(server);
|
|
|
|
const workerServer = newWorkerServer();
|
|
await expect(
|
|
workerServer.init({ health: false, overwrites: false, metrics: false }),
|
|
).rejects.toThrowError(AssertionError);
|
|
});
|
|
|
|
it('should call `worker.ready` external hook', async () => {
|
|
const server = mock<http.Server>();
|
|
jest.spyOn(http, 'createServer').mockReturnValue(server);
|
|
|
|
const workerServer = newWorkerServer();
|
|
|
|
server.listen.mockImplementation((...args: unknown[]) => {
|
|
const callback = args.find((arg) => typeof arg === 'function');
|
|
if (callback) callback();
|
|
return server;
|
|
});
|
|
|
|
await workerServer.init({ health: true, overwrites: true, metrics: true });
|
|
|
|
expect(externalHooks.run).toHaveBeenCalledWith('worker.ready');
|
|
});
|
|
});
|
|
});
|