Skip to content

Commit

Permalink
Add GoCardless bank integration for ING_INGDDEFF (actualbudget#294)
Browse files Browse the repository at this point in the history
including some test data

Co-authored-by: Matiss Janis Aboltins <[email protected]>
  • Loading branch information
2 people authored and MMichotte committed Sep 9, 2024
1 parent 4b1bdfe commit ed35b16
Show file tree
Hide file tree
Showing 4 changed files with 364 additions and 0 deletions.
2 changes: 2 additions & 0 deletions src/app-gocardless/bank-factory.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import AmericanExpressAesudef1 from './banks/american-express-aesudef1.js';
import IngIngddeff from './banks/ing-ingddeff.js';
import IngPlIngbplpw from './banks/ing-pl-ingbplpw.js';
import IntegrationBank from './banks/integration-bank.js';
import MbankRetailBrexplpw from './banks/mbank-retail-brexplpw.js';
Expand All @@ -11,6 +12,7 @@ import Belfius from './banks/belfius_gkccbebb.js';

const banks = [
AmericanExpressAesudef1,
IngIngddeff,
IngPlIngbplpw,
MbankRetailBrexplpw,
SandboxfinanceSfin0000,
Expand Down
56 changes: 56 additions & 0 deletions src/app-gocardless/banks/ing-ingddeff.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
import { printIban, amountToInteger } from '../utils.js';

/** @type {import('./bank.interface.js').IBank} */
export default {
institutionIds: ['ING_INGDDEFF'],

normalizeAccount(account) {
return {
account_id: account.id,
institution: account.institution,
mask: account.iban.slice(-4),
iban: account.iban,
name: [account.product, printIban(account)].join(' '),
official_name: account.product,
type: 'checking',
};
},

normalizeTransaction(transaction, _booked) {
const remittanceInformationMatch = /remittanceinformation:(.*)$/.exec(
transaction.remittanceInformationUnstructured,
);
const remittanceInformation = remittanceInformationMatch
? remittanceInformationMatch[1]
: transaction.remittanceInformationUnstructured;

return {
...transaction,
remittanceInformationUnstructured: remittanceInformation,
date: transaction.bookingDate || transaction.valueDate,
};
},

sortTransactions(transactions = []) {
return transactions.sort((a, b) => {
const diff =
+new Date(b.valueDate || b.bookingDate) -
+new Date(a.valueDate || a.bookingDate);
if (diff) return diff;
const idA = parseInt(a.transactionId);
const idB = parseInt(b.transactionId);
if (!isNaN(idA) && !isNaN(idB)) return idB - idA;
return 0;
});
},

calculateStartingBalance(sortedTransactions = [], balances = []) {
const currentBalance = balances.find(
(balance) => 'interimBooked' === balance.balanceType,
);

return sortedTransactions.reduce((total, trans) => {
return total - amountToInteger(trans.transactionAmount.amount);
}, amountToInteger(currentBalance.balanceAmount.amount));
},
};
300 changes: 300 additions & 0 deletions src/app-gocardless/banks/tests/ing-ingddeff.spec.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,300 @@
import IngIngddeff from '../ing-ingddeff.js';

describe('IngIngddeff', () => {
describe('#normalizeAccount', () => {
/** @type {import('../../gocardless.types.js').DetailedAccountWithInstitution} */
const accountRaw = {
resourceId: 'e896eec6-6096-4efc-a941-756bd9d74765',
iban: 'DE02500105170137075030',
currency: 'EUR',
ownerName: 'Jane Doe',
product: 'Girokonto',
id: 'a787ba27-02ee-4fd6-be86-73831adc5498',
created: '2023-12-29T14:17:11.630352Z',
last_accessed: '2023-12-29T14:19:42.709478Z',
institution_id: 'ING_INGDDEFF',
status: 'READY',
owner_name: 'Jane Doe',
institution: {
id: 'ING_INGDDEFF',
name: 'ING',
bic: 'INGDDEFFXXX',
transaction_total_days: '390',
countries: ['DE'],
logo: 'https://storage.googleapis.com/gc-prd-institution_icons-production/DE/PNG/ing.png',
supported_payments: {
'single-payment': ['SCT'],
},
supported_features: [
'account_selection',
'business_accounts',
'corporate_accounts',
'payments',
'pending_transactions',
'private_accounts',
],
/*identification_codes: [],*/
},
};

it('returns normalized account data returned to Frontend', () => {
expect(IngIngddeff.normalizeAccount(accountRaw)).toEqual({
account_id: 'a787ba27-02ee-4fd6-be86-73831adc5498',
iban: 'DE02500105170137075030',
institution: {
bic: 'INGDDEFFXXX',
countries: ['DE'],
id: 'ING_INGDDEFF',
logo: 'https://storage.googleapis.com/gc-prd-institution_icons-production/DE/PNG/ing.png',
name: 'ING',
supported_features: [
'account_selection',
'business_accounts',
'corporate_accounts',
'payments',
'pending_transactions',
'private_accounts',
],
supported_payments: {
'single-payment': ['SCT'],
},
transaction_total_days: '390',
},
mask: '5030',
name: 'Girokonto (XXX 5030)',
official_name: 'Girokonto',
type: 'checking',
});
});
});

const transactionsRaw = [
{
transactionId: '000010348081381',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-29',
valueDate: '2023-12-29',
transactionAmount: {
amount: '-4.00',
currency: 'EUR',
},
creditorName: 'VISA XXXXXXXXXXXXXXXXXXXX ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:NR XXXX 63053 51590342815 KAUFUMSATZ 24.90 2311825 ARN044873748454374484719431 Google Pay ',
proprietaryBankTransactionCode: 'Lastschrifteinzug',
internalTransactionId: '085179a2e5fa34b0ff71b3f2c9f4876f',
date: '2023-12-29',
},
{
transactionId: '000010348081380',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-29',
valueDate: '2023-12-29',
transactionAmount: {
amount: '-2.00',
currency: 'EUR',
},
creditorName: 'VISA XXXXXXXXXXXXXXXXXXXX ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:NR XXXX 8987 90671935362 KAUFUMSATZ 94.81 929614 ARN54795476045598005130492 Google Pay ',
proprietaryBankTransactionCode: 'Lastschrifteinzug',
internalTransactionId: '0707bbe2de27e5aabfd5dc614c584951',
date: '2023-12-29',
},
{
transactionId: '000010348081379',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-29',
valueDate: '2023-12-29',
transactionAmount: {
amount: '-6.00',
currency: 'EUR',
},
creditorName: 'VISA XXXXXXXXXXXXXXXXXXXX ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:NR XXXX 2206 17679024325 KAUFUMSATZ 55.25 819456 ARN08595270353806495555431 Google Pay ',
proprietaryBankTransactionCode: 'Lastschrifteinzug',
internalTransactionId: '4b15b590652c9ebdc3f974591b15b250',
date: '2023-12-29',
},
{
transactionId: '000010348081378',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-29',
valueDate: '2023-12-29',
transactionAmount: {
amount: '-12.99',
currency: 'EUR',
},
creditorName: 'VISA XXXXXXXXXXXXXXXXXXXX ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:NR XXXX 9437 535-182-825 LU KAUFUMSATZ 43.79 665448 ARN86236748928277201384604 ',
proprietaryBankTransactionCode: 'Lastschrifteinzug',
internalTransactionId: 'f930f8c153f3e37fb9906e4b3a2b4552',
date: '2023-12-29',
},
{
transactionId: '000010348081377',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-29',
valueDate: '2023-12-29',
transactionAmount: {
amount: '-9.00',
currency: 'EUR',
},
creditorName: 'VISA XXXXXXXXXXXXXXXXXXXX ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:NR XXXX 3582 98236826123 KAUFUMSATZ 88.90 477561 ARN64452564252952225664357 Google Pay ',
proprietaryBankTransactionCode: 'Lastschrifteinzug',
internalTransactionId: '1ce866282deb78cc4ff4cd108e11b8cc',
date: '2023-12-29',
},
{
transactionId: '000010347374680',
endToEndId: '9212020-0900000070-2023121711315956',
bookingDate: '2023-12-29',
valueDate: '2023-12-29',
transactionAmount: {
amount: '2892.61',
currency: 'EUR',
},
debtorName: 'XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:F22685813 Gehalt 80/6586',
proprietaryBankTransactionCode: 'Gehalt/Rente',
internalTransactionId: 'e731d8eb47f1ae96ccc11e1fb8b76a60',
date: '2023-12-29',
},
{
transactionId: '000010336959253',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-28',
valueDate: '2023-12-28',
transactionAmount: {
amount: '-85.80',
currency: 'EUR',
},
creditorName: 'VISA XXXXXXXXXXXXXXXXXXXX ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:NR XXXX 7082 FAUCOGNEY E FR KAUFUMSATZ 38.20 265113 ARN47998616225906149245029 ',
proprietaryBankTransactionCode: 'Lastschrifteinzug',
internalTransactionId: '2bbc054ae7ba299482a7849fded864f3',
date: '2023-12-28',
},
{
transactionId: '000010350537843',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-29',
valueDate: '2023-12-27',
transactionAmount: {
amount: '2.79',
currency: 'EUR',
},
debtorName: ' ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation: Zins/Dividende ISIN IE36B9RBWM04 VANG.FTSE',
proprietaryBankTransactionCode: 'Zins / Dividende WP',
internalTransactionId: '3bb7c58199d3fa5a44e85871d9001798',
date: '2023-12-29',
},
{
transactionId: '000010341786083',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-28',
valueDate: '2023-12-27',
transactionAmount: {
amount: '79.80',
currency: 'EUR',
},
debtorName: 'VISA XXXXXXXXXXXXXXXXXXXX ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:NR XXXX 4619 GUTSCHRIFTSBELEG 03.91 134870 ',
proprietaryBankTransactionCode: 'Gutschrift',
internalTransactionId: '5570eefb7213e39153a6c7fb97d7dc6f',
date: '2023-12-28',
},
{
transactionId: '000010328399902',
endToEndId: 'NOTPROVIDED',
bookingDate: '2023-12-27',
valueDate: '2023-12-27',
transactionAmount: {
amount: '-10.90',
currency: 'EUR',
},
debtorName: 'VISA XXXXXXXXXXXXXXXXXXXX ',
remittanceInformationUnstructured:
'mandatereference:,creditorid:,remittanceinformation:NR XXXX 3465 XXXXXXXXX KAUFUMSATZ 90.40 505416 ARN63639757770303957985044 Google Pay ',
proprietaryBankTransactionCode: 'Lastschrifteinzug',
internalTransactionId: '1b1bf30b23afb56ba4d41b9c65cf0efa',
date: '2023-12-27',
},
];

describe('#sortTransactions', () => {
it('handles empty arrays', () => {
const transactions = [];
const sortedTransactions = IngIngddeff.sortTransactions(transactions);
expect(sortedTransactions).toEqual([]);
});

it('returns empty array for undefined input', () => {
const sortedTransactions = IngIngddeff.sortTransactions(undefined);
expect(sortedTransactions).toEqual([]);
});

it('returns sorted array for unsorted inputs', () => {
const normalizeTransactions = transactionsRaw.map((tx) =>
IngIngddeff.normalizeTransaction(tx, true),
);
const originalOrder = Array.from(normalizeTransactions);
const swap = (a, b) => {
const swap = normalizeTransactions[a];
normalizeTransactions[a] = normalizeTransactions[b];
normalizeTransactions[b] = swap;
};
swap(1, 4);
swap(3, 6);
swap(0, 7);
const sortedTransactions = IngIngddeff.sortTransactions(
normalizeTransactions,
);
expect(sortedTransactions).toEqual(originalOrder);
});
});

describe('#countStartingBalance', () => {
/** @type {import('../../gocardless-node.types.js').Balance[]} */
const balances = [
{
balanceAmount: { amount: '3596.87', currency: 'EUR' },
balanceType: 'interimBooked',
lastChangeDateTime: '2023-12-29T16:44:06.479Z',
},
];

it('should calculate the starting balance correctly', () => {
const normalizeTransactions = transactionsRaw.map((tx) =>
IngIngddeff.normalizeTransaction(tx, true),
);
const sortedTransactions = IngIngddeff.sortTransactions(
normalizeTransactions,
);

const startingBalance = IngIngddeff.calculateStartingBalance(
sortedTransactions,
balances,
);

expect(startingBalance).toEqual(75236);
});

it('returns the same balance amount when no transactions', () => {
const transactions = [];

expect(
IngIngddeff.calculateStartingBalance(transactions, balances),
).toEqual(359687);
});
});
});
6 changes: 6 additions & 0 deletions upcoming-release-notes/294.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
---
category: Enhancements
authors: [t4cmyk]
---

Add GoCardless integration for ING (Germany).

0 comments on commit ed35b16

Please sign in to comment.