2021-03-11 13:08:41 +01:00
|
|
|
/*
|
|
|
|
This file is part of GNU Taler
|
|
|
|
(C) 2020 Taler Systems S.A.
|
|
|
|
|
|
|
|
GNU Taler is free software; you can redistribute it and/or modify it under the
|
|
|
|
terms of the GNU General Public License as published by the Free Software
|
|
|
|
Foundation; either version 3, or (at your option) any later version.
|
|
|
|
|
|
|
|
GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
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
|
|
|
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Imports.
|
|
|
|
*/
|
2021-03-17 17:56:37 +01:00
|
|
|
import { PreparePayResultType } from "@gnu-taler/taler-util";
|
2021-06-17 13:34:59 +02:00
|
|
|
import { WalletApiOperation } from "@gnu-taler/taler-wallet-core";
|
2021-04-07 19:29:51 +02:00
|
|
|
import { GlobalTestState, WalletCli, MerchantPrivateApi } from "./harness";
|
2021-03-11 13:08:41 +01:00
|
|
|
import {
|
|
|
|
createSimpleTestkudosEnvironment,
|
|
|
|
makeTestPayment,
|
|
|
|
withdrawViaBank,
|
|
|
|
} from "./helpers";
|
|
|
|
import { SyncService } from "./sync";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Run test for basic, bank-integrated withdrawal.
|
|
|
|
*/
|
|
|
|
export async function runWalletBackupDoublespendTest(t: GlobalTestState) {
|
|
|
|
// Set up test environment
|
|
|
|
|
|
|
|
const {
|
|
|
|
commonDb,
|
|
|
|
merchant,
|
|
|
|
wallet,
|
|
|
|
bank,
|
|
|
|
exchange,
|
|
|
|
} = await createSimpleTestkudosEnvironment(t);
|
|
|
|
|
|
|
|
const sync = await SyncService.create(t, {
|
|
|
|
currency: "TESTKUDOS",
|
|
|
|
annualFee: "TESTKUDOS:0.5",
|
|
|
|
database: commonDb.connStr,
|
|
|
|
fulfillmentUrl: "taler://fulfillment-success",
|
|
|
|
httpPort: 8089,
|
|
|
|
name: "sync1",
|
|
|
|
paymentBackendUrl: merchant.makeInstanceBaseUrl(),
|
|
|
|
uploadLimitMb: 10,
|
|
|
|
});
|
|
|
|
|
|
|
|
await sync.start();
|
|
|
|
await sync.pingUntilAvailable();
|
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
await wallet.client.call(WalletApiOperation.AddBackupProvider, {
|
2021-03-11 13:08:41 +01:00
|
|
|
backupProviderBaseUrl: sync.baseUrl,
|
|
|
|
activate: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
await withdrawViaBank(t, { wallet, bank, exchange, amount: "TESTKUDOS:10" });
|
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
await wallet.client.call(WalletApiOperation.RunBackupCycle, {});
|
2021-03-11 13:08:41 +01:00
|
|
|
await wallet.runUntilDone();
|
2021-06-17 13:34:59 +02:00
|
|
|
await wallet.client.call(WalletApiOperation.RunBackupCycle, {});
|
2021-03-11 13:08:41 +01:00
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
const backupRecovery = await wallet.client.call(
|
|
|
|
WalletApiOperation.ExportBackupRecovery,
|
|
|
|
{},
|
|
|
|
);
|
2021-03-11 13:08:41 +01:00
|
|
|
|
|
|
|
const wallet2 = new WalletCli(t, "wallet2");
|
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
await wallet2.client.call(WalletApiOperation.ImportBackupRecovery, {
|
|
|
|
recovery: backupRecovery,
|
|
|
|
});
|
2021-03-11 13:08:41 +01:00
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
await wallet2.client.call(WalletApiOperation.RunBackupCycle, {});
|
2021-03-11 13:08:41 +01:00
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
console.log(
|
|
|
|
"wallet1 balance before spend:",
|
|
|
|
await wallet.client.call(WalletApiOperation.GetBalances, {}),
|
|
|
|
);
|
2021-03-11 13:08:41 +01:00
|
|
|
|
|
|
|
await makeTestPayment(t, {
|
|
|
|
merchant,
|
|
|
|
wallet,
|
|
|
|
order: {
|
|
|
|
summary: "foo",
|
|
|
|
amount: "TESTKUDOS:7",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
await wallet.runUntilDone();
|
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
console.log(
|
|
|
|
"wallet1 balance after spend:",
|
|
|
|
await wallet.client.call(WalletApiOperation.GetBalances, {}),
|
|
|
|
);
|
2021-03-11 13:08:41 +01:00
|
|
|
|
|
|
|
{
|
2021-06-17 13:34:59 +02:00
|
|
|
console.log(
|
|
|
|
"wallet2 balance:",
|
|
|
|
await wallet2.client.call(WalletApiOperation.GetBalances, {}),
|
|
|
|
);
|
2021-03-11 13:08:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Now we double-spend with the second wallet
|
|
|
|
|
|
|
|
{
|
|
|
|
const instance = "default";
|
|
|
|
|
|
|
|
const orderResp = await MerchantPrivateApi.createOrder(merchant, instance, {
|
|
|
|
order: {
|
|
|
|
amount: "TESTKUDOS:8",
|
|
|
|
summary: "bla",
|
|
|
|
fulfillment_url: "taler://fulfillment-success",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
let orderStatus = await MerchantPrivateApi.queryPrivateOrderStatus(
|
|
|
|
merchant,
|
|
|
|
{
|
|
|
|
orderId: orderResp.order_id,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
t.assertTrue(orderStatus.order_status === "unpaid");
|
|
|
|
|
|
|
|
// Make wallet pay for the order
|
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
const preparePayResult = await wallet2.client.call(
|
|
|
|
WalletApiOperation.PreparePayForUri,
|
|
|
|
{
|
|
|
|
talerPayUri: orderStatus.taler_pay_uri,
|
|
|
|
},
|
|
|
|
);
|
2021-03-11 13:08:41 +01:00
|
|
|
|
|
|
|
t.assertTrue(
|
|
|
|
preparePayResult.status === PreparePayResultType.PaymentPossible,
|
|
|
|
);
|
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
const res = await wallet2.client.call(WalletApiOperation.ConfirmPay, {
|
2021-03-11 13:08:41 +01:00
|
|
|
proposalId: preparePayResult.proposalId,
|
|
|
|
});
|
|
|
|
|
|
|
|
console.log(res);
|
2021-04-07 19:29:51 +02:00
|
|
|
|
|
|
|
// FIXME: wait for a notification that indicates insufficient funds!
|
|
|
|
|
|
|
|
await withdrawViaBank(t, {
|
|
|
|
wallet: wallet2,
|
|
|
|
bank,
|
|
|
|
exchange,
|
|
|
|
amount: "TESTKUDOS:50",
|
|
|
|
});
|
|
|
|
|
2021-06-17 13:34:59 +02:00
|
|
|
const bal = await wallet2.client.call(WalletApiOperation.GetBalances, {});
|
2021-04-07 19:29:51 +02:00
|
|
|
console.log("bal", bal);
|
|
|
|
|
|
|
|
await wallet2.runUntilDone();
|
2021-03-11 13:08:41 +01:00
|
|
|
}
|
|
|
|
}
|
2021-05-21 11:47:20 +02:00
|
|
|
|
|
|
|
runWalletBackupDoublespendTest.suites = ["wallet", "wallet-backup"];
|