2019-10-09 12:52:45 -07:00
import { get } from 'lodash' ;
2019-06-23 03:35:23 -07:00
import { IExecuteFunctions } from 'n8n-core' ;
import {
GenericValue ,
INodeExecutionData ,
INodeType ,
INodeTypeDescription ,
2022-06-03 08:25:07 -07:00
IPairedItemData ,
2019-06-23 03:35:23 -07:00
} from 'n8n-workflow' ;
export class Merge implements INodeType {
description : INodeTypeDescription = {
displayName : 'Merge' ,
name : 'merge' ,
2019-07-26 02:27:46 -07:00
icon : 'fa:code-branch' ,
2019-06-23 03:35:23 -07:00
group : [ 'transform' ] ,
version : 1 ,
2019-08-01 09:22:48 -07:00
subtitle : '={{$parameter["mode"]}}' ,
2021-07-03 05:40:16 -07:00
description : 'Merges data of multiple streams once data from both is available' ,
2019-06-23 03:35:23 -07:00
defaults : {
name : 'Merge' ,
2019-07-26 02:41:08 -07:00
color : '#00bbcc' ,
2019-06-23 03:35:23 -07:00
} ,
inputs : [ 'main' , 'main' ] ,
outputs : [ 'main' ] ,
2019-08-02 06:56:05 -07:00
inputNames : [ 'Input 1' , 'Input 2' ] ,
2019-06-23 03:35:23 -07:00
properties : [
{
displayName : 'Mode' ,
name : 'mode' ,
type : 'options' ,
options : [
{
name : 'Append' ,
2019-07-11 05:54:18 -07:00
value : 'append' ,
2022-08-17 08:50:24 -07:00
description :
'Combines data of both inputs. The output will contain items of input 1 and input 2.' ,
2019-06-23 03:35:23 -07:00
} ,
2019-11-03 04:26:18 -08:00
{
name : 'Keep Key Matches' ,
value : 'keepKeyMatches' ,
2022-05-06 14:01:25 -07:00
description : 'Keeps data of input 1 if it does find a match with data of input 2' ,
2019-11-03 04:26:18 -08:00
} ,
2019-06-23 03:35:23 -07:00
{
2019-08-02 01:26:05 -07:00
name : 'Merge By Index' ,
value : 'mergeByIndex' ,
2022-08-17 08:50:24 -07:00
description :
'Merges data of both inputs. The output will contain items of input 1 merged with data of input 2. Merge happens depending on the index of the items. So first item of input 1 will be merged with first item of input 2 and so on.' ,
2019-08-02 01:26:05 -07:00
} ,
{
name : 'Merge By Key' ,
value : 'mergeByKey' ,
2022-08-17 08:50:24 -07:00
description :
'Merges data of both inputs. The output will contain items of input 1 merged with data of input 2. Merge happens depending on a defined key.' ,
2019-07-11 05:54:18 -07:00
} ,
2020-01-15 04:15:26 -08:00
{
name : 'Multiplex' ,
2020-01-15 05:53:27 -08:00
value : 'multiplex' ,
2022-08-17 08:50:24 -07:00
description :
'Merges each value of one input with each value of the other input. The output will contain (m * n) items where (m) and (n) are lengths of the inputs.' ,
2020-01-15 04:15:26 -08:00
} ,
2019-07-11 05:54:18 -07:00
{
2022-06-03 10:23:49 -07:00
name : 'Pass-Through' ,
2019-07-11 05:54:18 -07:00
value : 'passThrough' ,
2022-08-17 08:50:24 -07:00
description :
'Passes through data of one input. The output will contain only items of the defined input.' ,
2019-07-11 05:54:18 -07:00
} ,
2019-11-03 04:26:18 -08:00
{
name : 'Remove Key Matches' ,
value : 'removeKeyMatches' ,
2022-05-06 14:01:25 -07:00
description : 'Keeps data of input 1 if it does NOT find match with data of input 2' ,
2019-11-03 04:26:18 -08:00
} ,
2019-07-11 05:54:18 -07:00
{
name : 'Wait' ,
value : 'wait' ,
2022-08-17 08:50:24 -07:00
description :
'Waits till data of both inputs is available and will then output a single empty item. Source Nodes must connect to both Input 1 and 2. This node only supports 2 Sources, if you need more Sources, connect multiple Merge nodes in series. This node will not output any data.' ,
2019-06-23 03:35:23 -07:00
} ,
] ,
default : 'append' ,
2022-05-06 14:01:25 -07:00
description : 'How data of branches should be merged' ,
2019-08-02 01:26:05 -07:00
} ,
{
displayName : 'Join' ,
name : 'join' ,
type : 'options' ,
displayOptions : {
show : {
2022-08-17 08:50:24 -07:00
mode : [ 'mergeByIndex' ] ,
2019-08-02 01:26:05 -07:00
} ,
} ,
options : [
{
name : 'Inner Join' ,
value : 'inner' ,
2022-08-17 08:50:24 -07:00
description :
'Merges as many items as both inputs contain. (Example: Input1 = 5 items, Input2 = 3 items | Output will contain 3 items).' ,
2019-08-02 01:26:05 -07:00
} ,
{
name : 'Left Join' ,
value : 'left' ,
2022-08-17 08:50:24 -07:00
description :
'Merges as many items as first input contains. (Example: Input1 = 3 items, Input2 = 5 items | Output will contain 3 items).' ,
2019-08-02 01:26:05 -07:00
} ,
{
name : 'Outer Join' ,
value : 'outer' ,
2022-08-17 08:50:24 -07:00
description :
'Merges as many items as input contains with most items. (Example: Input1 = 3 items, Input2 = 5 items | Output will contain 5 items).' ,
2019-08-02 01:26:05 -07:00
} ,
] ,
default : 'left' ,
2022-08-17 08:50:24 -07:00
description :
'How many items the output will contain if inputs contain different amount of items' ,
2019-06-23 03:35:23 -07:00
} ,
{
displayName : 'Property Input 1' ,
name : 'propertyName1' ,
type : 'string' ,
default : '' ,
2022-01-27 22:55:25 -08:00
hint : 'The name of the field as text (e.g. “id”)' ,
2019-07-11 05:54:18 -07:00
required : true ,
2019-06-23 03:35:23 -07:00
displayOptions : {
show : {
2022-08-17 08:50:24 -07:00
mode : [ 'keepKeyMatches' , 'mergeByKey' , 'removeKeyMatches' ] ,
2019-06-23 03:35:23 -07:00
} ,
} ,
2022-05-06 14:01:25 -07:00
description : 'Name of property which decides which items to merge of input 1' ,
2019-06-23 03:35:23 -07:00
} ,
{
displayName : 'Property Input 2' ,
name : 'propertyName2' ,
type : 'string' ,
default : '' ,
2022-01-27 22:55:25 -08:00
hint : 'The name of the field as text (e.g. “id”)' ,
2019-07-11 05:54:18 -07:00
required : true ,
2019-06-23 03:35:23 -07:00
displayOptions : {
show : {
2022-08-17 08:50:24 -07:00
mode : [ 'keepKeyMatches' , 'mergeByKey' , 'removeKeyMatches' ] ,
2019-06-23 03:35:23 -07:00
} ,
} ,
2022-05-06 14:01:25 -07:00
description : 'Name of property which decides which items to merge of input 2' ,
2019-06-23 03:35:23 -07:00
} ,
2019-07-11 05:54:18 -07:00
{
displayName : 'Output Data' ,
name : 'output' ,
type : 'options' ,
displayOptions : {
show : {
2022-08-17 08:50:24 -07:00
mode : [ 'passThrough' ] ,
2019-07-11 05:54:18 -07:00
} ,
} ,
options : [
{
name : 'Input 1' ,
value : 'input1' ,
} ,
{
name : 'Input 2' ,
value : 'input2' ,
} ,
] ,
default : 'input1' ,
2022-05-06 14:01:25 -07:00
description : 'Defines of which input the data should be used as output of node' ,
2019-07-11 05:54:18 -07:00
} ,
2020-01-18 20:01:56 -08:00
{
displayName : 'Overwrite' ,
name : 'overwrite' ,
type : 'options' ,
displayOptions : {
show : {
2022-08-17 08:50:24 -07:00
mode : [ 'mergeByKey' ] ,
2020-01-18 20:01:56 -08:00
} ,
} ,
options : [
{
name : 'Always' ,
value : 'always' ,
2022-05-06 14:01:25 -07:00
description : 'Always overwrites everything' ,
2020-01-18 20:01:56 -08:00
} ,
{
2020-01-19 12:01:10 -08:00
name : 'If Blank' ,
value : 'blank' ,
2022-05-06 14:01:25 -07:00
description : 'Overwrites only values of "null", "undefined" or empty string' ,
2020-01-18 20:01:56 -08:00
} ,
{
2020-01-19 12:01:10 -08:00
name : 'If Missing' ,
value : 'undefined' ,
2022-05-06 14:01:25 -07:00
description : 'Only adds values which do not exist yet' ,
2020-01-18 20:01:56 -08:00
} ,
] ,
default : 'always' ,
2022-05-06 14:01:25 -07:00
description : 'Select when to overwrite the values from Input1 with values from Input 2' ,
2020-10-22 06:46:03 -07:00
} ,
] ,
2019-06-23 03:35:23 -07:00
} ;
async execute ( this : IExecuteFunctions ) : Promise < INodeExecutionData [ ] [ ] > {
const returnData : INodeExecutionData [ ] = [ ] ;
const mode = this . getNodeParameter ( 'mode' , 0 ) as string ;
if ( mode === 'append' ) {
// Simply appends the data
for ( let i = 0 ; i < 2 ; i ++ ) {
returnData . push . apply ( returnData , this . getInputData ( i ) ) ;
}
2019-08-02 01:26:05 -07:00
} else if ( mode === 'mergeByIndex' ) {
// Merges data by index
const join = this . getNodeParameter ( 'join' , 0 ) as string ;
const dataInput1 = this . getInputData ( 0 ) ;
const dataInput2 = this . getInputData ( 1 ) ;
if ( dataInput1 === undefined || dataInput1 . length === 0 ) {
if ( [ 'inner' , 'left' ] . includes ( join ) ) {
// When "inner" or "left" join return empty if first
// input does not contain any items
return [ returnData ] ;
}
// For "outer" return data of second input
return [ dataInput2 ] ;
}
if ( dataInput2 === undefined || dataInput2 . length === 0 ) {
if ( [ 'left' , 'outer' ] . includes ( join ) ) {
// When "left" or "outer" join return data of first input
return [ dataInput1 ] ;
}
// For "inner" return empty
return [ returnData ] ;
}
// Default "left"
let numEntries = dataInput1 . length ;
if ( join === 'inner' ) {
numEntries = Math . min ( dataInput1 . length , dataInput2 . length ) ;
} else if ( join === 'outer' ) {
numEntries = Math . max ( dataInput1 . length , dataInput2 . length ) ;
}
let newItem : INodeExecutionData ;
for ( let i = 0 ; i < numEntries ; i ++ ) {
if ( i >= dataInput1 . length ) {
returnData . push ( dataInput2 [ i ] ) ;
continue ;
}
if ( i >= dataInput2 . length ) {
returnData . push ( dataInput1 [ i ] ) ;
continue ;
}
newItem = {
json : { } ,
2022-06-03 08:25:07 -07:00
pairedItem : [
dataInput1 [ i ] . pairedItem as IPairedItemData ,
dataInput2 [ i ] . pairedItem as IPairedItemData ,
] ,
2019-08-02 01:26:05 -07:00
} ;
if ( dataInput1 [ i ] . binary !== undefined ) {
newItem . binary = { } ;
// Create a shallow copy of the binary data so that the old
// data references which do not get changed still stay behind
// but the incoming data does not get changed.
Object . assign ( newItem . binary , dataInput1 [ i ] . binary ) ;
}
// Create also a shallow copy of the json data
Object . assign ( newItem . json , dataInput1 [ i ] . json ) ;
// Copy json data
for ( const key of Object . keys ( dataInput2 [ i ] . json ) ) {
newItem . json [ key ] = dataInput2 [ i ] . json [ key ] ;
}
// Copy binary data
if ( dataInput2 [ i ] . binary !== undefined ) {
if ( newItem . binary === undefined ) {
newItem . binary = { } ;
}
for ( const key of Object . keys ( dataInput2 [ i ] . binary ! ) ) {
2022-01-09 01:39:48 -08:00
newItem . binary [ key ] = dataInput2 [ i ] . binary ! [ key ] ? ? newItem . binary [ key ] ;
2019-08-02 01:26:05 -07:00
}
}
returnData . push ( newItem ) ;
}
2020-01-15 05:53:27 -08:00
} else if ( mode === 'multiplex' ) {
2020-01-15 04:15:26 -08:00
const dataInput1 = this . getInputData ( 0 ) ;
const dataInput2 = this . getInputData ( 1 ) ;
if ( ! dataInput1 || ! dataInput2 ) {
return [ returnData ] ;
}
let entry1 : INodeExecutionData ;
let entry2 : INodeExecutionData ;
for ( entry1 of dataInput1 ) {
for ( entry2 of dataInput2 ) {
2022-06-03 08:25:07 -07:00
returnData . push ( {
json : {
2022-08-17 08:50:24 -07:00
. . . entry1 . json ,
. . . entry2 . json ,
2022-06-03 08:25:07 -07:00
} ,
pairedItem : [
entry1 . pairedItem as IPairedItemData ,
entry2 . pairedItem as IPairedItemData ,
] ,
} ) ;
2020-01-15 04:15:26 -08:00
}
}
return [ returnData ] ;
2019-11-03 04:26:18 -08:00
} else if ( [ 'keepKeyMatches' , 'mergeByKey' , 'removeKeyMatches' ] . includes ( mode ) ) {
2019-06-23 03:35:23 -07:00
const dataInput1 = this . getInputData ( 0 ) ;
if ( ! dataInput1 ) {
// If it has no input data from first input return nothing
return [ returnData ] ;
}
const propertyName1 = this . getNodeParameter ( 'propertyName1' , 0 ) as string ;
const propertyName2 = this . getNodeParameter ( 'propertyName2' , 0 ) as string ;
2020-01-19 12:24:44 -08:00
const overwrite = this . getNodeParameter ( 'overwrite' , 0 , 'always' ) as string ;
2019-06-23 03:35:23 -07:00
const dataInput2 = this . getInputData ( 1 ) ;
if ( ! dataInput2 || ! propertyName1 || ! propertyName2 ) {
2019-11-03 04:26:18 -08:00
// Second input does not have any data or the property names are not defined
if ( mode === 'keepKeyMatches' ) {
// For "keepKeyMatches" return nothing
return [ returnData ] ;
}
// For "mergeByKey" and "removeKeyMatches" return the data from the first input
2019-06-23 03:35:23 -07:00
return [ dataInput1 ] ;
}
// Get the data to copy
const copyData : {
[ key : string ] : INodeExecutionData ;
} = { } ;
let entry : INodeExecutionData ;
for ( entry of dataInput2 ) {
2019-10-09 12:52:45 -07:00
const key = get ( entry . json , propertyName2 ) ;
if ( ! entry . json || ! key ) {
2019-06-23 03:35:23 -07:00
// Entry does not have the property so skip it
continue ;
}
2019-10-09 12:52:45 -07:00
copyData [ key as string ] = entry ;
2019-06-23 03:35:23 -07:00
}
2019-11-03 04:26:18 -08:00
// Copy data on entries or add matching entries
2019-06-23 03:35:23 -07:00
let referenceValue : GenericValue ;
let key : string ;
for ( entry of dataInput1 ) {
2019-10-09 12:52:45 -07:00
referenceValue = get ( entry . json , propertyName1 ) ;
2019-06-23 03:35:23 -07:00
2019-11-03 04:26:18 -08:00
if ( referenceValue === undefined ) {
// Entry does not have the property
if ( mode === 'removeKeyMatches' ) {
// For "removeKeyMatches" add item
returnData . push ( entry ) ;
}
// For "mergeByKey" and "keepKeyMatches" skip item
2019-06-23 03:35:23 -07:00
continue ;
}
if ( ! [ 'string' , 'number' ] . includes ( typeof referenceValue ) ) {
2019-11-03 04:26:18 -08:00
if ( referenceValue !== null && referenceValue . constructor . name !== 'Data' ) {
// Reference value is not of comparable type
if ( mode === 'removeKeyMatches' ) {
// For "removeKeyMatches" add item
returnData . push ( entry ) ;
}
// For "mergeByKey" and "keepKeyMatches" skip item
continue ;
}
2019-06-23 03:35:23 -07:00
}
if ( typeof referenceValue === 'number' ) {
referenceValue = referenceValue . toString ( ) ;
2019-11-03 04:26:18 -08:00
} else if ( referenceValue !== null && referenceValue . constructor . name === 'Date' ) {
referenceValue = ( referenceValue as Date ) . toISOString ( ) ;
2019-06-23 03:35:23 -07:00
}
if ( copyData . hasOwnProperty ( referenceValue as string ) ) {
2019-11-03 04:26:18 -08:00
// Item with reference value got found
2019-06-23 03:35:23 -07:00
if ( [ 'null' , 'undefined' ] . includes ( typeof referenceValue ) ) {
2019-11-03 04:26:18 -08:00
// The reference value is null or undefined
if ( mode === 'removeKeyMatches' ) {
// For "removeKeyMatches" add item
returnData . push ( entry ) ;
}
// For "mergeByKey" and "keepKeyMatches" skip item
2019-06-23 03:35:23 -07:00
continue ;
}
2019-08-01 13:55:33 -07:00
2019-11-03 04:26:18 -08:00
// Match exists
if ( mode === 'removeKeyMatches' ) {
// For "removeKeyMatches" we can skip the item as it has a match
continue ;
} else if ( mode === 'mergeByKey' ) {
// Copy the entry as the data gets changed
entry = JSON . parse ( JSON . stringify ( entry ) ) ;
for ( key of Object . keys ( copyData [ referenceValue as string ] . json ) ) {
2020-01-19 12:01:10 -08:00
if ( key === propertyName2 ) {
continue ;
}
2020-01-18 20:01:56 -08:00
2020-01-19 12:01:10 -08:00
// TODO: Currently only copies json data and no binary one
const value = copyData [ referenceValue as string ] . json [ key ] ;
if (
overwrite === 'always' ||
( overwrite === 'undefined' && ! entry . json . hasOwnProperty ( key ) ) ||
( overwrite === 'blank' && [ null , undefined , '' ] . includes ( entry . json [ key ] as string ) )
) {
entry . json [ key ] = value ;
2020-01-18 20:01:56 -08:00
}
2019-11-03 04:26:18 -08:00
}
} else {
// For "keepKeyMatches" we add it as it is
returnData . push ( entry ) ;
continue ;
}
} else {
// No item for reference value got found
if ( mode === 'removeKeyMatches' ) {
// For "removeKeyMatches" we can add it if not match got found
returnData . push ( entry ) ;
continue ;
2019-06-23 03:35:23 -07:00
}
}
2019-11-03 04:26:18 -08:00
if ( mode === 'mergeByKey' ) {
// For "mergeByKey" we always add the entry anyway but then the unchanged one
returnData . push ( entry ) ;
}
2019-06-23 03:35:23 -07:00
}
2019-08-01 13:55:33 -07:00
return [ returnData ] ;
2019-07-11 05:54:18 -07:00
} else if ( mode === 'passThrough' ) {
const output = this . getNodeParameter ( 'output' , 0 ) as string ;
if ( output === 'input1' ) {
returnData . push . apply ( returnData , this . getInputData ( 0 ) ) ;
2022-08-17 08:50:24 -07:00
} else {
2019-07-11 05:54:18 -07:00
returnData . push . apply ( returnData , this . getInputData ( 1 ) ) ;
}
} else if ( mode === 'wait' ) {
returnData . push ( { json : { } } ) ;
2019-06-23 03:35:23 -07:00
}
return [ returnData ] ;
}
}