mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
125 lines
4.8 KiB
TypeScript
125 lines
4.8 KiB
TypeScript
import { returnJsonArray } from 'n8n-core';
|
|
import { captor, mock, type CalledWithMock } from 'jest-mock-extended';
|
|
import type { ICredentialDataDecryptedObject, ITriggerFunctions } from 'n8n-workflow';
|
|
|
|
import { RedisTrigger } from '../RedisTrigger.node';
|
|
import { type RedisClientType, setupRedisClient } from '../utils';
|
|
|
|
type GetCredentialsMock = CalledWithMock<Promise<ICredentialDataDecryptedObject>, [string]>;
|
|
|
|
jest.mock('../utils', () => {
|
|
const mockRedisClient = mock<RedisClientType>();
|
|
return {
|
|
setupRedisClient: jest.fn().mockReturnValue(mockRedisClient),
|
|
};
|
|
});
|
|
|
|
describe('Redis Trigger Node', () => {
|
|
const channel = 'testing';
|
|
const credentials = mock<ICredentialDataDecryptedObject>();
|
|
const triggerFunctions = mock<ITriggerFunctions>({
|
|
helpers: { returnJsonArray },
|
|
});
|
|
|
|
beforeEach(() => {
|
|
jest.clearAllMocks();
|
|
|
|
// Jest always uses the last fn overload's types, override to use the overload we need
|
|
(triggerFunctions.getCredentials as unknown as GetCredentialsMock)
|
|
.calledWith('redis')
|
|
.mockResolvedValue(credentials);
|
|
triggerFunctions.getNodeParameter.calledWith('channels').mockReturnValue(channel);
|
|
});
|
|
|
|
it('should emit in manual mode', async () => {
|
|
triggerFunctions.getMode.mockReturnValue('manual');
|
|
triggerFunctions.getNodeParameter.calledWith('options').mockReturnValue({});
|
|
|
|
const response = await new RedisTrigger().trigger.call(triggerFunctions);
|
|
expect(response.manualTriggerFunction).toBeDefined();
|
|
expect(response.closeFunction).toBeDefined();
|
|
|
|
expect(triggerFunctions.getCredentials).toHaveBeenCalledTimes(1);
|
|
expect(triggerFunctions.getNodeParameter).toHaveBeenCalledTimes(2);
|
|
|
|
const mockRedisClient = setupRedisClient(mock());
|
|
expect(mockRedisClient.connect).toHaveBeenCalledTimes(1);
|
|
expect(mockRedisClient.ping).toHaveBeenCalledTimes(1);
|
|
|
|
// manually trigger the node, like Workflow.runNode does
|
|
const triggerPromise = response.manualTriggerFunction!();
|
|
|
|
const onMessageCaptor = captor<(message: string, channel: string) => unknown>();
|
|
expect(mockRedisClient.pSubscribe).toHaveBeenCalledWith([channel], onMessageCaptor);
|
|
expect(triggerFunctions.emit).not.toHaveBeenCalled();
|
|
|
|
// simulate a message
|
|
const onMessage = onMessageCaptor.value;
|
|
onMessage('{"testing": true}', channel);
|
|
expect(triggerFunctions.emit).toHaveBeenCalledWith([
|
|
[{ json: { message: '{"testing": true}', channel } }],
|
|
]);
|
|
|
|
// wait for the promise to resolve
|
|
await new Promise((resolve) => setImmediate(resolve));
|
|
await expect(triggerPromise).resolves.toEqual(undefined);
|
|
|
|
expect(mockRedisClient.quit).not.toHaveBeenCalled();
|
|
await response.closeFunction!();
|
|
expect(mockRedisClient.pUnsubscribe).toHaveBeenCalledTimes(1);
|
|
expect(mockRedisClient.quit).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it('should emit in trigger mode', async () => {
|
|
triggerFunctions.getMode.mockReturnValue('trigger');
|
|
triggerFunctions.getNodeParameter.calledWith('options').mockReturnValue({});
|
|
|
|
const response = await new RedisTrigger().trigger.call(triggerFunctions);
|
|
expect(response.manualTriggerFunction).toBeDefined();
|
|
expect(response.closeFunction).toBeDefined();
|
|
|
|
expect(triggerFunctions.getCredentials).toHaveBeenCalledTimes(1);
|
|
expect(triggerFunctions.getNodeParameter).toHaveBeenCalledTimes(2);
|
|
|
|
const mockRedisClient = setupRedisClient(mock());
|
|
expect(mockRedisClient.connect).toHaveBeenCalledTimes(1);
|
|
expect(mockRedisClient.ping).toHaveBeenCalledTimes(1);
|
|
|
|
const onMessageCaptor = captor<(message: string, channel: string) => unknown>();
|
|
expect(mockRedisClient.pSubscribe).toHaveBeenCalledWith([channel], onMessageCaptor);
|
|
expect(triggerFunctions.emit).not.toHaveBeenCalled();
|
|
|
|
// simulate a message
|
|
const onMessage = onMessageCaptor.value;
|
|
onMessage('{"testing": true}', channel);
|
|
expect(triggerFunctions.emit).toHaveBeenCalledWith([
|
|
[{ json: { message: '{"testing": true}', channel } }],
|
|
]);
|
|
|
|
expect(mockRedisClient.quit).not.toHaveBeenCalled();
|
|
await response.closeFunction!();
|
|
expect(mockRedisClient.pUnsubscribe).toHaveBeenCalledTimes(1);
|
|
expect(mockRedisClient.quit).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it('should parse JSON messages when configured', async () => {
|
|
triggerFunctions.getMode.mockReturnValue('trigger');
|
|
triggerFunctions.getNodeParameter.calledWith('options').mockReturnValue({
|
|
jsonParseBody: true,
|
|
});
|
|
|
|
await new RedisTrigger().trigger.call(triggerFunctions);
|
|
|
|
const mockRedisClient = setupRedisClient(mock());
|
|
const onMessageCaptor = captor<(message: string, channel: string) => unknown>();
|
|
expect(mockRedisClient.pSubscribe).toHaveBeenCalledWith([channel], onMessageCaptor);
|
|
|
|
// simulate a message
|
|
const onMessage = onMessageCaptor.value;
|
|
onMessage('{"testing": true}', channel);
|
|
expect(triggerFunctions.emit).toHaveBeenCalledWith([
|
|
[{ json: { message: { testing: true }, channel } }],
|
|
]);
|
|
});
|
|
});
|