2024-11-26 07:04:24 -08:00
|
|
|
import { mockInstance } from 'n8n-core/test/utils';
|
2024-11-12 01:28:32 -08:00
|
|
|
import { Container } from 'typedi';
|
|
|
|
|
|
|
|
import type { AnnotationTagEntity } from '@/databases/entities/annotation-tag-entity.ee';
|
|
|
|
import type { User } from '@/databases/entities/user';
|
|
|
|
import type { WorkflowEntity } from '@/databases/entities/workflow-entity';
|
|
|
|
import { TestDefinitionRepository } from '@/databases/repositories/test-definition.repository.ee';
|
2024-11-26 07:04:24 -08:00
|
|
|
import { TestRunnerService } from '@/evaluation/test-runner/test-runner.service.ee';
|
2024-11-12 01:28:32 -08:00
|
|
|
import { createAnnotationTags } from '@test-integration/db/executions';
|
|
|
|
|
|
|
|
import { createUserShell } from './../shared/db/users';
|
|
|
|
import { createWorkflow } from './../shared/db/workflows';
|
|
|
|
import * as testDb from './../shared/test-db';
|
|
|
|
import type { SuperAgentTest } from './../shared/types';
|
|
|
|
import * as utils from './../shared/utils/';
|
|
|
|
|
2024-11-26 07:04:24 -08:00
|
|
|
const testRunner = mockInstance(TestRunnerService);
|
|
|
|
|
2024-11-12 01:28:32 -08:00
|
|
|
let authOwnerAgent: SuperAgentTest;
|
|
|
|
let workflowUnderTest: WorkflowEntity;
|
2024-11-26 06:01:00 -08:00
|
|
|
let workflowUnderTest2: WorkflowEntity;
|
2024-11-12 01:28:32 -08:00
|
|
|
let evaluationWorkflow: WorkflowEntity;
|
|
|
|
let otherWorkflow: WorkflowEntity;
|
|
|
|
let ownerShell: User;
|
|
|
|
let annotationTag: AnnotationTagEntity;
|
|
|
|
const testServer = utils.setupTestServer({ endpointGroups: ['evaluation'] });
|
|
|
|
|
|
|
|
beforeAll(async () => {
|
|
|
|
ownerShell = await createUserShell('global:owner');
|
|
|
|
authOwnerAgent = testServer.authAgentFor(ownerShell);
|
|
|
|
});
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
await testDb.truncate(['TestDefinition', 'Workflow', 'AnnotationTag']);
|
|
|
|
|
|
|
|
workflowUnderTest = await createWorkflow({ name: 'workflow-under-test' }, ownerShell);
|
2024-11-26 06:01:00 -08:00
|
|
|
workflowUnderTest2 = await createWorkflow({ name: 'workflow-under-test-2' }, ownerShell);
|
2024-11-12 01:28:32 -08:00
|
|
|
evaluationWorkflow = await createWorkflow({ name: 'evaluation-workflow' }, ownerShell);
|
|
|
|
otherWorkflow = await createWorkflow({ name: 'other-workflow' });
|
|
|
|
annotationTag = (await createAnnotationTags(['test-tag']))[0];
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('GET /evaluation/test-definitions', () => {
|
|
|
|
test('should retrieve empty test definitions list', async () => {
|
|
|
|
const resp = await authOwnerAgent.get('/evaluation/test-definitions');
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.count).toBe(0);
|
|
|
|
expect(resp.body.data.testDefinitions).toHaveLength(0);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should retrieve test definitions list', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.get('/evaluation/test-definitions');
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data).toEqual({
|
|
|
|
count: 1,
|
|
|
|
testDefinitions: [
|
|
|
|
expect.objectContaining({
|
|
|
|
name: 'test',
|
|
|
|
workflowId: workflowUnderTest.id,
|
|
|
|
evaluationWorkflowId: null,
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should retrieve test definitions list with pagination', async () => {
|
|
|
|
// Add a bunch of test definitions
|
|
|
|
const testDefinitions = [];
|
|
|
|
|
|
|
|
for (let i = 0; i < 15; i++) {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: `test-${i}`,
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
testDefinitions.push(newTest);
|
|
|
|
}
|
|
|
|
|
|
|
|
await Container.get(TestDefinitionRepository).save(testDefinitions);
|
|
|
|
|
|
|
|
// Fetch the first page
|
|
|
|
let resp = await authOwnerAgent.get('/evaluation/test-definitions?take=10');
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.count).toBe(15);
|
|
|
|
expect(resp.body.data.testDefinitions).toHaveLength(10);
|
|
|
|
|
|
|
|
// Fetch the second page
|
|
|
|
resp = await authOwnerAgent.get('/evaluation/test-definitions?take=10&skip=10');
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.count).toBe(15);
|
|
|
|
expect(resp.body.data.testDefinitions).toHaveLength(5);
|
|
|
|
});
|
2024-11-26 06:01:00 -08:00
|
|
|
|
|
|
|
test('should retrieve test definitions list for a workflow', async () => {
|
|
|
|
// Add a bunch of test definitions for two different workflows
|
|
|
|
const testDefinitions = [];
|
|
|
|
|
|
|
|
for (let i = 0; i < 15; i++) {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: `test-${i}`,
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
|
|
|
|
const newTest2 = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: `test-${i * 2}`,
|
|
|
|
workflow: { id: workflowUnderTest2.id },
|
|
|
|
});
|
|
|
|
|
|
|
|
testDefinitions.push(newTest, newTest2);
|
|
|
|
}
|
|
|
|
|
|
|
|
await Container.get(TestDefinitionRepository).save(testDefinitions);
|
|
|
|
|
|
|
|
// Fetch test definitions of a second workflow
|
|
|
|
let resp = await authOwnerAgent.get(
|
|
|
|
`/evaluation/test-definitions?filter=${JSON.stringify({ workflowId: workflowUnderTest2.id })}`,
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.count).toBe(15);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return error if user has no access to the workflowId specified in filter', async () => {
|
|
|
|
let resp = await authOwnerAgent.get(
|
|
|
|
`/evaluation/test-definitions?filter=${JSON.stringify({ workflowId: otherWorkflow.id })}`,
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(403);
|
|
|
|
expect(resp.body.message).toBe('User does not have access to the workflow');
|
|
|
|
});
|
2024-11-12 01:28:32 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('GET /evaluation/test-definitions/:id', () => {
|
|
|
|
test('should retrieve test definition', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.get(`/evaluation/test-definitions/${newTest.id}`);
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.name).toBe('test');
|
|
|
|
expect(resp.body.data.workflowId).toBe(workflowUnderTest.id);
|
|
|
|
expect(resp.body.data.evaluationWorkflowId).toBe(null);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return 404 for non-existent test definition', async () => {
|
|
|
|
const resp = await authOwnerAgent.get('/evaluation/test-definitions/123');
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(404);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should retrieve test definition with evaluation workflow', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
evaluationWorkflow: { id: evaluationWorkflow.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.get(`/evaluation/test-definitions/${newTest.id}`);
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.name).toBe('test');
|
|
|
|
expect(resp.body.data.workflowId).toBe(workflowUnderTest.id);
|
|
|
|
expect(resp.body.data.evaluationWorkflowId).toBe(evaluationWorkflow.id);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should not retrieve test definition if user does not have access to workflow under test', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: otherWorkflow.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.get(`/evaluation/test-definitions/${newTest.id}`);
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(404);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('POST /evaluation/test-definitions', () => {
|
|
|
|
test('should create test definition', async () => {
|
|
|
|
const resp = await authOwnerAgent.post('/evaluation/test-definitions').send({
|
|
|
|
name: 'test',
|
|
|
|
workflowId: workflowUnderTest.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.name).toBe('test');
|
|
|
|
expect(resp.body.data.workflowId).toBe(workflowUnderTest.id);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should create test definition with evaluation workflow', async () => {
|
|
|
|
const resp = await authOwnerAgent.post('/evaluation/test-definitions').send({
|
|
|
|
name: 'test',
|
|
|
|
workflowId: workflowUnderTest.id,
|
|
|
|
evaluationWorkflowId: evaluationWorkflow.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
2024-11-14 05:45:02 -08:00
|
|
|
expect(resp.body.data).toEqual(
|
|
|
|
expect.objectContaining({
|
|
|
|
name: 'test',
|
|
|
|
workflowId: workflowUnderTest.id,
|
|
|
|
evaluationWorkflowId: evaluationWorkflow.id,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should create test definition with all fields', async () => {
|
|
|
|
const resp = await authOwnerAgent.post('/evaluation/test-definitions').send({
|
|
|
|
name: 'test',
|
|
|
|
description: 'test description',
|
|
|
|
workflowId: workflowUnderTest.id,
|
|
|
|
evaluationWorkflowId: evaluationWorkflow.id,
|
|
|
|
annotationTagId: annotationTag.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data).toEqual(
|
|
|
|
expect.objectContaining({
|
|
|
|
name: 'test',
|
|
|
|
description: 'test description',
|
|
|
|
workflowId: workflowUnderTest.id,
|
|
|
|
evaluationWorkflowId: evaluationWorkflow.id,
|
|
|
|
annotationTag: expect.objectContaining({
|
|
|
|
id: annotationTag.id,
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
);
|
2024-11-12 01:28:32 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
test('should return error if name is empty', async () => {
|
|
|
|
const resp = await authOwnerAgent.post('/evaluation/test-definitions').send({
|
|
|
|
name: '',
|
|
|
|
workflowId: workflowUnderTest.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(400);
|
|
|
|
expect(resp.body.errors).toEqual(
|
|
|
|
expect.arrayContaining([
|
|
|
|
expect.objectContaining({
|
|
|
|
code: 'too_small',
|
|
|
|
path: ['name'],
|
|
|
|
}),
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return error if user has no access to the workflow', async () => {
|
|
|
|
const resp = await authOwnerAgent.post('/evaluation/test-definitions').send({
|
|
|
|
name: 'test',
|
|
|
|
workflowId: otherWorkflow.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(403);
|
|
|
|
expect(resp.body.message).toBe('User does not have access to the workflow');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return error if user has no access to the evaluation workflow', async () => {
|
|
|
|
const resp = await authOwnerAgent.post('/evaluation/test-definitions').send({
|
|
|
|
name: 'test',
|
|
|
|
workflowId: workflowUnderTest.id,
|
|
|
|
evaluationWorkflowId: otherWorkflow.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(403);
|
|
|
|
expect(resp.body.message).toBe('User does not have access to the evaluation workflow');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('PATCH /evaluation/test-definitions/:id', () => {
|
|
|
|
test('should update test definition', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.patch(`/evaluation/test-definitions/${newTest.id}`).send({
|
|
|
|
name: 'updated-test',
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.name).toBe('updated-test');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return 404 if user has no access to the workflow', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: otherWorkflow.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.patch(`/evaluation/test-definitions/${newTest.id}`).send({
|
|
|
|
name: 'updated-test',
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(404);
|
|
|
|
expect(resp.body.message).toBe('Test definition not found');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should update test definition with evaluation workflow', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.patch(`/evaluation/test-definitions/${newTest.id}`).send({
|
|
|
|
name: 'updated-test',
|
|
|
|
evaluationWorkflowId: evaluationWorkflow.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.name).toBe('updated-test');
|
|
|
|
expect(resp.body.data.evaluationWorkflowId).toBe(evaluationWorkflow.id);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return error if user has no access to the evaluation workflow', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.patch(`/evaluation/test-definitions/${newTest.id}`).send({
|
|
|
|
name: 'updated-test',
|
|
|
|
evaluationWorkflowId: otherWorkflow.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(403);
|
|
|
|
expect(resp.body.message).toBe('User does not have access to the evaluation workflow');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should disallow workflowId', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.patch(`/evaluation/test-definitions/${newTest.id}`).send({
|
|
|
|
name: 'updated-test',
|
|
|
|
workflowId: otherWorkflow.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(400);
|
|
|
|
expect(resp.body.errors).toEqual(
|
|
|
|
expect.arrayContaining([
|
|
|
|
expect.objectContaining({
|
|
|
|
code: 'unrecognized_keys',
|
|
|
|
keys: ['workflowId'],
|
|
|
|
}),
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should update annotationTagId', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.patch(`/evaluation/test-definitions/${newTest.id}`).send({
|
|
|
|
annotationTagId: annotationTag.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.annotationTag.id).toBe(annotationTag.id);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return error if annotationTagId is invalid', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.patch(`/evaluation/test-definitions/${newTest.id}`).send({
|
|
|
|
annotationTagId: '123',
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(400);
|
|
|
|
expect(resp.body.message).toBe('Annotation tag not found');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('DELETE /evaluation/test-definitions/:id', () => {
|
|
|
|
test('should delete test definition', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.delete(`/evaluation/test-definitions/${newTest.id}`);
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(200);
|
|
|
|
expect(resp.body.data.success).toBe(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return 404 if test definition does not exist', async () => {
|
|
|
|
const resp = await authOwnerAgent.delete('/evaluation/test-definitions/123');
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(404);
|
|
|
|
expect(resp.body.message).toBe('Test definition not found');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should return 404 if user has no access to the workflow', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: otherWorkflow.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.delete(`/evaluation/test-definitions/${newTest.id}`);
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(404);
|
|
|
|
expect(resp.body.message).toBe('Test definition not found');
|
|
|
|
});
|
|
|
|
});
|
2024-11-26 07:04:24 -08:00
|
|
|
|
|
|
|
describe('POST /evaluation/test-definitions/:id/run', () => {
|
|
|
|
test('should trigger the test run', async () => {
|
|
|
|
const newTest = Container.get(TestDefinitionRepository).create({
|
|
|
|
name: 'test',
|
|
|
|
workflow: { id: workflowUnderTest.id },
|
|
|
|
});
|
|
|
|
await Container.get(TestDefinitionRepository).save(newTest);
|
|
|
|
|
|
|
|
const resp = await authOwnerAgent.post(`/evaluation/test-definitions/${newTest.id}/run`);
|
|
|
|
|
|
|
|
expect(resp.statusCode).toBe(202);
|
|
|
|
expect(resp.body).toEqual(
|
|
|
|
expect.objectContaining({
|
|
|
|
success: true,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(testRunner.runTest).toHaveBeenCalledTimes(1);
|
|
|
|
});
|
|
|
|
});
|