2017-05-28 01:10:54 +02:00
|
|
|
/*
|
2021-03-15 13:43:53 +01:00
|
|
|
This file is part of GNU Taler
|
|
|
|
(C) 2021 Taler Systems S.A.
|
2017-05-28 01:10:54 +02:00
|
|
|
|
2021-03-15 13:43:53 +01:00
|
|
|
GNU Taler is free software; you can redistribute it and/or modify it under the
|
2017-05-28 01:10:54 +02:00
|
|
|
terms of the GNU General Public License as published by the Free Software
|
|
|
|
Foundation; either version 3, or (at your option) any later version.
|
|
|
|
|
2021-03-15 13:43:53 +01:00
|
|
|
GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY
|
2017-05-28 01:10:54 +02:00
|
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
|
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
2021-03-15 13:43:53 +01:00
|
|
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
2017-05-28 01:10:54 +02:00
|
|
|
*/
|
|
|
|
|
2021-03-15 13:43:53 +01:00
|
|
|
/**
|
|
|
|
* Imports.
|
|
|
|
*/
|
2021-06-15 18:52:43 +02:00
|
|
|
import test from "ava";
|
2021-03-17 17:56:37 +01:00
|
|
|
import { AmountJson, Amounts } from "@gnu-taler/taler-util";
|
2021-06-14 16:08:58 +02:00
|
|
|
import { AvailableCoinInfo, selectPayCoins } from "./coinSelection.js";
|
2017-05-28 01:10:54 +02:00
|
|
|
|
2018-01-03 14:42:06 +01:00
|
|
|
function a(x: string): AmountJson {
|
|
|
|
const amt = Amounts.parse(x);
|
2017-05-28 00:00:26 +02:00
|
|
|
if (!amt) {
|
|
|
|
throw Error("invalid amount");
|
|
|
|
}
|
|
|
|
return amt;
|
|
|
|
}
|
2016-11-14 03:37:47 +01:00
|
|
|
|
2020-03-30 12:22:04 +02:00
|
|
|
function fakeAci(current: string, feeDeposit: string): AvailableCoinInfo {
|
2017-05-28 00:00:26 +02:00
|
|
|
return {
|
2019-12-25 19:11:20 +01:00
|
|
|
availableAmount: a(current),
|
|
|
|
coinPub: "foobar",
|
|
|
|
denomPub: "foobar",
|
|
|
|
feeDeposit: a(feeDeposit),
|
2021-03-15 13:43:53 +01:00
|
|
|
exchangeBaseUrl: "https://example.com/",
|
2020-03-30 12:22:04 +02:00
|
|
|
};
|
2017-05-28 00:00:26 +02:00
|
|
|
}
|
|
|
|
|
2020-03-30 12:22:04 +02:00
|
|
|
test("coin selection 1", (t) => {
|
2019-12-25 19:11:20 +01:00
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.1"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.0"),
|
2017-05-28 00:00:26 +02:00
|
|
|
];
|
|
|
|
|
2021-03-15 13:43:53 +01:00
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:2.0"),
|
|
|
|
depositFeeLimit: a("EUR:0.1"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
|
|
|
|
2016-11-14 03:37:47 +01:00
|
|
|
if (!res) {
|
|
|
|
t.fail();
|
|
|
|
return;
|
|
|
|
}
|
2019-12-25 19:11:20 +01:00
|
|
|
t.true(res.coinPubs.length === 2);
|
2016-11-14 03:37:47 +01:00
|
|
|
t.pass();
|
|
|
|
});
|
|
|
|
|
2020-03-30 12:22:04 +02:00
|
|
|
test("coin selection 2", (t) => {
|
2019-12-25 19:11:20 +01:00
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.0"),
|
2017-05-28 00:00:26 +02:00
|
|
|
// Merchant covers the fee, this one shouldn't be used
|
2019-12-25 19:11:20 +01:00
|
|
|
fakeAci("EUR:1.0", "EUR:0.0"),
|
2017-05-28 00:00:26 +02:00
|
|
|
];
|
2021-03-15 13:43:53 +01:00
|
|
|
|
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:2.0"),
|
|
|
|
depositFeeLimit: a("EUR:0.5"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
|
|
|
|
2016-11-14 03:37:47 +01:00
|
|
|
if (!res) {
|
|
|
|
t.fail();
|
|
|
|
return;
|
|
|
|
}
|
2019-12-25 19:11:20 +01:00
|
|
|
t.true(res.coinPubs.length === 2);
|
2016-11-14 03:37:47 +01:00
|
|
|
t.pass();
|
|
|
|
});
|
|
|
|
|
2020-03-30 12:22:04 +02:00
|
|
|
test("coin selection 3", (t) => {
|
2019-12-25 19:11:20 +01:00
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
2017-05-28 00:00:26 +02:00
|
|
|
// this coin should be selected instead of previous one with fee
|
2019-12-25 19:11:20 +01:00
|
|
|
fakeAci("EUR:1.0", "EUR:0.0"),
|
2017-05-28 00:00:26 +02:00
|
|
|
];
|
2021-03-15 13:43:53 +01:00
|
|
|
|
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:2.0"),
|
|
|
|
depositFeeLimit: a("EUR:0.5"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
|
|
|
|
2016-11-14 03:37:47 +01:00
|
|
|
if (!res) {
|
|
|
|
t.fail();
|
|
|
|
return;
|
|
|
|
}
|
2019-12-25 19:11:20 +01:00
|
|
|
t.true(res.coinPubs.length === 2);
|
2016-11-14 03:37:47 +01:00
|
|
|
t.pass();
|
|
|
|
});
|
|
|
|
|
2020-03-30 12:22:04 +02:00
|
|
|
test("coin selection 4", (t) => {
|
2019-12-25 19:11:20 +01:00
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
2017-05-28 00:00:26 +02:00
|
|
|
];
|
2021-03-15 13:43:53 +01:00
|
|
|
|
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:2.0"),
|
|
|
|
depositFeeLimit: a("EUR:0.5"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
|
|
|
|
2016-11-14 03:37:47 +01:00
|
|
|
if (!res) {
|
|
|
|
t.fail();
|
|
|
|
return;
|
|
|
|
}
|
2019-12-25 19:11:20 +01:00
|
|
|
t.true(res.coinPubs.length === 3);
|
2016-11-14 03:37:47 +01:00
|
|
|
t.pass();
|
|
|
|
});
|
|
|
|
|
2020-03-30 12:22:04 +02:00
|
|
|
test("coin selection 5", (t) => {
|
2019-12-25 19:11:20 +01:00
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
2017-05-28 00:00:26 +02:00
|
|
|
];
|
2021-03-15 13:43:53 +01:00
|
|
|
|
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:4.0"),
|
|
|
|
depositFeeLimit: a("EUR:0.2"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
|
|
|
|
2017-05-27 19:20:27 +02:00
|
|
|
t.true(!res);
|
2016-11-14 03:37:47 +01:00
|
|
|
t.pass();
|
2017-05-28 00:00:26 +02:00
|
|
|
});
|
|
|
|
|
2020-03-30 12:22:04 +02:00
|
|
|
test("coin selection 6", (t) => {
|
2019-12-25 19:11:20 +01:00
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.5"),
|
2017-05-28 00:00:26 +02:00
|
|
|
];
|
2021-03-15 13:43:53 +01:00
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:2.0"),
|
|
|
|
depositFeeLimit: a("EUR:0.2"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
2017-05-28 00:00:26 +02:00
|
|
|
t.true(!res);
|
|
|
|
t.pass();
|
2016-11-14 03:37:47 +01:00
|
|
|
});
|
2021-03-27 20:48:44 +01:00
|
|
|
|
|
|
|
test("coin selection 7", (t) => {
|
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.1"),
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.1"),
|
|
|
|
];
|
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:2.0"),
|
|
|
|
depositFeeLimit: a("EUR:0.2"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
|
|
|
t.truthy(res);
|
|
|
|
t.true(Amounts.cmp(res!.customerDepositFees, "EUR:0.0") === 0);
|
|
|
|
t.true(
|
|
|
|
Amounts.cmp(Amounts.sum(res!.coinContributions).amount, "EUR:2.0") === 0,
|
|
|
|
);
|
|
|
|
t.pass();
|
|
|
|
});
|
|
|
|
|
|
|
|
test("coin selection 8", (t) => {
|
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.2"),
|
|
|
|
fakeAci("EUR:0.1", "EUR:0.2"),
|
|
|
|
fakeAci("EUR:0.05", "EUR:0.05"),
|
|
|
|
fakeAci("EUR:0.05", "EUR:0.05"),
|
|
|
|
];
|
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:1.1"),
|
|
|
|
depositFeeLimit: a("EUR:0.4"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
|
|
|
t.truthy(res);
|
|
|
|
t.true(res!.coinContributions.length === 3);
|
|
|
|
t.pass();
|
|
|
|
});
|
|
|
|
|
|
|
|
test("coin selection 9", (t) => {
|
|
|
|
const acis: AvailableCoinInfo[] = [
|
|
|
|
fakeAci("EUR:1.0", "EUR:0.2"),
|
|
|
|
fakeAci("EUR:0.2", "EUR:0.2"),
|
|
|
|
];
|
|
|
|
const res = selectPayCoins({
|
|
|
|
candidates: {
|
|
|
|
candidateCoins: acis,
|
|
|
|
wireFeesPerExchange: {},
|
|
|
|
},
|
|
|
|
contractTermsAmount: a("EUR:1.2"),
|
|
|
|
depositFeeLimit: a("EUR:0.4"),
|
|
|
|
wireFeeLimit: a("EUR:0"),
|
|
|
|
wireFeeAmortization: 1,
|
|
|
|
});
|
|
|
|
t.truthy(res);
|
|
|
|
t.true(res!.coinContributions.length === 2);
|
|
|
|
t.true(
|
|
|
|
Amounts.cmp(Amounts.sum(res!.coinContributions).amount, "EUR:1.2") === 0,
|
|
|
|
);
|
|
|
|
t.pass();
|
|
|
|
});
|