mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-14 14:28:14 -08:00
bed04ec122
## Summary Postgres columns can be - [generated as identity](https://www.postgresqltutorial.com/postgresql-tutorial/postgresql-identity-column/) - [generated by a custom expression](https://www.postgresql.org/docs/current/ddl-generated-columns.html) In these 2 cases, the column is not required when inserting a new row. This PR makes sure these types of column are not marked required in n8n. ### How to test 1. Create a Postgres table with all types of generated columns: for version >= 10 ```sql CREATE TABLE "public"."test_table" ( "id" int8 NOT NULL DEFAULT nextval('test_table_id_seq'::regclass), "identity_id" bigint GENERATED ALWAYS AS IDENTITY, "id_plus" numeric GENERATED ALWAYS AS (id + 5) STORED, "title" varchar NOT NULL, "created_at" timestamp DEFAULT now(), PRIMARY KEY ("id") ) ``` Before 10 you have to use serial or bigserial types: ```sql CREATE TABLE distributors ( did serial not null primary key, name varchar(40) NOT NULL CHECK (name <> '') ); ``` 2. Add a postgres node to canvas and try to insert data without the generated columns 3. Should successfully insert More info in Linear/Github issue ⬇️ ## Related tickets and issues - fixes #7084 - https://linear.app/n8n/issue/NODE-816/rmc-not-all-id-fields-should-be-required - https://linear.app/n8n/issue/NODE-681/postgres-cant-map-automatically-if-database-requires-a-field ## Review / Merge checklist - [ ] PR title and summary are descriptive. **Remember, the title automatically goes into the changelog. Use `(no-changelog)` otherwise.** ([conventions](https://github.com/n8n-io/n8n/blob/master/.github/pull_request_title_conventions.md)) - [ ] [Docs updated](https://github.com/n8n-io/n8n-docs) or follow-up ticket created. - [ ] Tests included. > A bug is not considered fixed, unless a test is added to prevent it from happening again. > A feature is not complete without tests. --------- Co-authored-by: Michael Kret <michael.k@radency.com>
513 lines
14 KiB
TypeScript
513 lines
14 KiB
TypeScript
import type {
|
|
IDataObject,
|
|
IExecuteFunctions,
|
|
INode,
|
|
INodeExecutionData,
|
|
INodePropertyOptions,
|
|
} from 'n8n-workflow';
|
|
import { NodeOperationError } from 'n8n-workflow';
|
|
|
|
import { generatePairedItemData } from '../../../../utils/utilities';
|
|
import type {
|
|
ColumnInfo,
|
|
EnumInfo,
|
|
PgpClient,
|
|
PgpDatabase,
|
|
QueryMode,
|
|
QueryValues,
|
|
QueryWithValues,
|
|
SortRule,
|
|
WhereClause,
|
|
} from './interfaces';
|
|
|
|
export function wrapData(data: IDataObject | IDataObject[]): INodeExecutionData[] {
|
|
if (!Array.isArray(data)) {
|
|
return [{ json: data }];
|
|
}
|
|
return data.map((item) => ({
|
|
json: item,
|
|
}));
|
|
}
|
|
|
|
export function prepareErrorItem(
|
|
items: INodeExecutionData[],
|
|
error: IDataObject | NodeOperationError | Error,
|
|
index: number,
|
|
) {
|
|
return {
|
|
json: { message: error.message, item: { ...items[index].json }, error: { ...error } },
|
|
pairedItem: { item: index },
|
|
} as INodeExecutionData;
|
|
}
|
|
|
|
export function parsePostgresError(
|
|
node: INode,
|
|
error: any,
|
|
queries: QueryWithValues[],
|
|
itemIndex?: number,
|
|
) {
|
|
if (error.message.includes('syntax error at or near') && queries.length) {
|
|
try {
|
|
const snippet = error.message.match(/syntax error at or near "(.*)"/)[1] as string;
|
|
const failedQureryIndex = queries.findIndex((query) => query.query.includes(snippet));
|
|
|
|
if (failedQureryIndex !== -1) {
|
|
if (!itemIndex) {
|
|
itemIndex = failedQureryIndex;
|
|
}
|
|
const failedQuery = queries[failedQureryIndex].query;
|
|
const lines = failedQuery.split('\n');
|
|
const lineIndex = lines.findIndex((line) => line.includes(snippet));
|
|
const errorMessage = `Syntax error at line ${lineIndex + 1} near "${snippet}"`;
|
|
error.message = errorMessage;
|
|
}
|
|
} catch {}
|
|
}
|
|
|
|
let message = error.message;
|
|
const errorDescription = error.description ? error.description : error.detail || error.hint;
|
|
let description = errorDescription;
|
|
|
|
if (!description && queries[itemIndex || 0]?.query) {
|
|
description = `Failed query: ${queries[itemIndex || 0].query}`;
|
|
}
|
|
|
|
if (error.message.includes('ECONNREFUSED')) {
|
|
message = 'Connection refused';
|
|
try {
|
|
description = error.message.split('ECONNREFUSED ')[1].trim();
|
|
} catch (e) {}
|
|
}
|
|
|
|
if (error.message.includes('ENOTFOUND')) {
|
|
message = 'Host not found';
|
|
try {
|
|
description = error.message.split('ENOTFOUND ')[1].trim();
|
|
} catch (e) {}
|
|
}
|
|
|
|
if (error.message.includes('ETIMEDOUT')) {
|
|
message = 'Connection timed out';
|
|
try {
|
|
description = error.message.split('ETIMEDOUT ')[1].trim();
|
|
} catch (e) {}
|
|
}
|
|
|
|
return new NodeOperationError(node, error as Error, {
|
|
message,
|
|
description,
|
|
itemIndex,
|
|
});
|
|
}
|
|
|
|
export function addWhereClauses(
|
|
node: INode,
|
|
itemIndex: number,
|
|
query: string,
|
|
clauses: WhereClause[],
|
|
replacements: QueryValues,
|
|
combineConditions: string,
|
|
): [string, QueryValues] {
|
|
if (clauses.length === 0) return [query, replacements];
|
|
|
|
let combineWith = 'AND';
|
|
|
|
if (combineConditions === 'OR') {
|
|
combineWith = 'OR';
|
|
}
|
|
|
|
let replacementIndex = replacements.length + 1;
|
|
|
|
let whereQuery = ' WHERE';
|
|
const values: QueryValues = [];
|
|
|
|
clauses.forEach((clause, index) => {
|
|
if (clause.condition === 'equal') {
|
|
clause.condition = '=';
|
|
}
|
|
if (['>', '<', '>=', '<='].includes(clause.condition)) {
|
|
const value = Number(clause.value);
|
|
|
|
if (Number.isNaN(value)) {
|
|
throw new NodeOperationError(
|
|
node,
|
|
`Operator in entry ${index + 1} of 'Select Rows' works with numbers, but value ${
|
|
clause.value
|
|
} is not a number`,
|
|
{
|
|
itemIndex,
|
|
},
|
|
);
|
|
}
|
|
|
|
clause.value = value;
|
|
}
|
|
const columnReplacement = `$${replacementIndex}:name`;
|
|
values.push(clause.column);
|
|
replacementIndex = replacementIndex + 1;
|
|
|
|
let valueReplacement = '';
|
|
if (clause.condition !== 'IS NULL') {
|
|
valueReplacement = ` $${replacementIndex}`;
|
|
values.push(clause.value);
|
|
replacementIndex = replacementIndex + 1;
|
|
}
|
|
|
|
const operator = index === clauses.length - 1 ? '' : ` ${combineWith}`;
|
|
|
|
whereQuery += ` ${columnReplacement} ${clause.condition}${valueReplacement}${operator}`;
|
|
});
|
|
|
|
return [`${query}${whereQuery}`, replacements.concat(...values)];
|
|
}
|
|
|
|
export function addSortRules(
|
|
query: string,
|
|
rules: SortRule[],
|
|
replacements: QueryValues,
|
|
): [string, QueryValues] {
|
|
if (rules.length === 0) return [query, replacements];
|
|
|
|
let replacementIndex = replacements.length + 1;
|
|
|
|
let orderByQuery = ' ORDER BY';
|
|
const values: string[] = [];
|
|
|
|
rules.forEach((rule, index) => {
|
|
const columnReplacement = `$${replacementIndex}:name`;
|
|
values.push(rule.column);
|
|
replacementIndex = replacementIndex + 1;
|
|
|
|
const endWith = index === rules.length - 1 ? '' : ',';
|
|
|
|
const sortDirection = rule.direction === 'DESC' ? 'DESC' : 'ASC';
|
|
|
|
orderByQuery += ` ${columnReplacement} ${sortDirection}${endWith}`;
|
|
});
|
|
|
|
return [`${query}${orderByQuery}`, replacements.concat(...values)];
|
|
}
|
|
|
|
export function addReturning(
|
|
query: string,
|
|
outputColumns: string[],
|
|
replacements: QueryValues,
|
|
): [string, QueryValues] {
|
|
if (outputColumns.includes('*')) return [`${query} RETURNING *`, replacements];
|
|
|
|
const replacementIndex = replacements.length + 1;
|
|
|
|
return [`${query} RETURNING $${replacementIndex}:name`, [...replacements, outputColumns]];
|
|
}
|
|
|
|
const isSelectQuery = (query: string) => {
|
|
return query
|
|
.replace(/\/\*.*?\*\//g, '') // remove multiline comments
|
|
.replace(/\n/g, '')
|
|
.split(';')
|
|
.filter((statement) => statement && !statement.startsWith('--')) // remove comments and empty statements
|
|
.every((statement) => statement.trim().toLowerCase().startsWith('select'));
|
|
};
|
|
|
|
export function configureQueryRunner(
|
|
this: IExecuteFunctions,
|
|
node: INode,
|
|
continueOnFail: boolean,
|
|
pgp: PgpClient,
|
|
db: PgpDatabase,
|
|
) {
|
|
return async (queries: QueryWithValues[], items: INodeExecutionData[], options: IDataObject) => {
|
|
let returnData: INodeExecutionData[] = [];
|
|
const emptyReturnData: INodeExecutionData[] =
|
|
options.operation === 'select' ? [] : [{ json: { success: true } }];
|
|
|
|
const queryBatching = (options.queryBatching as QueryMode) || 'single';
|
|
|
|
if (queryBatching === 'single') {
|
|
try {
|
|
returnData = (await db.multi(pgp.helpers.concat(queries)))
|
|
.map((result, i) => {
|
|
return this.helpers.constructExecutionMetaData(wrapData(result as IDataObject[]), {
|
|
itemData: { item: i },
|
|
});
|
|
})
|
|
.flat();
|
|
|
|
if (!returnData.length) {
|
|
const pairedItem = generatePairedItemData(queries.length);
|
|
|
|
if ((options?.nodeVersion as number) < 2.3) {
|
|
if (emptyReturnData.length) {
|
|
emptyReturnData[0].pairedItem = pairedItem;
|
|
}
|
|
returnData = emptyReturnData;
|
|
} else {
|
|
returnData = queries.every((query) => isSelectQuery(query.query))
|
|
? []
|
|
: [{ json: { success: true }, pairedItem }];
|
|
}
|
|
}
|
|
} catch (err) {
|
|
const error = parsePostgresError(node, err, queries);
|
|
if (!continueOnFail) throw error;
|
|
|
|
return [
|
|
{
|
|
json: {
|
|
message: error.message,
|
|
error: { ...error },
|
|
},
|
|
},
|
|
];
|
|
}
|
|
}
|
|
|
|
if (queryBatching === 'transaction') {
|
|
returnData = await db.tx(async (transaction) => {
|
|
const result: INodeExecutionData[] = [];
|
|
for (let i = 0; i < queries.length; i++) {
|
|
try {
|
|
const query = queries[i].query;
|
|
const values = queries[i].values;
|
|
|
|
let transactionResults;
|
|
if ((options?.nodeVersion as number) < 2.3) {
|
|
transactionResults = await transaction.any(query, values);
|
|
} else {
|
|
transactionResults = (await transaction.multi(query, values)).flat();
|
|
}
|
|
|
|
if (!transactionResults.length) {
|
|
if ((options?.nodeVersion as number) < 2.3) {
|
|
transactionResults = emptyReturnData;
|
|
} else {
|
|
transactionResults = isSelectQuery(query) ? [] : [{ success: true }];
|
|
}
|
|
}
|
|
|
|
const executionData = this.helpers.constructExecutionMetaData(
|
|
wrapData(transactionResults),
|
|
{ itemData: { item: i } },
|
|
);
|
|
|
|
result.push(...executionData);
|
|
} catch (err) {
|
|
const error = parsePostgresError(node, err, queries, i);
|
|
if (!continueOnFail) throw error;
|
|
result.push(prepareErrorItem(items, error, i));
|
|
return result;
|
|
}
|
|
}
|
|
return result;
|
|
});
|
|
}
|
|
|
|
if (queryBatching === 'independently') {
|
|
returnData = await db.task(async (task) => {
|
|
const result: INodeExecutionData[] = [];
|
|
for (let i = 0; i < queries.length; i++) {
|
|
try {
|
|
const query = queries[i].query;
|
|
const values = queries[i].values;
|
|
|
|
let transactionResults;
|
|
if ((options?.nodeVersion as number) < 2.3) {
|
|
transactionResults = await task.any(query, values);
|
|
} else {
|
|
transactionResults = (await task.multi(query, values)).flat();
|
|
}
|
|
|
|
if (!transactionResults.length) {
|
|
if ((options?.nodeVersion as number) < 2.3) {
|
|
transactionResults = emptyReturnData;
|
|
} else {
|
|
transactionResults = isSelectQuery(query) ? [] : [{ success: true }];
|
|
}
|
|
}
|
|
|
|
const executionData = this.helpers.constructExecutionMetaData(
|
|
wrapData(transactionResults),
|
|
{ itemData: { item: i } },
|
|
);
|
|
|
|
result.push(...executionData);
|
|
} catch (err) {
|
|
const error = parsePostgresError(node, err, queries, i);
|
|
if (!continueOnFail) throw error;
|
|
result.push(prepareErrorItem(items, error, i));
|
|
}
|
|
}
|
|
return result;
|
|
});
|
|
}
|
|
|
|
return returnData;
|
|
};
|
|
}
|
|
|
|
export function replaceEmptyStringsByNulls(
|
|
items: INodeExecutionData[],
|
|
replace?: boolean,
|
|
): INodeExecutionData[] {
|
|
if (!replace) return items;
|
|
|
|
const returnData: INodeExecutionData[] = items.map((item) => {
|
|
const newItem = { ...item };
|
|
const keys = Object.keys(newItem.json);
|
|
|
|
for (const key of keys) {
|
|
if (newItem.json[key] === '') {
|
|
newItem.json[key] = null;
|
|
}
|
|
}
|
|
|
|
return newItem;
|
|
});
|
|
|
|
return returnData;
|
|
}
|
|
|
|
export function prepareItem(values: IDataObject[]) {
|
|
const item = values.reduce((acc, { column, value }) => {
|
|
acc[column as string] = value;
|
|
return acc;
|
|
}, {} as IDataObject);
|
|
|
|
return item;
|
|
}
|
|
|
|
export async function columnFeatureSupport(
|
|
db: PgpDatabase,
|
|
): Promise<{ identity_generation: boolean; is_generated: boolean }> {
|
|
const result = await db.any(
|
|
`SELECT EXISTS (
|
|
SELECT 1 FROM information_schema.columns WHERE table_name = 'columns' AND table_schema = 'information_schema' AND column_name = 'is_generated'
|
|
) as is_generated,
|
|
EXISTS (
|
|
SELECT 1 FROM information_schema.columns WHERE table_name = 'columns' AND table_schema = 'information_schema' AND column_name = 'identity_generation'
|
|
) as identity_generation;`,
|
|
);
|
|
|
|
return result[0];
|
|
}
|
|
|
|
export async function getTableSchema(
|
|
db: PgpDatabase,
|
|
schema: string,
|
|
table: string,
|
|
options?: { getColumnsForResourceMapper?: boolean },
|
|
): Promise<ColumnInfo[]> {
|
|
const select = ['column_name', 'data_type', 'is_nullable', 'udt_name', 'column_default'];
|
|
|
|
if (options?.getColumnsForResourceMapper) {
|
|
// Check if columns exist before querying (identity_generation was added in v10, is_generated in v12)
|
|
const supported = await columnFeatureSupport(db);
|
|
|
|
if (supported.identity_generation) {
|
|
select.push('identity_generation');
|
|
}
|
|
|
|
if (supported.is_generated) {
|
|
select.push('is_generated');
|
|
}
|
|
}
|
|
|
|
const selectString = select.join(', ');
|
|
const columns = await db.any(
|
|
`SELECT ${selectString} FROM information_schema.columns WHERE table_schema = $1 AND table_name = $2`,
|
|
[schema, table],
|
|
);
|
|
|
|
return columns;
|
|
}
|
|
|
|
export async function uniqueColumns(db: PgpDatabase, table: string, schema = 'public') {
|
|
// Using the modified query from https://wiki.postgresql.org/wiki/Retrieve_primary_key_columns
|
|
// `quote_ident` - properly quote and escape an identifier
|
|
// `::regclass` - cast a string to a regclass (internal type for object names)
|
|
const unique = await db.any(
|
|
`
|
|
SELECT DISTINCT a.attname
|
|
FROM pg_index i JOIN pg_attribute a ON a.attrelid = i.indrelid AND a.attnum = ANY(i.indkey)
|
|
WHERE i.indrelid = (quote_ident($1) || '.' || quote_ident($2))::regclass
|
|
AND (i.indisprimary OR i.indisunique);
|
|
`,
|
|
[schema, table],
|
|
);
|
|
return unique as IDataObject[];
|
|
}
|
|
|
|
export async function getEnums(db: PgpDatabase): Promise<EnumInfo[]> {
|
|
const enumsData = await db.any(
|
|
'SELECT pg_type.typname, pg_enum.enumlabel FROM pg_type JOIN pg_enum ON pg_enum.enumtypid = pg_type.oid;',
|
|
);
|
|
return enumsData as EnumInfo[];
|
|
}
|
|
|
|
export function getEnumValues(enumInfo: EnumInfo[], enumName: string): INodePropertyOptions[] {
|
|
return enumInfo.reduce((acc, current) => {
|
|
if (current.typname === enumName) {
|
|
acc.push({ name: current.enumlabel, value: current.enumlabel });
|
|
}
|
|
return acc;
|
|
}, [] as INodePropertyOptions[]);
|
|
}
|
|
|
|
export async function doesRowExist(
|
|
db: PgpDatabase,
|
|
schema: string,
|
|
table: string,
|
|
values: string[],
|
|
): Promise<boolean> {
|
|
const where = [];
|
|
for (let i = 3; i < 3 + values.length; i += 2) {
|
|
where.push(`$${i}:name=$${i + 1}`);
|
|
}
|
|
const exists = await db.any(
|
|
`SELECT EXISTS(SELECT 1 FROM $1:name.$2:name WHERE ${where.join(' AND ')})`,
|
|
[schema, table, ...values],
|
|
);
|
|
return exists[0].exists;
|
|
}
|
|
|
|
export function checkItemAgainstSchema(
|
|
node: INode,
|
|
item: IDataObject,
|
|
columnsInfo: ColumnInfo[],
|
|
index: number,
|
|
) {
|
|
if (columnsInfo.length === 0) return item;
|
|
const schema = columnsInfo.reduce((acc, { column_name, data_type, is_nullable }) => {
|
|
acc[column_name] = { type: data_type.toUpperCase(), nullable: is_nullable === 'YES' };
|
|
return acc;
|
|
}, {} as IDataObject);
|
|
|
|
for (const key of Object.keys(item)) {
|
|
if (schema[key] === undefined) {
|
|
throw new NodeOperationError(node, `Column '${key}' does not exist in selected table`, {
|
|
itemIndex: index,
|
|
});
|
|
}
|
|
if (item[key] === null && !(schema[key] as IDataObject)?.nullable) {
|
|
throw new NodeOperationError(node, `Column '${key}' is not nullable`, {
|
|
itemIndex: index,
|
|
});
|
|
}
|
|
}
|
|
|
|
return item;
|
|
}
|
|
|
|
export const configureTableSchemaUpdater = (initialSchema: string, initialTable: string) => {
|
|
let currentSchema = initialSchema;
|
|
let currentTable = initialTable;
|
|
return async (db: PgpDatabase, tableSchema: ColumnInfo[], schema: string, table: string) => {
|
|
if (currentSchema !== schema || currentTable !== table) {
|
|
currentSchema = schema;
|
|
currentTable = table;
|
|
tableSchema = await getTableSchema(db, schema, table);
|
|
}
|
|
return tableSchema;
|
|
};
|
|
};
|