feat(Invoice Ninja Node): Add actions for bank transactions (#10389)

This commit is contained in:
CodeShakingSheep 2024-09-17 04:42:05 -05:00 committed by GitHub
parent e0c0ddee59
commit 5a2c7e00a0
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 398 additions and 3 deletions

View file

@ -0,0 +1,213 @@
import type { INodeProperties } from 'n8n-workflow';
export const bankTransactionOperations: INodeProperties[] = [
{
displayName: 'Operation',
name: 'operation',
type: 'options',
noDataExpression: true,
displayOptions: {
show: {
resource: ['bank_transaction'],
},
},
options: [
{
name: 'Create',
value: 'create',
description: 'Create a new bank transaction',
action: 'Create a bank transaction',
},
{
name: 'Delete',
value: 'delete',
description: 'Delete a bank transaction',
action: 'Delete a bank transaction',
},
{
name: 'Get',
value: 'get',
description: 'Get data of a bank transaction',
action: 'Get a bank transaction',
},
{
name: 'Get Many',
value: 'getAll',
description: 'Get data of many bank transactions',
action: 'Get many bank transactions',
},
{
name: 'Match Payment',
value: 'matchPayment',
description: 'Match payment to a bank transaction',
action: 'Match payment to a bank transaction',
},
],
default: 'create',
},
];
export const bankTransactionFields: INodeProperties[] = [
/* -------------------------------------------------------------------------- */
/* bankTransaction:create */
/* -------------------------------------------------------------------------- */
{
displayName: 'Additional Fields',
name: 'additionalFields',
type: 'collection',
placeholder: 'Add Field',
default: {},
displayOptions: {
show: {
operation: ['create'],
resource: ['bank_transaction'],
},
},
options: [
{
displayName: 'Amount',
name: 'amount',
type: 'number',
default: 0,
},
{
displayName: 'Bank Integration Name or ID',
name: 'bankIntegrationId',
type: 'options',
description:
'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code-examples/expressions/">expression</a>',
typeOptions: {
loadOptionsMethod: 'getBankIntegrations',
},
default: '',
},
{
displayName: 'Base Type',
name: 'baseType',
type: 'options',
options: [
{
name: 'Deposit',
value: 'CREDIT',
},
{
name: 'Withdrawal',
value: 'DEBIT',
},
],
default: '',
},
{
displayName: 'Date',
name: 'date',
type: 'dateTime',
default: '',
},
{
displayName: 'Description',
name: 'description',
type: 'string',
default: '',
},
],
},
/* -------------------------------------------------------------------------- */
/* bankTransaction:delete */
/* -------------------------------------------------------------------------- */
{
displayName: 'Bank Transaction ID',
name: 'bankTransactionId',
type: 'string',
default: '',
required: true,
displayOptions: {
show: {
resource: ['bank_transaction'],
operation: ['delete'],
},
},
},
/* -------------------------------------------------------------------------- */
/* bankTransaction:get */
/* -------------------------------------------------------------------------- */
{
displayName: 'Bank Transaction ID',
name: 'bankTransactionId',
type: 'string',
default: '',
required: true,
displayOptions: {
show: {
resource: ['bank_transaction'],
operation: ['get'],
},
},
},
/* -------------------------------------------------------------------------- */
/* bankTransaction:getAll */
/* -------------------------------------------------------------------------- */
{
displayName: 'Return All',
name: 'returnAll',
type: 'boolean',
displayOptions: {
show: {
resource: ['bank_transaction'],
operation: ['getAll'],
},
},
default: false,
description: 'Whether to return all results or only up to a given limit',
},
{
displayName: 'Limit',
name: 'limit',
type: 'number',
displayOptions: {
show: {
resource: ['bank_transaction'],
operation: ['getAll'],
returnAll: [false],
},
},
typeOptions: {
minValue: 1,
maxValue: 60,
},
default: 50,
description: 'Max number of results to return',
},
/* -------------------------------------------------------------------------- */
/* bankTransaction:matchPayment */
/* -------------------------------------------------------------------------- */
{
displayName: 'Bank Transaction ID',
name: 'bankTransactionId',
type: 'string',
default: '',
required: true,
displayOptions: {
show: {
resource: ['bank_transaction'],
operation: ['matchPayment'],
},
},
},
{
displayName: 'Payment Name or ID',
name: 'paymentId',
type: 'options',
description:
'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code-examples/expressions/">expression</a>',
typeOptions: {
loadOptionsMethod: 'getPayments',
},
default: '',
displayOptions: {
show: {
resource: ['bank_transaction'],
operation: ['matchPayment'],
},
},
},
];

View file

@ -0,0 +1,9 @@
export interface IBankTransaction {
amount?: number;
bank_integration_id?: number;
base_type?: string;
date?: string;
description?: string;
id?: string;
paymentId?: string;
}

View file

@ -36,6 +36,10 @@ import { quoteFields, quoteOperations } from './QuoteDescription';
import type { IQuote } from './QuoteInterface'; import type { IQuote } from './QuoteInterface';
import { isoCountryCodes } from '@utils/ISOCountryCodes'; import { isoCountryCodes } from '@utils/ISOCountryCodes';
import { bankTransactionFields, bankTransactionOperations } from './BankTransactionDescription';
import type { IBankTransaction } from './BankTransactionInterface';
export class InvoiceNinja implements INodeType { export class InvoiceNinja implements INodeType {
description: INodeTypeDescription = { description: INodeTypeDescription = {
displayName: 'Invoice Ninja', displayName: 'Invoice Ninja',
@ -107,6 +111,15 @@ export class InvoiceNinja implements INodeType {
type: 'options', type: 'options',
noDataExpression: true, noDataExpression: true,
options: [ options: [
{
name: 'Bank Transaction',
value: 'bank_transaction',
displayOptions: {
show: {
apiVersion: ['v5'],
},
},
},
{ {
name: 'Client', name: 'Client',
value: 'client', value: 'client',
@ -146,6 +159,8 @@ export class InvoiceNinja implements INodeType {
...expenseFields, ...expenseFields,
...quoteOperations, ...quoteOperations,
...quoteFields, ...quoteFields,
...bankTransactionOperations,
...bankTransactionFields,
], ],
}; };
@ -255,6 +270,58 @@ export class InvoiceNinja implements INodeType {
} }
return returnData; return returnData;
}, },
// Get all the available bank integrations to display them to user so that they can
// select them easily
async getBankIntegrations(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
const returnData: INodePropertyOptions[] = [];
let banks = await invoiceNinjaApiRequestAllItems.call(
this,
'data',
'GET',
'/bank_integrations',
);
banks = banks.filter((e) => !e.is_deleted);
for (const bank of banks) {
const providerName = bank.provider_name as string;
const accountName = bank.bank_account_name as string;
const bankId = bank.id as string;
returnData.push({
name:
providerName != accountName
? `${providerName} - ${accountName}`
: accountName || providerName,
value: bankId,
});
}
return returnData;
},
// Get all the available users to display them to user so that they can
// select them easily
async getPayments(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
const returnData: INodePropertyOptions[] = [];
const qs: IDataObject = {};
// Only select payments that can be matched to transactions
qs.match_transactions = true;
const payments = await invoiceNinjaApiRequestAllItems.call(
this,
'data',
'GET',
'/payments',
{},
qs,
);
for (const payment of payments) {
const paymentName = [payment.number, payment.date, payment.amount]
.filter((e) => e)
.join(' - ');
const paymentId = payment.id as string;
returnData.push({
name: paymentName,
value: paymentId,
});
}
return returnData;
},
}, },
}; };
@ -858,6 +925,106 @@ export class InvoiceNinja implements INodeType {
responseData = responseData.data; responseData = responseData.data;
} }
} }
if (resource === 'bank_transaction') {
const resourceEndpoint = '/bank_transactions';
if (operation === 'create') {
const additionalFields = this.getNodeParameter('additionalFields', i);
const body: IBankTransaction = {};
if (additionalFields.amount) {
body.amount = additionalFields.amount as number;
}
if (additionalFields.baseType) {
body.base_type = additionalFields.baseType as string;
}
if (additionalFields.bankIntegrationId) {
body.bank_integration_id = additionalFields.bankIntegrationId as number;
}
if (additionalFields.client) {
body.date = additionalFields.date as string;
}
if (additionalFields.email) {
body.description = additionalFields.description as string;
}
responseData = await invoiceNinjaApiRequest.call(
this,
'POST',
resourceEndpoint,
body as IDataObject,
);
responseData = responseData.data;
}
if (operation === 'get') {
const bankTransactionId = this.getNodeParameter('bankTransactionId', i) as string;
const options = this.getNodeParameter('options', i);
if (options.include) {
qs.include = options.include as string;
}
responseData = await invoiceNinjaApiRequest.call(
this,
'GET',
`${resourceEndpoint}/${bankTransactionId}`,
{},
qs,
);
responseData = responseData.data;
}
if (operation === 'getAll') {
const returnAll = this.getNodeParameter('returnAll', 0);
const options = this.getNodeParameter('options', i);
if (options.include) {
qs.include = options.include as string;
}
if (options.invoiceNumber) {
qs.invoice_number = options.invoiceNumber as string;
}
if (returnAll) {
responseData = await invoiceNinjaApiRequestAllItems.call(
this,
'data',
'GET',
resourceEndpoint,
{},
qs,
);
} else {
qs.per_page = this.getNodeParameter('limit', 0);
responseData = await invoiceNinjaApiRequest.call(
this,
'GET',
resourceEndpoint,
{},
qs,
);
responseData = responseData.data;
}
}
if (operation === 'delete') {
const bankTransactionId = this.getNodeParameter('bankTransactionId', i) as string;
responseData = await invoiceNinjaApiRequest.call(
this,
'DELETE',
`${resourceEndpoint}/${bankTransactionId}`,
);
responseData = responseData.data;
}
if (operation === 'matchPayment') {
const bankTransactionId = this.getNodeParameter('bankTransactionId', i) as string;
const paymentId = this.getNodeParameter('paymentId', i) as string;
const body: IBankTransaction = {};
if (bankTransactionId) {
body.id = bankTransactionId as string;
}
if (paymentId) {
body.paymentId = paymentId as string;
}
responseData = await invoiceNinjaApiRequest.call(
this,
'POST',
`${resourceEndpoint}/match`,
body as IDataObject,
);
}
}
if (resource === 'quote') { if (resource === 'quote') {
const resourceEndpoint = apiVersion === 'v4' ? '/invoices' : '/quotes'; const resourceEndpoint = apiVersion === 'v4' ? '/invoices' : '/quotes';
if (operation === 'create') { if (operation === 'create') {
@ -983,7 +1150,7 @@ export class InvoiceNinja implements INodeType {
responseData = await invoiceNinjaApiRequest.call( responseData = await invoiceNinjaApiRequest.call(
this, this,
'GET', 'GET',
`/quotes/${quoteId}/email`, `${resourceEndpoint}/${quoteId}/email`,
); );
} }
} }
@ -1016,13 +1183,19 @@ export class InvoiceNinja implements INodeType {
this, this,
'data', 'data',
'GET', 'GET',
'/quotes', resourceEndpoint,
{}, {},
qs, qs,
); );
} else { } else {
qs.per_page = this.getNodeParameter('limit', 0); qs.per_page = this.getNodeParameter('limit', 0);
responseData = await invoiceNinjaApiRequest.call(this, 'GET', '/quotes', {}, qs); responseData = await invoiceNinjaApiRequest.call(
this,
'GET',
resourceEndpoint,
{},
qs,
);
responseData = responseData.data; responseData = responseData.data;
} }
} }