mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-09 20:07:31 -08:00
847f6ac771
Statistics collection about the first time a workflow loads data simply attempts an insert to db, and if it fails, we just ignore. This was causing this query to fire against production workflows multiple times, and since we want to insert only and detect whether the insertion failed, performing a select first provides gains both in terms of performance, as it's usually faster than trying an insertion as well as preventing unnecessary noise in logs. Github issue / Community forum post (link here to close automatically): https://community.n8n.io/t/duplicate-key-value-violates-unique-constraint-workflow-statistics-pkey-still-happening/29283 https://github.com/n8n-io/n8n/issues/7256 https://community.n8n.io/t/error-log-arriving-in-postgres/30191 https://github.com/n8n-io/n8n/issues/7256 https://community.n8n.io/t/cant-launch-webhooks-unable-to-find-data-of-execution/31867 --------- Co-authored-by: कारतोफ्फेलस्क्रिप्ट™ <aditya@netroy.in>
54 lines
2 KiB
TypeScript
54 lines
2 KiB
TypeScript
import { WorkflowStatisticsRepository } from '@db/repositories/workflowStatistics.repository';
|
|
import { DataSource, EntityManager, InsertResult, QueryFailedError } from 'typeorm';
|
|
import { mockInstance } from '../../shared/mocking';
|
|
import { mock, mockClear } from 'jest-mock-extended';
|
|
import { StatisticsNames, WorkflowStatistics } from '@/databases/entities/WorkflowStatistics';
|
|
|
|
const entityManager = mockInstance(EntityManager);
|
|
const dataSource = mockInstance(DataSource, { manager: entityManager });
|
|
dataSource.getMetadata.mockReturnValue(mock());
|
|
Object.assign(entityManager, { connection: dataSource });
|
|
const workflowStatisticsRepository = new WorkflowStatisticsRepository(dataSource);
|
|
|
|
describe('insertWorkflowStatistics', () => {
|
|
beforeEach(() => {
|
|
mockClear(entityManager.insert);
|
|
});
|
|
it('Successfully inserts data when it is not yet present', async () => {
|
|
entityManager.findOne.mockResolvedValueOnce(null);
|
|
entityManager.insert.mockResolvedValueOnce(mockInstance(InsertResult));
|
|
|
|
const insertionResult = await workflowStatisticsRepository.insertWorkflowStatistics(
|
|
StatisticsNames.dataLoaded,
|
|
'workflowId',
|
|
);
|
|
|
|
expect(insertionResult).toBe('insert');
|
|
});
|
|
|
|
it('Does not insert when data is present', async () => {
|
|
entityManager.findOne.mockResolvedValueOnce(mockInstance(WorkflowStatistics));
|
|
const insertionResult = await workflowStatisticsRepository.insertWorkflowStatistics(
|
|
StatisticsNames.dataLoaded,
|
|
'workflowId',
|
|
);
|
|
|
|
expect(insertionResult).toBe('alreadyExists');
|
|
expect(entityManager.insert).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('throws an error when insertion fails', async () => {
|
|
entityManager.findOne.mockResolvedValueOnce(null);
|
|
entityManager.insert.mockImplementation(async () => {
|
|
throw new QueryFailedError('Query', [], 'driver error');
|
|
});
|
|
|
|
const insertionResult = await workflowStatisticsRepository.insertWorkflowStatistics(
|
|
StatisticsNames.dataLoaded,
|
|
'workflowId',
|
|
);
|
|
|
|
expect(insertionResult).toBe('failed');
|
|
});
|
|
});
|