mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-11 21:07:28 -08:00
feat(Invoice Ninja Node): Add actions for bank transactions (#10389)
This commit is contained in:
parent
e0c0ddee59
commit
5a2c7e00a0
|
@ -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'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
|
@ -0,0 +1,9 @@
|
||||||
|
export interface IBankTransaction {
|
||||||
|
amount?: number;
|
||||||
|
bank_integration_id?: number;
|
||||||
|
base_type?: string;
|
||||||
|
date?: string;
|
||||||
|
description?: string;
|
||||||
|
id?: string;
|
||||||
|
paymentId?: string;
|
||||||
|
}
|
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue