mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-07 02:47:32 -08:00
920 lines
20 KiB
TypeScript
920 lines
20 KiB
TypeScript
import type {
|
|
IDataObject,
|
|
IExecuteFunctions,
|
|
IGetNodeParameterOptions,
|
|
INode,
|
|
INodeParameters,
|
|
} from 'n8n-workflow';
|
|
|
|
import { get } from 'lodash';
|
|
import type { ColumnInfo, PgpDatabase, QueriesRunner } from '../../v2/helpers/interfaces';
|
|
|
|
import * as deleteTable from '../../v2/actions/database/deleteTable.operation';
|
|
import * as executeQuery from '../../v2/actions/database/executeQuery.operation';
|
|
import * as insert from '../../v2/actions/database/insert.operation';
|
|
import * as select from '../../v2/actions/database/select.operation';
|
|
import * as update from '../../v2/actions/database/update.operation';
|
|
import * as upsert from '../../v2/actions/database/upsert.operation';
|
|
|
|
const runQueries: QueriesRunner = jest.fn();
|
|
|
|
const node: INode = {
|
|
id: '1',
|
|
name: 'Postgres node',
|
|
typeVersion: 2,
|
|
type: 'n8n-nodes-base.postgres',
|
|
position: [60, 760],
|
|
parameters: {
|
|
operation: 'executeQuery',
|
|
},
|
|
};
|
|
|
|
const items = [{ json: {} }];
|
|
|
|
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() {
|
|
node.parameters = { ...node.parameters, ...(nodeParameters as INodeParameters) };
|
|
return node;
|
|
},
|
|
evaluateExpression(str: string, _: number) {
|
|
return str.replace('{{', '').replace('}}', '');
|
|
},
|
|
} as unknown as IExecuteFunctions;
|
|
return fakeExecuteFunction;
|
|
};
|
|
|
|
const createMockDb = (columnInfo: ColumnInfo[]) => {
|
|
return {
|
|
async any() {
|
|
return columnInfo;
|
|
},
|
|
} as unknown as PgpDatabase;
|
|
};
|
|
|
|
// if node parameters copied from canvas all default parameters has to be added manually as JSON would not have them
|
|
describe('Test PostgresV2, deleteTable operation', () => {
|
|
afterEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
it('deleteCommand: delete, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'deleteTable',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
cachedResultName: 'my_table',
|
|
},
|
|
deleteCommand: 'delete',
|
|
where: {
|
|
values: [
|
|
{
|
|
column: 'id',
|
|
condition: 'LIKE',
|
|
value: '1',
|
|
},
|
|
],
|
|
},
|
|
options: { nodeVersion: 2.1 },
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await deleteTable.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query: 'DELETE FROM $1:name.$2:name WHERE $3:name LIKE $4',
|
|
values: ['public', 'my_table', 'id', '1'],
|
|
},
|
|
],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('deleteCommand: truncate, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'deleteTable',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
cachedResultName: 'my_table',
|
|
},
|
|
deleteCommand: 'truncate',
|
|
restartSequences: true,
|
|
options: {
|
|
cascade: true,
|
|
},
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await deleteTable.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query: 'TRUNCATE TABLE $1:name.$2:name RESTART IDENTITY CASCADE',
|
|
values: ['public', 'my_table'],
|
|
},
|
|
],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('deleteCommand: drop, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'deleteTable',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
cachedResultName: 'my_table',
|
|
},
|
|
deleteCommand: 'drop',
|
|
options: { nodeVersion: 2.1 },
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await deleteTable.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query: 'DROP TABLE IF EXISTS $1:name.$2:name',
|
|
values: ['public', 'my_table'],
|
|
},
|
|
],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('Test PostgresV2, executeQuery operation', () => {
|
|
afterEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
it('should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'executeQuery',
|
|
query: 'select * from $1:name;',
|
|
options: {
|
|
queryReplacement: 'my_table',
|
|
},
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await executeQuery.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[{ query: 'select * from $1:name;', values: ['my_table'] }],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('should call runQueries and insert enclosed placeholder into values', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'executeQuery',
|
|
query: "select '$1';",
|
|
options: {},
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await executeQuery.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[{ query: 'select $1;', values: ['$1'] }],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('should call runQueries and not insert enclosed placeholder into values because queryReplacement is defined', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'executeQuery',
|
|
query: "select '$1';",
|
|
options: {
|
|
queryReplacement: 'my_table',
|
|
},
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await executeQuery.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[{ query: "select '$1';", values: ['my_table'] }],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('should call runQueries and insert enclosed placeholder into values because treatQueryParametersInSingleQuotesAsText is true', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'executeQuery',
|
|
query: "select '$1';",
|
|
options: {
|
|
queryReplacement: 'my_table',
|
|
treatQueryParametersInSingleQuotesAsText: true,
|
|
},
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await executeQuery.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[{ query: 'select $2;', values: ['my_table', '$1'] }],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('should call runQueries with falsy query replacements', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'executeQuery',
|
|
query: 'SELECT *\nFROM users\nWHERE username IN ($1, $2, $3)',
|
|
options: {
|
|
queryReplacement: '={{ 0 }}, {{ null }}, {{ 0 }}',
|
|
},
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
expect(async () => {
|
|
await executeQuery.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
}).not.toThrow();
|
|
});
|
|
});
|
|
|
|
describe('Test PostgresV2, insert operation', () => {
|
|
afterEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
it('dataMode: define, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
},
|
|
dataMode: 'defineBelow',
|
|
valuesToSend: {
|
|
values: [
|
|
{
|
|
column: 'json',
|
|
value: '{"test":15}',
|
|
},
|
|
{
|
|
column: 'foo',
|
|
value: 'select 5',
|
|
},
|
|
{
|
|
column: 'id',
|
|
value: '4',
|
|
},
|
|
],
|
|
},
|
|
options: { nodeVersion: 2.1 },
|
|
};
|
|
const columnsInfo: ColumnInfo[] = [
|
|
{ column_name: 'id', data_type: 'integer', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'json', data_type: 'json', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'foo', data_type: 'text', is_nullable: 'NO', udt_name: '' },
|
|
];
|
|
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await insert.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
createMockDb(columnsInfo),
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query: 'INSERT INTO $1:name.$2:name($3:name) VALUES($3:csv) RETURNING *',
|
|
values: ['public', 'my_table', { json: '{"test":15}', foo: 'select 5', id: '4' }],
|
|
},
|
|
],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('dataMode: autoMapInputData, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
},
|
|
dataMode: 'autoMapInputData',
|
|
options: { nodeVersion: 2.1 },
|
|
};
|
|
const columnsInfo: ColumnInfo[] = [
|
|
{ column_name: 'id', data_type: 'integer', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'json', data_type: 'json', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'foo', data_type: 'text', is_nullable: 'NO', udt_name: '' },
|
|
];
|
|
|
|
const inputItems = [
|
|
{
|
|
json: {
|
|
id: 1,
|
|
json: {
|
|
test: 15,
|
|
},
|
|
foo: 'data 1',
|
|
},
|
|
},
|
|
{
|
|
json: {
|
|
id: 2,
|
|
json: {
|
|
test: 10,
|
|
},
|
|
foo: 'data 2',
|
|
},
|
|
},
|
|
{
|
|
json: {
|
|
id: 3,
|
|
json: {
|
|
test: 5,
|
|
},
|
|
foo: 'data 3',
|
|
},
|
|
},
|
|
];
|
|
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await insert.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
inputItems,
|
|
nodeOptions,
|
|
createMockDb(columnsInfo),
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query: 'INSERT INTO $1:name.$2:name($3:name) VALUES($3:csv) RETURNING *',
|
|
values: ['public', 'my_table', { id: 1, json: { test: 15 }, foo: 'data 1' }],
|
|
},
|
|
{
|
|
query: 'INSERT INTO $1:name.$2:name($3:name) VALUES($3:csv) RETURNING *',
|
|
values: ['public', 'my_table', { id: 2, json: { test: 10 }, foo: 'data 2' }],
|
|
},
|
|
{
|
|
query: 'INSERT INTO $1:name.$2:name($3:name) VALUES($3:csv) RETURNING *',
|
|
values: ['public', 'my_table', { id: 3, json: { test: 5 }, foo: 'data 3' }],
|
|
},
|
|
],
|
|
inputItems,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('Test PostgresV2, select operation', () => {
|
|
afterEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
it('returnAll, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'select',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
cachedResultName: 'my_table',
|
|
},
|
|
returnAll: true,
|
|
options: {},
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await select.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query: 'SELECT * FROM $1:name.$2:name',
|
|
values: ['public', 'my_table'],
|
|
},
|
|
],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('limit, whereClauses, sortRules, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'select',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
cachedResultName: 'my_table',
|
|
},
|
|
limit: 5,
|
|
where: {
|
|
values: [
|
|
{
|
|
column: 'id',
|
|
condition: '>=',
|
|
value: 2,
|
|
},
|
|
{
|
|
column: 'foo',
|
|
condition: 'equal',
|
|
value: 'data 2',
|
|
},
|
|
],
|
|
},
|
|
sort: {
|
|
values: [
|
|
{
|
|
column: 'id',
|
|
},
|
|
],
|
|
},
|
|
options: {
|
|
outputColumns: ['json', 'id'],
|
|
},
|
|
};
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await select.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query:
|
|
'SELECT $3:name FROM $1:name.$2:name WHERE $4:name >= $5 AND $6:name = $7 ORDER BY $8:name ASC LIMIT 5',
|
|
values: ['public', 'my_table', ['json', 'id'], 'id', 2, 'foo', 'data 2', 'id'],
|
|
},
|
|
],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('Test PostgresV2, update operation', () => {
|
|
afterEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
it('dataMode: define, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'update',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
},
|
|
dataMode: 'defineBelow',
|
|
columnToMatchOn: 'id',
|
|
valueToMatchOn: '1',
|
|
valuesToSend: {
|
|
values: [
|
|
{
|
|
column: 'json',
|
|
value: { text: 'some text' },
|
|
},
|
|
{
|
|
column: 'foo',
|
|
value: 'updated',
|
|
},
|
|
],
|
|
},
|
|
options: {
|
|
outputColumns: ['json', 'foo'],
|
|
nodeVersion: 2.1,
|
|
},
|
|
};
|
|
const columnsInfo: ColumnInfo[] = [
|
|
{ column_name: 'id', data_type: 'integer', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'json', data_type: 'json', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'foo', data_type: 'text', is_nullable: 'NO', udt_name: '' },
|
|
];
|
|
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await update.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
createMockDb(columnsInfo),
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query:
|
|
'UPDATE $1:name.$2:name SET $5:name = $6, $7:name = $8 WHERE $3:name = $4 RETURNING $9:name',
|
|
values: [
|
|
'public',
|
|
'my_table',
|
|
'id',
|
|
'1',
|
|
'json',
|
|
{ text: 'some text' },
|
|
'foo',
|
|
'updated',
|
|
['json', 'foo'],
|
|
],
|
|
},
|
|
],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('dataMode: autoMapInputData, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'update',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
},
|
|
dataMode: 'autoMapInputData',
|
|
columnToMatchOn: 'id',
|
|
options: { nodeVersion: 2.1 },
|
|
};
|
|
const columnsInfo: ColumnInfo[] = [
|
|
{ column_name: 'id', data_type: 'integer', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'json', data_type: 'json', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'foo', data_type: 'text', is_nullable: 'NO', udt_name: '' },
|
|
];
|
|
|
|
const inputItems = [
|
|
{
|
|
json: {
|
|
id: 1,
|
|
json: {
|
|
test: 15,
|
|
},
|
|
foo: 'data 1',
|
|
},
|
|
},
|
|
{
|
|
json: {
|
|
id: 2,
|
|
json: {
|
|
test: 10,
|
|
},
|
|
foo: 'data 2',
|
|
},
|
|
},
|
|
{
|
|
json: {
|
|
id: 3,
|
|
json: {
|
|
test: 5,
|
|
},
|
|
foo: 'data 3',
|
|
},
|
|
},
|
|
];
|
|
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await update.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
inputItems,
|
|
nodeOptions,
|
|
createMockDb(columnsInfo),
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query:
|
|
'UPDATE $1:name.$2:name SET $5:name = $6, $7:name = $8 WHERE $3:name = $4 RETURNING *',
|
|
values: ['public', 'my_table', 'id', 1, 'json', { test: 15 }, 'foo', 'data 1'],
|
|
},
|
|
{
|
|
query:
|
|
'UPDATE $1:name.$2:name SET $5:name = $6, $7:name = $8 WHERE $3:name = $4 RETURNING *',
|
|
values: ['public', 'my_table', 'id', 2, 'json', { test: 10 }, 'foo', 'data 2'],
|
|
},
|
|
{
|
|
query:
|
|
'UPDATE $1:name.$2:name SET $5:name = $6, $7:name = $8 WHERE $3:name = $4 RETURNING *',
|
|
values: ['public', 'my_table', 'id', 3, 'json', { test: 5 }, 'foo', 'data 3'],
|
|
},
|
|
],
|
|
inputItems,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('Test PostgresV2, upsert operation', () => {
|
|
it('dataMode: define, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'upsert',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
},
|
|
dataMode: 'defineBelow',
|
|
columnToMatchOn: 'id',
|
|
valueToMatchOn: '5',
|
|
valuesToSend: {
|
|
values: [
|
|
{
|
|
column: 'json',
|
|
value: '{ "test": 5 }',
|
|
},
|
|
{
|
|
column: 'foo',
|
|
value: 'data 5',
|
|
},
|
|
],
|
|
},
|
|
options: {
|
|
outputColumns: ['json'],
|
|
nodeVersion: 2.1,
|
|
},
|
|
};
|
|
const columnsInfo: ColumnInfo[] = [
|
|
{ column_name: 'id', data_type: 'integer', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'json', data_type: 'json', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'foo', data_type: 'text', is_nullable: 'NO', udt_name: '' },
|
|
];
|
|
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await upsert.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
items,
|
|
nodeOptions,
|
|
createMockDb(columnsInfo),
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query:
|
|
'INSERT INTO $1:name.$2:name($4:name) VALUES($4:csv) ON CONFLICT ($3:name) DO UPDATE SET $5:name = $6, $7:name = $8 RETURNING $9:name',
|
|
values: [
|
|
'public',
|
|
'my_table',
|
|
'id',
|
|
{ json: '{ "test": 5 }', foo: 'data 5', id: '5' },
|
|
'json',
|
|
'{ "test": 5 }',
|
|
'foo',
|
|
'data 5',
|
|
['json'],
|
|
],
|
|
},
|
|
],
|
|
items,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
|
|
it('dataMode: autoMapInputData, should call runQueries with', async () => {
|
|
const nodeParameters: IDataObject = {
|
|
operation: 'upsert',
|
|
schema: {
|
|
__rl: true,
|
|
mode: 'list',
|
|
value: 'public',
|
|
},
|
|
table: {
|
|
__rl: true,
|
|
value: 'my_table',
|
|
mode: 'list',
|
|
},
|
|
dataMode: 'autoMapInputData',
|
|
columnToMatchOn: 'id',
|
|
options: { nodeVersion: 2.1 },
|
|
};
|
|
const columnsInfo: ColumnInfo[] = [
|
|
{ column_name: 'id', data_type: 'integer', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'json', data_type: 'json', is_nullable: 'NO', udt_name: '' },
|
|
{ column_name: 'foo', data_type: 'text', is_nullable: 'NO', udt_name: '' },
|
|
];
|
|
|
|
const inputItems = [
|
|
{
|
|
json: {
|
|
id: 1,
|
|
json: {
|
|
test: 15,
|
|
},
|
|
foo: 'data 1',
|
|
},
|
|
},
|
|
{
|
|
json: {
|
|
id: 2,
|
|
json: {
|
|
test: 10,
|
|
},
|
|
foo: 'data 2',
|
|
},
|
|
},
|
|
{
|
|
json: {
|
|
id: 3,
|
|
json: {
|
|
test: 5,
|
|
},
|
|
foo: 'data 3',
|
|
},
|
|
},
|
|
];
|
|
|
|
const nodeOptions = nodeParameters.options as IDataObject;
|
|
|
|
await upsert.execute.call(
|
|
createMockExecuteFunction(nodeParameters),
|
|
runQueries,
|
|
inputItems,
|
|
nodeOptions,
|
|
createMockDb(columnsInfo),
|
|
);
|
|
|
|
expect(runQueries).toHaveBeenCalledWith(
|
|
[
|
|
{
|
|
query:
|
|
'INSERT INTO $1:name.$2:name($4:name) VALUES($4:csv) ON CONFLICT ($3:name) DO UPDATE SET $5:name = $6, $7:name = $8 RETURNING *',
|
|
values: [
|
|
'public',
|
|
'my_table',
|
|
'id',
|
|
{ id: 1, json: { test: 15 }, foo: 'data 1' },
|
|
'json',
|
|
{ test: 15 },
|
|
'foo',
|
|
'data 1',
|
|
],
|
|
},
|
|
{
|
|
query:
|
|
'INSERT INTO $1:name.$2:name($4:name) VALUES($4:csv) ON CONFLICT ($3:name) DO UPDATE SET $5:name = $6, $7:name = $8 RETURNING *',
|
|
values: [
|
|
'public',
|
|
'my_table',
|
|
'id',
|
|
{ id: 2, json: { test: 10 }, foo: 'data 2' },
|
|
'json',
|
|
{ test: 10 },
|
|
'foo',
|
|
'data 2',
|
|
],
|
|
},
|
|
{
|
|
query:
|
|
'INSERT INTO $1:name.$2:name($4:name) VALUES($4:csv) ON CONFLICT ($3:name) DO UPDATE SET $5:name = $6, $7:name = $8 RETURNING *',
|
|
values: [
|
|
'public',
|
|
'my_table',
|
|
'id',
|
|
{ id: 3, json: { test: 5 }, foo: 'data 3' },
|
|
'json',
|
|
{ test: 5 },
|
|
'foo',
|
|
'data 3',
|
|
],
|
|
},
|
|
],
|
|
inputItems,
|
|
nodeOptions,
|
|
);
|
|
});
|
|
});
|