2019-08-21 09:44:10 -07:00
import { IExecuteFunctions } from 'n8n-core' ;
import {
IDataObject ,
INodeExecutionData ,
INodeType ,
2021-04-16 09:33:36 -07:00
INodeTypeDescription ,
NodeOperationError ,
2019-08-21 09:44:10 -07:00
} from 'n8n-workflow' ;
2022-04-08 14:32:08 -07:00
import pgPromise from 'pg-promise' ;
2019-08-21 09:44:10 -07:00
2020-07-08 05:36:40 -07:00
import { pgInsert , pgQuery , pgUpdate } from './Postgres.node.functions' ;
2019-08-21 09:44:10 -07:00
export class Postgres implements INodeType {
description : INodeTypeDescription = {
displayName : 'Postgres' ,
name : 'postgres' ,
2021-06-12 12:00:37 -07:00
icon : 'file:postgres.svg' ,
2019-08-21 09:44:10 -07:00
group : [ 'input' ] ,
version : 1 ,
2021-07-03 05:40:16 -07:00
description : 'Get, add and update data in Postgres' ,
2019-08-21 09:44:10 -07:00
defaults : {
name : 'Postgres' ,
} ,
inputs : [ 'main' ] ,
outputs : [ 'main' ] ,
credentials : [
{
name : 'postgres' ,
2020-07-08 03:01:16 -07:00
required : true ,
} ,
2019-08-21 09:44:10 -07:00
] ,
properties : [
{
displayName : 'Operation' ,
name : 'operation' ,
type : 'options' ,
options : [
{
name : 'Execute Query' ,
value : 'executeQuery' ,
2020-07-09 05:33:05 -07:00
description : 'Execute an SQL query' ,
2019-08-21 09:44:10 -07:00
} ,
{
name : 'Insert' ,
value : 'insert' ,
2020-07-09 05:33:05 -07:00
description : 'Insert rows in database' ,
2019-08-21 09:44:10 -07:00
} ,
{
name : 'Update' ,
value : 'update' ,
2020-07-09 05:33:05 -07:00
description : 'Update rows in database' ,
2020-07-08 03:01:16 -07:00
} ,
2019-08-21 09:44:10 -07:00
] ,
default : 'insert' ,
2020-07-08 03:01:16 -07:00
description : 'The operation to perform.' ,
2019-08-21 09:44:10 -07:00
} ,
// ----------------------------------
// executeQuery
// ----------------------------------
{
displayName : 'Query' ,
name : 'query' ,
type : 'string' ,
typeOptions : {
2021-09-11 03:58:48 -07:00
alwaysOpenEditWindow : true ,
2019-08-21 09:44:10 -07:00
} ,
displayOptions : {
show : {
2020-07-08 03:01:16 -07:00
operation : [ 'executeQuery' ] ,
} ,
2019-08-21 09:44:10 -07:00
} ,
default : '' ,
2021-04-30 15:35:34 -07:00
placeholder : 'SELECT id, name FROM product WHERE quantity > $1 AND price <= $2' ,
2019-08-21 09:44:10 -07:00
required : true ,
2021-04-30 15:35:34 -07:00
description : 'The SQL query to execute. You can use n8n expressions or $1 and $2 in conjunction with query parameters.' ,
2019-08-21 09:44:10 -07:00
} ,
// ----------------------------------
// insert
// ----------------------------------
2020-03-17 16:49:21 -07:00
{
displayName : 'Schema' ,
name : 'schema' ,
type : 'string' ,
displayOptions : {
show : {
2020-07-08 03:01:16 -07:00
operation : [ 'insert' ] ,
} ,
2020-03-17 16:49:21 -07:00
} ,
default : 'public' ,
required : true ,
2020-07-08 03:01:16 -07:00
description : 'Name of the schema the table belongs to' ,
2020-03-17 16:49:21 -07:00
} ,
2019-08-21 09:44:10 -07:00
{
displayName : 'Table' ,
name : 'table' ,
type : 'string' ,
displayOptions : {
show : {
2020-07-08 03:01:16 -07:00
operation : [ 'insert' ] ,
} ,
2019-08-21 09:44:10 -07:00
} ,
default : '' ,
required : true ,
2020-07-08 03:01:16 -07:00
description : 'Name of the table in which to insert data to.' ,
2019-08-21 09:44:10 -07:00
} ,
{
displayName : 'Columns' ,
name : 'columns' ,
type : 'string' ,
displayOptions : {
show : {
2020-07-08 03:01:16 -07:00
operation : [ 'insert' ] ,
} ,
2019-08-21 09:44:10 -07:00
} ,
default : '' ,
2021-04-03 07:53:47 -07:00
placeholder : 'id:int,name:text,description' ,
2022-04-22 09:29:51 -07:00
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. ` ,
2019-08-21 09:44:10 -07:00
} ,
// ----------------------------------
// update
// ----------------------------------
2020-09-29 12:57:22 -07:00
{
displayName : 'Schema' ,
name : 'schema' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'update' ] ,
} ,
} ,
default : 'public' ,
2021-04-24 13:55:14 -07:00
required : false ,
2020-09-29 12:57:22 -07:00
description : 'Name of the schema the table belongs to' ,
} ,
2019-08-21 09:44:10 -07:00
{
displayName : 'Table' ,
name : 'table' ,
type : 'string' ,
displayOptions : {
show : {
2020-07-08 03:01:16 -07:00
operation : [ 'update' ] ,
} ,
2019-08-21 09:44:10 -07:00
} ,
default : '' ,
required : true ,
2020-07-08 03:01:16 -07:00
description : 'Name of the table in which to update data in' ,
2019-08-21 09:44:10 -07:00
} ,
{
displayName : 'Update Key' ,
name : 'updateKey' ,
type : 'string' ,
displayOptions : {
show : {
2020-07-08 03:01:16 -07:00
operation : [ 'update' ] ,
} ,
2019-08-21 09:44:10 -07:00
} ,
default : 'id' ,
required : true ,
2022-04-22 09:29:51 -07:00
description : 'Comma-separated list of the properties which decides which rows in the database should be updated. Normally that would be "id".' ,
2019-08-21 09:44:10 -07:00
} ,
{
displayName : 'Columns' ,
name : 'columns' ,
type : 'string' ,
displayOptions : {
show : {
2020-07-08 03:01:16 -07:00
operation : [ 'update' ] ,
} ,
2019-08-21 09:44:10 -07:00
} ,
default : '' ,
2021-04-03 07:53:47 -07:00
placeholder : 'name:text,description' ,
2022-04-22 09:29:51 -07:00
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. ` ,
2020-07-08 03:01:16 -07:00
} ,
2021-04-24 13:55:14 -07:00
// ----------------------------------
// insert,update
// ----------------------------------
{
displayName : 'Return Fields' ,
name : 'returnFields' ,
type : 'string' ,
displayOptions : {
show : {
operation : [ 'insert' , 'update' ] ,
} ,
} ,
default : '*' ,
2022-04-22 09:29:51 -07:00
description : 'Comma-separated list of the fields that the operation will return' ,
2021-04-24 13:55:14 -07:00
} ,
// ----------------------------------
// 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' ,
2021-11-25 09:10:06 -08:00
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/nodes/n8n-nodes-base.postgres/">the docs</a> for more examples' ,
2021-04-24 13:55:14 -07:00
} ,
2021-04-30 15:35:34 -07:00
{
displayName : 'Query Parameters' ,
name : 'queryParams' ,
type : 'string' ,
displayOptions : {
show : {
'/operation' : [
'executeQuery' ,
] ,
} ,
} ,
default : '' ,
placeholder : 'quantity,price' ,
2022-04-22 09:29:51 -07:00
description : 'Comma-separated list of properties which should be used as query parameters.' ,
2021-04-30 15:35:34 -07:00
} ,
2021-04-24 13:55:14 -07:00
] ,
} ,
2020-07-08 03:01:16 -07:00
] ,
2019-08-21 09:44:10 -07:00
} ;
async execute ( this : IExecuteFunctions ) : Promise < INodeExecutionData [ ] [ ] > {
2021-08-20 09:57:30 -07:00
const credentials = await this . getCredentials ( 'postgres' ) ;
2019-08-21 09:44:10 -07:00
const pgp = pgPromise ( ) ;
2020-08-25 08:02:01 -07:00
const config : IDataObject = {
2020-01-10 01:21:45 -08:00
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 ,
} ;
2020-08-25 08:02:01 -07:00
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' ;
}
2020-01-10 01:21:45 -08:00
const db = pgp ( config ) ;
2019-08-21 09:44:10 -07:00
let returnItems = [ ] ;
const items = this . getInputData ( ) ;
const operation = this . getNodeParameter ( 'operation' , 0 ) as string ;
if ( operation === 'executeQuery' ) {
// ----------------------------------
// executeQuery
// ----------------------------------
2021-04-24 13:55:14 -07:00
const queryResult = await pgQuery ( this . getNodeParameter , pgp , db , items , this . continueOnFail ( ) ) ;
2019-08-21 09:44:10 -07:00
2021-04-24 13:55:14 -07:00
returnItems = this . helpers . returnJsonArray ( queryResult ) ;
2019-08-21 09:44:10 -07:00
} else if ( operation === 'insert' ) {
// ----------------------------------
// insert
// ----------------------------------
2021-04-24 13:55:14 -07:00
const insertData = await pgInsert ( this . getNodeParameter , pgp , db , items , this . continueOnFail ( ) ) ;
2019-08-21 09:44:10 -07:00
for ( let i = 0 ; i < insertData . length ; i ++ ) {
returnItems . push ( {
2021-04-24 13:55:14 -07:00
json : insertData [ i ] ,
2019-08-21 09:44:10 -07:00
} ) ;
}
} else if ( operation === 'update' ) {
// ----------------------------------
// update
// ----------------------------------
2021-04-24 13:55:14 -07:00
const updateItems = await pgUpdate ( this . getNodeParameter , pgp , db , items , this . continueOnFail ( ) ) ;
2019-08-21 09:44:10 -07:00
2020-07-08 05:36:40 -07:00
returnItems = this . helpers . returnJsonArray ( updateItems ) ;
2019-08-21 09:44:10 -07:00
} else {
2020-04-07 23:43:16 -07:00
await pgp . end ( ) ;
2021-04-16 09:33:36 -07:00
throw new NodeOperationError ( this . getNode ( ) , ` The operation " ${ operation } " is not supported! ` ) ;
2019-08-21 09:44:10 -07:00
}
2019-10-31 09:43:30 -07:00
// Close the connection
await pgp . end ( ) ;
2019-08-21 09:44:10 -07:00
return this . prepareOutputData ( returnItems ) ;
}
}