2023-04-03 08:18:01 -07:00
import type {
ICredentialsDecrypted ,
ICredentialTestFunctions ,
IDataObject ,
2023-07-12 02:15:38 -07:00
IExecuteFunctions ,
2023-04-03 08:18:01 -07:00
INodeCredentialTestResult ,
INodeExecutionData ,
INodeType ,
INodeTypeBaseDescription ,
INodeTypeDescription ,
} from 'n8n-workflow' ;
2024-08-29 06:55:53 -07:00
import { NodeConnectionType , NodeOperationError } from 'n8n-workflow' ;
2023-04-03 08:18:01 -07:00
import pgPromise from 'pg-promise' ;
import { pgInsertV2 , pgQueryV2 , pgUpdate , wrapData } from './genericFunctions' ;
2023-06-22 07:47:28 -07:00
import { oldVersionNotice } from '@utils/descriptions' ;
2023-05-02 01:45:04 -07:00
2023-04-03 08:18:01 -07:00
const versionDescription : INodeTypeDescription = {
displayName : 'Postgres' ,
name : 'postgres' ,
icon : 'file:postgres.svg' ,
group : [ 'input' ] ,
version : 1 ,
description : 'Get, add and update data in Postgres' ,
defaults : {
name : 'Postgres' ,
} ,
2024-08-29 06:55:53 -07:00
inputs : [ NodeConnectionType . Main ] ,
outputs : [ NodeConnectionType . Main ] ,
2023-04-03 08:18:01 -07:00
credentials : [
{
name : 'postgres' ,
required : true ,
testedBy : 'postgresConnectionTest' ,
} ,
] ,
properties : [
2023-05-02 01:45:04 -07:00
oldVersionNotice ,
2023-04-03 08:18:01 -07:00
{
displayName : 'Operation' ,
name : 'operation' ,
type : 'options' ,
noDataExpression : true ,
options : [
{
name : 'Execute Query' ,
value : 'executeQuery' ,
description : 'Execute an SQL query' ,
action : 'Execute a SQL query' ,
} ,
{
name : 'Insert' ,
value : 'insert' ,
description : 'Insert rows in database' ,
action : 'Insert rows in database' ,
} ,
{
name : 'Update' ,
value : 'update' ,
description : 'Update rows in database' ,
action : 'Update rows in database' ,
} ,
] ,
default : 'insert' ,
} ,
// ----------------------------------
// executeQuery
// ----------------------------------
{
displayName : 'Query' ,
name : 'query' ,
type : 'string' ,
2023-06-22 07:47:28 -07:00
noDataExpression : true ,
2023-04-25 09:18:27 -07:00
typeOptions : {
editor : 'sqlEditor' ,
2023-06-22 07:47:28 -07:00
sqlDialect : 'PostgreSQL' ,
2023-04-25 09:18:27 -07:00
} ,
2023-04-03 08:18:01 -07:00
displayOptions : {
show : {
operation : [ 'executeQuery' ] ,
} ,
} ,
default : '' ,
placeholder : 'SELECT id, name FROM product WHERE quantity > $1 AND price <= $2' ,
required : true ,
description :
'The SQL query to execute. You can use n8n expressions or $1 and $2 in conjunction with query parameters.' ,
} ,
// ----------------------------------
// insert
// ----------------------------------
{
displayName : 'Schema' ,
name : 'schema' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'insert' ] ,
} ,
} ,
default : 'public' ,
required : true ,
description : 'Name of the schema the table belongs to' ,
} ,
{
displayName : 'Table' ,
name : 'table' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'insert' ] ,
} ,
} ,
default : '' ,
required : true ,
description : 'Name of the table in which to insert data to' ,
} ,
{
displayName : 'Columns' ,
name : 'columns' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'insert' ] ,
} ,
} ,
default : '' ,
// eslint-disable-next-line n8n-nodes-base/node-param-placeholder-miscased-id
placeholder : 'id:int,name:text,description' ,
// eslint-disable-next-line n8n-nodes-base/node-param-description-miscased-id
description :
'Comma-separated list of the properties which should used as columns for the new rows. You can use type casting with colons (:) like id:int.' ,
} ,
// ----------------------------------
// update
// ----------------------------------
{
displayName : 'Schema' ,
name : 'schema' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'update' ] ,
} ,
} ,
default : 'public' ,
description : 'Name of the schema the table belongs to' ,
} ,
{
displayName : 'Table' ,
name : 'table' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'update' ] ,
} ,
} ,
default : '' ,
required : true ,
description : 'Name of the table in which to update data in' ,
} ,
{
displayName : 'Update Key' ,
name : 'updateKey' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'update' ] ,
} ,
} ,
default : 'id' ,
required : true ,
// eslint-disable-next-line n8n-nodes-base/node-param-description-miscased-id
description :
'Comma-separated list of the properties which decides which rows in the database should be updated. Normally that would be "id".' ,
} ,
{
displayName : 'Columns' ,
name : 'columns' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'update' ] ,
} ,
} ,
default : '' ,
placeholder : 'name:text,description' ,
// eslint-disable-next-line n8n-nodes-base/node-param-description-miscased-id
description :
'Comma-separated list of the properties which should used as columns for rows to update. You can use type casting with colons (:) like id:int.' ,
} ,
// ----------------------------------
// insert,update
// ----------------------------------
{
displayName : 'Return Fields' ,
name : 'returnFields' ,
type : 'string' ,
requiresDataPath : 'multiple' ,
displayOptions : {
show : {
operation : [ 'insert' , 'update' ] ,
} ,
} ,
default : '*' ,
description : 'Comma-separated list of the fields that the operation will return' ,
} ,
// ----------------------------------
// Additional fields
// ----------------------------------
{
displayName : 'Additional Fields' ,
name : 'additionalFields' ,
type : 'collection' ,
placeholder : 'Add Field' ,
default : { } ,
options : [
{
displayName : 'Mode' ,
name : 'mode' ,
type : 'options' ,
options : [
{
name : 'Independently' ,
value : 'independently' ,
description : 'Execute each query independently' ,
} ,
{
name : 'Multiple Queries' ,
value : 'multiple' ,
description : '<b>Default</b>. Sends multiple queries at once to database.' ,
} ,
{
name : 'Transaction' ,
value : 'transaction' ,
description : 'Executes all queries in a single transaction' ,
} ,
] ,
default : 'multiple' ,
description :
'The way queries should be sent to database. Can be used in conjunction with <b>Continue on Fail</b>. See <a href="https://docs.n8n.io/integrations/builtin/app-nodes/n8n-nodes-base.postgres/">the docs</a> for more examples' ,
} ,
{
displayName : 'Output Large-Format Numbers As' ,
name : 'largeNumbersOutput' ,
type : 'options' ,
options : [
{
name : 'Numbers' ,
value : 'numbers' ,
} ,
{
name : 'Text' ,
value : 'text' ,
description :
'Use this if you expect numbers longer than 16 digits (otherwise numbers may be incorrect)' ,
} ,
] ,
hint : 'Applies to NUMERIC and BIGINT columns only' ,
default : 'text' ,
} ,
{
displayName : 'Query Parameters' ,
name : 'queryParams' ,
type : 'string' ,
displayOptions : {
show : {
'/operation' : [ 'executeQuery' ] ,
} ,
} ,
default : '' ,
placeholder : 'quantity,price' ,
description :
'Comma-separated list of properties which should be used as query parameters' ,
} ,
] ,
} ,
] ,
} ;
export class PostgresV1 implements INodeType {
description : INodeTypeDescription ;
constructor ( baseDescription : INodeTypeBaseDescription ) {
this . description = {
. . . baseDescription ,
. . . versionDescription ,
} ;
}
methods = {
credentialTest : {
async postgresConnectionTest (
this : ICredentialTestFunctions ,
credential : ICredentialsDecrypted ,
) : Promise < INodeCredentialTestResult > {
const credentials = credential . data as IDataObject ;
try {
const pgp = pgPromise ( ) ;
const config : IDataObject = {
host : credentials.host as string ,
port : credentials.port as number ,
database : credentials.database as string ,
user : credentials.user as string ,
password : credentials.password as string ,
} ;
if ( credentials . allowUnauthorizedCerts === true ) {
config . ssl = {
rejectUnauthorized : false ,
} ;
} else {
config . ssl = ! [ 'disable' , undefined ] . includes ( credentials . ssl as string | undefined ) ;
config . sslmode = ( credentials . ssl as string ) || 'disable' ;
}
const db = pgp ( config ) ;
await db . connect ( ) ;
2023-10-12 02:10:14 -07:00
await db . $pool . end ( ) ;
2023-04-03 08:18:01 -07:00
} catch ( error ) {
return {
status : 'Error' ,
message : error.message ,
} ;
}
return {
status : 'OK' ,
message : 'Connection successful!' ,
} ;
} ,
} ,
} ;
async execute ( this : IExecuteFunctions ) : Promise < INodeExecutionData [ ] [ ] > {
const credentials = await this . getCredentials ( 'postgres' ) ;
const largeNumbersOutput = this . getNodeParameter (
'additionalFields.largeNumbersOutput' ,
0 ,
'' ,
) as string ;
const pgp = pgPromise ( ) ;
if ( largeNumbersOutput === 'numbers' ) {
pgp . pg . types . setTypeParser ( 20 , ( value : string ) = > {
return parseInt ( value , 10 ) ;
} ) ;
pgp . pg . types . setTypeParser ( 1700 , ( value : string ) = > {
return parseFloat ( value ) ;
} ) ;
}
const config : IDataObject = {
host : credentials.host as string ,
port : credentials.port as number ,
database : credentials.database as string ,
user : credentials.user as string ,
password : credentials.password as string ,
} ;
if ( credentials . allowUnauthorizedCerts === true ) {
config . ssl = {
rejectUnauthorized : false ,
} ;
} else {
config . ssl = ! [ 'disable' , undefined ] . includes ( credentials . ssl as string | undefined ) ;
config . sslmode = ( credentials . ssl as string ) || 'disable' ;
}
const db = pgp ( config ) ;
let returnItems : INodeExecutionData [ ] = [ ] ;
const items = this . getInputData ( ) ;
const operation = this . getNodeParameter ( 'operation' , 0 ) ;
if ( operation === 'executeQuery' ) {
// ----------------------------------
// executeQuery
// ----------------------------------
const queryResult = await pgQueryV2 . call ( this , pgp , db , items , this . continueOnFail ( ) ) ;
returnItems = queryResult as INodeExecutionData [ ] ;
} else if ( operation === 'insert' ) {
// ----------------------------------
// insert
// ----------------------------------
const insertData = await pgInsertV2 . call ( this , pgp , db , items , this . continueOnFail ( ) ) ;
// returnItems = this.helpers.returnJsonArray(insertData);
returnItems = insertData as INodeExecutionData [ ] ;
} else if ( operation === 'update' ) {
// ----------------------------------
// update
// ----------------------------------
const updateItems = await pgUpdate (
this . getNodeParameter ,
pgp ,
db ,
items ,
this . continueOnFail ( ) ,
) ;
returnItems = wrapData ( updateItems ) ;
} else {
2023-10-12 02:10:14 -07:00
await db . $pool . end ( ) ;
2023-04-03 08:18:01 -07:00
throw new NodeOperationError (
this . getNode ( ) ,
` The operation " ${ operation } " is not supported! ` ,
) ;
}
2023-10-12 02:10:14 -07:00
// shuts down the connection pool associated with the db object to allow the process to finish
await db . $pool . end ( ) ;
2023-04-03 08:18:01 -07:00
2023-09-05 03:59:02 -07:00
return [ returnItems ] ;
2023-04-03 08:18:01 -07:00
}
}