mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-15 14:58:53 -08:00
3a474552b2
Github issue / Community forum post (link here to close automatically): https://github.com/n8n-io/n8n/pull/6348 --------- Co-authored-by: Giulio Andreini <g.andreini@gmail.com> Co-authored-by: Marcus <marcus@n8n.io>
248 lines
5 KiB
TypeScript
248 lines
5 KiB
TypeScript
import type { IDataObject, IExecuteFunctions, IGetNodeParameterOptions, INode } from 'n8n-workflow';
|
|
import { constructExecutionMetaData } from 'n8n-core';
|
|
import get from 'lodash/get';
|
|
import { composeReturnItem, parseJsonParameter, validateEntry } from '../../v2/helpers/utils';
|
|
import type { SetNodeOptions } from '../../v2/helpers/interfaces';
|
|
|
|
export const node: INode = {
|
|
id: '11',
|
|
name: 'Edit Fields',
|
|
type: 'n8n-nodes-base.set',
|
|
typeVersion: 3,
|
|
position: [42, 42],
|
|
parameters: {
|
|
mode: 'manual',
|
|
fields: {
|
|
values: [],
|
|
},
|
|
include: 'none',
|
|
options: {},
|
|
},
|
|
};
|
|
|
|
export const createMockExecuteFunction = (nodeParameters: IDataObject) => {
|
|
const fakeExecuteFunction = {
|
|
getNodeParameter(
|
|
parameterName: string,
|
|
_itemIndex: number,
|
|
fallbackValue?: IDataObject | undefined,
|
|
options?: IGetNodeParameterOptions | undefined,
|
|
) {
|
|
const parameter = options?.extractValue ? `${parameterName}.value` : parameterName;
|
|
return get(nodeParameters, parameter, fallbackValue);
|
|
},
|
|
getNode() {
|
|
return node;
|
|
},
|
|
helpers: { constructExecutionMetaData },
|
|
continueOnFail: () => false,
|
|
} as unknown as IExecuteFunctions;
|
|
return fakeExecuteFunction;
|
|
};
|
|
|
|
describe('test Set2, composeReturnItem', () => {
|
|
it('should compose return item including no other fields', () => {
|
|
const fakeExecuteFunction = createMockExecuteFunction({});
|
|
|
|
const inputItem = {
|
|
json: {
|
|
input1: 'value1',
|
|
input2: 2,
|
|
input3: [1, 2, 3],
|
|
},
|
|
pairedItem: {
|
|
item: 0,
|
|
input: undefined,
|
|
},
|
|
};
|
|
|
|
const newData = {
|
|
num1: 55,
|
|
str1: '42',
|
|
arr1: ['foo', 'bar'],
|
|
obj: {
|
|
key: 'value',
|
|
},
|
|
};
|
|
|
|
const options: SetNodeOptions = {
|
|
include: 'none',
|
|
};
|
|
|
|
const result = composeReturnItem.call(fakeExecuteFunction, 0, inputItem, newData, options);
|
|
|
|
expect(result).toEqual({
|
|
json: {
|
|
num1: 55,
|
|
str1: '42',
|
|
arr1: ['foo', 'bar'],
|
|
obj: {
|
|
key: 'value',
|
|
},
|
|
},
|
|
pairedItem: {
|
|
item: 0,
|
|
},
|
|
});
|
|
});
|
|
|
|
it('should compose return item including selected fields', () => {
|
|
const fakeExecuteFunction = createMockExecuteFunction({ includeFields: 'input1, input2' });
|
|
|
|
const inputItem = {
|
|
json: {
|
|
input1: 'value1',
|
|
input2: 2,
|
|
input3: [1, 2, 3],
|
|
},
|
|
pairedItem: {
|
|
item: 0,
|
|
input: undefined,
|
|
},
|
|
};
|
|
|
|
const newData = {
|
|
num1: 55,
|
|
str1: '42',
|
|
arr1: ['foo', 'bar'],
|
|
obj: {
|
|
key: 'value',
|
|
},
|
|
};
|
|
|
|
const options: SetNodeOptions = {
|
|
include: 'selected',
|
|
};
|
|
|
|
const result = composeReturnItem.call(fakeExecuteFunction, 0, inputItem, newData, options);
|
|
|
|
expect(result).toEqual({
|
|
json: {
|
|
num1: 55,
|
|
str1: '42',
|
|
arr1: ['foo', 'bar'],
|
|
input1: 'value1',
|
|
input2: 2,
|
|
obj: {
|
|
key: 'value',
|
|
},
|
|
},
|
|
pairedItem: {
|
|
item: 0,
|
|
},
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('test Set2, parseJsonParameter', () => {
|
|
it('should parse valid JSON string', () => {
|
|
const result = parseJsonParameter('{"foo": "bar"}', node, 0, 'test');
|
|
|
|
expect(result).toEqual({
|
|
foo: 'bar',
|
|
});
|
|
});
|
|
|
|
it('should tolerate single quotes in string', () => {
|
|
const result = parseJsonParameter("{'foo': 'bar'}", node, 0, 'test');
|
|
|
|
expect(result).toEqual({
|
|
foo: 'bar',
|
|
});
|
|
});
|
|
|
|
it('should tolerate unquoted keys', () => {
|
|
const result = parseJsonParameter("{foo: 'bar'}", node, 0, 'test');
|
|
|
|
expect(result).toEqual({
|
|
foo: 'bar',
|
|
});
|
|
});
|
|
|
|
it('should tolerate trailing comma', () => {
|
|
const result = parseJsonParameter('{"foo": "bar"},', node, 0, 'test');
|
|
|
|
expect(result).toEqual({
|
|
foo: 'bar',
|
|
});
|
|
});
|
|
|
|
it('should tolerate trailing commas in objects', () => {
|
|
const result = parseJsonParameter("{foo: 'bar', baz: {'foo': 'bar',}, }", node, 0, 'test');
|
|
|
|
expect(result).toEqual({
|
|
foo: 'bar',
|
|
baz: {
|
|
foo: 'bar',
|
|
},
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('test Set2, validateEntry', () => {
|
|
it('should convert number to string', () => {
|
|
const result = validateEntry(
|
|
{ name: 'foo', type: 'stringValue', stringValue: 42 as unknown as string },
|
|
node,
|
|
0,
|
|
);
|
|
|
|
expect(result).toEqual({
|
|
name: 'foo',
|
|
value: '42',
|
|
});
|
|
});
|
|
|
|
it('should convert array to string', () => {
|
|
const result = validateEntry(
|
|
{ name: 'foo', type: 'stringValue', stringValue: [1, 2, 3] as unknown as string },
|
|
node,
|
|
0,
|
|
);
|
|
|
|
expect(result).toEqual({
|
|
name: 'foo',
|
|
value: '[1,2,3]',
|
|
});
|
|
});
|
|
|
|
it('should convert object to string', () => {
|
|
const result = validateEntry(
|
|
{ name: 'foo', type: 'stringValue', stringValue: { foo: 'bar' } as unknown as string },
|
|
node,
|
|
0,
|
|
);
|
|
|
|
expect(result).toEqual({
|
|
name: 'foo',
|
|
value: '{"foo":"bar"}',
|
|
});
|
|
});
|
|
|
|
it('should convert boolean to string', () => {
|
|
const result = validateEntry(
|
|
{ name: 'foo', type: 'stringValue', stringValue: true as unknown as string },
|
|
node,
|
|
0,
|
|
);
|
|
|
|
expect(result).toEqual({
|
|
name: 'foo',
|
|
value: 'true',
|
|
});
|
|
});
|
|
|
|
it('should convert undefined to string', () => {
|
|
const result = validateEntry(
|
|
{ name: 'foo', type: 'stringValue', stringValue: undefined as unknown as string },
|
|
node,
|
|
0,
|
|
);
|
|
|
|
expect(result).toEqual({
|
|
name: 'foo',
|
|
value: 'undefined',
|
|
});
|
|
});
|
|
});
|