mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-07 10:57:29 -08:00
317 lines
7.6 KiB
TypeScript
317 lines
7.6 KiB
TypeScript
|
import { mock } from 'jest-mock-extended';
|
||
|
import { NodeOperationError, type INode } from 'n8n-workflow';
|
||
|
|
||
|
import { testVersionedWebhookTriggerNode } from '@test/nodes/TriggerHelpers';
|
||
|
|
||
|
import { FormTrigger } from '../FormTrigger.node';
|
||
|
import type { FormField } from '../interfaces';
|
||
|
|
||
|
describe('FormTrigger', () => {
|
||
|
beforeEach(() => {
|
||
|
jest.clearAllMocks();
|
||
|
});
|
||
|
|
||
|
it('should render a form template with correct fields', async () => {
|
||
|
const formFields: FormField[] = [
|
||
|
{ fieldLabel: 'Name', fieldType: 'text', requiredField: true },
|
||
|
{ fieldLabel: 'Age', fieldType: 'number', requiredField: false },
|
||
|
{ fieldLabel: 'Notes', fieldType: 'textarea', requiredField: false },
|
||
|
{
|
||
|
fieldLabel: 'Gender',
|
||
|
fieldType: 'select',
|
||
|
requiredField: true,
|
||
|
fieldOptions: { values: [{ option: 'Male' }, { option: 'Female' }] },
|
||
|
},
|
||
|
{
|
||
|
fieldLabel: 'Resume',
|
||
|
fieldType: 'file',
|
||
|
requiredField: true,
|
||
|
acceptFileTypes: '.pdf,.doc',
|
||
|
multipleFiles: false,
|
||
|
},
|
||
|
];
|
||
|
|
||
|
const { response, responseData } = await testVersionedWebhookTriggerNode(FormTrigger, 2, {
|
||
|
mode: 'manual',
|
||
|
node: {
|
||
|
parameters: {
|
||
|
formTitle: 'Test Form',
|
||
|
formDescription: 'Test Description',
|
||
|
responseMode: 'onReceived',
|
||
|
formFields: { values: formFields },
|
||
|
options: {
|
||
|
appendAttribution: false,
|
||
|
respondWithOptions: { values: { respondWith: 'text' } },
|
||
|
},
|
||
|
},
|
||
|
},
|
||
|
});
|
||
|
|
||
|
expect(response.render).toHaveBeenCalledWith('form-trigger', {
|
||
|
appendAttribution: false,
|
||
|
formDescription: 'Test Description',
|
||
|
formFields: [
|
||
|
{
|
||
|
defaultValue: '',
|
||
|
errorId: 'error-field-0',
|
||
|
id: 'field-0',
|
||
|
inputRequired: 'form-required',
|
||
|
isInput: true,
|
||
|
label: 'Name',
|
||
|
placeholder: undefined,
|
||
|
type: 'text',
|
||
|
},
|
||
|
{
|
||
|
defaultValue: '',
|
||
|
errorId: 'error-field-1',
|
||
|
id: 'field-1',
|
||
|
inputRequired: '',
|
||
|
isInput: true,
|
||
|
label: 'Age',
|
||
|
placeholder: undefined,
|
||
|
type: 'number',
|
||
|
},
|
||
|
{
|
||
|
defaultValue: '',
|
||
|
errorId: 'error-field-2',
|
||
|
id: 'field-2',
|
||
|
inputRequired: '',
|
||
|
label: 'Notes',
|
||
|
placeholder: undefined,
|
||
|
isTextarea: true,
|
||
|
},
|
||
|
{
|
||
|
defaultValue: '',
|
||
|
errorId: 'error-field-3',
|
||
|
id: 'field-3',
|
||
|
inputRequired: 'form-required',
|
||
|
isInput: true,
|
||
|
label: 'Gender',
|
||
|
placeholder: undefined,
|
||
|
type: 'select',
|
||
|
},
|
||
|
{
|
||
|
acceptFileTypes: '.pdf,.doc',
|
||
|
defaultValue: '',
|
||
|
errorId: 'error-field-4',
|
||
|
id: 'field-4',
|
||
|
inputRequired: 'form-required',
|
||
|
isFileInput: true,
|
||
|
label: 'Resume',
|
||
|
multipleFiles: '',
|
||
|
placeholder: undefined,
|
||
|
},
|
||
|
],
|
||
|
formSubmittedText: 'Your response has been recorded',
|
||
|
formTitle: 'Test Form',
|
||
|
n8nWebsiteLink:
|
||
|
'https://n8n.io/?utm_source=n8n-internal&utm_medium=form-trigger&utm_campaign=instanceId',
|
||
|
testRun: true,
|
||
|
useResponseData: false,
|
||
|
validForm: true,
|
||
|
});
|
||
|
|
||
|
expect(responseData).toEqual({ noWebhookResponse: true });
|
||
|
});
|
||
|
|
||
|
it('should return workflowData on POST request', async () => {
|
||
|
const formFields: FormField[] = [
|
||
|
{ fieldLabel: 'Name', fieldType: 'text', requiredField: true },
|
||
|
{ fieldLabel: 'Age', fieldType: 'number', requiredField: false },
|
||
|
{ fieldLabel: 'Date', fieldType: 'date', formatDate: 'dd MMM', requiredField: false },
|
||
|
{ fieldLabel: 'Empty', fieldType: 'number', requiredField: false },
|
||
|
{
|
||
|
fieldLabel: 'Tags',
|
||
|
fieldType: 'select',
|
||
|
multiselect: true,
|
||
|
requiredField: false,
|
||
|
fieldOptions: { values: [{ option: 'Popular' }, { option: 'Recent' }] },
|
||
|
},
|
||
|
];
|
||
|
|
||
|
const bodyData = {
|
||
|
data: {
|
||
|
'field-0': 'John Doe',
|
||
|
'field-1': '30',
|
||
|
'field-2': '2024-08-31',
|
||
|
'field-4': '{}',
|
||
|
},
|
||
|
};
|
||
|
|
||
|
const { responseData } = await testVersionedWebhookTriggerNode(FormTrigger, 2, {
|
||
|
mode: 'manual',
|
||
|
node: {
|
||
|
parameters: {
|
||
|
formTitle: 'Test Form',
|
||
|
formDescription: 'Test Description',
|
||
|
responseMode: 'onReceived',
|
||
|
formFields: { values: formFields },
|
||
|
},
|
||
|
},
|
||
|
request: { method: 'POST' },
|
||
|
bodyData,
|
||
|
});
|
||
|
|
||
|
expect(responseData).toEqual({
|
||
|
webhookResponse: { status: 200 },
|
||
|
workflowData: [
|
||
|
[
|
||
|
{
|
||
|
json: {
|
||
|
Name: 'John Doe',
|
||
|
Age: 30,
|
||
|
Date: '31 Jan',
|
||
|
Empty: null,
|
||
|
Tags: {},
|
||
|
submittedAt: expect.any(String),
|
||
|
formMode: 'test',
|
||
|
},
|
||
|
},
|
||
|
],
|
||
|
],
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('Respond to Webhook', () => {
|
||
|
it('should throw when misconfigured', async () => {
|
||
|
await expect(
|
||
|
testVersionedWebhookTriggerNode(FormTrigger, 2, {
|
||
|
node: {
|
||
|
parameters: {
|
||
|
responseMode: 'responseNode',
|
||
|
},
|
||
|
},
|
||
|
request: { method: 'POST' },
|
||
|
childNodes: [],
|
||
|
}),
|
||
|
).rejects.toEqual(
|
||
|
new NodeOperationError(mock<INode>(), 'No Respond to Webhook node found in the workflow'),
|
||
|
);
|
||
|
|
||
|
await expect(
|
||
|
testVersionedWebhookTriggerNode(FormTrigger, 2, {
|
||
|
node: {
|
||
|
parameters: {
|
||
|
responseMode: 'onReceived',
|
||
|
},
|
||
|
},
|
||
|
request: { method: 'POST' },
|
||
|
childNodes: [
|
||
|
{
|
||
|
name: 'Test Respond To Webhook',
|
||
|
type: 'n8n-nodes-base.respondToWebhook',
|
||
|
typeVersion: 1,
|
||
|
},
|
||
|
],
|
||
|
}),
|
||
|
).rejects.toEqual(
|
||
|
new NodeOperationError(mock<INode>(), 'n8n Form Trigger node not correctly configured'),
|
||
|
);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should throw on invalid webhook authentication', async () => {
|
||
|
const formFields: FormField[] = [
|
||
|
{ fieldLabel: 'Name', fieldType: 'text', requiredField: true },
|
||
|
{ fieldLabel: 'Age', fieldType: 'number', requiredField: false },
|
||
|
];
|
||
|
|
||
|
const { responseData, response } = await testVersionedWebhookTriggerNode(FormTrigger, 2, {
|
||
|
mode: 'manual',
|
||
|
node: {
|
||
|
parameters: {
|
||
|
formTitle: 'Test Form',
|
||
|
formDescription: 'Test Description',
|
||
|
responseMode: 'onReceived',
|
||
|
formFields: { values: formFields },
|
||
|
authentication: 'basicAuth',
|
||
|
},
|
||
|
},
|
||
|
request: { method: 'POST' },
|
||
|
});
|
||
|
|
||
|
expect(responseData).toEqual({ noWebhookResponse: true });
|
||
|
expect(response.status).toHaveBeenCalledWith(401);
|
||
|
expect(response.setHeader).toHaveBeenCalledWith(
|
||
|
'WWW-Authenticate',
|
||
|
'Basic realm="Enter credentials"',
|
||
|
);
|
||
|
});
|
||
|
|
||
|
it('should handle files', async () => {
|
||
|
const formFields: FormField[] = [
|
||
|
{
|
||
|
fieldLabel: 'Resume',
|
||
|
fieldType: 'file',
|
||
|
requiredField: true,
|
||
|
acceptFileTypes: '.pdf,.doc',
|
||
|
multipleFiles: false,
|
||
|
},
|
||
|
{
|
||
|
fieldLabel: 'Attachments',
|
||
|
fieldType: 'file',
|
||
|
requiredField: true,
|
||
|
acceptFileTypes: '.pdf,.doc',
|
||
|
multipleFiles: true,
|
||
|
},
|
||
|
];
|
||
|
|
||
|
const bodyData = {
|
||
|
files: {
|
||
|
'field-0': {
|
||
|
originalFilename: 'resume.pdf',
|
||
|
mimetype: 'application/json',
|
||
|
filepath: '/resume.pdf',
|
||
|
size: 200,
|
||
|
},
|
||
|
'field-1': [
|
||
|
{
|
||
|
originalFilename: 'attachment1.pdf',
|
||
|
mimetype: 'application/json',
|
||
|
filepath: '/attachment1.pdf',
|
||
|
size: 201,
|
||
|
},
|
||
|
],
|
||
|
},
|
||
|
};
|
||
|
|
||
|
const { responseData } = await testVersionedWebhookTriggerNode(FormTrigger, 2, {
|
||
|
mode: 'trigger',
|
||
|
node: {
|
||
|
parameters: {
|
||
|
formTitle: 'Test Form',
|
||
|
formDescription: 'Test Description',
|
||
|
responseMode: 'onReceived',
|
||
|
formFields: { values: formFields },
|
||
|
},
|
||
|
},
|
||
|
request: { method: 'POST' },
|
||
|
bodyData,
|
||
|
});
|
||
|
|
||
|
expect(responseData?.webhookResponse).toEqual({ status: 200 });
|
||
|
expect(responseData?.workflowData).toEqual([
|
||
|
[
|
||
|
expect.objectContaining({
|
||
|
json: {
|
||
|
Resume: {
|
||
|
filename: 'resume.pdf',
|
||
|
mimetype: 'application/json',
|
||
|
size: 200,
|
||
|
},
|
||
|
Attachments: [
|
||
|
{
|
||
|
filename: 'attachment1.pdf',
|
||
|
mimetype: 'application/json',
|
||
|
size: 201,
|
||
|
},
|
||
|
],
|
||
|
formMode: 'production',
|
||
|
submittedAt: expect.any(String),
|
||
|
},
|
||
|
}),
|
||
|
],
|
||
|
]);
|
||
|
});
|
||
|
});
|