import type { INodeExecutionData, MultiPartFormData, IDataObject, IWebhookFunctions, } from 'n8n-workflow'; import { NodeOperationError, jsonParse } from 'n8n-workflow'; import type { FormField, FormTriggerData, FormTriggerInput } from './interfaces'; import { FORM_TRIGGER_AUTHENTICATION_PROPERTY } from './interfaces'; import { WebhookAuthorizationError } from '../Webhook/error'; import { validateWebhookAuthentication } from '../Webhook/utils'; import { DateTime } from 'luxon'; import isbot from 'isbot'; export function prepareFormData({ formTitle, formDescription, formSubmittedText, redirectUrl, formFields, testRun, query, instanceId, useResponseData, appendAttribution = true, }: { formTitle: string; formDescription: string; formSubmittedText: string | undefined; redirectUrl: string | undefined; formFields: FormField[]; testRun: boolean; query: IDataObject; instanceId?: string; useResponseData?: boolean; appendAttribution?: boolean; }) { const validForm = formFields.length > 0; const utm_campaign = instanceId ? `&utm_campaign=${instanceId}` : ''; const n8nWebsiteLink = `https://n8n.io/?utm_source=n8n-internal&utm_medium=form-trigger${utm_campaign}`; if (formSubmittedText === undefined) { formSubmittedText = 'Your response has been recorded'; } const formData: FormTriggerData = { testRun, validForm, formTitle, formDescription, formSubmittedText, n8nWebsiteLink, formFields: [], useResponseData, appendAttribution, }; if (redirectUrl) { if (!redirectUrl.includes('://')) { redirectUrl = `http://${redirectUrl}`; } formData.redirectUrl = redirectUrl; } if (!validForm) { return formData; } for (const [index, field] of formFields.entries()) { const { fieldType, requiredField, multiselect, placeholder } = field; const input: IDataObject = { id: `field-${index}`, errorId: `error-field-${index}`, label: field.fieldLabel, inputRequired: requiredField ? 'form-required' : '', defaultValue: query[field.fieldLabel] ?? '', placeholder, }; if (multiselect) { input.isMultiSelect = true; input.multiSelectOptions = field.fieldOptions?.values.map((e, i) => ({ id: `option${i}`, label: e.option, })) ?? []; } else if (fieldType === 'file') { input.isFileInput = true; input.acceptFileTypes = field.acceptFileTypes; input.multipleFiles = field.multipleFiles ? 'multiple' : ''; } else if (fieldType === 'dropdown') { input.isSelect = true; const fieldOptions = field.fieldOptions?.values ?? []; input.selectOptions = fieldOptions.map((e) => e.option); } else if (fieldType === 'textarea') { input.isTextarea = true; } else { input.isInput = true; input.type = fieldType as 'text' | 'number' | 'date' | 'email'; } formData.formFields.push(input as FormTriggerInput); } return formData; } const checkResponseModeConfiguration = (context: IWebhookFunctions) => { const responseMode = context.getNodeParameter('responseMode', 'onReceived') as string; const connectedNodes = context.getChildNodes(context.getNode().name); const isRespondToWebhookConnected = connectedNodes.some( (node) => node.type === 'n8n-nodes-base.respondToWebhook', ); if (!isRespondToWebhookConnected && responseMode === 'responseNode') { throw new NodeOperationError( context.getNode(), new Error('No Respond to Webhook node found in the workflow'), { description: 'Insert a Respond to Webhook node to your workflow to respond to the form submission or choose another option for the “Respond When” parameter', }, ); } if (isRespondToWebhookConnected && responseMode !== 'responseNode') { throw new NodeOperationError( context.getNode(), new Error(`${context.getNode().name} node not correctly configured`), { description: 'Set the “Respond When” parameter to “Using Respond to Webhook Node” or remove the Respond to Webhook node', }, ); } }; export async function formWebhook( context: IWebhookFunctions, authProperty = FORM_TRIGGER_AUTHENTICATION_PROPERTY, ) { const node = context.getNode(); const options = context.getNodeParameter('options', {}) as { ignoreBots?: boolean; respondWithOptions?: { values: { respondWith: 'text' | 'redirect'; formSubmittedText: string; redirectUrl: string; }; }; formSubmittedText?: string; useWorkflowTimezone?: boolean; appendAttribution?: boolean; }; const res = context.getResponseObject(); const req = context.getRequestObject(); try { if (options.ignoreBots && isbot(req.headers['user-agent'])) { throw new WebhookAuthorizationError(403); } await validateWebhookAuthentication(context, authProperty); } catch (error) { if (error instanceof WebhookAuthorizationError) { res.writeHead(error.responseCode, { 'WWW-Authenticate': 'Basic realm="Webhook"' }); res.end(error.message); return { noWebhookResponse: true }; } throw error; } const mode = context.getMode() === 'manual' ? 'test' : 'production'; const formFields = context.getNodeParameter('formFields.values', []) as FormField[]; const method = context.getRequestObject().method; checkResponseModeConfiguration(context); //Show the form on GET request if (method === 'GET') { const formTitle = context.getNodeParameter('formTitle', '') as string; const formDescription = (context.getNodeParameter('formDescription', '') as string) .replace(/\\n/g, '\n') .replace(/
/g, '\n'); const instanceId = context.getInstanceId(); const responseMode = context.getNodeParameter('responseMode', '') as string; let formSubmittedText; let redirectUrl; let appendAttribution = true; if (options.respondWithOptions) { const values = (options.respondWithOptions as IDataObject).values as IDataObject; if (values.respondWith === 'text') { formSubmittedText = values.formSubmittedText as string; } if (values.respondWith === 'redirect') { redirectUrl = values.redirectUrl as string; } } else { formSubmittedText = options.formSubmittedText as string; } if (options.appendAttribution === false) { appendAttribution = false; } const useResponseData = responseMode === 'responseNode'; const query = context.getRequestObject().query as IDataObject; const data = prepareFormData({ formTitle, formDescription, formSubmittedText, redirectUrl, formFields, testRun: mode === 'test', query, instanceId, useResponseData, appendAttribution, }); res.render('form-trigger', data); return { noWebhookResponse: true, }; } const bodyData = (context.getBodyData().data as IDataObject) ?? {}; const files = (context.getBodyData().files as IDataObject) ?? {}; const returnItem: INodeExecutionData = { json: {}, }; if (files && Object.keys(files).length) { returnItem.binary = {}; } for (const key of Object.keys(files)) { const processFiles: MultiPartFormData.File[] = []; let multiFile = false; const filesInput = files[key] as MultiPartFormData.File[] | MultiPartFormData.File; if (Array.isArray(filesInput)) { bodyData[key] = filesInput.map((file) => ({ filename: file.originalFilename, mimetype: file.mimetype, size: file.size, })); processFiles.push(...filesInput); multiFile = true; } else { bodyData[key] = { filename: filesInput.originalFilename, mimetype: filesInput.mimetype, size: filesInput.size, }; processFiles.push(filesInput); } const entryIndex = Number(key.replace(/field-/g, '')); const fieldLabel = isNaN(entryIndex) ? key : formFields[entryIndex].fieldLabel; let fileCount = 0; for (const file of processFiles) { let binaryPropertyName = fieldLabel.replace(/\W/g, '_'); if (multiFile) { binaryPropertyName += `_${fileCount++}`; } returnItem.binary![binaryPropertyName] = await context.nodeHelpers.copyBinaryFile( file.filepath, file.originalFilename ?? file.newFilename, file.mimetype, ); } } for (const [index, field] of formFields.entries()) { const key = `field-${index}`; let value = bodyData[key] ?? null; if (value === null) { returnItem.json[field.fieldLabel] = null; continue; } if (field.fieldType === 'number') { value = Number(value); } if (field.fieldType === 'text') { value = String(value).trim(); } if (field.multiselect && typeof value === 'string') { value = jsonParse(value); } if (field.fieldType === 'date' && value && field.formatDate !== '') { value = DateTime.fromFormat(String(value), 'yyyy-mm-dd').toFormat(field.formatDate as string); } if (field.fieldType === 'file' && field.multipleFiles && !Array.isArray(value)) { value = [value]; } returnItem.json[field.fieldLabel] = value; } let { useWorkflowTimezone } = options; if (useWorkflowTimezone === undefined && node.typeVersion > 2) { useWorkflowTimezone = true; } const timezone = useWorkflowTimezone ? context.getTimezone() : 'UTC'; returnItem.json.submittedAt = DateTime.now().setZone(timezone).toISO(); returnItem.json.formMode = mode; const webhookResponse: IDataObject = { status: 200 }; return { webhookResponse, workflowData: [[returnItem]], }; }