mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-07 10:57:29 -08:00
203 lines
5.4 KiB
TypeScript
203 lines
5.4 KiB
TypeScript
import type {
|
|
IDataObject,
|
|
IExecuteFunctions,
|
|
INodeExecutionData,
|
|
INodeProperties,
|
|
} from 'n8n-workflow';
|
|
|
|
import type { QueryRunner, QueryValues, QueryWithValues } from '../../helpers/interfaces';
|
|
import { AUTO_MAP, DATA_MODE } from '../../helpers/interfaces';
|
|
|
|
import { escapeSqlIdentifier, replaceEmptyStringsByNulls } from '../../helpers/utils';
|
|
|
|
import { optionsCollection } from '../common.descriptions';
|
|
import { updateDisplayOptions } from '@utils/utilities';
|
|
|
|
const properties: INodeProperties[] = [
|
|
{
|
|
displayName: 'Data Mode',
|
|
name: 'dataMode',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'Auto-Map Input Data to Columns',
|
|
value: DATA_MODE.AUTO_MAP,
|
|
description: 'Use when node input properties names exactly match the table column names',
|
|
},
|
|
{
|
|
name: 'Map Each Column Below',
|
|
value: DATA_MODE.MANUAL,
|
|
description: 'Set the value for each destination column manually',
|
|
},
|
|
],
|
|
default: AUTO_MAP,
|
|
description:
|
|
'Whether to map node input properties and the table data automatically or manually',
|
|
},
|
|
{
|
|
displayName: `
|
|
In this mode, make sure incoming data fields are named the same as the columns in your table. If needed, use an 'Edit Fields' node before this node to change the field names.
|
|
`,
|
|
name: 'notice',
|
|
type: 'notice',
|
|
default: '',
|
|
displayOptions: {
|
|
show: {
|
|
dataMode: [DATA_MODE.AUTO_MAP],
|
|
},
|
|
},
|
|
},
|
|
{
|
|
// eslint-disable-next-line n8n-nodes-base/node-param-display-name-wrong-for-dynamic-options
|
|
displayName: 'Column to Match On',
|
|
name: 'columnToMatchOn',
|
|
type: 'options',
|
|
required: true,
|
|
// eslint-disable-next-line n8n-nodes-base/node-param-description-wrong-for-dynamic-options
|
|
description:
|
|
'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code/expressions/" target="_blank">expression</a>',
|
|
typeOptions: {
|
|
loadOptionsMethod: 'getColumns',
|
|
loadOptionsDependsOn: ['schema.value', 'table.value'],
|
|
},
|
|
default: '',
|
|
hint: "Used to find the correct row to update. Doesn't get changed.",
|
|
},
|
|
{
|
|
displayName: 'Value of Column to Match On',
|
|
name: 'valueToMatchOn',
|
|
type: 'string',
|
|
default: '',
|
|
description:
|
|
'Rows with a value in the specified "Column to Match On" that corresponds to the value in this field will be updated',
|
|
displayOptions: {
|
|
show: {
|
|
dataMode: [DATA_MODE.MANUAL],
|
|
},
|
|
},
|
|
},
|
|
{
|
|
displayName: 'Values to Send',
|
|
name: 'valuesToSend',
|
|
placeholder: 'Add Value',
|
|
type: 'fixedCollection',
|
|
typeOptions: {
|
|
multipleValueButtonText: 'Add Value',
|
|
multipleValues: true,
|
|
},
|
|
displayOptions: {
|
|
show: {
|
|
dataMode: [DATA_MODE.MANUAL],
|
|
},
|
|
},
|
|
default: {},
|
|
options: [
|
|
{
|
|
displayName: 'Values',
|
|
name: 'values',
|
|
values: [
|
|
{
|
|
// eslint-disable-next-line n8n-nodes-base/node-param-display-name-wrong-for-dynamic-options
|
|
displayName: 'Column',
|
|
name: 'column',
|
|
type: 'options',
|
|
// eslint-disable-next-line n8n-nodes-base/node-param-description-wrong-for-dynamic-options
|
|
description:
|
|
'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code/expressions/" target="_blank">expression</a>',
|
|
typeOptions: {
|
|
loadOptionsMethod: 'getColumnsWithoutColumnToMatchOn',
|
|
loadOptionsDependsOn: ['schema.value', 'table.value'],
|
|
},
|
|
default: [],
|
|
},
|
|
{
|
|
displayName: 'Value',
|
|
name: 'value',
|
|
type: 'string',
|
|
default: '',
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
optionsCollection,
|
|
];
|
|
|
|
const displayOptions = {
|
|
show: {
|
|
resource: ['database'],
|
|
operation: ['update'],
|
|
},
|
|
hide: {
|
|
table: [''],
|
|
},
|
|
};
|
|
|
|
export const description = updateDisplayOptions(displayOptions, properties);
|
|
|
|
export async function execute(
|
|
this: IExecuteFunctions,
|
|
inputItems: INodeExecutionData[],
|
|
runQueries: QueryRunner,
|
|
nodeOptions: IDataObject,
|
|
): Promise<INodeExecutionData[]> {
|
|
let returnData: INodeExecutionData[] = [];
|
|
const items = replaceEmptyStringsByNulls(inputItems, nodeOptions.replaceEmptyStrings as boolean);
|
|
|
|
const queries: QueryWithValues[] = [];
|
|
|
|
for (let i = 0; i < items.length; i++) {
|
|
const table = this.getNodeParameter('table', i, undefined, {
|
|
extractValue: true,
|
|
}) as string;
|
|
|
|
const columnToMatchOn = this.getNodeParameter('columnToMatchOn', i) as string;
|
|
|
|
const dataMode = this.getNodeParameter('dataMode', i) as string;
|
|
|
|
let item: IDataObject = {};
|
|
let valueToMatchOn: string | IDataObject = '';
|
|
|
|
if (dataMode === DATA_MODE.AUTO_MAP) {
|
|
item = items[i].json;
|
|
valueToMatchOn = item[columnToMatchOn] as string;
|
|
}
|
|
|
|
if (dataMode === DATA_MODE.MANUAL) {
|
|
const valuesToSend = (this.getNodeParameter('valuesToSend', i, []) as IDataObject)
|
|
.values as IDataObject[];
|
|
|
|
item = valuesToSend.reduce((acc, { column, value }) => {
|
|
acc[column as string] = value;
|
|
return acc;
|
|
}, {} as IDataObject);
|
|
|
|
valueToMatchOn = this.getNodeParameter('valueToMatchOn', i) as string;
|
|
}
|
|
|
|
const values: QueryValues = [];
|
|
|
|
const updateColumns = Object.keys(item).filter((column) => column !== columnToMatchOn);
|
|
|
|
const updates: string[] = [];
|
|
|
|
for (const column of updateColumns) {
|
|
updates.push(`${escapeSqlIdentifier(column)} = ?`);
|
|
values.push(item[column] as string);
|
|
}
|
|
|
|
const condition = `${escapeSqlIdentifier(columnToMatchOn)} = ?`;
|
|
values.push(valueToMatchOn);
|
|
|
|
const query = `UPDATE ${escapeSqlIdentifier(table)} SET ${updates.join(
|
|
', ',
|
|
)} WHERE ${condition}`;
|
|
|
|
queries.push({ query, values });
|
|
}
|
|
|
|
returnData = await runQueries(queries);
|
|
|
|
return returnData;
|
|
}
|