mirror of
https://github.com/n8n-io/n8n.git
synced 2024-12-25 04:34:06 -08:00
7b8d388d17
* ✨ Add Webhook-Response-Node * ⚡ Replace callback function with promise * ✨ Add support for Bull and binary-data * ✨ Add string response option * ⚡ Remove some comments * ✨ Make more generically possible & fix issue multi call in queue mode * ⚡ Fix startup and eslint issues * ⚡ Improvements to webhook response node and functionality * ⚡ Replace data with more generic type * ⚡ Make statusMessage optional * ⚡ Change parameter order * ⚡ Move Response Code underneath options * ⚡ Hide Response Code on Webhook node if mode responseNode got selected * ⚡ Minor improvements * ⚡ Add missing file and fix lint issue * ⚡ Fix some node linting issues * ⚡ Apply feedback * ⚡ Minor improvements
279 lines
6.6 KiB
TypeScript
279 lines
6.6 KiB
TypeScript
import {
|
|
BINARY_ENCODING,
|
|
} from 'n8n-core';
|
|
|
|
import {
|
|
IDataObject,
|
|
IExecuteFunctions,
|
|
IN8nHttpFullResponse,
|
|
IN8nHttpResponse,
|
|
INodeExecutionData,
|
|
INodeType,
|
|
INodeTypeDescription,
|
|
NodeOperationError,
|
|
} from 'n8n-workflow';
|
|
|
|
export class RespondToWebhook implements INodeType {
|
|
description: INodeTypeDescription = {
|
|
displayName: 'Respond to Webhook',
|
|
icon: 'file:webhook.svg',
|
|
name: 'respondToWebhook',
|
|
group: ['transform'],
|
|
version: 1,
|
|
description: 'Returns data for Webhook',
|
|
defaults: {
|
|
name: 'Respond to Webhook',
|
|
color: '#885577',
|
|
},
|
|
inputs: ['main'],
|
|
outputs: ['main'],
|
|
credentials: [
|
|
],
|
|
properties: [
|
|
{
|
|
displayName: 'Respond With',
|
|
name: 'respondWith',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'First Incoming Item',
|
|
value: 'firstIncomingItem',
|
|
},
|
|
{
|
|
name: 'Text',
|
|
value: 'text',
|
|
},
|
|
{
|
|
name: 'JSON',
|
|
value: 'json',
|
|
},
|
|
{
|
|
name: 'Binary',
|
|
value: 'binary',
|
|
},
|
|
{
|
|
name: 'No Data',
|
|
value: 'noData',
|
|
},
|
|
],
|
|
default: 'firstIncomingItem',
|
|
description: 'The data that should be returned',
|
|
},
|
|
{
|
|
displayName: 'When using expressions, note that this node will only run for the first item in the input data.',
|
|
name: 'webhookNotice',
|
|
type: 'notice',
|
|
displayOptions: {
|
|
show: {
|
|
respondWith: [
|
|
'json',
|
|
'text',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
},
|
|
{
|
|
displayName: 'Response Body',
|
|
name: 'responseBody',
|
|
type: 'json',
|
|
displayOptions: {
|
|
show: {
|
|
respondWith: [
|
|
'json',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
placeholder: '{ "key": "value" }',
|
|
description: 'The HTTP Response JSON data',
|
|
},
|
|
{
|
|
displayName: 'Response Body',
|
|
name: 'responseBody',
|
|
type: 'string',
|
|
displayOptions: {
|
|
show: {
|
|
respondWith: [
|
|
'text',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
placeholder: 'e.g. Workflow started',
|
|
description: 'The HTTP Response text data',
|
|
},
|
|
{
|
|
displayName: 'Response Data Source',
|
|
name: 'responseDataSource',
|
|
type: 'options',
|
|
displayOptions: {
|
|
show: {
|
|
respondWith: [
|
|
'binary',
|
|
],
|
|
},
|
|
},
|
|
options: [
|
|
{
|
|
name: 'Choose Automatically From Input',
|
|
value: 'automatically',
|
|
description: 'Use if input data will contain a single piece of binary data',
|
|
},
|
|
{
|
|
name: 'Specify Myself',
|
|
value: 'set',
|
|
description: 'Enter the name of the input field the binary data will be in',
|
|
},
|
|
],
|
|
default: 'automatically',
|
|
},
|
|
{
|
|
displayName: 'Input Field Name',
|
|
name: 'inputFieldName',
|
|
type: 'string',
|
|
required: true,
|
|
default: 'data',
|
|
displayOptions: {
|
|
show: {
|
|
respondWith: [
|
|
'binary',
|
|
],
|
|
responseDataSource: [
|
|
'set',
|
|
],
|
|
},
|
|
},
|
|
description: 'The name of the node input field with the binary data',
|
|
},
|
|
|
|
{
|
|
displayName: 'Options',
|
|
name: 'options',
|
|
type: 'collection',
|
|
placeholder: 'Add Option',
|
|
default: {},
|
|
options: [
|
|
{
|
|
displayName: 'Response Code',
|
|
name: 'responseCode',
|
|
type: 'number',
|
|
typeOptions: {
|
|
minValue: 100,
|
|
maxValue: 599,
|
|
},
|
|
default: 200,
|
|
description: 'The HTTP Response code to return. Defaults to 200.',
|
|
},
|
|
{
|
|
displayName: 'Response Headers',
|
|
name: 'responseHeaders',
|
|
placeholder: 'Add Response Header',
|
|
description: 'Add headers to the webhook response',
|
|
type: 'fixedCollection',
|
|
typeOptions: {
|
|
multipleValues: true,
|
|
},
|
|
default: {},
|
|
options: [
|
|
{
|
|
name: 'entries',
|
|
displayName: 'Entries',
|
|
values: [
|
|
{
|
|
displayName: 'Name',
|
|
name: 'name',
|
|
type: 'string',
|
|
default: '',
|
|
description: 'Name of the header',
|
|
},
|
|
{
|
|
displayName: 'Value',
|
|
name: 'value',
|
|
type: 'string',
|
|
default: '',
|
|
description: 'Value of the header',
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
],
|
|
};
|
|
|
|
execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
|
const items = this.getInputData();
|
|
|
|
const respondWith = this.getNodeParameter('respondWith', 0) as string;
|
|
const options = this.getNodeParameter('options', 0, {}) as IDataObject;
|
|
|
|
const headers = {} as IDataObject;
|
|
if (options.responseHeaders) {
|
|
for (const header of (options.responseHeaders as IDataObject).entries as IDataObject[]) {
|
|
if (typeof header.name !== 'string') {
|
|
header.name = header.name?.toString();
|
|
}
|
|
headers[header.name?.toLowerCase() as string] = header.value?.toString();
|
|
}
|
|
}
|
|
|
|
let responseBody: IN8nHttpResponse;
|
|
if (respondWith === 'json') {
|
|
const responseBodyParameter = this.getNodeParameter('responseBody', 0) as string;
|
|
if (responseBodyParameter) {
|
|
responseBody = JSON.parse(responseBodyParameter);
|
|
}
|
|
} else if (respondWith === 'firstIncomingItem') {
|
|
responseBody = items[0].json;
|
|
} else if (respondWith === 'text') {
|
|
responseBody = this.getNodeParameter('responseBody', 0) as string;
|
|
} else if (respondWith === 'binary') {
|
|
const item = this.getInputData()[0];
|
|
|
|
if (item.binary === undefined) {
|
|
throw new NodeOperationError(this.getNode(), 'No binary data exists on the first item!');
|
|
}
|
|
|
|
let responseBinaryPropertyName: string;
|
|
|
|
const responseDataSource = this.getNodeParameter('responseDataSource', 0) as string;
|
|
|
|
if (responseDataSource === 'set') {
|
|
responseBinaryPropertyName = this.getNodeParameter('inputFieldName', 0) as string;
|
|
} else {
|
|
const binaryKeys = Object.keys(item.binary);
|
|
if (binaryKeys.length === 0) {
|
|
throw new NodeOperationError(this.getNode(), 'No binary data exists on the first item!');
|
|
}
|
|
responseBinaryPropertyName = binaryKeys[0];
|
|
}
|
|
|
|
const binaryData = item.binary[responseBinaryPropertyName];
|
|
|
|
if (binaryData === undefined) {
|
|
throw new NodeOperationError(this.getNode(), `No binary data property "${responseBinaryPropertyName}" does not exists on item!`);
|
|
}
|
|
|
|
if (headers['content-type']) {
|
|
headers['content-type'] = binaryData.mimeType;
|
|
}
|
|
responseBody = Buffer.from(binaryData.data, BINARY_ENCODING);
|
|
} else if (respondWith !== 'noData') {
|
|
throw new NodeOperationError(this.getNode(), `The Response Data option "${respondWith}" is not supported!`);
|
|
}
|
|
|
|
const response: IN8nHttpFullResponse = {
|
|
body: responseBody,
|
|
headers,
|
|
statusCode: options.responseCode as number || 200,
|
|
};
|
|
|
|
this.sendResponse(response);
|
|
|
|
return this.prepareOutputData(items);
|
|
}
|
|
|
|
}
|