n8n/packages/cli/test/integration/controllers/dynamic-node-parameters.controller.test.ts

361 lines
11 KiB
TypeScript

import { randomCredentialPayload, testDb } from '@n8n/backend-test-utils';
import { Container } from '@n8n/di';
import { mock } from 'jest-mock-extended';
import type {
INodeListSearchResult,
IWorkflowExecuteAdditionalData,
ResourceMapperFields,
NodeParameterValueType,
} from 'n8n-workflow';
import { DynamicNodeParametersService } from '@/services/dynamic-node-parameters.service';
import * as AdditionalData from '@/workflow-execute-additional-data';
import { saveCredential } from '../shared/db/credentials';
import { createMember, createOwner } from '../shared/db/users';
import type { SuperAgentTest } from '../shared/types';
import { setupTestServer } from '../shared/utils';
describe('DynamicNodeParametersController', () => {
const additionalData = mock<IWorkflowExecuteAdditionalData>();
const testServer = setupTestServer({ endpointGroups: ['dynamic-node-parameters'] });
let ownerAgent: SuperAgentTest;
let service: DynamicNodeParametersService;
beforeAll(() => {
service = Container.get(DynamicNodeParametersService);
});
beforeEach(() => {
jest.clearAllMocks();
jest.spyOn(AdditionalData, 'getBase').mockResolvedValue(additionalData);
});
afterEach(() => {
jest.restoreAllMocks();
});
const commonRequestParams = {
credentials: {},
currentNodeParameters: {},
nodeTypeAndVersion: { name: 'TestNode', version: 1 },
path: 'path',
};
describe('POST /dynamic-node-parameters/options', () => {
it('should take params via body', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
jest.spyOn(service, 'getOptionsViaMethodName').mockResolvedValue([]);
await ownerAgent
.post('/dynamic-node-parameters/options')
.send({
...commonRequestParams,
methodName: 'testMethod',
})
.expect(200);
});
it('should take params with loadOptions', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
const expectedResult = [{ name: 'Test Option', value: 'test' }];
jest.spyOn(service, 'getOptionsViaLoadOptions').mockResolvedValue(expectedResult);
const response = await ownerAgent
.post('/dynamic-node-parameters/options')
.send({
...commonRequestParams,
loadOptions: { type: 'test' },
})
.expect(200);
expect(response.body).toEqual({ data: expectedResult });
});
it('should return empty array when no method or loadOptions provided', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
const response = await ownerAgent
.post('/dynamic-node-parameters/options')
.send({
...commonRequestParams,
})
.expect(200);
expect(response.body).toEqual({ data: [] });
});
});
describe('POST /dynamic-node-parameters/resource-locator-results', () => {
it('should return resource locator results', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
const expectedResult: INodeListSearchResult = { results: [] };
jest.spyOn(service, 'getResourceLocatorResults').mockResolvedValue(expectedResult);
const response = await ownerAgent
.post('/dynamic-node-parameters/resource-locator-results')
.send({
...commonRequestParams,
methodName: 'testMethod',
filter: 'testFilter',
paginationToken: 'testToken',
})
.expect(200);
expect(response.body).toEqual({ data: expectedResult });
});
it('should handle resource locator results without pagination', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
jest.spyOn(service, 'getResourceLocatorResults').mockResolvedValue({ results: [] });
await ownerAgent
.post('/dynamic-node-parameters/resource-locator-results')
.send({
methodName: 'testMethod',
...commonRequestParams,
})
.expect(200);
});
it('should return a 400 if methodName is not defined', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
await ownerAgent
.post('/dynamic-node-parameters/resource-locator-results')
.send(commonRequestParams)
.expect(400);
});
});
describe('POST /dynamic-node-parameters/resource-mapper-fields', () => {
it('should return resource mapper fields', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
const expectedResult: ResourceMapperFields = { fields: [] };
jest.spyOn(service, 'getResourceMappingFields').mockResolvedValue(expectedResult);
const response = await ownerAgent
.post('/dynamic-node-parameters/resource-mapper-fields')
.send({
...commonRequestParams,
methodName: 'testMethod',
loadOptions: 'testLoadOptions',
})
.expect(200);
expect(response.body).toEqual({ data: expectedResult });
});
it('should return a 400 if methodName is not defined', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
await ownerAgent
.post('/dynamic-node-parameters/resource-mapper-fields')
.send(commonRequestParams)
.expect(400);
});
});
describe('POST /dynamic-node-parameters/local-resource-mapper-fields', () => {
it('should return local resource mapper fields', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
const expectedResult: ResourceMapperFields = { fields: [] };
jest.spyOn(service, 'getLocalResourceMappingFields').mockResolvedValue(expectedResult);
const response = await ownerAgent
.post('/dynamic-node-parameters/local-resource-mapper-fields')
.send({
...commonRequestParams,
methodName: 'testMethod',
})
.expect(200);
expect(response.body).toEqual({ data: expectedResult });
});
it('should return a 400 if methodName is not defined', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
await ownerAgent
.post('/dynamic-node-parameters/local-resource-mapper-fields')
.send(commonRequestParams)
.expect(400);
});
});
describe('POST /dynamic-node-parameters/action-result', () => {
it('should return action result with handler', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
const expectedResult: NodeParameterValueType = { test: true };
jest.spyOn(service, 'getActionResult').mockResolvedValue(expectedResult);
const response = await ownerAgent
.post('/dynamic-node-parameters/action-result')
.send({
...commonRequestParams,
handler: 'testHandler',
payload: { someData: 'test' },
})
.expect(200);
expect(response.body).toEqual({ data: expectedResult });
});
it('should return a 400 if handler is not defined', async () => {
const owner = await createOwner();
ownerAgent = testServer.authAgentFor(owner);
await ownerAgent
.post('/dynamic-node-parameters/action-result')
.send({
...commonRequestParams,
payload: { someData: 'test' },
})
.expect(400);
});
});
describe('credential access enforcement', () => {
let member: Awaited<ReturnType<typeof createMember>>;
let memberAgent: SuperAgentTest;
beforeEach(async () => {
await testDb.truncate(['SharedCredentials', 'CredentialsEntity']);
member = await createMember();
memberAgent = testServer.authAgentFor(member);
});
it('should return 403 when user does not have access to supplied credential on /options', async () => {
const owner = await createOwner();
const ownerCredential = await saveCredential(randomCredentialPayload(), {
user: owner,
role: 'credential:owner',
});
await memberAgent
.post('/dynamic-node-parameters/options')
.send({
...commonRequestParams,
credentials: { httpBasicAuth: { id: ownerCredential.id, name: ownerCredential.name } },
})
.expect(403);
});
it('should return 403 when user does not have access to supplied credential on /resource-locator-results', async () => {
const owner = await createOwner();
const ownerCredential = await saveCredential(randomCredentialPayload(), {
user: owner,
role: 'credential:owner',
});
await memberAgent
.post('/dynamic-node-parameters/resource-locator-results')
.send({
...commonRequestParams,
methodName: 'searchItems',
credentials: { httpBasicAuth: { id: ownerCredential.id, name: ownerCredential.name } },
})
.expect(403);
});
it('should return 403 when user does not have access to supplied credential on /resource-mapper-fields', async () => {
const owner = await createOwner();
const ownerCredential = await saveCredential(randomCredentialPayload(), {
user: owner,
role: 'credential:owner',
});
await memberAgent
.post('/dynamic-node-parameters/resource-mapper-fields')
.send({
...commonRequestParams,
methodName: 'getFields',
credentials: { httpBasicAuth: { id: ownerCredential.id, name: ownerCredential.name } },
})
.expect(403);
});
it('should return 403 when user does not have access to supplied credential on /action-result', async () => {
const owner = await createOwner();
const ownerCredential = await saveCredential(randomCredentialPayload(), {
user: owner,
role: 'credential:owner',
});
await memberAgent
.post('/dynamic-node-parameters/action-result')
.send({
...commonRequestParams,
handler: 'handleAction',
payload: {},
credentials: { httpBasicAuth: { id: ownerCredential.id, name: ownerCredential.name } },
})
.expect(403);
});
it('should return 403 when workflowId is absent but credential is inaccessible', async () => {
const owner = await createOwner();
const ownerCredential = await saveCredential(randomCredentialPayload(), {
user: owner,
role: 'credential:owner',
});
// No workflowId — ensures the early-return bypass path is also blocked
await memberAgent
.post('/dynamic-node-parameters/options')
.send({
...commonRequestParams,
credentials: { httpBasicAuth: { id: ownerCredential.id, name: ownerCredential.name } },
})
.expect(403);
});
it('should not return 403 when user has access to the supplied credential', async () => {
const ownCredential = await saveCredential(randomCredentialPayload(), {
user: member,
role: 'credential:owner',
});
jest.spyOn(service, 'getOptionsViaMethodName').mockResolvedValue([]);
const { status } = await memberAgent.post('/dynamic-node-parameters/options').send({
...commonRequestParams,
methodName: 'testMethod',
credentials: { httpBasicAuth: { id: ownCredential.id, name: ownCredential.name } },
});
expect(status).not.toBe(403);
});
it('should not return 403 when no credentials are supplied', async () => {
jest.spyOn(service, 'getOptionsViaMethodName').mockResolvedValue([]);
const { status } = await memberAgent.post('/dynamic-node-parameters/options').send({
...commonRequestParams,
methodName: 'testMethod',
});
expect(status).not.toBe(403);
});
});
});