fix(Postgres Node): Stop marking autogenerated columns as required (#8230)

## 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>
This commit is contained in:
Elias Meire 2024-01-05 12:37:33 +01:00 committed by GitHub
parent 048b588852
commit bed04ec122
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 45 additions and 4 deletions

View file

@ -17,6 +17,8 @@ export type ColumnInfo = {
is_nullable: string; is_nullable: string;
udt_name?: string; udt_name?: string;
column_default?: string; column_default?: string;
is_generated?: 'ALWAYS' | 'NEVER';
identity_generation?: 'ALWAYS' | 'NEVER';
}; };
export type EnumInfo = { export type EnumInfo = {
typname: string; typname: string;

View file

@ -376,13 +376,45 @@ export function prepareItem(values: IDataObject[]) {
return item; 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( export async function getTableSchema(
db: PgpDatabase, db: PgpDatabase,
schema: string, schema: string,
table: string, table: string,
options?: { getColumnsForResourceMapper?: boolean },
): Promise<ColumnInfo[]> { ): 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( const columns = await db.any(
'SELECT column_name, data_type, is_nullable, udt_name, column_default FROM information_schema.columns WHERE table_schema = $1 AND table_name = $2', `SELECT ${selectString} FROM information_schema.columns WHERE table_schema = $1 AND table_name = $2`,
[schema, table], [schema, table],
); );

View file

@ -62,7 +62,7 @@ export async function getMappingColumns(
}) as string; }) as string;
try { try {
const columns = await getTableSchema(db, schema, table); const columns = await getTableSchema(db, schema, table, { getColumnsForResourceMapper: true });
const unique = operation === 'upsert' ? await uniqueColumns(db, table, schema) : []; const unique = operation === 'upsert' ? await uniqueColumns(db, table, schema) : [];
const enumInfo = await getEnums(db); const enumInfo = await getEnums(db);
const fields = await Promise.all( const fields = await Promise.all(
@ -72,11 +72,13 @@ export async function getMappingColumns(
const type = mapPostgresType(col.data_type); const type = mapPostgresType(col.data_type);
const options = const options =
type === 'options' ? getEnumValues(enumInfo, col.udt_name as string) : undefined; type === 'options' ? getEnumValues(enumInfo, col.udt_name as string) : undefined;
const isAutoIncrement = col.column_default?.startsWith('nextval'); const hasDefault = Boolean(col.column_default);
const isGenerated = col.is_generated === 'ALWAYS' || col.identity_generation === 'ALWAYS';
const nullable = col.is_nullable === 'YES';
return { return {
id: col.column_name, id: col.column_name,
displayName: col.column_name, displayName: col.column_name,
required: col.is_nullable !== 'YES' && !isAutoIncrement, required: !nullable && !hasDefault && !isGenerated,
defaultMatch: (col.column_name === 'id' && canBeUsedToMatch) || false, defaultMatch: (col.column_name === 'id' && canBeUsedToMatch) || false,
display: true, display: true,
type, type,

View file

@ -1230,6 +1230,11 @@ export const validateResourceMapperParameter = (
value: ResourceMapperValue, value: ResourceMapperValue,
skipRequiredCheck = false, skipRequiredCheck = false,
): Record<string, string[]> => { ): Record<string, string[]> => {
// No issues to raise in automatic mapping mode, no user input to validate
if (value.mappingMode === 'autoMapInputData') {
return {};
}
const issues: Record<string, string[]> = {}; const issues: Record<string, string[]> = {};
let fieldWordSingular = let fieldWordSingular =
nodeProperties.typeOptions?.resourceMapper?.fieldWords?.singular || 'Field'; nodeProperties.typeOptions?.resourceMapper?.fieldWords?.singular || 'Field';