n8n/packages/nodes-base/nodes/DataTable/actions/router.ts

123 lines
3.8 KiB
TypeScript

import type {
IExecuteFunctions,
INodeExecutionData,
AllEntities,
DataTableRowOperation,
DataTableTableOperation,
} from 'n8n-workflow';
import { NodeApiError, NodeOperationError } from 'n8n-workflow';
import * as row from './row/Row.resource';
import * as table from './table/Table.resource';
import { DATA_TABLE_ID_FIELD } from '../common/fields';
import { getDataTableProxyExecute } from '../common/utils';
type DataTableNodeType = AllEntities<{
row: DataTableRowOperation;
table: DataTableTableOperation;
}>;
const BULK_OPERATIONS = ['insert'] as const;
function hasBulkExecute(operation: string): operation is (typeof BULK_OPERATIONS)[number] {
return (BULK_OPERATIONS as readonly string[]).includes(operation);
}
function hasComplexId(ctx: IExecuteFunctions) {
const dataTableIdExpr = ctx.getNodeParameter(`${DATA_TABLE_ID_FIELD}.value`, 0, undefined, {
rawExpressions: true,
});
return typeof dataTableIdExpr === 'string' && dataTableIdExpr.includes('{');
}
export async function router(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
let operationResult: INodeExecutionData[] = [];
let responseData: INodeExecutionData[] = [];
const items = this.getInputData();
const resource = this.getNodeParameter('resource', 0);
const operation = this.getNodeParameter('operation', 0);
const dataTableNodeData = {
resource,
operation,
} as DataTableNodeType;
if (dataTableNodeData.resource === 'table') {
// Table operations
for (let i = 0; i < items.length; i++) {
try {
const tableOperation =
dataTableNodeData.operation === 'delete'
? table.deleteTable
: table[dataTableNodeData.operation];
responseData = await tableOperation.execute.call(this, i);
const executionData = this.helpers.constructExecutionMetaData(responseData, {
itemData: { item: i },
});
operationResult = operationResult.concat(executionData);
} catch (error) {
if (this.continueOnFail()) {
const inputData = this.getInputData(i)[0].json;
if (error instanceof NodeApiError || error instanceof NodeOperationError) {
operationResult.push({ json: inputData, error });
} else {
operationResult.push({
json: inputData,
error: new NodeOperationError(this.getNode(), error as Error),
});
}
} else {
throw error;
}
}
}
} else if (hasBulkExecute(dataTableNodeData.operation) && !hasComplexId(this)) {
// Row bulk operations
try {
const proxy = await getDataTableProxyExecute(this);
responseData = await row[dataTableNodeData.operation]['executeBulk'].call(this, proxy);
operationResult = responseData;
} catch (error) {
if (this.continueOnFail()) {
if (error instanceof NodeApiError || error instanceof NodeOperationError) {
operationResult = this.getInputData().map((json) => ({ json, error }));
} else {
operationResult = this.getInputData().map((json) => ({ json }));
}
} else {
throw error;
}
}
} else {
// Row operations
for (let i = 0; i < items.length; i++) {
try {
responseData = await row[dataTableNodeData.operation].execute.call(this, i);
const executionData = this.helpers.constructExecutionMetaData(responseData, {
itemData: { item: i },
});
// pushing here risks stack overflows for very high numbers (~100k) of results on filter-based queries (update, get, etc.)
operationResult = operationResult.concat(executionData);
} catch (error) {
if (this.continueOnFail()) {
const inputData = this.getInputData(i)[0].json;
if (error instanceof NodeApiError || error instanceof NodeOperationError) {
operationResult.push({ json: inputData, error });
} else {
operationResult.push({ json: inputData });
}
} else {
throw error;
}
}
}
}
return [operationResult];
}