2023-01-30 08:34:26 -08:00
|
|
|
import { LoggerProxy, WorkflowExecuteMode } from 'n8n-workflow';
|
|
|
|
import { QueryFailedError } from 'typeorm';
|
2022-12-06 06:55:40 -08:00
|
|
|
import config from '@/config';
|
2023-01-30 08:34:26 -08:00
|
|
|
import { Db } from '@/index';
|
2022-12-22 01:14:15 -08:00
|
|
|
import { nodeFetchedData, workflowExecutionCompleted } from '@/events/WorkflowStatistics';
|
2022-12-20 01:52:01 -08:00
|
|
|
import { getLogger } from '@/Logger';
|
2023-02-21 10:21:56 -08:00
|
|
|
import * as UserManagementHelper from '@/UserManagement/UserManagementHelper';
|
|
|
|
import { InternalHooks } from '@/InternalHooks';
|
|
|
|
import { mockInstance } from '../integration/shared/utils';
|
2022-12-06 06:55:40 -08:00
|
|
|
|
|
|
|
const FAKE_USER_ID = 'abcde-fghij';
|
|
|
|
|
2022-12-22 01:14:15 -08:00
|
|
|
jest.mock('@/Db', () => {
|
2022-12-06 06:55:40 -08:00
|
|
|
return {
|
|
|
|
collections: {
|
|
|
|
WorkflowStatistics: {
|
2023-01-30 08:34:26 -08:00
|
|
|
insert: jest.fn((...args) => {}),
|
2022-12-06 06:55:40 -08:00
|
|
|
update: jest.fn((...args) => {}),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
2023-02-21 10:21:56 -08:00
|
|
|
jest.spyOn(UserManagementHelper, 'getWorkflowOwner').mockImplementation(async (_workflowId) => {
|
2023-01-30 08:34:26 -08:00
|
|
|
return { id: FAKE_USER_ID };
|
2022-12-06 06:55:40 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('Events', () => {
|
2023-02-21 10:21:56 -08:00
|
|
|
const internalHooks = mockInstance(InternalHooks);
|
|
|
|
|
2022-12-06 06:55:40 -08:00
|
|
|
beforeAll(() => {
|
|
|
|
config.set('diagnostics.enabled', true);
|
|
|
|
config.set('deployment.type', 'n8n-testing');
|
2022-12-20 01:52:01 -08:00
|
|
|
LoggerProxy.init(getLogger());
|
2022-12-06 06:55:40 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
afterAll(() => {
|
|
|
|
jest.clearAllTimers();
|
|
|
|
jest.useRealTimers();
|
|
|
|
});
|
|
|
|
|
|
|
|
beforeEach(() => {
|
2023-02-21 10:21:56 -08:00
|
|
|
internalHooks.onFirstProductionWorkflowSuccess.mockClear();
|
|
|
|
internalHooks.onFirstWorkflowDataLoad.mockClear();
|
2022-12-06 06:55:40 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {});
|
|
|
|
|
|
|
|
describe('workflowExecutionCompleted', () => {
|
|
|
|
test('should create metrics for production successes', async () => {
|
|
|
|
// Call the function with a production success result, ensure metrics hook gets called
|
|
|
|
const workflow = {
|
|
|
|
id: '1',
|
|
|
|
name: '',
|
|
|
|
active: false,
|
|
|
|
createdAt: new Date(),
|
|
|
|
updatedAt: new Date(),
|
|
|
|
nodes: [],
|
|
|
|
connections: {},
|
|
|
|
};
|
|
|
|
const runData = {
|
|
|
|
finished: true,
|
|
|
|
data: { resultData: { runData: {} } },
|
|
|
|
mode: 'internal' as WorkflowExecuteMode,
|
|
|
|
startedAt: new Date(),
|
|
|
|
};
|
|
|
|
await workflowExecutionCompleted(workflow, runData);
|
2023-02-21 10:21:56 -08:00
|
|
|
expect(internalHooks.onFirstProductionWorkflowSuccess).toBeCalledTimes(1);
|
|
|
|
expect(internalHooks.onFirstProductionWorkflowSuccess).toHaveBeenNthCalledWith(1, {
|
2022-12-06 06:55:40 -08:00
|
|
|
user_id: FAKE_USER_ID,
|
2023-01-02 08:42:32 -08:00
|
|
|
workflow_id: workflow.id,
|
2022-12-06 06:55:40 -08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should only create metrics for production successes', async () => {
|
|
|
|
// Call the function with a non production success result, ensure metrics hook is never called
|
|
|
|
const workflow = {
|
|
|
|
id: '1',
|
|
|
|
name: '',
|
|
|
|
active: false,
|
|
|
|
createdAt: new Date(),
|
|
|
|
updatedAt: new Date(),
|
|
|
|
nodes: [],
|
|
|
|
connections: {},
|
|
|
|
};
|
|
|
|
const runData = {
|
|
|
|
finished: false,
|
|
|
|
data: { resultData: { runData: {} } },
|
|
|
|
mode: 'internal' as WorkflowExecuteMode,
|
|
|
|
startedAt: new Date(),
|
|
|
|
};
|
|
|
|
await workflowExecutionCompleted(workflow, runData);
|
2023-02-21 10:21:56 -08:00
|
|
|
expect(internalHooks.onFirstProductionWorkflowSuccess).toBeCalledTimes(0);
|
2022-12-06 06:55:40 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
test('should not send metrics for updated entries', async () => {
|
2023-01-30 08:34:26 -08:00
|
|
|
// Call the function with a fail insert, ensure update is called *and* metrics aren't sent
|
2023-02-21 10:21:56 -08:00
|
|
|
Db.collections.WorkflowStatistics.insert.mockImplementationOnce(() => {
|
2023-01-30 08:34:26 -08:00
|
|
|
throw new QueryFailedError('invalid insert', [], '');
|
|
|
|
});
|
2022-12-06 06:55:40 -08:00
|
|
|
const workflow = {
|
2023-01-30 08:34:26 -08:00
|
|
|
id: '1',
|
2022-12-06 06:55:40 -08:00
|
|
|
name: '',
|
|
|
|
active: false,
|
|
|
|
createdAt: new Date(),
|
|
|
|
updatedAt: new Date(),
|
|
|
|
nodes: [],
|
|
|
|
connections: {},
|
|
|
|
};
|
|
|
|
const runData = {
|
|
|
|
finished: true,
|
|
|
|
data: { resultData: { runData: {} } },
|
|
|
|
mode: 'internal' as WorkflowExecuteMode,
|
|
|
|
startedAt: new Date(),
|
|
|
|
};
|
|
|
|
await workflowExecutionCompleted(workflow, runData);
|
2023-02-21 10:21:56 -08:00
|
|
|
expect(internalHooks.onFirstProductionWorkflowSuccess).toBeCalledTimes(0);
|
2022-12-06 06:55:40 -08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('nodeFetchedData', () => {
|
|
|
|
test('should create metrics when the db is updated', async () => {
|
|
|
|
// Call the function with a production success result, ensure metrics hook gets called
|
|
|
|
const workflowId = '1';
|
|
|
|
const node = {
|
|
|
|
id: 'abcde',
|
|
|
|
name: 'test node',
|
|
|
|
typeVersion: 1,
|
|
|
|
type: '',
|
|
|
|
position: [0, 0] as [number, number],
|
|
|
|
parameters: {},
|
|
|
|
};
|
|
|
|
await nodeFetchedData(workflowId, node);
|
2023-02-21 10:21:56 -08:00
|
|
|
expect(internalHooks.onFirstWorkflowDataLoad).toBeCalledTimes(1);
|
|
|
|
expect(internalHooks.onFirstWorkflowDataLoad).toHaveBeenNthCalledWith(1, {
|
2022-12-06 06:55:40 -08:00
|
|
|
user_id: FAKE_USER_ID,
|
2023-01-02 08:42:32 -08:00
|
|
|
workflow_id: workflowId,
|
2022-12-06 06:55:40 -08:00
|
|
|
node_type: node.type,
|
|
|
|
node_id: node.id,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should create metrics with credentials when the db is updated', async () => {
|
|
|
|
// Call the function with a production success result, ensure metrics hook gets called
|
|
|
|
const workflowId = '1';
|
|
|
|
const node = {
|
|
|
|
id: 'abcde',
|
|
|
|
name: 'test node',
|
|
|
|
typeVersion: 1,
|
|
|
|
type: '',
|
|
|
|
position: [0, 0] as [number, number],
|
|
|
|
parameters: {},
|
|
|
|
credentials: {
|
|
|
|
testCredentials: {
|
|
|
|
id: '1',
|
|
|
|
name: 'Test Credentials',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
await nodeFetchedData(workflowId, node);
|
2023-02-21 10:21:56 -08:00
|
|
|
expect(internalHooks.onFirstWorkflowDataLoad).toBeCalledTimes(1);
|
|
|
|
expect(internalHooks.onFirstWorkflowDataLoad).toHaveBeenNthCalledWith(1, {
|
2022-12-06 06:55:40 -08:00
|
|
|
user_id: FAKE_USER_ID,
|
2023-01-02 08:42:32 -08:00
|
|
|
workflow_id: workflowId,
|
2022-12-06 06:55:40 -08:00
|
|
|
node_type: node.type,
|
|
|
|
node_id: node.id,
|
|
|
|
credential_type: 'testCredentials',
|
|
|
|
credential_id: node.credentials.testCredentials.id,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should not send metrics for entries that already have the flag set', async () => {
|
|
|
|
// Fetch data for workflow 2 which is set up to not be altered in the mocks
|
2023-02-21 10:21:56 -08:00
|
|
|
Db.collections.WorkflowStatistics.insert.mockImplementationOnce(() => {
|
2023-01-30 08:34:26 -08:00
|
|
|
throw new QueryFailedError('invalid insert', [], '');
|
|
|
|
});
|
|
|
|
const workflowId = '1';
|
2022-12-06 06:55:40 -08:00
|
|
|
const node = {
|
|
|
|
id: 'abcde',
|
|
|
|
name: 'test node',
|
|
|
|
typeVersion: 1,
|
|
|
|
type: '',
|
|
|
|
position: [0, 0] as [number, number],
|
|
|
|
parameters: {},
|
|
|
|
};
|
|
|
|
await nodeFetchedData(workflowId, node);
|
2023-02-21 10:21:56 -08:00
|
|
|
expect(internalHooks.onFirstWorkflowDataLoad).toBeCalledTimes(0);
|
2022-12-06 06:55:40 -08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|