2015-12-25 22:42:14 +01:00
|
|
|
/*
|
|
|
|
This file is part of TALER
|
|
|
|
(C) 2015 GNUnet e.V.
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
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
|
2016-07-07 17:59:29 +02:00
|
|
|
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
2015-12-25 22:42:14 +01:00
|
|
|
*/
|
|
|
|
|
2016-01-05 15:42:46 +01:00
|
|
|
/**
|
|
|
|
* High-level wallet operations that should be indepentent from the underlying
|
|
|
|
* browser extension interface.
|
|
|
|
*/
|
|
|
|
|
2017-05-24 16:52:00 +02:00
|
|
|
/**
|
|
|
|
* Imports.
|
|
|
|
*/
|
2019-08-15 19:10:23 +02:00
|
|
|
import { CryptoApi, CryptoWorkerFactory } from "./crypto/cryptoApi";
|
2017-05-28 01:10:54 +02:00
|
|
|
import {
|
|
|
|
amountToPretty,
|
|
|
|
canonicalJson,
|
|
|
|
canonicalizeBaseUrl,
|
|
|
|
getTalerStampSec,
|
2018-01-04 11:35:04 +01:00
|
|
|
strcmp,
|
2019-11-20 19:48:43 +01:00
|
|
|
extractTalerStamp,
|
2017-05-28 01:10:54 +02:00
|
|
|
} from "./helpers";
|
2019-11-21 23:09:43 +01:00
|
|
|
import { HttpRequestLibrary } from "./http";
|
2017-06-04 20:16:09 +02:00
|
|
|
import * as LibtoolVersion from "./libtoolVersion";
|
2017-05-28 01:10:54 +02:00
|
|
|
import {
|
2019-11-21 23:09:43 +01:00
|
|
|
TransactionAbort,
|
2019-11-20 19:48:43 +01:00
|
|
|
oneShotPut,
|
|
|
|
oneShotGet,
|
|
|
|
runWithWriteTransaction,
|
|
|
|
oneShotIter,
|
|
|
|
oneShotIterIndex,
|
|
|
|
oneShotGetIndexed,
|
|
|
|
oneShotMutate,
|
2017-05-28 01:10:54 +02:00
|
|
|
} from "./query";
|
2018-01-03 14:42:06 +01:00
|
|
|
|
|
|
|
import { AmountJson } from "./amounts";
|
|
|
|
import * as Amounts from "./amounts";
|
|
|
|
|
2018-01-04 11:35:04 +01:00
|
|
|
import URI = require("urijs");
|
|
|
|
|
2016-05-24 01:53:56 +02:00
|
|
|
import {
|
2017-05-28 01:10:54 +02:00
|
|
|
CoinRecord,
|
|
|
|
CoinStatus,
|
2018-01-04 11:35:04 +01:00
|
|
|
CoinsReturnRecord,
|
2017-05-28 01:10:54 +02:00
|
|
|
CurrencyRecord,
|
|
|
|
DenominationRecord,
|
|
|
|
DenominationStatus,
|
2016-11-15 15:07:17 +01:00
|
|
|
ExchangeRecord,
|
|
|
|
PreCoinRecord,
|
2018-01-17 03:49:54 +01:00
|
|
|
ProposalDownloadRecord,
|
2017-08-30 17:08:54 +02:00
|
|
|
PurchaseRecord,
|
2017-12-09 03:37:21 +01:00
|
|
|
RefreshPreCoinRecord,
|
2016-11-15 15:07:17 +01:00
|
|
|
RefreshSessionRecord,
|
2018-01-03 14:42:06 +01:00
|
|
|
ReserveRecord,
|
2018-01-04 11:35:04 +01:00
|
|
|
Stores,
|
2018-01-03 14:42:06 +01:00
|
|
|
TipRecord,
|
|
|
|
WireFee,
|
2019-11-19 16:16:12 +01:00
|
|
|
WithdrawalRecord,
|
2019-11-20 19:48:43 +01:00
|
|
|
ExchangeDetails,
|
|
|
|
ExchangeUpdateStatus,
|
2019-11-21 23:09:43 +01:00
|
|
|
ReserveRecordStatus,
|
2018-01-03 14:42:06 +01:00
|
|
|
} from "./dbTypes";
|
|
|
|
import {
|
|
|
|
Auditor,
|
|
|
|
ContractTerms,
|
|
|
|
Denomination,
|
|
|
|
ExchangeHandle,
|
2019-05-08 04:53:26 +02:00
|
|
|
ExchangeWireJson,
|
2018-01-04 11:35:04 +01:00
|
|
|
KeysJson,
|
2018-01-29 16:41:17 +01:00
|
|
|
MerchantRefundPermission,
|
|
|
|
MerchantRefundResponse,
|
2018-01-03 14:42:06 +01:00
|
|
|
PayReq,
|
|
|
|
PaybackConfirmation,
|
2018-01-17 03:49:54 +01:00
|
|
|
Proposal,
|
2018-01-29 16:41:17 +01:00
|
|
|
RefundRequest,
|
2018-01-29 22:58:47 +01:00
|
|
|
ReserveStatus,
|
2018-01-03 14:42:06 +01:00
|
|
|
TipPlanchetDetail,
|
|
|
|
TipResponse,
|
2019-08-28 02:49:27 +02:00
|
|
|
WithdrawOperationStatusResponse,
|
2019-08-30 17:27:59 +02:00
|
|
|
TipPickupGetResponse,
|
2018-01-03 14:42:06 +01:00
|
|
|
} from "./talerTypes";
|
|
|
|
import {
|
2018-01-04 11:35:04 +01:00
|
|
|
Badge,
|
2018-09-20 02:56:13 +02:00
|
|
|
BenchmarkResult,
|
2018-01-03 14:42:06 +01:00
|
|
|
CoinSelectionResult,
|
|
|
|
CoinWithDenom,
|
|
|
|
ConfirmPayResult,
|
|
|
|
ConfirmReserveRequest,
|
|
|
|
CreateReserveRequest,
|
|
|
|
CreateReserveResponse,
|
2019-11-21 23:09:43 +01:00
|
|
|
HistoryEvent,
|
2018-01-18 02:50:18 +01:00
|
|
|
NextUrlResult,
|
2018-01-03 14:42:06 +01:00
|
|
|
Notifier,
|
|
|
|
PayCoinInfo,
|
2016-11-16 01:59:39 +01:00
|
|
|
ReserveCreationInfo,
|
2017-08-14 04:16:12 +02:00
|
|
|
ReturnCoinsRequest,
|
|
|
|
SenderWireInfos,
|
2017-11-30 04:07:36 +01:00
|
|
|
TipStatus,
|
2016-11-13 08:16:12 +01:00
|
|
|
WalletBalance,
|
|
|
|
WalletBalanceEntry,
|
2019-08-20 17:58:01 +02:00
|
|
|
PreparePayResult,
|
2019-08-28 02:49:27 +02:00
|
|
|
DownloadedWithdrawInfo,
|
2019-08-29 23:12:55 +02:00
|
|
|
WithdrawDetails,
|
|
|
|
AcceptWithdrawalResponse,
|
2019-08-31 13:27:12 +02:00
|
|
|
PurchaseDetails,
|
2019-11-19 16:16:12 +01:00
|
|
|
PendingOperationInfo,
|
|
|
|
PendingOperationsResponse,
|
|
|
|
HistoryQuery,
|
2019-11-20 19:48:43 +01:00
|
|
|
getTimestampNow,
|
|
|
|
OperationError,
|
2019-11-21 23:09:43 +01:00
|
|
|
Timestamp,
|
2018-01-03 14:42:06 +01:00
|
|
|
} from "./walletTypes";
|
2019-08-31 13:27:12 +02:00
|
|
|
import {
|
|
|
|
parsePayUri,
|
|
|
|
parseWithdrawUri,
|
|
|
|
parseTipUri,
|
|
|
|
parseRefundUri,
|
|
|
|
} from "./taleruri";
|
2019-11-02 00:22:55 +01:00
|
|
|
import { isFirefox } from "./webex/compat";
|
2019-11-21 23:09:43 +01:00
|
|
|
import { Logger } from "./logging";
|
2015-12-13 23:47:30 +01:00
|
|
|
|
2017-12-12 21:54:14 +01:00
|
|
|
interface SpeculativePayData {
|
|
|
|
payCoinInfo: PayCoinInfo;
|
|
|
|
exchangeUrl: string;
|
|
|
|
proposalId: number;
|
2018-01-17 03:49:54 +01:00
|
|
|
proposal: ProposalDownloadRecord;
|
2017-12-12 21:54:14 +01:00
|
|
|
}
|
|
|
|
|
2017-06-05 03:36:33 +02:00
|
|
|
/**
|
|
|
|
* Wallet protocol version spoken with the exchange
|
|
|
|
* and merchant.
|
|
|
|
*
|
|
|
|
* Uses libtool's current:revision:age versioning.
|
|
|
|
*/
|
2019-05-08 07:01:17 +02:00
|
|
|
export const WALLET_PROTOCOL_VERSION = "3:0:0";
|
2017-06-04 18:46:32 +02:00
|
|
|
|
2019-09-06 12:34:05 +02:00
|
|
|
const WALLET_CACHE_BREAKER_CLIENT_VERSION = "2";
|
2019-08-31 22:07:16 +02:00
|
|
|
|
2017-03-24 16:59:23 +01:00
|
|
|
const builtinCurrencies: CurrencyRecord[] = [
|
|
|
|
{
|
|
|
|
auditors: [
|
|
|
|
{
|
2017-06-04 19:41:43 +02:00
|
|
|
auditorPub: "BW9DC48PHQY4NH011SHHX36DZZ3Q22Y6X7FZ1VD1CMZ2PTFZ6PN0",
|
2017-04-28 23:28:27 +02:00
|
|
|
baseUrl: "https://auditor.demo.taler.net/",
|
2019-06-26 15:30:32 +02:00
|
|
|
expirationStamp: new Date(2027, 1).getTime(),
|
2017-03-24 16:59:23 +01:00
|
|
|
},
|
2017-04-12 17:47:14 +02:00
|
|
|
],
|
|
|
|
exchanges: [],
|
2017-05-28 01:10:54 +02:00
|
|
|
fractionalDigits: 2,
|
|
|
|
name: "KUDOS",
|
2017-03-24 16:59:23 +01:00
|
|
|
},
|
|
|
|
];
|
|
|
|
|
2016-11-16 01:59:39 +01:00
|
|
|
function isWithdrawableDenom(d: DenominationRecord) {
|
2019-06-26 15:30:32 +02:00
|
|
|
const nowSec = new Date().getTime() / 1000;
|
2017-05-28 01:10:54 +02:00
|
|
|
const stampWithdrawSec = getTalerStampSec(d.stampExpireWithdraw);
|
|
|
|
if (stampWithdrawSec === null) {
|
2017-04-20 03:09:25 +02:00
|
|
|
return false;
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
const stampStartSec = getTalerStampSec(d.stampStart);
|
|
|
|
if (stampStartSec === null) {
|
2017-04-20 03:09:25 +02:00
|
|
|
return false;
|
|
|
|
}
|
2016-02-10 02:03:31 +01:00
|
|
|
// Withdraw if still possible to withdraw within a minute
|
2019-06-26 15:30:32 +02:00
|
|
|
if (stampWithdrawSec + 60 > nowSec && nowSec >= stampStartSec) {
|
2016-02-10 02:03:31 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-08-30 17:08:54 +02:00
|
|
|
interface SelectPayCoinsResult {
|
|
|
|
cds: CoinWithDenom[];
|
|
|
|
totalFees: AmountJson;
|
|
|
|
}
|
|
|
|
|
2017-05-28 16:27:34 +02:00
|
|
|
/**
|
2017-08-30 17:08:54 +02:00
|
|
|
* Get the amount that we lose when refreshing a coin of the given denomination
|
|
|
|
* with a certain amount left.
|
|
|
|
*
|
|
|
|
* If the amount left is zero, then the refresh cost
|
|
|
|
* is also considered to be zero. If a refresh isn't possible (e.g. due to lack of
|
|
|
|
* the right denominations), then the cost is the full amount left.
|
|
|
|
*
|
|
|
|
* Considers refresh fees, withdrawal fees after refresh and amounts too small
|
|
|
|
* to refresh.
|
2017-05-28 16:27:34 +02:00
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
export function getTotalRefreshCost(
|
|
|
|
denoms: DenominationRecord[],
|
|
|
|
refreshedDenom: DenominationRecord,
|
|
|
|
amountLeft: AmountJson,
|
|
|
|
): AmountJson {
|
|
|
|
const withdrawAmount = Amounts.sub(amountLeft, refreshedDenom.feeRefresh)
|
|
|
|
.amount;
|
2017-08-30 17:08:54 +02:00
|
|
|
const withdrawDenoms = getWithdrawDenomList(withdrawAmount, denoms);
|
2019-06-26 15:30:32 +02:00
|
|
|
const resultingAmount = Amounts.add(
|
|
|
|
Amounts.getZero(withdrawAmount.currency),
|
|
|
|
...withdrawDenoms.map(d => d.value),
|
|
|
|
).amount;
|
2017-08-30 17:08:54 +02:00
|
|
|
const totalCost = Amounts.sub(amountLeft, resultingAmount).amount;
|
2019-08-28 02:49:27 +02:00
|
|
|
Wallet.enableTracing &&
|
|
|
|
console.log(
|
|
|
|
"total refresh cost for",
|
|
|
|
amountToPretty(amountLeft),
|
|
|
|
"is",
|
|
|
|
amountToPretty(totalCost),
|
|
|
|
);
|
2017-08-30 17:08:54 +02:00
|
|
|
return totalCost;
|
|
|
|
}
|
|
|
|
|
2017-05-28 16:27:34 +02:00
|
|
|
/**
|
|
|
|
* Select coins for a payment under the merchant's constraints.
|
2017-08-30 17:08:54 +02:00
|
|
|
*
|
|
|
|
* @param denoms all available denoms, used to compute refresh fees
|
2017-05-28 16:27:34 +02:00
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
export function selectPayCoins(
|
|
|
|
denoms: DenominationRecord[],
|
|
|
|
cds: CoinWithDenom[],
|
|
|
|
paymentAmount: AmountJson,
|
|
|
|
depositFeeLimit: AmountJson,
|
|
|
|
): SelectPayCoinsResult | undefined {
|
2017-05-28 01:10:54 +02:00
|
|
|
if (cds.length === 0) {
|
2016-11-14 03:01:42 +01:00
|
|
|
return undefined;
|
|
|
|
}
|
2017-08-30 17:08:54 +02:00
|
|
|
// Sort by ascending deposit fee and denomPub if deposit fee is the same
|
|
|
|
// (to guarantee deterministic results)
|
2019-06-26 15:30:32 +02:00
|
|
|
cds.sort(
|
|
|
|
(o1, o2) =>
|
|
|
|
Amounts.cmp(o1.denom.feeDeposit, o2.denom.feeDeposit) ||
|
|
|
|
strcmp(o1.denom.denomPub, o2.denom.denomPub),
|
|
|
|
);
|
2017-05-28 01:10:54 +02:00
|
|
|
const currency = cds[0].denom.value.currency;
|
|
|
|
const cdsResult: CoinWithDenom[] = [];
|
2017-08-30 17:08:54 +02:00
|
|
|
let accDepositFee: AmountJson = Amounts.getZero(currency);
|
2016-11-14 03:01:42 +01:00
|
|
|
let accAmount: AmountJson = Amounts.getZero(currency);
|
2019-06-26 15:30:32 +02:00
|
|
|
for (const { coin, denom } of cds) {
|
2016-11-18 00:09:43 +01:00
|
|
|
if (coin.suspended) {
|
|
|
|
continue;
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
if (coin.status !== CoinStatus.Fresh) {
|
2016-11-18 00:09:43 +01:00
|
|
|
continue;
|
|
|
|
}
|
2016-11-16 01:59:39 +01:00
|
|
|
if (Amounts.cmp(denom.feeDeposit, coin.currentAmount) >= 0) {
|
2016-11-14 03:01:42 +01:00
|
|
|
continue;
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
cdsResult.push({ coin, denom });
|
2017-08-30 17:08:54 +02:00
|
|
|
accDepositFee = Amounts.add(denom.feeDeposit, accDepositFee).amount;
|
2019-06-26 15:30:32 +02:00
|
|
|
let leftAmount = Amounts.sub(
|
|
|
|
coin.currentAmount,
|
|
|
|
Amounts.sub(paymentAmount, accAmount).amount,
|
|
|
|
).amount;
|
2016-11-14 03:01:42 +01:00
|
|
|
accAmount = Amounts.add(coin.currentAmount, accAmount).amount;
|
2017-08-30 17:08:54 +02:00
|
|
|
const coversAmount = Amounts.cmp(accAmount, paymentAmount) >= 0;
|
2019-06-26 15:30:32 +02:00
|
|
|
const coversAmountWithFee =
|
|
|
|
Amounts.cmp(
|
|
|
|
accAmount,
|
|
|
|
Amounts.add(paymentAmount, denom.feeDeposit).amount,
|
|
|
|
) >= 0;
|
2017-08-30 17:08:54 +02:00
|
|
|
const isBelowFee = Amounts.cmp(accDepositFee, depositFeeLimit) <= 0;
|
2017-08-14 04:16:12 +02:00
|
|
|
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing &&
|
|
|
|
console.log("candidate coin selection", {
|
|
|
|
coversAmount,
|
|
|
|
isBelowFee,
|
|
|
|
accDepositFee,
|
|
|
|
accAmount,
|
|
|
|
paymentAmount,
|
|
|
|
});
|
2017-08-27 05:57:39 +02:00
|
|
|
|
2016-11-14 03:01:42 +01:00
|
|
|
if ((coversAmount && isBelowFee) || coversAmountWithFee) {
|
2019-06-26 15:30:32 +02:00
|
|
|
const depositFeeToCover = Amounts.sub(accDepositFee, depositFeeLimit)
|
|
|
|
.amount;
|
2017-08-30 17:08:54 +02:00
|
|
|
leftAmount = Amounts.sub(leftAmount, depositFeeToCover).amount;
|
2019-08-28 02:49:27 +02:00
|
|
|
Wallet.enableTracing &&
|
|
|
|
console.log("deposit fee to cover", amountToPretty(depositFeeToCover));
|
2017-08-30 17:08:54 +02:00
|
|
|
|
|
|
|
let totalFees: AmountJson = Amounts.getZero(currency);
|
|
|
|
if (coversAmountWithFee && !isBelowFee) {
|
|
|
|
// these are the fees the customer has to pay
|
|
|
|
// because the merchant doesn't cover them
|
|
|
|
totalFees = Amounts.sub(depositFeeLimit, accDepositFee).amount;
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
totalFees = Amounts.add(
|
|
|
|
totalFees,
|
|
|
|
getTotalRefreshCost(denoms, denom, leftAmount),
|
|
|
|
).amount;
|
2017-08-30 17:08:54 +02:00
|
|
|
return { cds: cdsResult, totalFees };
|
2016-11-14 03:01:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2016-02-11 18:17:02 +01:00
|
|
|
/**
|
|
|
|
* Get a list of denominations (with repetitions possible)
|
|
|
|
* whose total value is as close as possible to the available
|
|
|
|
* amount, but never larger.
|
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
function getWithdrawDenomList(
|
|
|
|
amountAvailable: AmountJson,
|
|
|
|
denoms: DenominationRecord[],
|
|
|
|
): DenominationRecord[] {
|
2016-02-22 23:13:28 +01:00
|
|
|
let remaining = Amounts.copy(amountAvailable);
|
2016-11-16 01:59:39 +01:00
|
|
|
const ds: DenominationRecord[] = [];
|
2016-02-11 18:17:02 +01:00
|
|
|
|
|
|
|
denoms = denoms.filter(isWithdrawableDenom);
|
2016-02-22 23:13:28 +01:00
|
|
|
denoms.sort((d1, d2) => Amounts.cmp(d2.value, d1.value));
|
|
|
|
|
2016-02-11 18:17:02 +01:00
|
|
|
// This is an arbitrary number of coins
|
|
|
|
// we can withdraw in one go. It's not clear if this limit
|
|
|
|
// is useful ...
|
|
|
|
for (let i = 0; i < 1000; i++) {
|
|
|
|
let found = false;
|
2017-05-28 01:10:54 +02:00
|
|
|
for (const d of denoms) {
|
|
|
|
const cost = Amounts.add(d.value, d.feeWithdraw).amount;
|
2016-02-22 23:13:28 +01:00
|
|
|
if (Amounts.cmp(remaining, cost) < 0) {
|
2016-02-11 18:17:02 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
found = true;
|
2016-02-22 23:13:28 +01:00
|
|
|
remaining = Amounts.sub(remaining, cost).amount;
|
2016-02-11 18:17:02 +01:00
|
|
|
ds.push(d);
|
2016-02-22 23:13:28 +01:00
|
|
|
break;
|
2016-02-11 18:17:02 +01:00
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ds;
|
|
|
|
}
|
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
interface CoinsForPaymentArgs {
|
|
|
|
allowedAuditors: Auditor[];
|
|
|
|
allowedExchanges: ExchangeHandle[];
|
|
|
|
depositFeeLimit: AmountJson;
|
|
|
|
paymentAmount: AmountJson;
|
|
|
|
wireFeeAmortization: number;
|
|
|
|
wireFeeLimit: AmountJson;
|
2019-11-21 23:09:43 +01:00
|
|
|
wireFeeTime: Timestamp;
|
2017-05-28 01:10:54 +02:00
|
|
|
wireMethod: string;
|
2016-10-18 01:16:31 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
/**
|
|
|
|
* This error is thrown when an
|
|
|
|
*/
|
2019-11-21 23:09:43 +01:00
|
|
|
export class OperationFailedAndReportedError extends Error {
|
|
|
|
constructor(message: string) {
|
|
|
|
super(message);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
|
|
|
// Set the prototype explicitly.
|
|
|
|
Object.setPrototypeOf(this, OperationFailedAndReportedError.prototype);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const logger = new Logger("wallet.ts");
|
|
|
|
|
2017-05-28 16:27:34 +02:00
|
|
|
/**
|
|
|
|
* The platform-independent wallet implementation.
|
|
|
|
*/
|
2016-01-10 20:07:42 +01:00
|
|
|
export class Wallet {
|
2017-06-05 02:00:03 +02:00
|
|
|
/**
|
|
|
|
* IndexedDB database used by the wallet.
|
|
|
|
*/
|
|
|
|
db: IDBDatabase;
|
2019-08-18 23:06:27 +02:00
|
|
|
static enableTracing = false;
|
2016-01-06 15:39:22 +01:00
|
|
|
private http: HttpRequestLibrary;
|
|
|
|
private badge: Badge;
|
2016-02-18 23:41:29 +01:00
|
|
|
private notifier: Notifier;
|
2017-05-28 16:27:34 +02:00
|
|
|
private cryptoApi: CryptoApi;
|
2017-12-12 21:54:14 +01:00
|
|
|
private speculativePayData: SpeculativePayData | undefined;
|
2018-01-18 02:50:18 +01:00
|
|
|
private cachedNextUrl: { [fulfillmentUrl: string]: NextUrlResult } = {};
|
2016-02-11 18:17:02 +01:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
constructor(
|
|
|
|
db: IDBDatabase,
|
|
|
|
http: HttpRequestLibrary,
|
|
|
|
badge: Badge,
|
|
|
|
notifier: Notifier,
|
2019-08-15 19:10:23 +02:00
|
|
|
cryptoWorkerFactory: CryptoWorkerFactory,
|
2019-06-26 15:30:32 +02:00
|
|
|
) {
|
2016-01-06 15:39:22 +01:00
|
|
|
this.db = db;
|
|
|
|
this.http = http;
|
|
|
|
this.badge = badge;
|
2016-02-18 23:41:29 +01:00
|
|
|
this.notifier = notifier;
|
2019-08-15 19:10:23 +02:00
|
|
|
this.cryptoApi = new CryptoApi(cryptoWorkerFactory);
|
2019-11-20 19:48:43 +01:00
|
|
|
}
|
2017-06-05 02:00:03 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
/**
|
|
|
|
* Process pending operations.
|
|
|
|
*/
|
|
|
|
public async runPending(): Promise<void> {
|
|
|
|
// FIXME: maybe prioritize pending operations by their urgency?
|
2019-11-20 20:02:48 +01:00
|
|
|
const exchangeBaseUrlList = await oneShotIter(
|
|
|
|
this.db,
|
|
|
|
Stores.exchanges,
|
|
|
|
).map(x => x.baseUrl);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
|
|
|
for (let exchangeBaseUrl of exchangeBaseUrlList) {
|
|
|
|
await this.updateExchangeFromUrl(exchangeBaseUrl);
|
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
|
|
|
|
const reservesPubList = await oneShotIter(this.db, Stores.reserves).map(
|
|
|
|
x => x.reservePub,
|
|
|
|
);
|
|
|
|
|
|
|
|
for (let reservePub of reservesPubList) {
|
|
|
|
await this.processReserve(reservePub);
|
|
|
|
}
|
|
|
|
|
|
|
|
const refreshSessionList = await oneShotIter(this.db, Stores.refresh).map(
|
|
|
|
x => x.refreshSessionId,
|
|
|
|
);
|
|
|
|
for (let rs of refreshSessionList) {
|
|
|
|
await this.processRefreshSession(rs);
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-11-21 23:09:43 +01:00
|
|
|
* Process pending operations and wait for scheduled operations in
|
|
|
|
* a loop until the wallet is stopped explicitly.
|
2019-11-20 19:48:43 +01:00
|
|
|
*/
|
2019-11-21 23:09:43 +01:00
|
|
|
public async runUntilStopped(): Promise<void> {
|
|
|
|
throw Error("not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Run until all coins have been withdrawn from the given reserve,
|
|
|
|
* or an error has occured.
|
|
|
|
*/
|
|
|
|
public async runUntilReserveDepleted(reservePub: string) {
|
|
|
|
while (true) {
|
|
|
|
let reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
|
|
|
|
if (!reserve) {
|
|
|
|
throw Error("Reserve does not exist.");
|
|
|
|
}
|
|
|
|
if (reserve.lastError !== undefined) {
|
|
|
|
throw Error("Reserve error: " + reserve.lastError.message);
|
|
|
|
}
|
|
|
|
if (reserve.reserveStatus === ReserveRecordStatus.UNCONFIRMED) {
|
|
|
|
throw Error("Reserve is not confirmed.");
|
|
|
|
}
|
|
|
|
if (reserve.reserveStatus === ReserveRecordStatus.DORMANT) {
|
|
|
|
// Check if all withdraws are done!
|
|
|
|
const precoins = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.precoins.byReservePub,
|
|
|
|
reservePub,
|
|
|
|
).toArray();
|
|
|
|
for (const pc of precoins) {
|
|
|
|
await this.processPreCoin(pc.coinPub);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
await this.processReserve(reservePub);
|
|
|
|
}
|
2016-05-24 01:18:23 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
/**
|
|
|
|
* Insert the hard-coded defaults for exchanges, coins and
|
|
|
|
* auditors into the database, unless these defaults have
|
|
|
|
* already been applied.
|
|
|
|
*/
|
|
|
|
async fillDefaults() {
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.config, Stores.currencies],
|
|
|
|
async tx => {
|
2019-11-20 19:48:43 +01:00
|
|
|
let applied = false;
|
2019-11-20 20:02:48 +01:00
|
|
|
await tx.iter(Stores.config).forEach(x => {
|
2019-11-20 19:48:43 +01:00
|
|
|
if (x.key == "currencyDefaultsApplied" && x.value == true) {
|
|
|
|
applied = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (!applied) {
|
|
|
|
for (let c of builtinCurrencies) {
|
|
|
|
await tx.put(Stores.currencies, c);
|
|
|
|
}
|
|
|
|
}
|
2019-11-20 20:02:48 +01:00
|
|
|
},
|
|
|
|
);
|
2017-03-24 16:59:23 +01:00
|
|
|
}
|
|
|
|
|
2016-10-20 01:37:00 +02:00
|
|
|
async updateExchanges(): Promise<void> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const exchangeUrls = await oneShotIter(this.db, Stores.exchanges).map(
|
|
|
|
e => e.baseUrl,
|
|
|
|
);
|
2016-10-19 23:55:58 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
for (const url of exchangeUrls) {
|
2019-06-26 15:30:32 +02:00
|
|
|
this.updateExchangeFromUrl(url).catch(e => {
|
|
|
|
console.error("updating exchange failed", e);
|
|
|
|
});
|
2016-10-19 23:55:58 +02:00
|
|
|
}
|
2016-05-24 17:30:27 +02:00
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
private async getCoinsForReturn(
|
|
|
|
exchangeBaseUrl: string,
|
|
|
|
amount: AmountJson,
|
|
|
|
): Promise<CoinWithDenom[] | undefined> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const exchange = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.exchanges,
|
|
|
|
exchangeBaseUrl,
|
|
|
|
);
|
2017-08-14 04:16:12 +02:00
|
|
|
if (!exchange) {
|
|
|
|
throw Error(`Exchange ${exchangeBaseUrl} not known to the wallet`);
|
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const coins: CoinRecord[] = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.coins.exchangeBaseUrlIndex,
|
|
|
|
exchange.baseUrl,
|
|
|
|
).toArray();
|
2017-08-14 04:16:12 +02:00
|
|
|
|
|
|
|
if (!coins || !coins.length) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const denoms = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.denominations.exchangeBaseUrlIndex,
|
|
|
|
exchange.baseUrl,
|
|
|
|
).toArray();
|
2017-08-30 17:08:54 +02:00
|
|
|
|
2017-08-14 04:16:12 +02:00
|
|
|
// Denomination of the first coin, we assume that all other
|
|
|
|
// coins have the same currency
|
2019-11-20 19:48:43 +01:00
|
|
|
const firstDenom = await oneShotGet(this.db, Stores.denominations, [
|
2019-06-26 15:30:32 +02:00
|
|
|
exchange.baseUrl,
|
|
|
|
coins[0].denomPub,
|
|
|
|
]);
|
2017-08-14 04:16:12 +02:00
|
|
|
if (!firstDenom) {
|
|
|
|
throw Error("db inconsistent");
|
|
|
|
}
|
|
|
|
const currency = firstDenom.value.currency;
|
|
|
|
|
|
|
|
const cds: CoinWithDenom[] = [];
|
|
|
|
for (const coin of coins) {
|
2019-11-20 19:48:43 +01:00
|
|
|
const denom = await oneShotGet(this.db, Stores.denominations, [
|
2019-06-26 15:30:32 +02:00
|
|
|
exchange.baseUrl,
|
|
|
|
coin.denomPub,
|
|
|
|
]);
|
2017-08-14 04:16:12 +02:00
|
|
|
if (!denom) {
|
|
|
|
throw Error("db inconsistent");
|
|
|
|
}
|
|
|
|
if (denom.value.currency !== currency) {
|
2019-06-26 15:30:32 +02:00
|
|
|
console.warn(
|
2019-08-26 01:39:13 +02:00
|
|
|
`same pubkey for different currencies at exchange ${exchange.baseUrl}`,
|
2019-06-26 15:30:32 +02:00
|
|
|
);
|
2017-08-14 04:16:12 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (coin.suspended) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (coin.status !== CoinStatus.Fresh) {
|
|
|
|
continue;
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
cds.push({ coin, denom });
|
2017-08-14 04:16:12 +02:00
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
console.log("coin return: selecting from possible coins", { cds, amount });
|
2017-08-27 05:57:39 +02:00
|
|
|
|
2017-08-30 17:08:54 +02:00
|
|
|
const res = selectPayCoins(denoms, cds, amount, amount);
|
|
|
|
if (res) {
|
|
|
|
return res.cds;
|
|
|
|
}
|
2017-10-15 19:28:35 +02:00
|
|
|
return undefined;
|
2017-08-14 04:16:12 +02:00
|
|
|
}
|
|
|
|
|
2016-01-06 15:39:22 +01:00
|
|
|
/**
|
2019-08-15 23:34:08 +02:00
|
|
|
* Get exchanges and associated coins that are still spendable, but only
|
|
|
|
* if the sum the coins' remaining value covers the payment amount and fees.
|
2016-01-06 15:39:22 +01:00
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
private async getCoinsForPayment(
|
|
|
|
args: CoinsForPaymentArgs,
|
|
|
|
): Promise<CoinSelectionResult | undefined> {
|
2017-05-28 01:10:54 +02:00
|
|
|
const {
|
|
|
|
allowedAuditors,
|
|
|
|
allowedExchanges,
|
|
|
|
depositFeeLimit,
|
|
|
|
paymentAmount,
|
|
|
|
wireFeeAmortization,
|
|
|
|
wireFeeLimit,
|
|
|
|
wireFeeTime,
|
|
|
|
wireMethod,
|
|
|
|
} = args;
|
|
|
|
|
|
|
|
let remainingAmount = paymentAmount;
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchanges = await oneShotIter(this.db, Stores.exchanges).toArray();
|
2017-05-28 01:10:54 +02:00
|
|
|
|
|
|
|
for (const exchange of exchanges) {
|
2017-04-26 14:11:35 +02:00
|
|
|
let isOkay: boolean = false;
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchangeDetails = exchange.details;
|
|
|
|
if (!exchangeDetails) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const exchangeFees = exchange.wireInfo;
|
|
|
|
if (!exchangeFees) {
|
|
|
|
continue;
|
|
|
|
}
|
2017-04-26 14:11:35 +02:00
|
|
|
|
|
|
|
// is the exchange explicitly allowed?
|
2017-05-28 01:10:54 +02:00
|
|
|
for (const allowedExchange of allowedExchanges) {
|
2019-11-20 19:48:43 +01:00
|
|
|
if (allowedExchange.master_pub === exchangeDetails.masterPublicKey) {
|
2017-04-26 14:11:35 +02:00
|
|
|
isOkay = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// is the exchange allowed because of one of its auditors?
|
|
|
|
if (!isOkay) {
|
2017-05-28 01:10:54 +02:00
|
|
|
for (const allowedAuditor of allowedAuditors) {
|
2019-11-20 19:48:43 +01:00
|
|
|
for (const auditor of exchangeDetails.auditors) {
|
2017-05-28 01:10:54 +02:00
|
|
|
if (auditor.auditor_pub === allowedAuditor.auditor_pub) {
|
2017-04-26 14:11:35 +02:00
|
|
|
isOkay = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (isOkay) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isOkay) {
|
2016-11-14 02:52:29 +01:00
|
|
|
continue;
|
2016-05-24 17:30:27 +02:00
|
|
|
}
|
2017-04-26 14:11:35 +02:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const coins = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.coins.exchangeBaseUrlIndex,
|
|
|
|
exchange.baseUrl,
|
|
|
|
).toArray();
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const denoms = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.denominations.exchangeBaseUrlIndex,
|
|
|
|
exchange.baseUrl,
|
|
|
|
).toArray();
|
2019-08-15 23:34:08 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
if (!coins || coins.length === 0) {
|
2016-11-14 02:52:29 +01:00
|
|
|
continue;
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
2017-08-14 04:16:12 +02:00
|
|
|
|
2016-11-14 02:52:29 +01:00
|
|
|
// Denomination of the first coin, we assume that all other
|
|
|
|
// coins have the same currency
|
2019-11-20 19:48:43 +01:00
|
|
|
const firstDenom = await oneShotGet(this.db, Stores.denominations, [
|
2019-06-26 15:30:32 +02:00
|
|
|
exchange.baseUrl,
|
|
|
|
coins[0].denomPub,
|
|
|
|
]);
|
2016-11-14 02:52:29 +01:00
|
|
|
if (!firstDenom) {
|
|
|
|
throw Error("db inconsistent");
|
2016-02-19 13:03:45 +01:00
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
const currency = firstDenom.value.currency;
|
|
|
|
const cds: CoinWithDenom[] = [];
|
|
|
|
for (const coin of coins) {
|
2019-11-20 19:48:43 +01:00
|
|
|
const denom = await oneShotGet(this.db, Stores.denominations, [
|
2019-06-26 15:30:32 +02:00
|
|
|
exchange.baseUrl,
|
|
|
|
coin.denomPub,
|
|
|
|
]);
|
2016-11-14 02:52:29 +01:00
|
|
|
if (!denom) {
|
|
|
|
throw Error("db inconsistent");
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
if (denom.value.currency !== currency) {
|
2019-06-26 15:30:32 +02:00
|
|
|
console.warn(
|
2019-08-26 01:39:13 +02:00
|
|
|
`same pubkey for different currencies at exchange ${exchange.baseUrl}`,
|
2019-06-26 15:30:32 +02:00
|
|
|
);
|
2016-11-14 02:52:29 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (coin.suspended) {
|
|
|
|
continue;
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
if (coin.status !== CoinStatus.Fresh) {
|
2016-11-18 00:09:43 +01:00
|
|
|
continue;
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
cds.push({ coin, denom });
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
2016-03-05 01:36:38 +01:00
|
|
|
|
2017-08-30 17:08:54 +02:00
|
|
|
let totalFees = Amounts.getZero(currency);
|
2019-06-26 15:30:32 +02:00
|
|
|
let wireFee: AmountJson | undefined;
|
2019-11-20 19:48:43 +01:00
|
|
|
for (const fee of exchangeFees.feesForType[wireMethod] || []) {
|
2018-01-17 05:33:06 +01:00
|
|
|
if (fee.startStamp <= wireFeeTime && fee.endStamp >= wireFeeTime) {
|
2017-04-27 04:06:48 +02:00
|
|
|
wireFee = fee.wireFee;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wireFee) {
|
2017-05-28 01:10:54 +02:00
|
|
|
const amortizedWireFee = Amounts.divide(wireFee, wireFeeAmortization);
|
2017-04-27 04:06:48 +02:00
|
|
|
if (Amounts.cmp(wireFeeLimit, amortizedWireFee) < 0) {
|
2017-08-30 17:08:54 +02:00
|
|
|
totalFees = Amounts.add(amortizedWireFee, totalFees).amount;
|
2019-06-26 15:30:32 +02:00
|
|
|
remainingAmount = Amounts.add(amortizedWireFee, remainingAmount)
|
|
|
|
.amount;
|
2017-04-27 04:06:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-30 17:08:54 +02:00
|
|
|
const res = selectPayCoins(denoms, cds, remainingAmount, depositFeeLimit);
|
2019-08-15 23:34:08 +02:00
|
|
|
|
2016-11-14 03:01:42 +01:00
|
|
|
if (res) {
|
2017-08-30 17:08:54 +02:00
|
|
|
totalFees = Amounts.add(totalFees, res.totalFees).amount;
|
2016-11-14 03:01:42 +01:00
|
|
|
return {
|
2017-08-30 17:08:54 +02:00
|
|
|
cds: res.cds,
|
2017-05-28 01:10:54 +02:00
|
|
|
exchangeUrl: exchange.baseUrl,
|
2018-01-17 05:33:06 +01:00
|
|
|
totalAmount: remainingAmount,
|
2017-08-30 17:08:54 +02:00
|
|
|
totalFees,
|
2017-05-28 01:10:54 +02:00
|
|
|
};
|
2016-03-05 01:36:38 +01:00
|
|
|
}
|
2016-11-14 02:52:29 +01:00
|
|
|
}
|
|
|
|
return undefined;
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
2015-12-13 23:47:30 +01:00
|
|
|
|
2016-02-10 02:03:31 +01:00
|
|
|
/**
|
|
|
|
* Record all information that is necessary to
|
2017-06-01 18:46:07 +02:00
|
|
|
* pay for a proposal in the wallet's database.
|
2016-02-10 02:03:31 +01:00
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
private async recordConfirmPay(
|
|
|
|
proposal: ProposalDownloadRecord,
|
|
|
|
payCoinInfo: PayCoinInfo,
|
|
|
|
chosenExchange: string,
|
|
|
|
): Promise<PurchaseRecord> {
|
2017-05-28 01:10:54 +02:00
|
|
|
const payReq: PayReq = {
|
2017-12-12 21:54:14 +01:00
|
|
|
coins: payCoinInfo.sigs,
|
2017-05-31 16:04:14 +02:00
|
|
|
merchant_pub: proposal.contractTerms.merchant_pub,
|
2018-01-04 13:22:23 +01:00
|
|
|
mode: "pay",
|
2017-05-31 16:04:14 +02:00
|
|
|
order_id: proposal.contractTerms.order_id,
|
2016-10-17 15:58:36 +02:00
|
|
|
};
|
2017-08-27 03:56:19 +02:00
|
|
|
const t: PurchaseRecord = {
|
2018-01-29 16:41:17 +01:00
|
|
|
abortDone: false,
|
|
|
|
abortRequested: false,
|
2017-06-01 18:46:07 +02:00
|
|
|
contractTerms: proposal.contractTerms,
|
|
|
|
contractTermsHash: proposal.contractTermsHash,
|
2016-10-19 22:49:03 +02:00
|
|
|
finished: false,
|
2018-01-18 02:50:18 +01:00
|
|
|
lastSessionId: undefined,
|
2017-05-31 16:04:14 +02:00
|
|
|
merchantSig: proposal.merchantSig,
|
2017-05-28 01:10:54 +02:00
|
|
|
payReq,
|
2017-08-27 03:56:19 +02:00
|
|
|
refundsDone: {},
|
|
|
|
refundsPending: {},
|
2019-11-21 23:09:43 +01:00
|
|
|
timestamp: getTimestampNow(),
|
|
|
|
timestamp_refund: undefined,
|
2016-01-24 02:29:13 +01:00
|
|
|
};
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.coins, Stores.purchases],
|
|
|
|
async tx => {
|
|
|
|
await tx.put(Stores.purchases, t);
|
|
|
|
for (let c of payCoinInfo.updatedCoins) {
|
|
|
|
await tx.put(Stores.coins, c);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2017-12-12 16:39:55 +01:00
|
|
|
this.badge.showNotification();
|
2016-09-28 18:00:13 +02:00
|
|
|
this.notifier.notify();
|
2018-01-17 03:49:54 +01:00
|
|
|
return t;
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
2015-12-14 16:54:47 +01:00
|
|
|
|
2019-08-29 23:12:55 +02:00
|
|
|
getNextUrl(contractTerms: ContractTerms): string {
|
|
|
|
const fu = new URI(contractTerms.fulfillment_url);
|
|
|
|
fu.addSearch("order_id", contractTerms.order_id);
|
|
|
|
return fu.href();
|
|
|
|
}
|
|
|
|
|
2019-09-06 09:48:00 +02:00
|
|
|
/**
|
|
|
|
* Check if a payment for the given taler://pay/ URI is possible.
|
2019-09-06 11:06:28 +02:00
|
|
|
*
|
2019-09-06 09:48:00 +02:00
|
|
|
* If the payment is possible, the signature are already generated but not
|
|
|
|
* yet send to the merchant.
|
|
|
|
*/
|
|
|
|
async preparePay(talerPayUri: string): Promise<PreparePayResult> {
|
|
|
|
const uriResult = parsePayUri(talerPayUri);
|
2019-08-28 02:49:27 +02:00
|
|
|
|
|
|
|
if (!uriResult) {
|
|
|
|
return {
|
|
|
|
status: "error",
|
|
|
|
error: "URI not supported",
|
|
|
|
};
|
2019-08-20 17:58:01 +02:00
|
|
|
}
|
2019-08-28 02:49:27 +02:00
|
|
|
|
2019-08-20 17:58:01 +02:00
|
|
|
let proposalId: number;
|
|
|
|
try {
|
2019-08-28 02:49:27 +02:00
|
|
|
proposalId = await this.downloadProposal(
|
|
|
|
uriResult.downloadUrl,
|
|
|
|
uriResult.sessionId,
|
|
|
|
);
|
2019-08-20 17:58:01 +02:00
|
|
|
} catch (e) {
|
|
|
|
return {
|
|
|
|
status: "error",
|
|
|
|
error: e.toString(),
|
2019-08-24 18:42:00 +02:00
|
|
|
};
|
2019-08-20 17:58:01 +02:00
|
|
|
}
|
|
|
|
const proposal = await this.getProposal(proposalId);
|
|
|
|
if (!proposal) {
|
|
|
|
throw Error("could not get proposal");
|
|
|
|
}
|
2019-08-28 02:49:27 +02:00
|
|
|
|
|
|
|
console.log("proposal", proposal);
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const differentPurchase = await oneShotGetIndexed(
|
|
|
|
this.db,
|
2019-09-06 11:06:28 +02:00
|
|
|
Stores.purchases.fulfillmentUrlIndex,
|
|
|
|
proposal.contractTerms.fulfillment_url,
|
|
|
|
);
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
let fulfillmentUrl = proposal.contractTerms.fulfillment_url;
|
|
|
|
let doublePurchaseDetection = false;
|
|
|
|
if (fulfillmentUrl.startsWith("http")) {
|
|
|
|
doublePurchaseDetection = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (differentPurchase && doublePurchaseDetection) {
|
2019-09-06 11:06:28 +02:00
|
|
|
// We do this check to prevent merchant B to find out if we bought a
|
|
|
|
// digital product with merchant A by abusing the existing payment
|
|
|
|
// redirect feature.
|
|
|
|
if (
|
|
|
|
differentPurchase.contractTerms.merchant_pub !=
|
|
|
|
proposal.contractTerms.merchant_pub
|
|
|
|
) {
|
|
|
|
console.warn(
|
|
|
|
"merchant with different public key offered contract with same fulfillment URL as an existing purchase",
|
|
|
|
);
|
|
|
|
} else {
|
2019-09-06 11:45:53 +02:00
|
|
|
if (uriResult.sessionId) {
|
2019-09-06 12:34:05 +02:00
|
|
|
await this.submitPay(
|
|
|
|
differentPurchase.contractTermsHash,
|
|
|
|
uriResult.sessionId,
|
|
|
|
);
|
2019-09-06 11:45:53 +02:00
|
|
|
}
|
2019-09-06 11:06:28 +02:00
|
|
|
return {
|
|
|
|
status: "paid",
|
|
|
|
contractTerms: differentPurchase.contractTerms,
|
|
|
|
nextUrl: this.getNextUrl(differentPurchase.contractTerms),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-06 09:48:00 +02:00
|
|
|
// First check if we already payed for it.
|
2019-11-20 20:02:48 +01:00
|
|
|
const purchase = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.purchases,
|
|
|
|
proposal.contractTermsHash,
|
|
|
|
);
|
2019-09-06 09:48:00 +02:00
|
|
|
|
|
|
|
if (!purchase) {
|
|
|
|
const paymentAmount = Amounts.parseOrThrow(proposal.contractTerms.amount);
|
|
|
|
let wireFeeLimit;
|
|
|
|
if (proposal.contractTerms.max_wire_fee) {
|
2019-09-06 11:06:28 +02:00
|
|
|
wireFeeLimit = Amounts.parseOrThrow(
|
|
|
|
proposal.contractTerms.max_wire_fee,
|
|
|
|
);
|
2019-09-06 09:48:00 +02:00
|
|
|
} else {
|
|
|
|
wireFeeLimit = Amounts.getZero(paymentAmount.currency);
|
|
|
|
}
|
|
|
|
// If not already payed, check if we could pay for it.
|
|
|
|
const res = await this.getCoinsForPayment({
|
|
|
|
allowedAuditors: proposal.contractTerms.auditors,
|
|
|
|
allowedExchanges: proposal.contractTerms.exchanges,
|
|
|
|
depositFeeLimit: Amounts.parseOrThrow(proposal.contractTerms.max_fee),
|
|
|
|
paymentAmount,
|
|
|
|
wireFeeAmortization: proposal.contractTerms.wire_fee_amortization || 1,
|
|
|
|
wireFeeLimit,
|
2019-11-21 23:09:43 +01:00
|
|
|
// FIXME: parse this properly
|
|
|
|
wireFeeTime: extractTalerStamp(proposal.contractTerms.timestamp) || {
|
|
|
|
t_ms: 0,
|
|
|
|
},
|
2019-09-06 09:48:00 +02:00
|
|
|
wireMethod: proposal.contractTerms.wire_method,
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
console.log("not confirming payment, insufficient coins");
|
2019-08-28 02:49:27 +02:00
|
|
|
return {
|
2019-09-06 09:48:00 +02:00
|
|
|
status: "insufficient-balance",
|
|
|
|
contractTerms: proposal.contractTerms,
|
|
|
|
proposalId: proposal.id!,
|
2019-08-28 02:49:27 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-09-06 09:48:00 +02:00
|
|
|
// Only create speculative signature if we don't already have one for this proposal
|
|
|
|
if (
|
|
|
|
!this.speculativePayData ||
|
|
|
|
(this.speculativePayData &&
|
|
|
|
this.speculativePayData.proposalId !== proposalId)
|
|
|
|
) {
|
|
|
|
const { exchangeUrl, cds, totalAmount } = res;
|
|
|
|
const payCoinInfo = await this.cryptoApi.signDeposit(
|
|
|
|
proposal.contractTerms,
|
|
|
|
cds,
|
|
|
|
totalAmount,
|
|
|
|
);
|
|
|
|
this.speculativePayData = {
|
|
|
|
exchangeUrl,
|
|
|
|
payCoinInfo,
|
|
|
|
proposal,
|
|
|
|
proposalId,
|
|
|
|
};
|
|
|
|
Wallet.enableTracing &&
|
|
|
|
console.log("created speculative pay data for payment");
|
|
|
|
}
|
|
|
|
|
2019-08-20 17:58:01 +02:00
|
|
|
return {
|
|
|
|
status: "payment-possible",
|
|
|
|
contractTerms: proposal.contractTerms,
|
|
|
|
proposalId: proposal.id!,
|
2019-09-06 09:48:00 +02:00
|
|
|
totalFees: res.totalFees,
|
2019-08-20 17:58:01 +02:00
|
|
|
};
|
|
|
|
}
|
2019-09-06 09:48:00 +02:00
|
|
|
|
|
|
|
if (uriResult.sessionId) {
|
2019-09-06 11:06:28 +02:00
|
|
|
await this.submitPay(purchase.contractTermsHash, uriResult.sessionId);
|
2019-09-06 09:48:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
status: "paid",
|
|
|
|
contractTerms: proposal.contractTerms,
|
|
|
|
nextUrl: this.getNextUrl(purchase.contractTerms),
|
|
|
|
};
|
2019-08-20 17:58:01 +02:00
|
|
|
}
|
|
|
|
|
2017-05-31 16:04:14 +02:00
|
|
|
/**
|
2018-01-17 03:49:54 +01:00
|
|
|
* Download a proposal and store it in the database.
|
|
|
|
* Returns an id for it to retrieve it later.
|
2019-08-24 18:42:00 +02:00
|
|
|
*
|
|
|
|
* @param sessionId Current session ID, if the proposal is being
|
|
|
|
* downloaded in the context of a session ID.
|
2017-05-31 16:04:14 +02:00
|
|
|
*/
|
2019-08-24 18:42:00 +02:00
|
|
|
async downloadProposal(url: string, sessionId?: string): Promise<number> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const oldProposal = await oneShotGetIndexed(
|
|
|
|
this.db,
|
|
|
|
Stores.proposals.urlIndex,
|
|
|
|
url,
|
|
|
|
);
|
2018-01-18 01:37:30 +01:00
|
|
|
if (oldProposal) {
|
|
|
|
return oldProposal.id!;
|
|
|
|
}
|
|
|
|
|
2018-01-17 03:49:54 +01:00
|
|
|
const { priv, pub } = await this.cryptoApi.createEddsaKeypair();
|
|
|
|
const parsed_url = new URI(url);
|
2018-01-18 01:37:30 +01:00
|
|
|
const urlWithNonce = parsed_url.setQuery({ nonce: pub }).href();
|
|
|
|
console.log("downloading contract from '" + urlWithNonce + "'");
|
2018-01-17 03:49:54 +01:00
|
|
|
let resp;
|
|
|
|
try {
|
2019-08-22 23:36:36 +02:00
|
|
|
resp = await this.http.get(urlWithNonce);
|
2018-01-17 03:49:54 +01:00
|
|
|
} catch (e) {
|
|
|
|
console.log("contract download failed", e);
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
2019-08-22 23:36:36 +02:00
|
|
|
const proposal = Proposal.checked(resp.responseJson);
|
2018-01-17 03:49:54 +01:00
|
|
|
|
|
|
|
const contractTermsHash = await this.hashContract(proposal.contract_terms);
|
|
|
|
|
|
|
|
const proposalRecord: ProposalDownloadRecord = {
|
|
|
|
contractTerms: proposal.contract_terms,
|
|
|
|
contractTermsHash,
|
|
|
|
merchantSig: proposal.sig,
|
|
|
|
noncePriv: priv,
|
2019-11-21 23:09:43 +01:00
|
|
|
timestamp: getTimestampNow(),
|
2018-01-17 03:49:54 +01:00
|
|
|
url,
|
2019-08-24 18:42:00 +02:00
|
|
|
downloadSessionId: sessionId,
|
2018-01-17 03:49:54 +01:00
|
|
|
};
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const id = await oneShotPut(this.db, Stores.proposals, proposalRecord);
|
2016-11-13 10:17:39 +01:00
|
|
|
this.notifier.notify();
|
|
|
|
if (typeof id !== "number") {
|
|
|
|
throw Error("db schema wrong");
|
|
|
|
}
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2018-01-29 16:41:17 +01:00
|
|
|
async refundFailedPay(proposalId: number) {
|
|
|
|
console.log(`refunding failed payment with proposal id ${proposalId}`);
|
2019-11-20 19:48:43 +01:00
|
|
|
const proposal = await oneShotGet(this.db, Stores.proposals, proposalId);
|
2018-01-29 16:41:17 +01:00
|
|
|
if (!proposal) {
|
|
|
|
throw Error(`proposal with id ${proposalId} not found`);
|
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const purchase = await oneShotGet(
|
|
|
|
this.db,
|
2019-06-26 15:30:32 +02:00
|
|
|
Stores.purchases,
|
2019-11-20 20:02:48 +01:00
|
|
|
proposal.contractTermsHash,
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2018-01-29 16:41:17 +01:00
|
|
|
if (!purchase) {
|
|
|
|
throw Error("purchase not found for proposal");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (purchase.finished) {
|
|
|
|
throw Error("can't auto-refund finished purchase");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
async submitPay(
|
|
|
|
contractTermsHash: string,
|
|
|
|
sessionId: string | undefined,
|
|
|
|
): Promise<ConfirmPayResult> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const purchase = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.purchases,
|
|
|
|
contractTermsHash,
|
|
|
|
);
|
2018-01-19 01:27:27 +01:00
|
|
|
if (!purchase) {
|
|
|
|
throw Error("Purchase not found: " + contractTermsHash);
|
|
|
|
}
|
2018-01-29 16:41:17 +01:00
|
|
|
if (purchase.abortRequested) {
|
|
|
|
throw Error("not submitting payment for aborted purchase");
|
|
|
|
}
|
2018-01-17 03:49:54 +01:00
|
|
|
let resp;
|
|
|
|
const payReq = { ...purchase.payReq, session_id: sessionId };
|
2018-01-29 16:41:17 +01:00
|
|
|
|
2019-11-02 00:22:55 +01:00
|
|
|
const payUrl = new URI("pay")
|
|
|
|
.absoluteTo(purchase.contractTerms.merchant_base_url)
|
|
|
|
.href();
|
2019-11-01 18:39:23 +01:00
|
|
|
|
2018-01-17 03:49:54 +01:00
|
|
|
try {
|
2019-11-01 18:39:23 +01:00
|
|
|
resp = await this.http.postJson(payUrl, payReq);
|
2018-01-17 03:49:54 +01:00
|
|
|
} catch (e) {
|
|
|
|
// Gives the user the option to retry / abort and refresh
|
|
|
|
console.log("payment failed", e);
|
|
|
|
throw e;
|
|
|
|
}
|
2019-08-22 23:36:36 +02:00
|
|
|
const merchantResp = resp.responseJson;
|
2019-11-01 18:39:23 +01:00
|
|
|
console.log("got success from pay URL");
|
2018-01-23 16:19:03 +01:00
|
|
|
|
|
|
|
const merchantPub = purchase.contractTerms.merchant_pub;
|
2019-06-26 15:30:32 +02:00
|
|
|
const valid: boolean = await this.cryptoApi.isValidPaymentSignature(
|
|
|
|
merchantResp.sig,
|
|
|
|
contractTermsHash,
|
|
|
|
merchantPub,
|
2018-01-23 16:19:03 +01:00
|
|
|
);
|
|
|
|
if (!valid) {
|
|
|
|
console.error("merchant payment signature invalid");
|
|
|
|
// FIXME: properly display error
|
|
|
|
throw Error("merchant payment signature invalid");
|
|
|
|
}
|
|
|
|
purchase.finished = true;
|
|
|
|
const modifiedCoins: CoinRecord[] = [];
|
|
|
|
for (const pc of purchase.payReq.coins) {
|
2019-11-20 19:48:43 +01:00
|
|
|
const c = await oneShotGet(this.db, Stores.coins, pc.coin_pub);
|
2018-01-23 16:19:03 +01:00
|
|
|
if (!c) {
|
|
|
|
console.error("coin not found");
|
|
|
|
throw Error("coin used in payment not found");
|
|
|
|
}
|
|
|
|
c.status = CoinStatus.Dirty;
|
|
|
|
modifiedCoins.push(c);
|
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.coins, Stores.purchases],
|
|
|
|
async tx => {
|
|
|
|
for (let c of modifiedCoins) {
|
|
|
|
tx.put(Stores.coins, c);
|
|
|
|
}
|
|
|
|
tx.put(Stores.purchases, purchase);
|
|
|
|
},
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2018-01-23 16:19:03 +01:00
|
|
|
for (const c of purchase.payReq.coins) {
|
|
|
|
this.refresh(c.coin_pub);
|
|
|
|
}
|
|
|
|
|
2019-08-29 23:12:55 +02:00
|
|
|
const nextUrl = this.getNextUrl(purchase.contractTerms);
|
2019-06-26 15:30:32 +02:00
|
|
|
this.cachedNextUrl[purchase.contractTerms.fulfillment_url] = {
|
|
|
|
nextUrl,
|
|
|
|
lastSessionId: sessionId,
|
|
|
|
};
|
2019-09-04 15:28:19 +02:00
|
|
|
|
2018-01-17 03:49:54 +01:00
|
|
|
return { nextUrl };
|
|
|
|
}
|
|
|
|
|
2019-09-04 15:28:19 +02:00
|
|
|
/**
|
|
|
|
* Refresh all dirty coins.
|
|
|
|
* The returned promise resolves only after all refresh
|
|
|
|
* operations have completed.
|
|
|
|
*/
|
|
|
|
async refreshDirtyCoins(): Promise<{ numRefreshed: number }> {
|
|
|
|
let n = 0;
|
2019-11-20 19:48:43 +01:00
|
|
|
const coins = await oneShotIter(this.db, Stores.coins).toArray();
|
2019-09-04 15:28:19 +02:00
|
|
|
for (let coin of coins) {
|
|
|
|
if (coin.status == CoinStatus.Dirty) {
|
|
|
|
try {
|
|
|
|
await this.refresh(coin.coinPub);
|
|
|
|
} catch (e) {
|
|
|
|
console.log("error during refresh");
|
|
|
|
}
|
|
|
|
|
|
|
|
n += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return { numRefreshed: n };
|
|
|
|
}
|
|
|
|
|
2016-02-10 02:03:31 +01:00
|
|
|
/**
|
2018-01-29 16:41:17 +01:00
|
|
|
* Add a contract to the wallet and sign coins, and send them.
|
2016-02-10 02:03:31 +01:00
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
async confirmPay(
|
|
|
|
proposalId: number,
|
2019-08-24 18:42:00 +02:00
|
|
|
sessionIdOverride: string | undefined,
|
2019-06-26 15:30:32 +02:00
|
|
|
): Promise<ConfirmPayResult> {
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing &&
|
|
|
|
console.log(
|
|
|
|
`executing confirmPay with proposalId ${proposalId} and sessionIdOverride ${sessionIdOverride}`,
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
const proposal = await oneShotGet(this.db, Stores.proposals, proposalId);
|
2017-05-31 16:04:14 +02:00
|
|
|
|
|
|
|
if (!proposal) {
|
|
|
|
throw Error(`proposal with id ${proposalId} not found`);
|
|
|
|
}
|
2016-02-22 21:52:53 +01:00
|
|
|
|
2019-08-24 18:42:00 +02:00
|
|
|
const sessionId = sessionIdOverride || proposal.downloadSessionId;
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
let purchase = await oneShotGet(
|
|
|
|
this.db,
|
2019-06-26 15:30:32 +02:00
|
|
|
Stores.purchases,
|
2019-11-20 20:02:48 +01:00
|
|
|
proposal.contractTermsHash,
|
|
|
|
);
|
2016-05-24 01:53:56 +02:00
|
|
|
|
2017-08-27 03:56:19 +02:00
|
|
|
if (purchase) {
|
2018-01-19 01:27:27 +01:00
|
|
|
return this.submitPay(purchase.contractTermsHash, sessionId);
|
2016-09-28 18:54:48 +02:00
|
|
|
}
|
|
|
|
|
2018-01-29 22:58:47 +01:00
|
|
|
const contractAmount = Amounts.parseOrThrow(proposal.contractTerms.amount);
|
|
|
|
|
|
|
|
let wireFeeLimit;
|
|
|
|
if (!proposal.contractTerms.max_wire_fee) {
|
|
|
|
wireFeeLimit = Amounts.getZero(contractAmount.currency);
|
|
|
|
} else {
|
|
|
|
wireFeeLimit = Amounts.parseOrThrow(proposal.contractTerms.max_wire_fee);
|
|
|
|
}
|
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const res = await this.getCoinsForPayment({
|
2017-05-31 16:04:14 +02:00
|
|
|
allowedAuditors: proposal.contractTerms.auditors,
|
|
|
|
allowedExchanges: proposal.contractTerms.exchanges,
|
2018-01-29 22:58:47 +01:00
|
|
|
depositFeeLimit: Amounts.parseOrThrow(proposal.contractTerms.max_fee),
|
|
|
|
paymentAmount: Amounts.parseOrThrow(proposal.contractTerms.amount),
|
2017-05-31 16:04:14 +02:00
|
|
|
wireFeeAmortization: proposal.contractTerms.wire_fee_amortization || 1,
|
2018-01-29 22:58:47 +01:00
|
|
|
wireFeeLimit,
|
2019-11-21 23:09:43 +01:00
|
|
|
// FIXME: parse this properly
|
|
|
|
wireFeeTime: extractTalerStamp(proposal.contractTerms.timestamp) || {
|
|
|
|
t_ms: 0,
|
|
|
|
},
|
2017-05-31 16:04:14 +02:00
|
|
|
wireMethod: proposal.contractTerms.wire_method,
|
2017-05-28 01:10:54 +02:00
|
|
|
});
|
2016-09-28 18:54:48 +02:00
|
|
|
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("coin selection result", res);
|
2016-11-20 04:15:49 +01:00
|
|
|
|
2016-11-14 02:52:29 +01:00
|
|
|
if (!res) {
|
2018-01-17 03:49:54 +01:00
|
|
|
// Should not happen, since checkPay should be called first
|
2016-09-28 18:54:48 +02:00
|
|
|
console.log("not confirming payment, insufficient coins");
|
2018-01-17 03:49:54 +01:00
|
|
|
throw Error("insufficient balance");
|
2016-09-28 18:54:48 +02:00
|
|
|
}
|
|
|
|
|
2017-12-12 21:54:14 +01:00
|
|
|
const sd = await this.getSpeculativePayData(proposalId);
|
|
|
|
if (!sd) {
|
2018-01-17 05:33:06 +01:00
|
|
|
const { exchangeUrl, cds, totalAmount } = res;
|
2019-06-26 15:30:32 +02:00
|
|
|
const payCoinInfo = await this.cryptoApi.signDeposit(
|
|
|
|
proposal.contractTerms,
|
|
|
|
cds,
|
|
|
|
totalAmount,
|
|
|
|
);
|
|
|
|
purchase = await this.recordConfirmPay(
|
|
|
|
proposal,
|
|
|
|
payCoinInfo,
|
|
|
|
exchangeUrl,
|
|
|
|
);
|
2017-12-12 21:54:14 +01:00
|
|
|
} else {
|
2019-06-26 15:30:32 +02:00
|
|
|
purchase = await this.recordConfirmPay(
|
|
|
|
sd.proposal,
|
|
|
|
sd.payCoinInfo,
|
|
|
|
sd.exchangeUrl,
|
|
|
|
);
|
2017-12-12 21:54:14 +01:00
|
|
|
}
|
|
|
|
|
2018-01-19 01:27:27 +01:00
|
|
|
return this.submitPay(purchase.contractTermsHash, sessionId);
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
2015-12-16 00:38:36 +01:00
|
|
|
|
2017-12-12 21:54:14 +01:00
|
|
|
/**
|
|
|
|
* Get the speculative pay data, but only if coins have not changed in between.
|
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
async getSpeculativePayData(
|
|
|
|
proposalId: number,
|
|
|
|
): Promise<SpeculativePayData | undefined> {
|
2017-12-12 21:54:14 +01:00
|
|
|
const sp = this.speculativePayData;
|
|
|
|
if (!sp) {
|
|
|
|
return;
|
|
|
|
}
|
2018-01-03 14:42:06 +01:00
|
|
|
if (sp.proposalId !== proposalId) {
|
2017-12-12 21:54:14 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const coinKeys = sp.payCoinInfo.updatedCoins.map(x => x.coinPub);
|
2019-11-20 19:48:43 +01:00
|
|
|
const coins: CoinRecord[] = [];
|
|
|
|
for (let coinKey of coinKeys) {
|
|
|
|
const cc = await oneShotGet(this.db, Stores.coins, coinKey);
|
|
|
|
if (cc) {
|
|
|
|
coins.push(cc);
|
|
|
|
}
|
|
|
|
}
|
2017-12-12 21:54:14 +01:00
|
|
|
for (let i = 0; i < coins.length; i++) {
|
|
|
|
const specCoin = sp.payCoinInfo.originalCoins[i];
|
|
|
|
const currentCoin = coins[i];
|
|
|
|
|
|
|
|
// Coin does not exist anymore!
|
|
|
|
if (!currentCoin) {
|
|
|
|
return;
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
if (
|
|
|
|
Amounts.cmp(specCoin.currentAmount, currentCoin.currentAmount) !== 0
|
|
|
|
) {
|
2018-01-03 14:42:06 +01:00
|
|
|
return;
|
2017-12-12 21:54:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return sp;
|
|
|
|
}
|
2016-02-10 02:03:31 +01:00
|
|
|
|
2019-11-19 16:16:12 +01:00
|
|
|
/**
|
2019-11-21 23:09:43 +01:00
|
|
|
* Send reserve details to the bank.
|
2019-11-19 16:16:12 +01:00
|
|
|
*/
|
2019-08-29 23:12:55 +02:00
|
|
|
private async sendReserveInfoToBank(reservePub: string) {
|
2019-11-20 20:02:48 +01:00
|
|
|
const reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
|
2019-08-29 23:12:55 +02:00
|
|
|
if (!reserve) {
|
|
|
|
throw Error("reserve not in db");
|
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
if (reserve.reserveStatus != ReserveRecordStatus.REGISTERING_BANK) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-08-29 23:12:55 +02:00
|
|
|
const bankStatusUrl = reserve.bankWithdrawStatusUrl;
|
|
|
|
if (!bankStatusUrl) {
|
2019-11-21 23:09:43 +01:00
|
|
|
throw Error("no bank withdraw status URL available.");
|
2019-08-29 23:12:55 +02:00
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const now = getTimestampNow();
|
2019-08-29 23:12:55 +02:00
|
|
|
let status;
|
|
|
|
try {
|
|
|
|
const statusResp = await this.http.get(bankStatusUrl);
|
|
|
|
status = WithdrawOperationStatusResponse.checked(statusResp.responseJson);
|
|
|
|
} catch (e) {
|
|
|
|
console.log("bank error response", e);
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status.transfer_done) {
|
2019-11-20 20:02:48 +01:00
|
|
|
await oneShotMutate(this.db, Stores.reserves, reservePub, r => {
|
2019-11-21 23:09:43 +01:00
|
|
|
r.timestampConfirmed = now;
|
2019-08-29 23:12:55 +02:00
|
|
|
return r;
|
|
|
|
});
|
2019-11-21 23:09:43 +01:00
|
|
|
} else if (reserve.timestampReserveInfoPosted === undefined) {
|
2019-08-29 23:12:55 +02:00
|
|
|
try {
|
|
|
|
if (!status.selection_done) {
|
|
|
|
const bankResp = await this.http.postJson(bankStatusUrl, {
|
|
|
|
reserve_pub: reservePub,
|
|
|
|
selected_exchange: reserve.exchangeWire,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.log("bank error response", e);
|
|
|
|
throw e;
|
|
|
|
}
|
2019-11-20 20:02:48 +01:00
|
|
|
await oneShotMutate(this.db, Stores.reserves, reservePub, r => {
|
2019-11-21 23:09:43 +01:00
|
|
|
r.timestampReserveInfoPosted = now;
|
2019-08-29 23:12:55 +02:00
|
|
|
return r;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-09 21:56:06 +01:00
|
|
|
/**
|
|
|
|
* First fetch information requred to withdraw from the reserve,
|
|
|
|
* then deplete the reserve, withdrawing coins until it is empty.
|
2019-11-21 23:09:43 +01:00
|
|
|
*
|
|
|
|
* The returned promise resolves once the reserve is set to the
|
|
|
|
* state DORMANT.
|
2016-02-09 21:56:06 +01:00
|
|
|
*/
|
2019-08-01 23:21:15 +02:00
|
|
|
async processReserve(reservePub: string): Promise<void> {
|
2019-11-21 23:09:43 +01:00
|
|
|
const reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
|
|
|
|
if (!reserve) {
|
|
|
|
console.log("not processing reserve: reserve does not exist");
|
|
|
|
return;
|
2019-08-01 23:21:15 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
logger.trace(
|
|
|
|
`Processing reserve ${reservePub} with status ${reserve.reserveStatus}`,
|
|
|
|
);
|
|
|
|
switch (reserve.reserveStatus) {
|
|
|
|
case ReserveRecordStatus.UNCONFIRMED:
|
|
|
|
// nothing to do
|
|
|
|
break;
|
|
|
|
case ReserveRecordStatus.REGISTERING_BANK:
|
|
|
|
await this.sendReserveInfoToBank(reservePub);
|
|
|
|
return this.processReserve(reservePub);
|
|
|
|
case ReserveRecordStatus.QUERYING_STATUS:
|
|
|
|
await this.updateReserve(reservePub);
|
|
|
|
return this.processReserve(reservePub);
|
|
|
|
case ReserveRecordStatus.WITHDRAWING:
|
|
|
|
await this.depleteReserve(reservePub);
|
|
|
|
break;
|
|
|
|
case ReserveRecordStatus.DORMANT:
|
|
|
|
// nothing to do
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
console.warn("unknown reserve record status:", reserve.reserveStatus);
|
|
|
|
break;
|
2016-09-28 18:54:48 +02:00
|
|
|
}
|
2016-05-24 01:18:23 +02:00
|
|
|
}
|
|
|
|
|
2017-10-15 18:30:02 +02:00
|
|
|
/**
|
|
|
|
* Given a planchet, withdraw a coin from the exchange.
|
|
|
|
*/
|
2019-08-01 23:21:15 +02:00
|
|
|
private async processPreCoin(preCoinPub: string): Promise<void> {
|
2019-11-21 23:09:43 +01:00
|
|
|
console.log("processPreCoin", preCoinPub);
|
|
|
|
const preCoin = await oneShotGet(this.db, Stores.precoins, preCoinPub);
|
|
|
|
if (!preCoin) {
|
|
|
|
console.log("processPreCoin: preCoinPub not found");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const exchange = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.exchanges,
|
|
|
|
preCoin.exchangeBaseUrl,
|
|
|
|
);
|
|
|
|
if (!exchange) {
|
|
|
|
console.error("db inconsistent: exchange for precoin not found");
|
|
|
|
return;
|
2016-10-19 22:59:24 +02:00
|
|
|
}
|
2019-08-01 23:21:15 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const denom = await oneShotGet(this.db, Stores.denominations, [
|
|
|
|
preCoin.exchangeBaseUrl,
|
|
|
|
preCoin.denomPub,
|
|
|
|
]);
|
2016-10-19 22:59:24 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
if (!denom) {
|
|
|
|
console.error("db inconsistent: denom for precoin not found");
|
|
|
|
return;
|
|
|
|
}
|
2016-10-19 22:59:24 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const wd: any = {};
|
|
|
|
wd.denom_pub_hash = preCoin.denomPubHash;
|
|
|
|
wd.reserve_pub = preCoin.reservePub;
|
|
|
|
wd.reserve_sig = preCoin.withdrawSig;
|
|
|
|
wd.coin_ev = preCoin.coinEv;
|
|
|
|
const reqUrl = new URI("reserve/withdraw").absoluteTo(exchange.baseUrl);
|
|
|
|
const resp = await this.http.postJson(reqUrl.href(), wd);
|
2017-11-30 04:07:36 +01:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const r = resp.responseJson;
|
2019-08-01 23:21:15 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const denomSig = await this.cryptoApi.rsaUnblind(
|
|
|
|
r.ev_sig,
|
|
|
|
preCoin.blindingKey,
|
|
|
|
preCoin.denomPub,
|
|
|
|
);
|
2017-11-30 04:07:36 +01:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const coin: CoinRecord = {
|
|
|
|
blindingKey: preCoin.blindingKey,
|
|
|
|
coinPriv: preCoin.coinPriv,
|
|
|
|
coinPub: preCoin.coinPub,
|
|
|
|
currentAmount: preCoin.coinValue,
|
|
|
|
denomPub: preCoin.denomPub,
|
|
|
|
denomPubHash: preCoin.denomPubHash,
|
|
|
|
denomSig,
|
|
|
|
exchangeBaseUrl: preCoin.exchangeBaseUrl,
|
|
|
|
reservePub: preCoin.reservePub,
|
|
|
|
status: CoinStatus.Fresh,
|
|
|
|
};
|
2019-06-26 15:30:32 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const mutateReserve = (r: ReserveRecord) => {
|
|
|
|
const x = Amounts.sub(
|
|
|
|
r.precoinAmount,
|
|
|
|
preCoin.coinValue,
|
|
|
|
denom.feeWithdraw,
|
|
|
|
);
|
|
|
|
if (x.saturated) {
|
|
|
|
console.error("database inconsistent");
|
|
|
|
throw TransactionAbort;
|
2019-08-01 23:21:15 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
r.precoinAmount = x.amount;
|
|
|
|
return r;
|
2019-08-01 23:21:15 +02:00
|
|
|
};
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.reserves, Stores.precoins, Stores.coins],
|
|
|
|
async tx => {
|
|
|
|
const currentPc = await tx.get(Stores.precoins, coin.coinPub);
|
|
|
|
if (!currentPc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
await tx.mutate(Stores.reserves, preCoin.reservePub, mutateReserve);
|
|
|
|
await tx.delete(Stores.precoins, coin.coinPub);
|
|
|
|
await tx.add(Stores.coins, coin);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
logger.trace(`withdraw of one coin ${coin.coinPub} finished`);
|
2016-01-24 19:57:09 +01:00
|
|
|
}
|
|
|
|
|
2016-02-09 21:56:06 +01:00
|
|
|
/**
|
|
|
|
* Create a reserve, but do not flag it as confirmed yet.
|
2017-04-28 23:28:27 +02:00
|
|
|
*
|
|
|
|
* Adds the corresponding exchange as a trusted exchange if it is neither
|
|
|
|
* audited nor trusted already.
|
2016-02-09 21:56:06 +01:00
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
async createReserve(
|
|
|
|
req: CreateReserveRequest,
|
|
|
|
): Promise<CreateReserveResponse> {
|
2017-05-28 01:10:54 +02:00
|
|
|
const keypair = await this.cryptoApi.createEddsaKeypair();
|
2019-11-21 23:09:43 +01:00
|
|
|
const now = getTimestampNow();
|
2016-09-28 18:54:48 +02:00
|
|
|
const canonExchange = canonicalizeBaseUrl(req.exchange);
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
let reserveStatus;
|
|
|
|
if (req.bankWithdrawStatusUrl) {
|
|
|
|
reserveStatus = ReserveRecordStatus.REGISTERING_BANK;
|
|
|
|
} else {
|
|
|
|
reserveStatus = ReserveRecordStatus.UNCONFIRMED;
|
|
|
|
}
|
|
|
|
|
2016-09-28 23:41:34 +02:00
|
|
|
const reserveRecord: ReserveRecord = {
|
2016-09-28 18:54:48 +02:00
|
|
|
created: now,
|
2019-11-21 23:09:43 +01:00
|
|
|
currentAmount: null,
|
|
|
|
exchangeBaseUrl: canonExchange,
|
2017-05-28 01:10:54 +02:00
|
|
|
hasPayback: false,
|
2019-11-21 23:09:43 +01:00
|
|
|
precoinAmount: Amounts.getZero(req.amount.currency),
|
|
|
|
requestedAmount: req.amount,
|
|
|
|
reservePriv: keypair.priv,
|
|
|
|
reservePub: keypair.pub,
|
2017-08-14 04:16:12 +02:00
|
|
|
senderWire: req.senderWire,
|
2019-11-21 23:09:43 +01:00
|
|
|
timestampConfirmed: undefined,
|
|
|
|
timestampReserveInfoPosted: undefined,
|
2019-08-28 02:49:27 +02:00
|
|
|
bankWithdrawStatusUrl: req.bankWithdrawStatusUrl,
|
|
|
|
exchangeWire: req.exchangeWire,
|
2019-11-21 23:09:43 +01:00
|
|
|
reserveStatus,
|
2016-09-28 18:54:48 +02:00
|
|
|
};
|
2016-01-06 15:39:22 +01:00
|
|
|
|
2018-01-04 11:08:39 +01:00
|
|
|
const senderWire = req.senderWire;
|
2019-05-08 04:53:26 +02:00
|
|
|
if (senderWire) {
|
2018-01-04 11:08:39 +01:00
|
|
|
const rec = {
|
2019-05-08 04:53:26 +02:00
|
|
|
paytoUri: senderWire,
|
2018-01-04 11:08:39 +01:00
|
|
|
};
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.senderWires, rec);
|
2018-01-04 11:08:39 +01:00
|
|
|
}
|
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const exchangeInfo = await this.updateExchangeFromUrl(req.exchange);
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchangeDetails = exchangeInfo.details;
|
|
|
|
if (!exchangeDetails) {
|
|
|
|
throw Error("exchange not updated");
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
const { isAudited, isTrusted } = await this.getExchangeTrust(exchangeInfo);
|
2019-11-20 20:02:48 +01:00
|
|
|
let currencyRecord = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.currencies,
|
|
|
|
exchangeDetails.currency,
|
|
|
|
);
|
2017-04-28 23:28:27 +02:00
|
|
|
if (!currencyRecord) {
|
|
|
|
currencyRecord = {
|
|
|
|
auditors: [],
|
2017-05-28 01:10:54 +02:00
|
|
|
exchanges: [],
|
|
|
|
fractionalDigits: 2,
|
2019-11-20 19:48:43 +01:00
|
|
|
name: exchangeDetails.currency,
|
2017-05-28 01:10:54 +02:00
|
|
|
};
|
2017-04-28 23:28:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!isAudited && !isTrusted) {
|
2019-06-26 15:30:32 +02:00
|
|
|
currencyRecord.exchanges.push({
|
|
|
|
baseUrl: req.exchange,
|
2019-11-20 19:48:43 +01:00
|
|
|
exchangePub: exchangeDetails.masterPublicKey,
|
2019-06-26 15:30:32 +02:00
|
|
|
});
|
2017-04-28 23:28:27 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const cr: CurrencyRecord = currencyRecord;
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
await runWithWriteTransaction(
|
2019-11-20 20:02:48 +01:00
|
|
|
this.db,
|
|
|
|
[Stores.currencies, Stores.reserves],
|
|
|
|
async tx => {
|
|
|
|
await tx.put(Stores.currencies, cr);
|
|
|
|
await tx.put(Stores.reserves, reserveRecord);
|
|
|
|
},
|
|
|
|
);
|
2016-02-11 18:17:02 +01:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
this.processReserve(keypair.pub).catch(e => {
|
|
|
|
console.error("Processing reserve failed:", e);
|
|
|
|
});
|
2019-08-28 02:49:27 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const r: CreateReserveResponse = {
|
2016-09-28 18:54:48 +02:00
|
|
|
exchange: canonExchange,
|
|
|
|
reservePub: keypair.pub,
|
|
|
|
};
|
|
|
|
return r;
|
2016-02-09 21:56:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Mark an existing reserve as confirmed. The wallet will start trying
|
|
|
|
* to withdraw from that reserve. This may not immediately succeed,
|
2016-03-01 19:39:17 +01:00
|
|
|
* since the exchange might not know about the reserve yet, even though the
|
2016-02-09 21:56:06 +01:00
|
|
|
* bank confirmed its creation.
|
|
|
|
*
|
|
|
|
* A confirmed reserve should be shown to the user in the UI, while
|
|
|
|
* an unconfirmed reserve should be hidden.
|
|
|
|
*/
|
2016-09-28 18:54:48 +02:00
|
|
|
async confirmReserve(req: ConfirmReserveRequest): Promise<void> {
|
2019-11-21 23:09:43 +01:00
|
|
|
const now = getTimestampNow();
|
|
|
|
await oneShotMutate(this.db, Stores.reserves, req.reservePub, reserve => {
|
|
|
|
if (reserve.reserveStatus !== ReserveRecordStatus.UNCONFIRMED) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
reserve.timestampConfirmed = now;
|
|
|
|
reserve.reserveStatus = ReserveRecordStatus.QUERYING_STATUS;
|
|
|
|
return reserve;
|
|
|
|
});
|
2016-01-05 01:10:31 +01:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
this.notifier.notify();
|
2016-09-28 18:54:48 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
this.processReserve(req.reservePub).catch(e => {
|
|
|
|
console.log("processing reserve failed:", e);
|
|
|
|
});
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
2015-12-13 23:47:30 +01:00
|
|
|
|
2016-01-06 15:39:22 +01:00
|
|
|
/**
|
|
|
|
* Withdraw coins from a reserve until it is empty.
|
2017-10-15 18:30:02 +02:00
|
|
|
*
|
|
|
|
* When finished, marks the reserve as depleted by setting
|
|
|
|
* the depleted timestamp.
|
2016-01-06 15:39:22 +01:00
|
|
|
*/
|
2019-11-21 23:09:43 +01:00
|
|
|
private async depleteReserve(reservePub: string): Promise<void> {
|
|
|
|
const reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
|
|
|
|
if (!reserve) {
|
|
|
|
return;
|
2016-10-20 01:37:00 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
if (reserve.reserveStatus !== ReserveRecordStatus.WITHDRAWING) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
logger.trace(`depleting reserve ${reservePub}`);
|
|
|
|
|
|
|
|
const withdrawAmount = reserve.currentAmount;
|
2017-05-28 01:10:54 +02:00
|
|
|
if (!withdrawAmount) {
|
2019-11-21 23:09:43 +01:00
|
|
|
throw Error("BUG: reserveStatus=WITHDRAWING, but currentAmount is empty");
|
2016-11-16 01:59:39 +01:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const denomsForWithdraw = await this.getVerifiedWithdrawDenomList(
|
2019-11-21 23:09:43 +01:00
|
|
|
reserve.exchangeBaseUrl,
|
2019-06-26 15:30:32 +02:00
|
|
|
withdrawAmount,
|
|
|
|
);
|
2019-11-21 23:09:43 +01:00
|
|
|
if (denomsForWithdraw.length === 0) {
|
|
|
|
const m = `Unable to withdraw from reserve, no denominations are available to withdraw.`;
|
|
|
|
await this.setReserveError(reserve.reservePub, {
|
|
|
|
type: "internal",
|
|
|
|
message: m,
|
|
|
|
details: {},
|
|
|
|
});
|
|
|
|
throw new OperationFailedAndReportedError(m);
|
|
|
|
}
|
2016-10-20 01:37:00 +02:00
|
|
|
|
2019-11-19 16:16:12 +01:00
|
|
|
const withdrawalRecord: WithdrawalRecord = {
|
2019-11-21 23:09:43 +01:00
|
|
|
reservePub: reserve.reservePub,
|
2019-11-19 16:16:12 +01:00
|
|
|
withdrawalAmount: Amounts.toString(withdrawAmount),
|
2019-11-21 23:09:43 +01:00
|
|
|
startTimestamp: getTimestampNow(),
|
|
|
|
numCoinsTotal: denomsForWithdraw.length,
|
|
|
|
numCoinsWithdrawn: 0,
|
2019-11-20 19:48:43 +01:00
|
|
|
};
|
2017-10-15 18:30:02 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const preCoinRecords: PreCoinRecord[] = await Promise.all(
|
|
|
|
denomsForWithdraw.map(async denom => {
|
|
|
|
return await this.cryptoApi.createPreCoin(denom, reserve);
|
|
|
|
}),
|
|
|
|
);
|
2016-10-20 01:37:00 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const totalCoinValue = Amounts.sum(denomsForWithdraw.map(x => x.value))
|
|
|
|
.amount;
|
|
|
|
const totalCoinWithdrawFee = Amounts.sum(
|
|
|
|
denomsForWithdraw.map(x => x.feeWithdraw),
|
|
|
|
).amount;
|
|
|
|
const totalWithdrawAmount = Amounts.add(
|
|
|
|
totalCoinValue,
|
|
|
|
totalCoinWithdrawFee,
|
|
|
|
).amount;
|
2019-08-28 02:49:27 +02:00
|
|
|
|
2019-11-19 16:16:12 +01:00
|
|
|
function mutateReserve(r: ReserveRecord): ReserveRecord {
|
2019-11-21 23:09:43 +01:00
|
|
|
const currentAmount = r.currentAmount;
|
2019-11-19 16:16:12 +01:00
|
|
|
if (!currentAmount) {
|
|
|
|
throw Error("can't withdraw when amount is unknown");
|
2019-08-01 23:21:15 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
r.precoinAmount = Amounts.add(
|
|
|
|
r.precoinAmount,
|
2019-11-20 19:48:43 +01:00
|
|
|
totalWithdrawAmount,
|
|
|
|
).amount;
|
2019-11-19 16:16:12 +01:00
|
|
|
const result = Amounts.sub(currentAmount, totalWithdrawAmount);
|
|
|
|
if (result.saturated) {
|
|
|
|
console.error("can't create precoins, saturated");
|
2019-11-21 23:09:43 +01:00
|
|
|
throw TransactionAbort;
|
2019-11-19 16:16:12 +01:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
r.currentAmount = result.amount;
|
|
|
|
r.reserveStatus = ReserveRecordStatus.DORMANT;
|
2019-11-19 16:16:12 +01:00
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const success = await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.precoins, Stores.withdrawals, Stores.reserves],
|
|
|
|
async tx => {
|
|
|
|
const myReserve = await tx.get(Stores.reserves, reservePub);
|
|
|
|
if (!myReserve) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (myReserve.reserveStatus !== ReserveRecordStatus.WITHDRAWING) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (let pcr of preCoinRecords) {
|
|
|
|
await tx.put(Stores.precoins, pcr);
|
|
|
|
}
|
|
|
|
await tx.mutate(Stores.reserves, reserve.reservePub, mutateReserve);
|
|
|
|
await tx.put(Stores.withdrawals, withdrawalRecord);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
);
|
2019-11-19 16:16:12 +01:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
if (success) {
|
|
|
|
logger.trace(`withdrawing ${preCoinRecords.length} coins`);
|
|
|
|
for (let x of preCoinRecords) {
|
|
|
|
await this.processPreCoin(x.coinPub);
|
|
|
|
}
|
2019-11-19 16:16:12 +01:00
|
|
|
}
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
|
|
|
|
2016-02-11 18:17:02 +01:00
|
|
|
/**
|
|
|
|
* Update the information about a reserve that is stored in the wallet
|
2016-03-01 19:39:17 +01:00
|
|
|
* by quering the reserve's exchange.
|
2016-02-11 18:17:02 +01:00
|
|
|
*/
|
2019-11-21 23:09:43 +01:00
|
|
|
private async updateReserve(reservePub: string): Promise<void> {
|
2019-11-20 19:48:43 +01:00
|
|
|
const reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
|
2016-10-13 02:23:24 +02:00
|
|
|
if (!reserve) {
|
|
|
|
throw Error("reserve not in db");
|
|
|
|
}
|
2019-08-28 02:49:27 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
if (reserve.timestampConfirmed === undefined) {
|
|
|
|
throw Error("reserve not confirmed yet");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (reserve.reserveStatus !== ReserveRecordStatus.QUERYING_STATUS) {
|
|
|
|
return;
|
2019-08-28 02:49:27 +02:00
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const reqUrl = new URI("reserve/status").absoluteTo(
|
2019-11-21 23:09:43 +01:00
|
|
|
reserve.exchangeBaseUrl,
|
2019-06-26 15:30:32 +02:00
|
|
|
);
|
|
|
|
reqUrl.query({ reserve_pub: reservePub });
|
2019-11-21 23:09:43 +01:00
|
|
|
let resp;
|
|
|
|
try {
|
|
|
|
resp = await this.http.get(reqUrl.href());
|
|
|
|
} catch (e) {
|
|
|
|
if (e.response?.status === 404) {
|
|
|
|
console.log("Reserve now known to exchange (yet).");
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
const m = e.message;
|
|
|
|
this.setReserveError(reservePub, {
|
|
|
|
type: "network",
|
|
|
|
details: {},
|
|
|
|
message: m,
|
|
|
|
});
|
|
|
|
throw new OperationFailedAndReportedError(m);
|
|
|
|
}
|
2016-09-28 19:09:10 +02:00
|
|
|
}
|
2019-07-31 01:33:56 +02:00
|
|
|
const reserveInfo = ReserveStatus.checked(resp.responseJson);
|
2019-11-21 23:09:43 +01:00
|
|
|
await oneShotMutate(this.db, Stores.reserves, reserve.reservePub, r => {
|
|
|
|
if (r.reserveStatus !== ReserveRecordStatus.QUERYING_STATUS) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
reserve.currentAmount = Amounts.parseOrThrow(reserveInfo.balance);
|
|
|
|
reserve.reserveStatus = ReserveRecordStatus.WITHDRAWING;
|
|
|
|
return r;
|
|
|
|
});
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.reserves, reserve);
|
2016-10-19 22:49:03 +02:00
|
|
|
this.notifier.notify();
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
async getPossibleDenoms(
|
|
|
|
exchangeBaseUrl: string,
|
|
|
|
): Promise<DenominationRecord[]> {
|
|
|
|
return await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.denominations.exchangeBaseUrlIndex,
|
|
|
|
exchangeBaseUrl,
|
|
|
|
).filter(d => {
|
|
|
|
return (
|
|
|
|
d.status === DenominationStatus.Unverified ||
|
|
|
|
d.status === DenominationStatus.VerifiedGood
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
});
|
2016-11-16 01:59:39 +01:00
|
|
|
}
|
|
|
|
|
2017-10-15 18:30:02 +02:00
|
|
|
/**
|
|
|
|
* Compute the smallest withdrawable amount possible, based on verified denominations.
|
|
|
|
*
|
|
|
|
* Writes to the DB in order to record the result from verifying
|
|
|
|
* denominations.
|
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
async getVerifiedSmallestWithdrawAmount(
|
|
|
|
exchangeBaseUrl: string,
|
|
|
|
): Promise<AmountJson> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const exchange = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.exchanges,
|
|
|
|
exchangeBaseUrl,
|
|
|
|
);
|
2017-10-15 18:30:02 +02:00
|
|
|
if (!exchange) {
|
|
|
|
throw Error(`exchange ${exchangeBaseUrl} not found`);
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchangeDetails = exchange.details;
|
|
|
|
if (!exchangeDetails) {
|
|
|
|
throw Error(`exchange ${exchangeBaseUrl} details not available`);
|
|
|
|
}
|
|
|
|
|
|
|
|
const possibleDenoms = await this.getPossibleDenoms(exchange.baseUrl);
|
2017-10-15 18:30:02 +02:00
|
|
|
|
|
|
|
possibleDenoms.sort((d1, d2) => {
|
2017-10-15 19:28:35 +02:00
|
|
|
const a1 = Amounts.add(d1.feeWithdraw, d1.value).amount;
|
|
|
|
const a2 = Amounts.add(d2.feeWithdraw, d2.value).amount;
|
2017-10-15 18:30:02 +02:00
|
|
|
return Amounts.cmp(a1, a2);
|
|
|
|
});
|
|
|
|
|
2017-10-15 19:28:35 +02:00
|
|
|
for (const denom of possibleDenoms) {
|
2017-10-15 18:30:02 +02:00
|
|
|
if (denom.status === DenominationStatus.VerifiedGood) {
|
|
|
|
return Amounts.add(denom.feeWithdraw, denom.value).amount;
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
const valid = await this.cryptoApi.isValidDenom(
|
|
|
|
denom,
|
2019-11-20 19:48:43 +01:00
|
|
|
exchangeDetails.masterPublicKey,
|
2019-06-26 15:30:32 +02:00
|
|
|
);
|
2017-10-15 18:30:02 +02:00
|
|
|
if (!valid) {
|
|
|
|
denom.status = DenominationStatus.VerifiedBad;
|
|
|
|
} else {
|
|
|
|
denom.status = DenominationStatus.VerifiedGood;
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.denominations, denom);
|
2017-10-15 18:30:02 +02:00
|
|
|
if (valid) {
|
|
|
|
return Amounts.add(denom.feeWithdraw, denom.value).amount;
|
|
|
|
}
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
return Amounts.getZero(exchangeDetails.currency);
|
2017-10-15 18:30:02 +02:00
|
|
|
}
|
|
|
|
|
2016-11-16 01:59:39 +01:00
|
|
|
/**
|
|
|
|
* Get a list of denominations to withdraw from the given exchange for the
|
|
|
|
* given amount, making sure that all denominations' signatures are verified.
|
|
|
|
*
|
|
|
|
* Writes to the DB in order to record the result from verifying
|
|
|
|
* denominations.
|
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
async getVerifiedWithdrawDenomList(
|
|
|
|
exchangeBaseUrl: string,
|
|
|
|
amount: AmountJson,
|
|
|
|
): Promise<DenominationRecord[]> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const exchange = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.exchanges,
|
|
|
|
exchangeBaseUrl,
|
|
|
|
);
|
2016-11-16 01:59:39 +01:00
|
|
|
if (!exchange) {
|
|
|
|
throw Error(`exchange ${exchangeBaseUrl} not found`);
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchangeDetails = exchange.details;
|
|
|
|
if (!exchangeDetails) {
|
|
|
|
throw Error(`exchange ${exchangeBaseUrl} details not available`);
|
|
|
|
}
|
2016-11-16 01:59:39 +01:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const possibleDenoms = await this.getPossibleDenoms(exchange.baseUrl);
|
2016-11-16 01:59:39 +01:00
|
|
|
|
|
|
|
let allValid = false;
|
|
|
|
|
|
|
|
let selectedDenoms: DenominationRecord[];
|
|
|
|
|
|
|
|
do {
|
|
|
|
allValid = true;
|
2017-05-28 01:10:54 +02:00
|
|
|
const nextPossibleDenoms = [];
|
2016-11-16 01:59:39 +01:00
|
|
|
selectedDenoms = getWithdrawDenomList(amount, possibleDenoms);
|
2017-05-28 01:10:54 +02:00
|
|
|
for (const denom of selectedDenoms || []) {
|
|
|
|
if (denom.status === DenominationStatus.Unverified) {
|
2019-06-26 15:30:32 +02:00
|
|
|
const valid = await this.cryptoApi.isValidDenom(
|
|
|
|
denom,
|
2019-11-20 19:48:43 +01:00
|
|
|
exchangeDetails.masterPublicKey,
|
2019-06-26 15:30:32 +02:00
|
|
|
);
|
2016-11-16 01:59:39 +01:00
|
|
|
if (!valid) {
|
|
|
|
denom.status = DenominationStatus.VerifiedBad;
|
|
|
|
allValid = false;
|
|
|
|
} else {
|
|
|
|
denom.status = DenominationStatus.VerifiedGood;
|
|
|
|
nextPossibleDenoms.push(denom);
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.denominations, denom);
|
2016-11-16 01:59:39 +01:00
|
|
|
} else {
|
|
|
|
nextPossibleDenoms.push(denom);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (selectedDenoms.length > 0 && !allValid);
|
|
|
|
|
|
|
|
return selectedDenoms;
|
|
|
|
}
|
|
|
|
|
2017-04-28 23:28:27 +02:00
|
|
|
/**
|
|
|
|
* Check if and how an exchange is trusted and/or audited.
|
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
async getExchangeTrust(
|
|
|
|
exchangeInfo: ExchangeRecord,
|
|
|
|
): Promise<{ isTrusted: boolean; isAudited: boolean }> {
|
2017-04-28 23:28:27 +02:00
|
|
|
let isTrusted = false;
|
|
|
|
let isAudited = false;
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchangeDetails = exchangeInfo.details;
|
|
|
|
if (!exchangeDetails) {
|
|
|
|
throw Error(`exchange ${exchangeInfo.baseUrl} details not available`);
|
|
|
|
}
|
2019-11-20 20:02:48 +01:00
|
|
|
const currencyRecord = await oneShotGet(
|
|
|
|
this.db,
|
2019-06-26 15:30:32 +02:00
|
|
|
Stores.currencies,
|
2019-11-20 20:02:48 +01:00
|
|
|
exchangeDetails.currency,
|
|
|
|
);
|
2017-04-28 23:28:27 +02:00
|
|
|
if (currencyRecord) {
|
2017-05-28 01:10:54 +02:00
|
|
|
for (const trustedExchange of currencyRecord.exchanges) {
|
2019-11-20 19:48:43 +01:00
|
|
|
if (trustedExchange.exchangePub === exchangeDetails.masterPublicKey) {
|
2017-04-28 23:28:27 +02:00
|
|
|
isTrusted = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
for (const trustedAuditor of currencyRecord.auditors) {
|
2019-11-20 19:48:43 +01:00
|
|
|
for (const exchangeAuditor of exchangeDetails.auditors) {
|
2017-06-04 20:25:28 +02:00
|
|
|
if (trustedAuditor.auditorPub === exchangeAuditor.auditor_pub) {
|
2017-04-28 23:28:27 +02:00
|
|
|
isAudited = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
return { isTrusted, isAudited };
|
2017-04-28 23:28:27 +02:00
|
|
|
}
|
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
async getWithdrawDetailsForUri(
|
|
|
|
talerWithdrawUri: string,
|
2019-08-29 23:12:55 +02:00
|
|
|
maybeSelectedExchange?: string,
|
|
|
|
): Promise<WithdrawDetails> {
|
2019-09-01 01:05:38 +02:00
|
|
|
const info = await this.getWithdrawalInfo(talerWithdrawUri);
|
2019-08-29 23:12:55 +02:00
|
|
|
let rci: ReserveCreationInfo | undefined = undefined;
|
|
|
|
if (maybeSelectedExchange) {
|
2019-08-30 17:27:59 +02:00
|
|
|
rci = await this.getWithdrawDetailsForAmount(
|
2019-08-29 23:12:55 +02:00
|
|
|
maybeSelectedExchange,
|
|
|
|
info.amount,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
withdrawInfo: info,
|
|
|
|
reserveCreationInfo: rci,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
async getWithdrawDetailsForAmount(
|
2019-06-26 15:30:32 +02:00
|
|
|
baseUrl: string,
|
|
|
|
amount: AmountJson,
|
|
|
|
): Promise<ReserveCreationInfo> {
|
2017-05-28 01:10:54 +02:00
|
|
|
const exchangeInfo = await this.updateExchangeFromUrl(baseUrl);
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchangeDetails = exchangeInfo.details;
|
|
|
|
if (!exchangeDetails) {
|
|
|
|
throw Error(`exchange ${exchangeInfo.baseUrl} details not available`);
|
|
|
|
}
|
|
|
|
const exchangeWireInfo = exchangeInfo.wireInfo;
|
|
|
|
if (!exchangeWireInfo) {
|
|
|
|
throw Error(
|
|
|
|
`exchange ${exchangeInfo.baseUrl} wire details not available`,
|
|
|
|
);
|
|
|
|
}
|
2016-04-06 02:06:57 +02:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const selectedDenoms = await this.getVerifiedWithdrawDenomList(
|
|
|
|
baseUrl,
|
|
|
|
amount,
|
|
|
|
);
|
2016-09-28 18:54:48 +02:00
|
|
|
let acc = Amounts.getZero(amount.currency);
|
2017-05-28 01:10:54 +02:00
|
|
|
for (const d of selectedDenoms) {
|
2016-11-16 01:59:39 +01:00
|
|
|
acc = Amounts.add(acc, d.feeWithdraw).amount;
|
2016-09-28 18:54:48 +02:00
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
const actualCoinCost = selectedDenoms
|
2019-06-26 15:30:32 +02:00
|
|
|
.map(
|
|
|
|
(d: DenominationRecord) => Amounts.add(d.value, d.feeWithdraw).amount,
|
|
|
|
)
|
2016-09-28 18:54:48 +02:00
|
|
|
.reduce((a, b) => Amounts.add(a, b).amount);
|
|
|
|
|
2019-05-08 04:53:26 +02:00
|
|
|
const exchangeWireAccounts: string[] = [];
|
2019-11-20 19:48:43 +01:00
|
|
|
for (let account of exchangeWireInfo.accounts) {
|
2019-05-08 04:53:26 +02:00
|
|
|
exchangeWireAccounts.push(account.url);
|
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const { isTrusted, isAudited } = await this.getExchangeTrust(exchangeInfo);
|
2017-04-28 23:28:27 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
let earliestDepositExpiration = Infinity;
|
|
|
|
for (const denom of selectedDenoms) {
|
|
|
|
const expireDeposit = getTalerStampSec(denom.stampExpireDeposit)!;
|
2017-04-28 23:42:14 +02:00
|
|
|
if (expireDeposit < earliestDepositExpiration) {
|
|
|
|
earliestDepositExpiration = expireDeposit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const possibleDenoms = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.denominations.exchangeBaseUrlIndex,
|
2019-11-20 20:02:48 +01:00
|
|
|
baseUrl,
|
|
|
|
).filter(d => d.isOffered);
|
2017-06-04 18:46:32 +02:00
|
|
|
|
2017-06-04 19:41:43 +02:00
|
|
|
const trustedAuditorPubs = [];
|
2019-11-20 20:02:48 +01:00
|
|
|
const currencyRecord = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.currencies,
|
|
|
|
amount.currency,
|
|
|
|
);
|
2017-06-04 19:41:43 +02:00
|
|
|
if (currencyRecord) {
|
2019-06-26 15:30:32 +02:00
|
|
|
trustedAuditorPubs.push(
|
|
|
|
...currencyRecord.auditors.map(a => a.auditorPub),
|
|
|
|
);
|
2017-06-04 18:46:32 +02:00
|
|
|
}
|
|
|
|
|
2017-06-04 20:16:09 +02:00
|
|
|
let versionMatch;
|
2019-11-20 19:48:43 +01:00
|
|
|
if (exchangeDetails.protocolVersion) {
|
2019-06-26 15:30:32 +02:00
|
|
|
versionMatch = LibtoolVersion.compare(
|
|
|
|
WALLET_PROTOCOL_VERSION,
|
2019-11-20 19:48:43 +01:00
|
|
|
exchangeDetails.protocolVersion,
|
2019-06-26 15:30:32 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
if (
|
|
|
|
versionMatch &&
|
|
|
|
!versionMatch.compatible &&
|
|
|
|
versionMatch.currentCmp === -1
|
|
|
|
) {
|
2019-11-02 00:22:55 +01:00
|
|
|
console.warn(
|
2019-11-21 23:09:43 +01:00
|
|
|
`wallet version ${WALLET_PROTOCOL_VERSION} might be outdated ` +
|
|
|
|
`(exchange has ${exchangeDetails.protocolVersion}), checking for updates`,
|
2019-11-02 00:22:55 +01:00
|
|
|
);
|
2019-11-02 00:24:18 +01:00
|
|
|
if (isFirefox()) {
|
2019-11-20 19:48:43 +01:00
|
|
|
console.log("skipping update check on Firefox");
|
2019-11-02 00:22:55 +01:00
|
|
|
} else {
|
|
|
|
chrome.runtime.requestUpdateCheck((status, details) => {
|
|
|
|
console.log("update check status:", status);
|
|
|
|
});
|
|
|
|
}
|
2017-06-04 20:16:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const ret: ReserveCreationInfo = {
|
|
|
|
earliestDepositExpiration,
|
2016-09-28 18:54:48 +02:00
|
|
|
exchangeInfo,
|
2019-05-08 04:53:26 +02:00
|
|
|
exchangeWireAccounts,
|
2019-11-20 19:48:43 +01:00
|
|
|
exchangeVersion: exchangeDetails.protocolVersion || "unknown",
|
2017-04-28 23:28:27 +02:00
|
|
|
isAudited,
|
|
|
|
isTrusted,
|
2017-06-04 18:46:32 +02:00
|
|
|
numOfferedDenoms: possibleDenoms.length,
|
2016-09-28 18:54:48 +02:00
|
|
|
overhead: Amounts.sub(amount, actualCoinCost).amount,
|
2017-05-28 01:10:54 +02:00
|
|
|
selectedDenoms,
|
2017-06-04 18:46:32 +02:00
|
|
|
trustedAuditorPubs,
|
2017-10-15 19:28:35 +02:00
|
|
|
versionMatch,
|
2018-01-03 14:42:06 +01:00
|
|
|
walletVersion: WALLET_PROTOCOL_VERSION,
|
2019-11-20 19:48:43 +01:00
|
|
|
wireFees: exchangeWireInfo,
|
2017-05-28 01:10:54 +02:00
|
|
|
withdrawFee: acc,
|
2016-09-28 18:54:48 +02:00
|
|
|
};
|
|
|
|
return ret;
|
2016-02-18 22:50:17 +01:00
|
|
|
}
|
|
|
|
|
2019-08-01 23:21:15 +02:00
|
|
|
async getExchangePaytoUri(
|
|
|
|
exchangeBaseUrl: string,
|
|
|
|
supportedTargetTypes: string[],
|
|
|
|
): Promise<string> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const exchangeRecord = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.exchanges,
|
|
|
|
exchangeBaseUrl,
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
if (!exchangeRecord) {
|
|
|
|
throw Error(`Exchange '${exchangeBaseUrl}' not found.`);
|
|
|
|
}
|
|
|
|
const exchangeWireInfo = exchangeRecord.wireInfo;
|
|
|
|
if (!exchangeWireInfo) {
|
|
|
|
throw Error(`Exchange wire info for '${exchangeBaseUrl}' not found.`);
|
|
|
|
}
|
|
|
|
for (let account of exchangeWireInfo.accounts) {
|
2019-07-31 01:33:56 +02:00
|
|
|
const paytoUri = new URI(account.url);
|
|
|
|
if (supportedTargetTypes.includes(paytoUri.authority())) {
|
|
|
|
return account.url;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw Error("no matching exchange account found");
|
|
|
|
}
|
|
|
|
|
2016-01-06 15:39:22 +01:00
|
|
|
/**
|
2019-11-20 19:48:43 +01:00
|
|
|
* Update or add exchange DB entry by fetching the /keys and /wire information.
|
2016-01-06 15:39:22 +01:00
|
|
|
* Optionally link the reserve entry to the new or existing
|
2016-03-01 19:39:17 +01:00
|
|
|
* exchange entry in then DB.
|
2016-01-06 15:39:22 +01:00
|
|
|
*/
|
2019-11-20 19:48:43 +01:00
|
|
|
async updateExchangeFromUrl(
|
2019-06-26 15:30:32 +02:00
|
|
|
baseUrl: string,
|
2019-11-20 19:48:43 +01:00
|
|
|
force: boolean = false,
|
2019-06-26 15:30:32 +02:00
|
|
|
): Promise<ExchangeRecord> {
|
2019-11-20 19:48:43 +01:00
|
|
|
const now = getTimestampNow();
|
|
|
|
baseUrl = canonicalizeBaseUrl(baseUrl);
|
2016-09-28 18:00:13 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const r = await oneShotGet(this.db, Stores.exchanges, baseUrl);
|
2016-09-28 18:00:13 +02:00
|
|
|
if (!r) {
|
2019-11-20 19:48:43 +01:00
|
|
|
const newExchangeRecord: ExchangeRecord = {
|
|
|
|
baseUrl: baseUrl,
|
|
|
|
details: undefined,
|
|
|
|
wireInfo: undefined,
|
|
|
|
updateStatus: ExchangeUpdateStatus.FETCH_KEYS,
|
|
|
|
updateStarted: now,
|
2019-11-21 23:09:43 +01:00
|
|
|
updateReason: "initial",
|
|
|
|
timestampAdded: getTimestampNow(),
|
2016-09-28 18:00:13 +02:00
|
|
|
};
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.exchanges, newExchangeRecord);
|
2016-09-28 18:00:13 +02:00
|
|
|
} else {
|
2019-11-21 23:09:43 +01:00
|
|
|
await runWithWriteTransaction(this.db, [Stores.exchanges], async t => {
|
2019-11-20 19:48:43 +01:00
|
|
|
const rec = await t.get(Stores.exchanges, baseUrl);
|
|
|
|
if (!rec) {
|
|
|
|
return;
|
2017-04-27 03:09:29 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
if (rec.updateStatus != ExchangeUpdateStatus.FETCH_KEYS && !force) {
|
2019-11-20 19:48:43 +01:00
|
|
|
return;
|
2017-04-27 03:09:29 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
if (rec.updateStatus != ExchangeUpdateStatus.FETCH_KEYS && force) {
|
|
|
|
rec.updateReason = "forced";
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
rec.updateStarted = now;
|
|
|
|
rec.updateStatus = ExchangeUpdateStatus.FETCH_KEYS;
|
2019-11-21 23:09:43 +01:00
|
|
|
rec.lastError = undefined;
|
2019-11-20 19:48:43 +01:00
|
|
|
t.put(Stores.exchanges, rec);
|
|
|
|
});
|
2017-04-27 03:09:29 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
await this.updateExchangeWithKeys(baseUrl);
|
|
|
|
await this.updateExchangeWithWireInfo(baseUrl);
|
2017-04-27 03:09:29 +02:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const updatedExchange = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.exchanges,
|
|
|
|
baseUrl,
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
|
|
|
if (!updatedExchange) {
|
|
|
|
// This should practically never happen
|
|
|
|
throw Error("exchange not found");
|
2017-05-01 04:05:16 +02:00
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
return updatedExchange;
|
|
|
|
}
|
2017-05-01 04:05:16 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
private async setExchangeError(
|
|
|
|
baseUrl: string,
|
|
|
|
err: OperationError,
|
|
|
|
): Promise<void> {
|
|
|
|
const mut = (exchange: ExchangeRecord) => {
|
|
|
|
exchange.lastError = err;
|
|
|
|
return exchange;
|
|
|
|
};
|
|
|
|
await oneShotMutate(this.db, Stores.exchanges, baseUrl, mut);
|
2016-05-24 17:30:27 +02:00
|
|
|
}
|
2016-02-18 22:50:17 +01:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
private async setReserveError(
|
|
|
|
reservePub: string,
|
|
|
|
err: OperationError,
|
|
|
|
): Promise<void> {
|
|
|
|
const mut = (reserve: ReserveRecord) => {
|
|
|
|
reserve.lastError = err;
|
|
|
|
return reserve;
|
|
|
|
};
|
|
|
|
await oneShotMutate(this.db, Stores.reserves, reservePub, mut);
|
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
/**
|
|
|
|
* Fetch the exchange's /keys and update our database accordingly.
|
|
|
|
*
|
|
|
|
* Exceptions thrown in this method must be caught and reported
|
|
|
|
* in the pending operations.
|
|
|
|
*/
|
|
|
|
private async updateExchangeWithKeys(baseUrl: string): Promise<void> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const existingExchangeRecord = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.exchanges,
|
|
|
|
baseUrl,
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
if (
|
|
|
|
existingExchangeRecord?.updateStatus != ExchangeUpdateStatus.FETCH_KEYS
|
|
|
|
) {
|
2019-11-20 19:48:43 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const keysUrl = new URI("keys")
|
|
|
|
.absoluteTo(baseUrl)
|
|
|
|
.addQuery("cacheBreaker", WALLET_CACHE_BREAKER_CLIENT_VERSION);
|
|
|
|
let keysResp;
|
|
|
|
try {
|
|
|
|
keysResp = await this.http.get(keysUrl.href());
|
|
|
|
} catch (e) {
|
2019-11-21 23:09:43 +01:00
|
|
|
const m = `Fetching keys failed: ${e.message}`;
|
2019-11-20 19:48:43 +01:00
|
|
|
await this.setExchangeError(baseUrl, {
|
|
|
|
type: "network",
|
2019-11-21 23:09:43 +01:00
|
|
|
details: {
|
|
|
|
requestUrl: e.config?.url,
|
|
|
|
},
|
|
|
|
message: m,
|
2019-11-20 19:48:43 +01:00
|
|
|
});
|
2019-11-21 23:09:43 +01:00
|
|
|
throw new OperationFailedAndReportedError(m);
|
2019-11-20 19:48:43 +01:00
|
|
|
}
|
|
|
|
let exchangeKeysJson: KeysJson;
|
|
|
|
try {
|
|
|
|
exchangeKeysJson = KeysJson.checked(keysResp.responseJson);
|
|
|
|
} catch (e) {
|
2019-11-21 23:09:43 +01:00
|
|
|
const m = `Parsing /keys response failed: ${e.message}`;
|
2019-11-20 19:48:43 +01:00
|
|
|
await this.setExchangeError(baseUrl, {
|
|
|
|
type: "protocol-violation",
|
|
|
|
details: {},
|
2019-11-21 23:09:43 +01:00
|
|
|
message: m,
|
2019-11-20 19:48:43 +01:00
|
|
|
});
|
2019-11-21 23:09:43 +01:00
|
|
|
throw new OperationFailedAndReportedError(m);
|
2016-05-24 17:30:27 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const lastUpdateTimestamp = extractTalerStamp(
|
|
|
|
exchangeKeysJson.list_issue_date,
|
|
|
|
);
|
|
|
|
if (!lastUpdateTimestamp) {
|
|
|
|
const m = `Parsing /keys response failed: invalid list_issue_date.`;
|
|
|
|
await this.setExchangeError(baseUrl, {
|
|
|
|
type: "protocol-violation",
|
|
|
|
details: {},
|
|
|
|
message: m,
|
|
|
|
});
|
2019-11-21 23:09:43 +01:00
|
|
|
throw new OperationFailedAndReportedError(m);
|
2019-11-20 19:48:43 +01:00
|
|
|
}
|
2016-05-24 17:30:27 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
if (exchangeKeysJson.denoms.length === 0) {
|
|
|
|
const m = "exchange doesn't offer any denominations";
|
|
|
|
await this.setExchangeError(baseUrl, {
|
|
|
|
type: "protocol-violation",
|
|
|
|
details: {},
|
|
|
|
message: m,
|
|
|
|
});
|
2019-11-21 23:09:43 +01:00
|
|
|
throw new OperationFailedAndReportedError(m);
|
2019-11-20 19:48:43 +01:00
|
|
|
}
|
2016-05-24 17:30:27 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const protocolVersion = exchangeKeysJson.version;
|
|
|
|
if (!protocolVersion) {
|
|
|
|
const m = "outdate exchange, no version in /keys response";
|
|
|
|
await this.setExchangeError(baseUrl, {
|
|
|
|
type: "protocol-violation",
|
|
|
|
details: {},
|
|
|
|
message: m,
|
|
|
|
});
|
2019-11-21 23:09:43 +01:00
|
|
|
throw new OperationFailedAndReportedError(m);
|
2016-11-16 01:59:39 +01:00
|
|
|
}
|
2016-05-24 17:30:27 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const currency = Amounts.parseOrThrow(exchangeKeysJson.denoms[0].value)
|
|
|
|
.currency;
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
const newDenominations = await Promise.all(
|
|
|
|
exchangeKeysJson.denoms.map(d =>
|
|
|
|
this.denominationRecordFromKeys(baseUrl, d),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.exchanges, Stores.denominations],
|
|
|
|
async tx => {
|
|
|
|
const r = await tx.get(Stores.exchanges, baseUrl);
|
|
|
|
if (!r) {
|
|
|
|
console.warn(`exchange ${baseUrl} no longer present`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (r.details) {
|
|
|
|
// FIXME: We need to do some consistency checks!
|
|
|
|
}
|
|
|
|
r.details = {
|
|
|
|
auditors: exchangeKeysJson.auditors,
|
|
|
|
currency: currency,
|
|
|
|
lastUpdateTime: lastUpdateTimestamp,
|
|
|
|
masterPublicKey: exchangeKeysJson.master_public_key,
|
|
|
|
protocolVersion: protocolVersion,
|
|
|
|
};
|
|
|
|
r.updateStatus = ExchangeUpdateStatus.FETCH_WIRE;
|
|
|
|
r.lastError = undefined;
|
|
|
|
await tx.put(Stores.exchanges, r);
|
|
|
|
|
|
|
|
for (const newDenom of newDenominations) {
|
|
|
|
const oldDenom = await tx.get(Stores.denominations, [
|
|
|
|
baseUrl,
|
|
|
|
newDenom.denomPub,
|
|
|
|
]);
|
|
|
|
if (oldDenom) {
|
|
|
|
// FIXME: Do consistency check
|
|
|
|
} else {
|
|
|
|
await tx.put(Stores.denominations, newDenom);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
}
|
2016-09-28 19:09:10 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
/**
|
|
|
|
* Fetch wire information for an exchange and store it in the database.
|
|
|
|
*
|
|
|
|
* @param exchangeBaseUrl Exchange base URL, assumed to be already normalized.
|
|
|
|
*/
|
2019-11-20 19:48:43 +01:00
|
|
|
private async updateExchangeWithWireInfo(exchangeBaseUrl: string) {
|
2019-11-21 23:09:43 +01:00
|
|
|
const exchange = await this.findExchange(exchangeBaseUrl);
|
|
|
|
if (!exchange) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (exchange.updateStatus != ExchangeUpdateStatus.FETCH_WIRE) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
const reqUrl = new URI("wire")
|
|
|
|
.absoluteTo(exchangeBaseUrl)
|
|
|
|
.addQuery("cacheBreaker", WALLET_CACHE_BREAKER_CLIENT_VERSION);
|
|
|
|
const resp = await this.http.get(reqUrl.href());
|
|
|
|
|
|
|
|
const wiJson = resp.responseJson;
|
|
|
|
if (!wiJson) {
|
|
|
|
throw Error("/wire response malformed");
|
|
|
|
}
|
|
|
|
const wireInfo = ExchangeWireJson.checked(wiJson);
|
2019-11-21 23:09:43 +01:00
|
|
|
const feesForType: { [wireMethod: string]: WireFee[] } = {};
|
|
|
|
for (const wireMethod of Object.keys(wireInfo.fees)) {
|
|
|
|
const feeList: WireFee[] = [];
|
|
|
|
for (const x of wireInfo.fees[wireMethod]) {
|
|
|
|
const startStamp = extractTalerStamp(x.start_date);
|
|
|
|
if (!startStamp) {
|
|
|
|
throw Error("wrong date format");
|
|
|
|
}
|
|
|
|
const endStamp = extractTalerStamp(x.end_date);
|
|
|
|
if (!endStamp) {
|
|
|
|
throw Error("wrong date format");
|
|
|
|
}
|
|
|
|
feeList.push({
|
|
|
|
closingFee: Amounts.parseOrThrow(x.closing_fee),
|
|
|
|
endStamp,
|
|
|
|
sig: x.sig,
|
|
|
|
startStamp,
|
|
|
|
wireFee: Amounts.parseOrThrow(x.wire_fee),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
feesForType[wireMethod] = feeList;
|
|
|
|
}
|
|
|
|
|
|
|
|
await runWithWriteTransaction(this.db, [Stores.exchanges], async tx => {
|
|
|
|
const r = await tx.get(Stores.exchanges, exchangeBaseUrl);
|
|
|
|
if (!r) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (r.updateStatus != ExchangeUpdateStatus.FETCH_WIRE) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
r.wireInfo = {
|
|
|
|
accounts: wireInfo.accounts,
|
|
|
|
feesForType: feesForType,
|
|
|
|
};
|
|
|
|
r.updateStatus = ExchangeUpdateStatus.FINISHED;
|
|
|
|
r.lastError = undefined;
|
|
|
|
await tx.put(Stores.exchanges, r);
|
|
|
|
});
|
2016-01-06 15:39:22 +01:00
|
|
|
}
|
2016-01-05 01:10:31 +01:00
|
|
|
|
2016-02-11 18:17:02 +01:00
|
|
|
/**
|
2017-08-14 04:16:12 +02:00
|
|
|
* Get detailed balance information, sliced by exchange and by currency.
|
2016-02-11 18:17:02 +01:00
|
|
|
*/
|
2016-10-19 18:40:29 +02:00
|
|
|
async getBalances(): Promise<WalletBalance> {
|
2017-08-14 04:16:12 +02:00
|
|
|
/**
|
|
|
|
* Add amount to a balance field, both for
|
|
|
|
* the slicing by exchange and currency.
|
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
function addTo(
|
|
|
|
balance: WalletBalance,
|
|
|
|
field: keyof WalletBalanceEntry,
|
|
|
|
amount: AmountJson,
|
|
|
|
exchange: string,
|
|
|
|
): void {
|
2017-08-14 04:16:12 +02:00
|
|
|
const z = Amounts.getZero(amount.currency);
|
2019-06-26 15:30:32 +02:00
|
|
|
const balanceIdentity = {
|
|
|
|
available: z,
|
|
|
|
paybackAmount: z,
|
|
|
|
pendingIncoming: z,
|
|
|
|
pendingPayment: z,
|
2019-11-02 00:22:55 +01:00
|
|
|
pendingIncomingDirty: z,
|
|
|
|
pendingIncomingRefresh: z,
|
|
|
|
pendingIncomingWithdraw: z,
|
2019-06-26 15:30:32 +02:00
|
|
|
};
|
2017-08-14 04:16:12 +02:00
|
|
|
let entryCurr = balance.byCurrency[amount.currency];
|
|
|
|
if (!entryCurr) {
|
2019-06-26 15:30:32 +02:00
|
|
|
balance.byCurrency[amount.currency] = entryCurr = {
|
|
|
|
...balanceIdentity,
|
|
|
|
};
|
2017-08-14 04:16:12 +02:00
|
|
|
}
|
|
|
|
let entryEx = balance.byExchange[exchange];
|
|
|
|
if (!entryEx) {
|
|
|
|
balance.byExchange[exchange] = entryEx = { ...balanceIdentity };
|
2016-10-19 18:40:29 +02:00
|
|
|
}
|
2017-08-14 04:16:12 +02:00
|
|
|
entryCurr[field] = Amounts.add(entryCurr[field], amount).amount;
|
|
|
|
entryEx[field] = Amounts.add(entryEx[field], amount).amount;
|
2016-10-19 18:40:29 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const balanceStore = {
|
|
|
|
byCurrency: {},
|
|
|
|
byExchange: {},
|
|
|
|
};
|
2017-05-01 04:05:16 +02:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.coins, Stores.refresh, Stores.reserves, Stores.purchases],
|
|
|
|
async tx => {
|
|
|
|
await tx.iter(Stores.coins).forEach(c => {
|
|
|
|
if (c.suspended) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (c.status === CoinStatus.Fresh) {
|
|
|
|
addTo(
|
|
|
|
balanceStore,
|
|
|
|
"available",
|
|
|
|
c.currentAmount,
|
|
|
|
c.exchangeBaseUrl,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (c.status === CoinStatus.Dirty) {
|
|
|
|
addTo(
|
|
|
|
balanceStore,
|
|
|
|
"pendingIncoming",
|
|
|
|
c.currentAmount,
|
|
|
|
c.exchangeBaseUrl,
|
|
|
|
);
|
|
|
|
addTo(
|
|
|
|
balanceStore,
|
|
|
|
"pendingIncomingDirty",
|
|
|
|
c.currentAmount,
|
|
|
|
c.exchangeBaseUrl,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
await tx.iter(Stores.refresh).forEach(r => {
|
|
|
|
// Don't count finished refreshes, since the refresh already resulted
|
|
|
|
// in coins being added to the wallet.
|
|
|
|
if (r.finished) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
addTo(
|
|
|
|
balanceStore,
|
2019-11-20 20:02:48 +01:00
|
|
|
"pendingIncoming",
|
|
|
|
r.valueOutput,
|
|
|
|
r.exchangeBaseUrl,
|
2019-11-20 19:48:43 +01:00
|
|
|
);
|
2019-11-20 20:02:48 +01:00
|
|
|
addTo(
|
|
|
|
balanceStore,
|
|
|
|
"pendingIncomingRefresh",
|
|
|
|
r.valueOutput,
|
|
|
|
r.exchangeBaseUrl,
|
|
|
|
);
|
|
|
|
});
|
2016-10-19 18:40:29 +02:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await tx.iter(Stores.reserves).forEach(r => {
|
2019-11-21 23:09:43 +01:00
|
|
|
if (!r.timestampConfirmed) {
|
2019-11-20 20:02:48 +01:00
|
|
|
return;
|
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
let amount = Amounts.getZero(r.requestedAmount.currency);
|
|
|
|
amount = Amounts.add(amount, r.precoinAmount).amount;
|
|
|
|
addTo(balanceStore, "pendingIncoming", amount, r.exchangeBaseUrl);
|
2019-11-20 20:02:48 +01:00
|
|
|
addTo(
|
|
|
|
balanceStore,
|
|
|
|
"pendingIncomingWithdraw",
|
|
|
|
amount,
|
2019-11-21 23:09:43 +01:00
|
|
|
r.exchangeBaseUrl,
|
2019-11-20 20:02:48 +01:00
|
|
|
);
|
|
|
|
});
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await tx.iter(Stores.reserves).forEach(r => {
|
|
|
|
if (!r.hasPayback) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
addTo(
|
|
|
|
balanceStore,
|
2019-11-20 20:02:48 +01:00
|
|
|
"paybackAmount",
|
2019-11-21 23:09:43 +01:00
|
|
|
r.currentAmount!,
|
|
|
|
r.exchangeBaseUrl,
|
2019-11-20 19:48:43 +01:00
|
|
|
);
|
2019-11-20 20:02:48 +01:00
|
|
|
return balanceStore;
|
|
|
|
});
|
|
|
|
|
|
|
|
await tx.iter(Stores.purchases).forEach(t => {
|
|
|
|
if (t.finished) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (const c of t.payReq.coins) {
|
|
|
|
addTo(
|
|
|
|
balanceStore,
|
|
|
|
"pendingPayment",
|
|
|
|
Amounts.parseOrThrow(c.contribution),
|
|
|
|
c.exchange_url,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
|
|
|
Wallet.enableTracing && console.log("computed balances:", balanceStore);
|
2017-10-15 19:28:35 +02:00
|
|
|
return balanceStore;
|
2016-10-13 02:23:24 +02:00
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
async refresh(oldCoinPub: string, force: boolean = false): Promise<void> {
|
2019-11-20 19:48:43 +01:00
|
|
|
const coin = await oneShotGet(this.db, Stores.coins, oldCoinPub);
|
2016-10-13 02:23:24 +02:00
|
|
|
if (!coin) {
|
2019-11-21 23:09:43 +01:00
|
|
|
console.warn("can't refresh, coin not in database");
|
|
|
|
return;
|
2016-10-13 02:23:24 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
switch (coin.status) {
|
|
|
|
case CoinStatus.Dirty:
|
|
|
|
break;
|
|
|
|
case CoinStatus.Dormant:
|
|
|
|
return;
|
|
|
|
case CoinStatus.Fresh:
|
|
|
|
if (!force) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
2017-04-13 15:05:38 +02:00
|
|
|
}
|
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const exchange = await this.updateExchangeFromUrl(coin.exchangeBaseUrl);
|
2016-10-13 02:23:24 +02:00
|
|
|
if (!exchange) {
|
2019-11-21 23:09:43 +01:00
|
|
|
throw Error("db inconsistent: exchange of coin not found");
|
2016-10-13 02:23:24 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const oldDenom = await oneShotGet(this.db, Stores.denominations, [
|
2019-06-26 15:30:32 +02:00
|
|
|
exchange.baseUrl,
|
|
|
|
coin.denomPub,
|
|
|
|
]);
|
2016-10-13 02:23:24 +02:00
|
|
|
|
|
|
|
if (!oldDenom) {
|
2019-11-21 23:09:43 +01:00
|
|
|
throw Error("db inconsistent: denomination for coin not found");
|
2016-10-13 02:23:24 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const availableDenoms: DenominationRecord[] = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.denominations.exchangeBaseUrlIndex,
|
|
|
|
exchange.baseUrl,
|
|
|
|
).toArray();
|
2016-10-13 02:23:24 +02:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const availableAmount = Amounts.sub(coin.currentAmount, oldDenom.feeRefresh)
|
|
|
|
.amount;
|
2016-10-14 02:13:06 +02:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const newCoinDenoms = getWithdrawDenomList(
|
|
|
|
availableAmount,
|
|
|
|
availableDenoms,
|
|
|
|
);
|
2016-10-13 02:23:24 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
if (newCoinDenoms.length === 0) {
|
2019-11-21 23:09:43 +01:00
|
|
|
logger.trace(
|
|
|
|
`not refreshing, available amount ${amountToPretty(
|
|
|
|
availableAmount,
|
|
|
|
)} too small`,
|
|
|
|
);
|
|
|
|
await oneShotMutate(this.db, Stores.coins, oldCoinPub, x => {
|
|
|
|
if (x.status != coin.status) {
|
|
|
|
// Concurrent modification?
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
x.status = CoinStatus.Dormant;
|
|
|
|
return x;
|
|
|
|
});
|
2017-08-30 17:08:54 +02:00
|
|
|
this.notifier.notify();
|
2019-11-21 23:09:43 +01:00
|
|
|
return;
|
2016-10-17 15:58:36 +02:00
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const refreshSession: RefreshSessionRecord = await this.cryptoApi.createRefreshSession(
|
|
|
|
exchange.baseUrl,
|
|
|
|
3,
|
|
|
|
coin,
|
|
|
|
newCoinDenoms,
|
|
|
|
oldDenom.feeRefresh,
|
|
|
|
);
|
2016-10-17 15:58:36 +02:00
|
|
|
|
2016-11-15 15:07:17 +01:00
|
|
|
function mutateCoin(c: CoinRecord): CoinRecord {
|
2019-06-26 15:30:32 +02:00
|
|
|
const r = Amounts.sub(c.currentAmount, refreshSession.valueWithFee);
|
2016-10-19 23:55:58 +02:00
|
|
|
if (r.saturated) {
|
|
|
|
// Something else must have written the coin value
|
2019-11-21 23:09:43 +01:00
|
|
|
throw TransactionAbort;
|
2016-10-19 23:55:58 +02:00
|
|
|
}
|
|
|
|
c.currentAmount = r.amount;
|
2019-11-21 23:09:43 +01:00
|
|
|
c.status = CoinStatus.Dormant;
|
2016-10-19 23:55:58 +02:00
|
|
|
return c;
|
|
|
|
}
|
2016-10-17 15:58:36 +02:00
|
|
|
|
2017-04-13 15:05:38 +02:00
|
|
|
// Store refresh session and subtract refreshed amount from
|
|
|
|
// coin in the same transaction.
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.refresh, Stores.coins],
|
|
|
|
async tx => {
|
2019-11-21 23:09:43 +01:00
|
|
|
await tx.put(Stores.refresh, refreshSession);
|
2019-11-20 20:02:48 +01:00
|
|
|
await tx.mutate(Stores.coins, coin.coinPub, mutateCoin);
|
|
|
|
},
|
|
|
|
);
|
2019-11-21 23:09:43 +01:00
|
|
|
logger.info(`created refresh session ${refreshSession.refreshSessionId}`);
|
2017-08-30 17:08:54 +02:00
|
|
|
this.notifier.notify();
|
2017-08-27 05:42:46 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
await this.processRefreshSession(refreshSession.refreshSessionId);
|
2016-10-17 23:49:04 +02:00
|
|
|
}
|
2016-10-13 02:23:24 +02:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
async processRefreshSession(refreshSessionId: string) {
|
|
|
|
const refreshSession = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.refresh,
|
|
|
|
refreshSessionId,
|
|
|
|
);
|
|
|
|
if (!refreshSession) {
|
|
|
|
return;
|
2016-10-17 23:49:04 +02:00
|
|
|
}
|
|
|
|
if (refreshSession.finished) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (typeof refreshSession.norevealIndex !== "number") {
|
2019-11-21 23:09:43 +01:00
|
|
|
await this.refreshMelt(refreshSession.refreshSessionId);
|
2016-10-17 23:49:04 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
await this.refreshReveal(refreshSession.refreshSessionId);
|
|
|
|
logger.trace("refresh finished");
|
2016-10-17 15:58:36 +02:00
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
async refreshMelt(refreshSessionId: string): Promise<void> {
|
|
|
|
const refreshSession = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.refresh,
|
|
|
|
refreshSessionId,
|
|
|
|
);
|
|
|
|
if (!refreshSession) {
|
|
|
|
return;
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
if (refreshSession.norevealIndex !== undefined) {
|
2016-10-17 15:58:36 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const coin = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.coins,
|
|
|
|
refreshSession.meltCoinPub,
|
|
|
|
);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2016-10-17 15:58:36 +02:00
|
|
|
if (!coin) {
|
|
|
|
console.error("can't melt coin, it does not exist");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const reqUrl = new URI("refresh/melt").absoluteTo(
|
|
|
|
refreshSession.exchangeBaseUrl,
|
|
|
|
);
|
2017-12-09 03:37:21 +01:00
|
|
|
const meltReq = {
|
2016-10-14 02:13:06 +02:00
|
|
|
coin_pub: coin.coinPub,
|
2017-05-28 01:10:54 +02:00
|
|
|
confirm_sig: refreshSession.confirmSig,
|
2019-05-08 07:01:17 +02:00
|
|
|
denom_pub_hash: coin.denomPubHash,
|
2016-10-14 02:13:06 +02:00
|
|
|
denom_sig: coin.denomSig,
|
2017-12-09 03:37:21 +01:00
|
|
|
rc: refreshSession.hash,
|
2016-10-14 02:13:06 +02:00
|
|
|
value_with_fee: refreshSession.valueWithFee,
|
|
|
|
};
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("melt request:", meltReq);
|
2017-12-09 03:37:21 +01:00
|
|
|
const resp = await this.http.postJson(reqUrl.href(), meltReq);
|
2016-10-13 02:36:33 +02:00
|
|
|
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("melt response:", resp.responseJson);
|
2016-09-28 18:00:13 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
if (resp.status !== 200) {
|
2019-07-31 01:33:56 +02:00
|
|
|
console.error(resp.responseJson);
|
2016-10-14 02:13:06 +02:00
|
|
|
throw Error("refresh failed");
|
|
|
|
}
|
|
|
|
|
2019-07-31 01:33:56 +02:00
|
|
|
const respJson = resp.responseJson;
|
2016-10-14 02:13:06 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const norevealIndex = respJson.noreveal_index;
|
2016-10-14 02:13:06 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
if (typeof norevealIndex !== "number") {
|
2016-10-14 02:13:06 +02:00
|
|
|
throw Error("invalid response");
|
|
|
|
}
|
|
|
|
|
|
|
|
refreshSession.norevealIndex = norevealIndex;
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
await oneShotMutate(this.db, Stores.refresh, refreshSessionId, rs => {
|
|
|
|
if (rs.norevealIndex !== undefined) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (rs.finished) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
rs.norevealIndex = norevealIndex;
|
|
|
|
return rs;
|
|
|
|
});
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2017-08-30 17:08:54 +02:00
|
|
|
this.notifier.notify();
|
2016-01-05 01:10:31 +01:00
|
|
|
}
|
2016-01-24 02:29:13 +01:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
private async refreshReveal(refreshSessionId: string): Promise<void> {
|
|
|
|
const refreshSession = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.refresh,
|
|
|
|
refreshSessionId,
|
|
|
|
);
|
|
|
|
if (!refreshSession) {
|
|
|
|
return;
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
const norevealIndex = refreshSession.norevealIndex;
|
|
|
|
if (norevealIndex === undefined) {
|
2016-10-14 02:13:06 +02:00
|
|
|
throw Error("can't reveal without melting first");
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
const privs = Array.from(refreshSession.transferPrivs);
|
2016-10-14 02:13:06 +02:00
|
|
|
privs.splice(norevealIndex, 1);
|
|
|
|
|
2017-12-09 03:37:21 +01:00
|
|
|
const preCoins = refreshSession.preCoinsForGammas[norevealIndex];
|
|
|
|
if (!preCoins) {
|
|
|
|
throw Error("refresh index error");
|
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const meltCoinRecord = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.coins,
|
|
|
|
refreshSession.meltCoinPub,
|
|
|
|
);
|
2019-06-26 15:30:32 +02:00
|
|
|
if (!meltCoinRecord) {
|
|
|
|
throw Error("inconsistent database");
|
|
|
|
}
|
|
|
|
|
2017-12-09 03:37:21 +01:00
|
|
|
const evs = preCoins.map((x: RefreshPreCoinRecord) => x.coinEv);
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const linkSigs: string[] = [];
|
|
|
|
for (let i = 0; i < refreshSession.newDenoms.length; i++) {
|
|
|
|
const linkSig = await this.cryptoApi.signCoinLink(
|
|
|
|
meltCoinRecord.coinPriv,
|
|
|
|
refreshSession.newDenomHashes[i],
|
|
|
|
refreshSession.meltCoinPub,
|
|
|
|
refreshSession.transferPubs[norevealIndex],
|
|
|
|
preCoins[i].coinEv,
|
|
|
|
);
|
|
|
|
linkSigs.push(linkSig);
|
|
|
|
}
|
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const req = {
|
2017-12-09 03:37:21 +01:00
|
|
|
coin_evs: evs,
|
|
|
|
new_denoms_h: refreshSession.newDenomHashes,
|
|
|
|
rc: refreshSession.hash,
|
2017-05-28 01:10:54 +02:00
|
|
|
transfer_privs: privs,
|
2017-12-09 03:37:21 +01:00
|
|
|
transfer_pub: refreshSession.transferPubs[norevealIndex],
|
2019-06-26 15:30:32 +02:00
|
|
|
link_sigs: linkSigs,
|
2016-10-14 02:13:06 +02:00
|
|
|
};
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const reqUrl = new URI("refresh/reveal").absoluteTo(
|
|
|
|
refreshSession.exchangeBaseUrl,
|
|
|
|
);
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("reveal request:", req);
|
2019-08-28 02:49:27 +02:00
|
|
|
|
|
|
|
let resp;
|
|
|
|
try {
|
|
|
|
resp = await this.http.postJson(reqUrl.href(), req);
|
|
|
|
} catch (e) {
|
|
|
|
console.error("got error during /refresh/reveal request");
|
|
|
|
return;
|
|
|
|
}
|
2016-10-14 02:13:06 +02:00
|
|
|
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("session:", refreshSession);
|
|
|
|
Wallet.enableTracing && console.log("reveal response:", resp);
|
2016-10-17 15:58:36 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
if (resp.status !== 200) {
|
2019-08-26 01:39:13 +02:00
|
|
|
console.error("error: /refresh/reveal returned status " + resp.status);
|
2016-10-17 15:58:36 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-07-31 01:33:56 +02:00
|
|
|
const respJson = resp.responseJson;
|
2016-10-17 15:58:36 +02:00
|
|
|
|
|
|
|
if (!respJson.ev_sigs || !Array.isArray(respJson.ev_sigs)) {
|
2019-08-26 01:39:13 +02:00
|
|
|
console.error("/refresh/reveal did not contain ev_sigs");
|
|
|
|
return;
|
2016-10-17 15:58:36 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchange = await this.findExchange(refreshSession.exchangeBaseUrl);
|
2016-10-17 15:58:36 +02:00
|
|
|
if (!exchange) {
|
|
|
|
console.error(`exchange ${refreshSession.exchangeBaseUrl} not found`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const coins: CoinRecord[] = [];
|
2016-10-17 23:49:04 +02:00
|
|
|
|
2016-10-17 15:58:36 +02:00
|
|
|
for (let i = 0; i < respJson.ev_sigs.length; i++) {
|
2019-11-20 19:48:43 +01:00
|
|
|
const denom = await oneShotGet(this.db, Stores.denominations, [
|
2019-06-26 15:30:32 +02:00
|
|
|
refreshSession.exchangeBaseUrl,
|
|
|
|
refreshSession.newDenoms[i],
|
|
|
|
]);
|
2016-10-17 15:58:36 +02:00
|
|
|
if (!denom) {
|
|
|
|
console.error("denom not found");
|
|
|
|
continue;
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
const pc =
|
|
|
|
refreshSession.preCoinsForGammas[refreshSession.norevealIndex!][i];
|
|
|
|
const denomSig = await this.cryptoApi.rsaUnblind(
|
|
|
|
respJson.ev_sigs[i].ev_sig,
|
|
|
|
pc.blindingKey,
|
|
|
|
denom.denomPub,
|
|
|
|
);
|
2017-05-28 01:10:54 +02:00
|
|
|
const coin: CoinRecord = {
|
2017-05-01 04:33:47 +02:00
|
|
|
blindingKey: pc.blindingKey,
|
2016-10-17 15:58:36 +02:00
|
|
|
coinPriv: pc.privateKey,
|
2017-05-28 01:10:54 +02:00
|
|
|
coinPub: pc.publicKey,
|
2016-10-17 15:58:36 +02:00
|
|
|
currentAmount: denom.value,
|
2017-05-28 01:10:54 +02:00
|
|
|
denomPub: denom.denomPub,
|
2019-05-08 07:01:17 +02:00
|
|
|
denomPubHash: denom.denomPubHash,
|
2017-05-28 01:10:54 +02:00
|
|
|
denomSig,
|
2016-10-17 15:58:36 +02:00
|
|
|
exchangeBaseUrl: refreshSession.exchangeBaseUrl,
|
2017-05-28 01:10:54 +02:00
|
|
|
reservePub: undefined,
|
2017-04-13 16:08:41 +02:00
|
|
|
status: CoinStatus.Fresh,
|
2016-10-17 15:58:36 +02:00
|
|
|
};
|
|
|
|
|
2016-10-17 23:49:04 +02:00
|
|
|
coins.push(coin);
|
2016-10-17 15:58:36 +02:00
|
|
|
}
|
2016-10-17 23:49:04 +02:00
|
|
|
|
|
|
|
refreshSession.finished = true;
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.coins, Stores.refresh],
|
|
|
|
async tx => {
|
2019-11-21 23:09:43 +01:00
|
|
|
const rs = await tx.get(Stores.refresh, refreshSessionId);
|
|
|
|
if (!rs) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (rs.finished) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-20 20:02:48 +01:00
|
|
|
for (let coin of coins) {
|
|
|
|
await tx.put(Stores.coins, coin);
|
|
|
|
}
|
|
|
|
await tx.put(Stores.refresh, refreshSession);
|
|
|
|
},
|
|
|
|
);
|
2017-08-30 17:08:54 +02:00
|
|
|
this.notifier.notify();
|
2016-10-14 02:13:06 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
async findExchange(
|
|
|
|
exchangeBaseUrl: string,
|
|
|
|
): Promise<ExchangeRecord | undefined> {
|
2019-11-20 19:48:43 +01:00
|
|
|
return await oneShotGet(this.db, Stores.exchanges, exchangeBaseUrl);
|
|
|
|
}
|
|
|
|
|
2016-02-11 18:17:02 +01:00
|
|
|
/**
|
|
|
|
* Retrive the full event history for this wallet.
|
|
|
|
*/
|
2019-11-20 19:48:43 +01:00
|
|
|
async getHistory(
|
|
|
|
historyQuery?: HistoryQuery,
|
2019-11-21 23:09:43 +01:00
|
|
|
): Promise<{ history: HistoryEvent[] }> {
|
|
|
|
const history: HistoryEvent[] = [];
|
2017-10-15 18:30:02 +02:00
|
|
|
|
|
|
|
// FIXME: do pagination instead of generating the full history
|
|
|
|
|
2019-08-22 23:36:36 +02:00
|
|
|
// We uniquely identify history rows via their timestamp.
|
|
|
|
// This works as timestamps are guaranteed to be monotonically
|
2019-08-24 18:42:00 +02:00
|
|
|
// increasing even
|
2019-08-22 23:36:36 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const proposals = await oneShotIter(this.db, Stores.proposals).toArray();
|
2017-10-15 19:28:35 +02:00
|
|
|
for (const p of proposals) {
|
2017-10-15 18:30:02 +02:00
|
|
|
history.push({
|
|
|
|
detail: {
|
|
|
|
contractTermsHash: p.contractTermsHash,
|
|
|
|
merchantName: p.contractTerms.merchant.name,
|
|
|
|
},
|
2017-10-15 19:28:35 +02:00
|
|
|
timestamp: p.timestamp,
|
2019-11-19 16:16:12 +01:00
|
|
|
type: "claim-order",
|
2019-11-21 23:09:43 +01:00
|
|
|
explicit: false,
|
2019-11-19 16:16:12 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const withdrawals = await oneShotIter(
|
|
|
|
this.db,
|
|
|
|
Stores.withdrawals,
|
|
|
|
).toArray();
|
2019-11-19 16:16:12 +01:00
|
|
|
for (const w of withdrawals) {
|
|
|
|
history.push({
|
|
|
|
detail: {
|
|
|
|
withdrawalAmount: w.withdrawalAmount,
|
|
|
|
},
|
|
|
|
timestamp: w.startTimestamp,
|
|
|
|
type: "withdraw",
|
2019-11-21 23:09:43 +01:00
|
|
|
explicit: false,
|
2017-10-15 18:30:02 +02:00
|
|
|
});
|
2016-01-24 02:29:13 +01:00
|
|
|
}
|
2016-01-26 17:21:17 +01:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const purchases = await oneShotIter(this.db, Stores.purchases).toArray();
|
2017-10-15 19:28:35 +02:00
|
|
|
for (const p of purchases) {
|
2017-10-15 18:30:02 +02:00
|
|
|
history.push({
|
|
|
|
detail: {
|
|
|
|
amount: p.contractTerms.amount,
|
|
|
|
contractTermsHash: p.contractTermsHash,
|
|
|
|
fulfillmentUrl: p.contractTerms.fulfillment_url,
|
|
|
|
merchantName: p.contractTerms.merchant.name,
|
|
|
|
},
|
2017-10-15 19:28:35 +02:00
|
|
|
timestamp: p.timestamp,
|
|
|
|
type: "pay",
|
2019-11-21 23:09:43 +01:00
|
|
|
explicit: false,
|
2017-10-15 18:30:02 +02:00
|
|
|
});
|
|
|
|
if (p.timestamp_refund) {
|
2018-01-29 22:58:47 +01:00
|
|
|
const contractAmount = Amounts.parseOrThrow(p.contractTerms.amount);
|
2019-06-26 15:30:32 +02:00
|
|
|
const amountsPending = Object.keys(p.refundsPending).map(x =>
|
|
|
|
Amounts.parseOrThrow(p.refundsPending[x].refund_amount),
|
2018-01-29 22:58:47 +01:00
|
|
|
);
|
2019-06-26 15:30:32 +02:00
|
|
|
const amountsDone = Object.keys(p.refundsDone).map(x =>
|
|
|
|
Amounts.parseOrThrow(p.refundsDone[x].refund_amount),
|
2018-01-29 22:58:47 +01:00
|
|
|
);
|
2017-10-15 18:30:02 +02:00
|
|
|
const amounts: AmountJson[] = amountsPending.concat(amountsDone);
|
2019-06-26 15:30:32 +02:00
|
|
|
const amount = Amounts.add(
|
|
|
|
Amounts.getZero(contractAmount.currency),
|
|
|
|
...amounts,
|
|
|
|
).amount;
|
2017-10-15 18:30:02 +02:00
|
|
|
|
|
|
|
history.push({
|
|
|
|
detail: {
|
|
|
|
contractTermsHash: p.contractTermsHash,
|
|
|
|
fulfillmentUrl: p.contractTerms.fulfillment_url,
|
|
|
|
merchantName: p.contractTerms.merchant.name,
|
2017-10-15 19:28:35 +02:00
|
|
|
refundAmount: amount,
|
2017-10-15 18:30:02 +02:00
|
|
|
},
|
2017-10-15 19:28:35 +02:00
|
|
|
timestamp: p.timestamp_refund,
|
|
|
|
type: "refund",
|
2019-11-21 23:09:43 +01:00
|
|
|
explicit: false,
|
2017-10-15 18:30:02 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2016-09-28 17:52:36 +02:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const reserves = await oneShotIter(this.db, Stores.reserves).toArray();
|
|
|
|
|
2017-10-15 19:28:35 +02:00
|
|
|
for (const r of reserves) {
|
2019-11-21 23:09:43 +01:00
|
|
|
const reserveType = r.bankWithdrawStatusUrl ? "taler-bank" : "manual";
|
2017-10-15 18:30:02 +02:00
|
|
|
history.push({
|
|
|
|
detail: {
|
2019-11-21 23:09:43 +01:00
|
|
|
exchangeBaseUrl: r.exchangeBaseUrl,
|
|
|
|
requestedAmount: Amounts.toString(r.requestedAmount),
|
|
|
|
reservePub: r.reservePub,
|
|
|
|
reserveType,
|
|
|
|
bankWithdrawStatusUrl: r.bankWithdrawStatusUrl,
|
2017-10-15 18:30:02 +02:00
|
|
|
},
|
2017-10-15 19:28:35 +02:00
|
|
|
timestamp: r.created,
|
2019-11-21 23:09:43 +01:00
|
|
|
type: "reserve-created",
|
|
|
|
explicit: false,
|
2017-10-15 18:30:02 +02:00
|
|
|
});
|
2019-11-21 23:09:43 +01:00
|
|
|
if (r.timestampConfirmed) {
|
2017-10-15 18:30:02 +02:00
|
|
|
history.push({
|
|
|
|
detail: {
|
2019-11-21 23:09:43 +01:00
|
|
|
exchangeBaseUrl: r.exchangeBaseUrl,
|
|
|
|
requestedAmount: Amounts.toString(r.requestedAmount),
|
|
|
|
reservePub: r.reservePub,
|
|
|
|
reserveType,
|
|
|
|
bankWithdrawStatusUrl: r.bankWithdrawStatusUrl,
|
2017-10-15 18:30:02 +02:00
|
|
|
},
|
2019-11-21 23:09:43 +01:00
|
|
|
timestamp: r.created,
|
|
|
|
type: "reserve-confirmed",
|
|
|
|
explicit: false,
|
2017-10-15 18:30:02 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
const tips: TipRecord[] = await oneShotIter(this.db, Stores.tips).toArray();
|
2017-12-12 15:38:03 +01:00
|
|
|
for (const tip of tips) {
|
|
|
|
history.push({
|
|
|
|
detail: {
|
2017-12-12 16:51:13 +01:00
|
|
|
accepted: tip.accepted,
|
2018-01-03 14:42:06 +01:00
|
|
|
amount: tip.amount,
|
|
|
|
merchantDomain: tip.merchantDomain,
|
2017-12-12 15:38:03 +01:00
|
|
|
tipId: tip.tipId,
|
|
|
|
},
|
|
|
|
timestamp: tip.timestamp,
|
2019-11-21 23:09:43 +01:00
|
|
|
explicit: false,
|
2017-12-12 15:38:03 +01:00
|
|
|
type: "tip",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
await oneShotIter(this.db, Stores.exchanges).forEach(exchange => {
|
|
|
|
history.push({
|
|
|
|
type: "exchange-added",
|
|
|
|
explicit: false,
|
|
|
|
timestamp: exchange.timestampAdded,
|
|
|
|
detail: {
|
|
|
|
exchangeBaseUrl: exchange.baseUrl,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
history.sort((h1, h2) => Math.sign(h1.timestamp.t_ms - h2.timestamp.t_ms));
|
2017-10-15 19:28:35 +02:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
return { history };
|
2016-10-12 02:55:53 +02:00
|
|
|
}
|
|
|
|
|
2019-11-19 16:16:12 +01:00
|
|
|
async getPendingOperations(): Promise<PendingOperationsResponse> {
|
2019-11-20 19:48:43 +01:00
|
|
|
const pendingOperations: PendingOperationInfo[] = [];
|
|
|
|
const exchanges = await this.getExchanges();
|
|
|
|
for (let e of exchanges) {
|
|
|
|
switch (e.updateStatus) {
|
2019-11-21 23:09:43 +01:00
|
|
|
case ExchangeUpdateStatus.FINISHED:
|
|
|
|
if (e.lastError) {
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "bug",
|
|
|
|
message:
|
|
|
|
"Exchange record is in FINISHED state but has lastError set",
|
|
|
|
details: {
|
|
|
|
exchangeBaseUrl: e.baseUrl,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
if (!e.details) {
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "bug",
|
|
|
|
message:
|
|
|
|
"Exchange record does not have details, but no update in progress.",
|
|
|
|
details: {
|
|
|
|
exchangeBaseUrl: e.baseUrl,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
if (!e.wireInfo) {
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "bug",
|
|
|
|
message:
|
|
|
|
"Exchange record does not have wire info, but no update in progress.",
|
|
|
|
details: {
|
|
|
|
exchangeBaseUrl: e.baseUrl,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
break;
|
|
|
|
case ExchangeUpdateStatus.FETCH_KEYS:
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "exchange-update",
|
|
|
|
stage: "fetch-keys",
|
|
|
|
exchangeBaseUrl: e.baseUrl,
|
2019-11-21 23:09:43 +01:00
|
|
|
lastError: e.lastError,
|
|
|
|
reason: e.updateReason || "unknown",
|
2019-11-20 19:48:43 +01:00
|
|
|
});
|
|
|
|
break;
|
|
|
|
case ExchangeUpdateStatus.FETCH_WIRE:
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "exchange-update",
|
|
|
|
stage: "fetch-wire",
|
|
|
|
exchangeBaseUrl: e.baseUrl,
|
2019-11-21 23:09:43 +01:00
|
|
|
lastError: e.lastError,
|
|
|
|
reason: e.updateReason || "unknown",
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "bug",
|
|
|
|
message: "Unknown exchangeUpdateStatus",
|
|
|
|
details: {
|
|
|
|
exchangeBaseUrl: e.baseUrl,
|
|
|
|
exchangeUpdateStatus: e.updateStatus,
|
|
|
|
},
|
2019-11-20 19:48:43 +01:00
|
|
|
});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
await oneShotIter(this.db, Stores.reserves).forEach(reserve => {
|
|
|
|
const reserveType = reserve.bankWithdrawStatusUrl
|
|
|
|
? "taler-bank"
|
|
|
|
: "manual";
|
|
|
|
switch (reserve.reserveStatus) {
|
|
|
|
case ReserveRecordStatus.DORMANT:
|
|
|
|
// nothing to report as pending
|
|
|
|
break;
|
|
|
|
case ReserveRecordStatus.WITHDRAWING:
|
|
|
|
case ReserveRecordStatus.UNCONFIRMED:
|
|
|
|
case ReserveRecordStatus.QUERYING_STATUS:
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "reserve",
|
|
|
|
stage: reserve.reserveStatus,
|
|
|
|
timestampCreated: reserve.created,
|
|
|
|
reserveType,
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "bug",
|
|
|
|
message: "Unknown reserve record status",
|
|
|
|
details: {
|
|
|
|
reservePub: reserve.reservePub,
|
|
|
|
reserveStatus: reserve.reserveStatus,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
await oneShotIter(this.db, Stores.refresh).forEach(r => {
|
|
|
|
if (r.finished) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let refreshStatus: string;
|
|
|
|
if (r.norevealIndex === undefined) {
|
|
|
|
refreshStatus = "melt";
|
|
|
|
} else {
|
|
|
|
refreshStatus = "reveal";
|
|
|
|
}
|
|
|
|
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "refresh",
|
|
|
|
oldCoinPub: r.meltCoinPub,
|
|
|
|
refreshStatus,
|
|
|
|
refreshOutputSize: r.newDenoms.length,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
await oneShotIter(this.db, Stores.precoins).forEach(pc => {
|
|
|
|
pendingOperations.push({
|
|
|
|
type: "withdraw",
|
|
|
|
stage: "planchet",
|
|
|
|
reservePub: pc.reservePub,
|
|
|
|
});
|
|
|
|
});
|
2019-11-19 16:16:12 +01:00
|
|
|
return {
|
2019-11-20 19:48:43 +01:00
|
|
|
pendingOperations,
|
2019-11-19 16:16:12 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-11-16 01:59:39 +01:00
|
|
|
async getDenoms(exchangeUrl: string): Promise<DenominationRecord[]> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const denoms = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.denominations.exchangeBaseUrlIndex,
|
|
|
|
exchangeUrl,
|
|
|
|
).toArray();
|
2016-11-16 01:59:39 +01:00
|
|
|
return denoms;
|
|
|
|
}
|
2016-11-13 10:17:39 +01:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
async getProposal(
|
|
|
|
proposalId: number,
|
|
|
|
): Promise<ProposalDownloadRecord | undefined> {
|
2019-11-20 19:48:43 +01:00
|
|
|
const proposal = await oneShotGet(this.db, Stores.proposals, proposalId);
|
2017-05-31 16:04:14 +02:00
|
|
|
return proposal;
|
2016-11-13 10:17:39 +01:00
|
|
|
}
|
|
|
|
|
2016-11-15 15:07:17 +01:00
|
|
|
async getExchanges(): Promise<ExchangeRecord[]> {
|
2019-11-20 19:48:43 +01:00
|
|
|
return await oneShotIter(this.db, Stores.exchanges).toArray();
|
2016-01-24 02:29:13 +01:00
|
|
|
}
|
2016-02-23 14:07:53 +01:00
|
|
|
|
2017-03-24 17:54:22 +01:00
|
|
|
async getCurrencies(): Promise<CurrencyRecord[]> {
|
2019-11-20 19:48:43 +01:00
|
|
|
return await oneShotIter(this.db, Stores.currencies).toArray();
|
2017-03-24 17:54:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
async updateCurrency(currencyRecord: CurrencyRecord): Promise<void> {
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("updating currency to", currencyRecord);
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.currencies, currencyRecord);
|
2017-03-24 17:54:22 +01:00
|
|
|
this.notifier.notify();
|
|
|
|
}
|
|
|
|
|
2016-10-13 02:23:24 +02:00
|
|
|
async getReserves(exchangeBaseUrl: string): Promise<ReserveRecord[]> {
|
2019-11-20 20:02:48 +01:00
|
|
|
return await oneShotIter(this.db, Stores.reserves).filter(
|
2019-11-21 23:09:43 +01:00
|
|
|
r => r.exchangeBaseUrl === exchangeBaseUrl,
|
2019-11-20 20:02:48 +01:00
|
|
|
);
|
2016-10-12 02:55:53 +02:00
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
async getCoinsForExchange(exchangeBaseUrl: string): Promise<CoinRecord[]> {
|
2019-11-20 20:02:48 +01:00
|
|
|
return await oneShotIter(this.db, Stores.coins).filter(
|
|
|
|
c => c.exchangeBaseUrl === exchangeBaseUrl,
|
|
|
|
);
|
2016-10-12 02:55:53 +02:00
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
async getCoins(): Promise<CoinRecord[]> {
|
|
|
|
return await oneShotIter(this.db, Stores.coins).toArray();
|
|
|
|
}
|
|
|
|
|
2016-11-15 15:07:17 +01:00
|
|
|
async getPreCoins(exchangeBaseUrl: string): Promise<PreCoinRecord[]> {
|
2019-11-20 20:02:48 +01:00
|
|
|
return await oneShotIter(this.db, Stores.precoins).filter(
|
|
|
|
c => c.exchangeBaseUrl === exchangeBaseUrl,
|
|
|
|
);
|
2016-10-12 02:55:53 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
private async hashContract(contract: ContractTerms): Promise<string> {
|
2016-09-28 23:41:34 +02:00
|
|
|
return this.cryptoApi.hashString(canonicalJson(contract));
|
|
|
|
}
|
|
|
|
|
2017-05-01 04:05:16 +02:00
|
|
|
async payback(coinPub: string): Promise<void> {
|
2019-11-20 19:48:43 +01:00
|
|
|
let coin = await oneShotGet(this.db, Stores.coins, coinPub);
|
2017-05-01 04:05:16 +02:00
|
|
|
if (!coin) {
|
|
|
|
throw Error(`Coin ${coinPub} not found, can't request payback`);
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
const reservePub = coin.reservePub;
|
2017-05-01 04:33:47 +02:00
|
|
|
if (!reservePub) {
|
|
|
|
throw Error(`Can't request payback for a refreshed coin`);
|
2017-05-01 04:05:16 +02:00
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
const reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
|
2017-05-01 04:05:16 +02:00
|
|
|
if (!reserve) {
|
|
|
|
throw Error(`Reserve of coin ${coinPub} not found`);
|
|
|
|
}
|
|
|
|
switch (coin.status) {
|
2019-11-21 23:09:43 +01:00
|
|
|
case CoinStatus.Dormant:
|
|
|
|
throw Error(`Can't do payback for coin ${coinPub} since it's dormant`);
|
2017-05-01 04:05:16 +02:00
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
coin.status = CoinStatus.Dormant;
|
2017-05-01 04:05:16 +02:00
|
|
|
// Even if we didn't get the payback yet, we suspend withdrawal, since
|
|
|
|
// technically we might update reserve status before we get the response
|
|
|
|
// from the reserve for the payback request.
|
|
|
|
reserve.hasPayback = true;
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.coins, Stores.reserves],
|
|
|
|
async tx => {
|
|
|
|
await tx.put(Stores.coins, coin!!);
|
|
|
|
await tx.put(Stores.reserves, reserve);
|
|
|
|
},
|
|
|
|
);
|
2017-08-30 17:08:54 +02:00
|
|
|
this.notifier.notify();
|
2017-05-01 04:05:16 +02:00
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const paybackRequest = await this.cryptoApi.createPaybackRequest(coin);
|
|
|
|
const reqUrl = new URI("payback").absoluteTo(coin.exchangeBaseUrl);
|
|
|
|
const resp = await this.http.postJson(reqUrl.href(), paybackRequest);
|
|
|
|
if (resp.status !== 200) {
|
2017-05-01 04:05:16 +02:00
|
|
|
throw Error();
|
|
|
|
}
|
2019-07-31 01:33:56 +02:00
|
|
|
const paybackConfirmation = PaybackConfirmation.checked(resp.responseJson);
|
2017-05-28 01:10:54 +02:00
|
|
|
if (paybackConfirmation.reserve_pub !== coin.reservePub) {
|
2017-05-01 04:05:16 +02:00
|
|
|
throw Error(`Coin's reserve doesn't match reserve on payback`);
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
coin = await oneShotGet(this.db, Stores.coins, coinPub);
|
2017-05-01 04:05:16 +02:00
|
|
|
if (!coin) {
|
|
|
|
throw Error(`Coin ${coinPub} not found, can't confirm payback`);
|
|
|
|
}
|
2019-11-21 23:09:43 +01:00
|
|
|
coin.status = CoinStatus.Dormant;
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.coins, coin);
|
2017-08-30 17:08:54 +02:00
|
|
|
this.notifier.notify();
|
2017-05-01 04:33:47 +02:00
|
|
|
await this.updateReserve(reservePub!);
|
2017-05-01 04:05:16 +02:00
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
private async denominationRecordFromKeys(
|
2019-06-26 15:30:32 +02:00
|
|
|
exchangeBaseUrl: string,
|
|
|
|
denomIn: Denomination,
|
|
|
|
): Promise<DenominationRecord> {
|
2017-05-28 01:10:54 +02:00
|
|
|
const denomPubHash = await this.cryptoApi.hashDenomPub(denomIn.denom_pub);
|
|
|
|
const d: DenominationRecord = {
|
2017-05-01 04:05:16 +02:00
|
|
|
denomPub: denomIn.denom_pub,
|
2017-05-28 01:10:54 +02:00
|
|
|
denomPubHash,
|
|
|
|
exchangeBaseUrl,
|
2018-01-29 22:58:47 +01:00
|
|
|
feeDeposit: Amounts.parseOrThrow(denomIn.fee_deposit),
|
|
|
|
feeRefresh: Amounts.parseOrThrow(denomIn.fee_refresh),
|
|
|
|
feeRefund: Amounts.parseOrThrow(denomIn.fee_refund),
|
|
|
|
feeWithdraw: Amounts.parseOrThrow(denomIn.fee_withdraw),
|
2017-05-28 01:10:54 +02:00
|
|
|
isOffered: true,
|
|
|
|
masterSig: denomIn.master_sig,
|
2017-05-01 04:05:16 +02:00
|
|
|
stampExpireDeposit: denomIn.stamp_expire_deposit,
|
|
|
|
stampExpireLegal: denomIn.stamp_expire_legal,
|
|
|
|
stampExpireWithdraw: denomIn.stamp_expire_withdraw,
|
|
|
|
stampStart: denomIn.stamp_start,
|
|
|
|
status: DenominationStatus.Unverified,
|
2018-01-29 22:58:47 +01:00
|
|
|
value: Amounts.parseOrThrow(denomIn.value),
|
2017-05-01 04:05:16 +02:00
|
|
|
};
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
|
|
|
async withdrawPaybackReserve(reservePub: string): Promise<void> {
|
2019-11-20 19:48:43 +01:00
|
|
|
const reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
|
2017-05-01 04:05:16 +02:00
|
|
|
if (!reserve) {
|
|
|
|
throw Error(`Reserve ${reservePub} does not exist`);
|
|
|
|
}
|
|
|
|
reserve.hasPayback = false;
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.reserves, reserve);
|
2019-11-21 23:09:43 +01:00
|
|
|
this.depleteReserve(reserve.reservePub).catch(e => {
|
|
|
|
console.error("Error depleting reserve after payback", e);
|
|
|
|
});
|
2017-05-01 04:05:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async getPaybackReserves(): Promise<ReserveRecord[]> {
|
2019-11-20 20:02:48 +01:00
|
|
|
return await oneShotIter(this.db, Stores.reserves).filter(
|
|
|
|
r => r.hasPayback,
|
|
|
|
);
|
2017-05-01 04:05:16 +02:00
|
|
|
}
|
|
|
|
|
2017-06-05 02:00:03 +02:00
|
|
|
/**
|
|
|
|
* Stop ongoing processing.
|
|
|
|
*/
|
|
|
|
stop() {
|
2019-11-21 23:09:43 +01:00
|
|
|
//this.timerGroup.stopCurrentAndFutureTimers();
|
2019-08-01 23:21:15 +02:00
|
|
|
this.cryptoApi.stop();
|
2017-06-05 02:00:03 +02:00
|
|
|
}
|
2017-08-14 04:16:12 +02:00
|
|
|
|
|
|
|
async getSenderWireInfos(): Promise<SenderWireInfos> {
|
|
|
|
const m: { [url: string]: Set<string> } = {};
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await oneShotIter(this.db, Stores.exchanges).forEach(x => {
|
2019-11-20 19:48:43 +01:00
|
|
|
const wi = x.wireInfo;
|
|
|
|
if (!wi) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const s = (m[x.baseUrl] = m[x.baseUrl] || new Set());
|
|
|
|
Object.keys(wi.feesForType).map(k => s.add(k));
|
|
|
|
});
|
|
|
|
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log(m);
|
2017-08-14 04:16:12 +02:00
|
|
|
const exchangeWireTypes: { [url: string]: string[] } = {};
|
2019-06-26 15:30:32 +02:00
|
|
|
Object.keys(m).map(e => {
|
|
|
|
exchangeWireTypes[e] = Array.from(m[e]);
|
|
|
|
});
|
2017-08-14 04:16:12 +02:00
|
|
|
|
2019-07-21 23:50:10 +02:00
|
|
|
const senderWiresSet: Set<string> = new Set();
|
2019-11-20 20:02:48 +01:00
|
|
|
await oneShotIter(this.db, Stores.senderWires).forEach(x => {
|
2019-11-20 19:48:43 +01:00
|
|
|
senderWiresSet.add(x.paytoUri);
|
|
|
|
});
|
|
|
|
|
2019-07-21 23:50:10 +02:00
|
|
|
const senderWires: string[] = Array.from(senderWiresSet);
|
2017-08-14 04:16:12 +02:00
|
|
|
|
|
|
|
return {
|
|
|
|
exchangeWireTypes,
|
|
|
|
senderWires,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Trigger paying coins back into the user's account.
|
|
|
|
*/
|
|
|
|
async returnCoins(req: ReturnCoinsRequest): Promise<void> {
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("got returnCoins request", req);
|
2017-08-14 04:16:12 +02:00
|
|
|
const wireType = (req.senderWire as any).type;
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("wireType", wireType);
|
2017-08-14 04:16:12 +02:00
|
|
|
if (!wireType || typeof wireType !== "string") {
|
|
|
|
console.error(`wire type must be a non-empty string, not ${wireType}`);
|
|
|
|
return;
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
const stampSecNow = Math.floor(new Date().getTime() / 1000);
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchange = await this.findExchange(req.exchange);
|
2017-08-14 04:16:12 +02:00
|
|
|
if (!exchange) {
|
|
|
|
console.error(`Exchange ${req.exchange} not known to the wallet`);
|
|
|
|
return;
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
const exchangeDetails = exchange.details;
|
|
|
|
if (!exchangeDetails) {
|
|
|
|
throw Error("exchange information needs to be updated first.");
|
|
|
|
}
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("selecting coins for return:", req);
|
2017-08-14 04:16:12 +02:00
|
|
|
const cds = await this.getCoinsForReturn(req.exchange, req.amount);
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log(cds);
|
2017-08-14 04:16:12 +02:00
|
|
|
|
|
|
|
if (!cds) {
|
|
|
|
throw Error("coin return impossible, can't select coins");
|
|
|
|
}
|
|
|
|
|
|
|
|
const { priv, pub } = await this.cryptoApi.createEddsaKeypair();
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const wireHash = await this.cryptoApi.hashString(
|
|
|
|
canonicalJson(req.senderWire),
|
|
|
|
);
|
2017-08-14 04:16:12 +02:00
|
|
|
|
|
|
|
const contractTerms: ContractTerms = {
|
|
|
|
H_wire: wireHash,
|
2018-01-29 22:58:47 +01:00
|
|
|
amount: Amounts.toString(req.amount),
|
2017-08-14 04:16:12 +02:00
|
|
|
auditors: [],
|
2019-06-26 15:30:32 +02:00
|
|
|
exchanges: [
|
2019-11-20 19:48:43 +01:00
|
|
|
{ master_pub: exchangeDetails.masterPublicKey, url: exchange.baseUrl },
|
2019-06-26 15:30:32 +02:00
|
|
|
],
|
2017-10-15 19:28:35 +02:00
|
|
|
extra: {},
|
|
|
|
fulfillment_url: "",
|
2017-08-14 04:16:12 +02:00
|
|
|
locations: [],
|
2018-01-29 22:58:47 +01:00
|
|
|
max_fee: Amounts.toString(req.amount),
|
2017-08-14 04:16:12 +02:00
|
|
|
merchant: {},
|
|
|
|
merchant_pub: pub,
|
2017-10-15 19:28:35 +02:00
|
|
|
order_id: "none",
|
2019-11-02 00:22:55 +01:00
|
|
|
pay_deadline: `/Date(${stampSecNow + 30 * 5})/`,
|
|
|
|
wire_transfer_deadline: `/Date(${stampSecNow + 60 * 5})/`,
|
2019-11-01 18:39:23 +01:00
|
|
|
merchant_base_url: "taler://return-to-account",
|
2017-08-14 04:16:12 +02:00
|
|
|
products: [],
|
|
|
|
refund_deadline: `/Date(${stampSecNow + 60 * 5})/`,
|
|
|
|
timestamp: `/Date(${stampSecNow})/`,
|
2017-10-15 19:28:35 +02:00
|
|
|
wire_method: wireType,
|
2017-08-14 04:16:12 +02:00
|
|
|
};
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const contractTermsHash = await this.cryptoApi.hashString(
|
|
|
|
canonicalJson(contractTerms),
|
|
|
|
);
|
2017-08-14 04:16:12 +02:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const payCoinInfo = await this.cryptoApi.signDeposit(
|
|
|
|
contractTerms,
|
|
|
|
cds,
|
|
|
|
Amounts.parseOrThrow(contractTerms.amount),
|
2018-01-29 22:58:47 +01:00
|
|
|
);
|
2017-08-14 04:16:12 +02:00
|
|
|
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("pci", payCoinInfo);
|
2017-08-14 04:16:12 +02:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
const coins = payCoinInfo.sigs.map(s => ({ coinPaySig: s }));
|
2017-08-14 04:16:12 +02:00
|
|
|
|
|
|
|
const coinsReturnRecord: CoinsReturnRecord = {
|
|
|
|
coins,
|
|
|
|
contractTerms,
|
|
|
|
contractTermsHash,
|
2017-10-15 19:28:35 +02:00
|
|
|
exchange: exchange.baseUrl,
|
2017-08-14 04:16:12 +02:00
|
|
|
merchantPriv: priv,
|
|
|
|
wire: req.senderWire,
|
2017-10-15 19:28:35 +02:00
|
|
|
};
|
2017-08-14 04:16:12 +02:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.coinsReturns, Stores.coins],
|
|
|
|
async tx => {
|
|
|
|
await tx.put(Stores.coinsReturns, coinsReturnRecord);
|
|
|
|
for (let c of payCoinInfo.updatedCoins) {
|
|
|
|
await tx.put(Stores.coins, c);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
2017-12-12 16:39:55 +01:00
|
|
|
this.badge.showNotification();
|
2017-08-30 17:08:54 +02:00
|
|
|
this.notifier.notify();
|
2017-08-14 04:16:12 +02:00
|
|
|
|
|
|
|
this.depositReturnedCoins(coinsReturnRecord);
|
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
async depositReturnedCoins(
|
|
|
|
coinsReturnRecord: CoinsReturnRecord,
|
|
|
|
): Promise<void> {
|
2017-08-14 04:16:12 +02:00
|
|
|
for (const c of coinsReturnRecord.coins) {
|
|
|
|
if (c.depositedSig) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const req = {
|
|
|
|
H_wire: coinsReturnRecord.contractTerms.H_wire,
|
|
|
|
coin_pub: c.coinPaySig.coin_pub,
|
2017-10-15 19:28:35 +02:00
|
|
|
coin_sig: c.coinPaySig.coin_sig,
|
2018-01-01 18:56:22 +01:00
|
|
|
contribution: c.coinPaySig.contribution,
|
2018-01-03 14:42:06 +01:00
|
|
|
denom_pub: c.coinPaySig.denom_pub,
|
2017-10-15 19:28:35 +02:00
|
|
|
h_contract_terms: coinsReturnRecord.contractTermsHash,
|
|
|
|
merchant_pub: coinsReturnRecord.contractTerms.merchant_pub,
|
2017-08-14 04:16:12 +02:00
|
|
|
pay_deadline: coinsReturnRecord.contractTerms.pay_deadline,
|
|
|
|
refund_deadline: coinsReturnRecord.contractTerms.refund_deadline,
|
2017-10-15 19:28:35 +02:00
|
|
|
timestamp: coinsReturnRecord.contractTerms.timestamp,
|
|
|
|
ub_sig: c.coinPaySig.ub_sig,
|
|
|
|
wire: coinsReturnRecord.wire,
|
|
|
|
wire_transfer_deadline: coinsReturnRecord.contractTerms.pay_deadline,
|
2017-08-14 04:16:12 +02:00
|
|
|
};
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("req", req);
|
2019-06-26 15:30:32 +02:00
|
|
|
const reqUrl = new URI("deposit").absoluteTo(coinsReturnRecord.exchange);
|
2017-08-14 04:16:12 +02:00
|
|
|
const resp = await this.http.postJson(reqUrl.href(), req);
|
|
|
|
if (resp.status !== 200) {
|
|
|
|
console.error("deposit failed due to status code", resp);
|
|
|
|
continue;
|
|
|
|
}
|
2019-07-31 01:33:56 +02:00
|
|
|
const respJson = resp.responseJson;
|
2017-08-14 04:16:12 +02:00
|
|
|
if (respJson.status !== "DEPOSIT_OK") {
|
|
|
|
console.error("deposit failed", resp);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!respJson.sig) {
|
|
|
|
console.error("invalid 'sig' field", resp);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: verify signature
|
|
|
|
|
|
|
|
// For every successful deposit, we replace the old record with an updated one
|
2019-11-20 20:02:48 +01:00
|
|
|
const currentCrr = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.coinsReturns,
|
|
|
|
coinsReturnRecord.contractTermsHash,
|
|
|
|
);
|
2017-08-14 04:16:12 +02:00
|
|
|
if (!currentCrr) {
|
|
|
|
console.error("database inconsistent");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
for (const nc of currentCrr.coins) {
|
|
|
|
if (nc.coinPaySig.coin_pub === c.coinPaySig.coin_pub) {
|
|
|
|
nc.depositedSig = respJson.sig;
|
|
|
|
}
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.coinsReturns, currentCrr);
|
2017-08-30 17:08:54 +02:00
|
|
|
this.notifier.notify();
|
2017-08-14 04:16:12 +02:00
|
|
|
}
|
|
|
|
}
|
2017-08-27 03:56:19 +02:00
|
|
|
|
2019-08-31 11:49:36 +02:00
|
|
|
private async acceptRefundResponse(
|
2019-06-26 15:30:32 +02:00
|
|
|
refundResponse: MerchantRefundResponse,
|
|
|
|
): Promise<string> {
|
2018-01-29 16:41:17 +01:00
|
|
|
const refundPermissions = refundResponse.refund_permissions;
|
2018-01-17 03:49:54 +01:00
|
|
|
|
2017-08-27 03:56:19 +02:00
|
|
|
if (!refundPermissions.length) {
|
|
|
|
console.warn("got empty refund list");
|
2018-01-17 03:49:54 +01:00
|
|
|
throw Error("empty refund");
|
2017-08-27 03:56:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add refund to purchase if not already added.
|
|
|
|
*/
|
2019-06-26 15:30:32 +02:00
|
|
|
function f(t: PurchaseRecord | undefined): PurchaseRecord | undefined {
|
2017-08-27 03:56:19 +02:00
|
|
|
if (!t) {
|
|
|
|
console.error("purchase not found, not adding refunds");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
t.timestamp_refund = getTimestampNow();
|
2017-10-15 18:30:02 +02:00
|
|
|
|
2017-08-27 03:56:19 +02:00
|
|
|
for (const perm of refundPermissions) {
|
2019-06-26 15:30:32 +02:00
|
|
|
if (
|
|
|
|
!t.refundsPending[perm.merchant_sig] &&
|
|
|
|
!t.refundsDone[perm.merchant_sig]
|
|
|
|
) {
|
2017-08-27 03:56:19 +02:00
|
|
|
t.refundsPending[perm.merchant_sig] = perm;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2018-01-29 16:41:17 +01:00
|
|
|
const hc = refundResponse.h_contract_terms;
|
|
|
|
|
2017-08-27 03:56:19 +02:00
|
|
|
// Add the refund permissions to the purchase within a DB transaction
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotMutate(this.db, Stores.purchases, hc, f);
|
2017-08-27 03:56:19 +02:00
|
|
|
this.notifier.notify();
|
|
|
|
|
2019-08-31 11:49:36 +02:00
|
|
|
await this.submitRefunds(hc);
|
2018-01-17 03:49:54 +01:00
|
|
|
|
2018-01-29 16:41:17 +01:00
|
|
|
return hc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Accept a refund, return the contract hash for the contract
|
|
|
|
* that was involved in the refund.
|
|
|
|
*/
|
2019-08-31 11:49:36 +02:00
|
|
|
async applyRefund(talerRefundUri: string): Promise<string> {
|
|
|
|
const parseResult = parseRefundUri(talerRefundUri);
|
|
|
|
|
|
|
|
if (!parseResult) {
|
|
|
|
throw Error("invalid refund URI");
|
|
|
|
}
|
|
|
|
|
|
|
|
const refundUrl = parseResult.refundUrl;
|
|
|
|
|
2019-08-26 01:39:13 +02:00
|
|
|
Wallet.enableTracing && console.log("processing refund");
|
2018-01-29 16:41:17 +01:00
|
|
|
let resp;
|
|
|
|
try {
|
2019-08-22 23:36:36 +02:00
|
|
|
resp = await this.http.get(refundUrl);
|
2018-01-29 16:41:17 +01:00
|
|
|
} catch (e) {
|
2019-08-26 01:39:13 +02:00
|
|
|
console.error("error downloading refund permission", e);
|
2018-01-29 16:41:17 +01:00
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
2019-08-22 23:36:36 +02:00
|
|
|
const refundResponse = MerchantRefundResponse.checked(resp.responseJson);
|
2018-01-29 16:41:17 +01:00
|
|
|
return this.acceptRefundResponse(refundResponse);
|
2017-08-27 03:56:19 +02:00
|
|
|
}
|
|
|
|
|
2018-01-22 01:12:08 +01:00
|
|
|
private async submitRefunds(contractTermsHash: string): Promise<void> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const purchase = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.purchases,
|
|
|
|
contractTermsHash,
|
|
|
|
);
|
2017-08-27 03:56:19 +02:00
|
|
|
if (!purchase) {
|
2019-06-26 15:30:32 +02:00
|
|
|
console.error(
|
|
|
|
"not submitting refunds, contract terms not found:",
|
|
|
|
contractTermsHash,
|
|
|
|
);
|
2017-08-27 03:56:19 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const pendingKeys = Object.keys(purchase.refundsPending);
|
|
|
|
if (pendingKeys.length === 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (const pk of pendingKeys) {
|
|
|
|
const perm = purchase.refundsPending[pk];
|
2018-01-29 16:41:17 +01:00
|
|
|
const req: RefundRequest = {
|
|
|
|
coin_pub: perm.coin_pub,
|
|
|
|
h_contract_terms: purchase.contractTermsHash,
|
|
|
|
merchant_pub: purchase.contractTerms.merchant_pub,
|
|
|
|
merchant_sig: perm.merchant_sig,
|
|
|
|
refund_amount: perm.refund_amount,
|
|
|
|
refund_fee: perm.refund_fee,
|
|
|
|
rtransaction_id: perm.rtransaction_id,
|
|
|
|
};
|
2017-08-27 03:56:19 +02:00
|
|
|
console.log("sending refund permission", perm);
|
2018-01-04 13:22:23 +01:00
|
|
|
// FIXME: not correct once we support multiple exchanges per payment
|
|
|
|
const exchangeUrl = purchase.payReq.coins[0].exchange_url;
|
2019-06-26 15:30:32 +02:00
|
|
|
const reqUrl = new URI("refund").absoluteTo(exchangeUrl);
|
2018-01-29 16:41:17 +01:00
|
|
|
const resp = await this.http.postJson(reqUrl.href(), req);
|
2017-08-27 03:56:19 +02:00
|
|
|
if (resp.status !== 200) {
|
|
|
|
console.error("refund failed", resp);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Transactionally mark successful refunds as done
|
2019-06-26 15:30:32 +02:00
|
|
|
const transformPurchase = (
|
|
|
|
t: PurchaseRecord | undefined,
|
|
|
|
): PurchaseRecord | undefined => {
|
2017-08-27 03:56:19 +02:00
|
|
|
if (!t) {
|
|
|
|
console.warn("purchase not found, not updating refund");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (t.refundsPending[pk]) {
|
|
|
|
t.refundsDone[pk] = t.refundsPending[pk];
|
|
|
|
delete t.refundsPending[pk];
|
|
|
|
}
|
|
|
|
return t;
|
|
|
|
};
|
2019-06-26 15:30:32 +02:00
|
|
|
const transformCoin = (
|
|
|
|
c: CoinRecord | undefined,
|
|
|
|
): CoinRecord | undefined => {
|
2017-08-27 03:56:19 +02:00
|
|
|
if (!c) {
|
|
|
|
console.warn("coin not found, can't apply refund");
|
|
|
|
return;
|
|
|
|
}
|
2018-01-29 22:58:47 +01:00
|
|
|
const refundAmount = Amounts.parseOrThrow(perm.refund_amount);
|
|
|
|
const refundFee = Amounts.parseOrThrow(perm.refund_fee);
|
2017-08-27 03:56:19 +02:00
|
|
|
c.status = CoinStatus.Dirty;
|
2018-01-29 22:58:47 +01:00
|
|
|
c.currentAmount = Amounts.add(c.currentAmount, refundAmount).amount;
|
|
|
|
c.currentAmount = Amounts.sub(c.currentAmount, refundFee).amount;
|
2017-08-27 03:56:19 +02:00
|
|
|
|
|
|
|
return c;
|
|
|
|
};
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(
|
|
|
|
this.db,
|
|
|
|
[Stores.purchases, Stores.coins],
|
|
|
|
async tx => {
|
|
|
|
await tx.mutate(
|
|
|
|
Stores.purchases,
|
|
|
|
contractTermsHash,
|
|
|
|
transformPurchase,
|
|
|
|
);
|
|
|
|
await tx.mutate(Stores.coins, perm.coin_pub, transformCoin);
|
|
|
|
},
|
|
|
|
);
|
2017-08-27 03:56:19 +02:00
|
|
|
this.refresh(perm.coin_pub);
|
|
|
|
}
|
|
|
|
|
2017-12-12 16:39:55 +01:00
|
|
|
this.badge.showNotification();
|
2017-08-27 03:56:19 +02:00
|
|
|
this.notifier.notify();
|
|
|
|
}
|
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
async getPurchase(
|
|
|
|
contractTermsHash: string,
|
|
|
|
): Promise<PurchaseRecord | undefined> {
|
2019-11-20 19:48:43 +01:00
|
|
|
return oneShotGet(this.db, Stores.purchases, contractTermsHash);
|
2017-08-27 03:56:19 +02:00
|
|
|
}
|
2017-08-30 17:08:54 +02:00
|
|
|
|
2019-06-26 15:30:32 +02:00
|
|
|
async getFullRefundFees(
|
|
|
|
refundPermissions: MerchantRefundPermission[],
|
|
|
|
): Promise<AmountJson> {
|
2017-08-30 17:08:54 +02:00
|
|
|
if (refundPermissions.length === 0) {
|
|
|
|
throw Error("no refunds given");
|
|
|
|
}
|
2019-11-20 20:02:48 +01:00
|
|
|
const coin0 = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.coins,
|
|
|
|
refundPermissions[0].coin_pub,
|
|
|
|
);
|
2017-08-30 17:08:54 +02:00
|
|
|
if (!coin0) {
|
|
|
|
throw Error("coin not found");
|
|
|
|
}
|
2019-06-26 15:30:32 +02:00
|
|
|
let feeAcc = Amounts.getZero(
|
|
|
|
Amounts.parseOrThrow(refundPermissions[0].refund_amount).currency,
|
|
|
|
);
|
2017-08-30 17:08:54 +02:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
const denoms = await oneShotIterIndex(
|
|
|
|
this.db,
|
|
|
|
Stores.denominations.exchangeBaseUrlIndex,
|
|
|
|
coin0.exchangeBaseUrl,
|
|
|
|
).toArray();
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2017-08-30 17:08:54 +02:00
|
|
|
for (const rp of refundPermissions) {
|
2019-11-20 19:48:43 +01:00
|
|
|
const coin = await oneShotGet(this.db, Stores.coins, rp.coin_pub);
|
2017-08-30 17:08:54 +02:00
|
|
|
if (!coin) {
|
|
|
|
throw Error("coin not found");
|
|
|
|
}
|
2019-11-20 19:48:43 +01:00
|
|
|
const denom = await oneShotGet(this.db, Stores.denominations, [
|
2019-06-26 15:30:32 +02:00
|
|
|
coin0.exchangeBaseUrl,
|
|
|
|
coin.denomPub,
|
|
|
|
]);
|
2017-08-30 17:08:54 +02:00
|
|
|
if (!denom) {
|
|
|
|
throw Error(`denom not found (${coin.denomPub})`);
|
|
|
|
}
|
|
|
|
// FIXME: this assumes that the refund already happened.
|
|
|
|
// When it hasn't, the refresh cost is inaccurate. To fix this,
|
|
|
|
// we need introduce a flag to tell if a coin was refunded or
|
|
|
|
// refreshed normally (and what about incremental refunds?)
|
2018-01-29 22:58:47 +01:00
|
|
|
const refundAmount = Amounts.parseOrThrow(rp.refund_amount);
|
|
|
|
const refundFee = Amounts.parseOrThrow(rp.refund_fee);
|
2019-06-26 15:30:32 +02:00
|
|
|
const refreshCost = getTotalRefreshCost(
|
|
|
|
denoms,
|
|
|
|
denom,
|
|
|
|
Amounts.sub(refundAmount, refundFee).amount,
|
|
|
|
);
|
2018-01-29 22:58:47 +01:00
|
|
|
feeAcc = Amounts.add(feeAcc, refreshCost, refundFee).amount;
|
2017-08-30 17:08:54 +02:00
|
|
|
}
|
|
|
|
return feeAcc;
|
|
|
|
}
|
2017-11-30 04:07:36 +01:00
|
|
|
|
2019-11-21 23:09:43 +01:00
|
|
|
async acceptTip(
|
|
|
|
talerTipUri: string,
|
2019-08-30 17:27:59 +02:00
|
|
|
): Promise<void> {
|
2019-11-21 23:09:43 +01:00
|
|
|
const { tipId, merchantOrigin } = await this.getTipStatus(talerTipUri);
|
2019-11-20 20:02:48 +01:00
|
|
|
let tipRecord = await oneShotGet(this.db, Stores.tips, [
|
|
|
|
tipId,
|
|
|
|
merchantOrigin,
|
|
|
|
]);
|
2019-08-30 17:27:59 +02:00
|
|
|
if (!tipRecord) {
|
|
|
|
throw Error("tip not in database");
|
2018-01-17 03:49:54 +01:00
|
|
|
}
|
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
tipRecord.accepted = true;
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.tips, tipRecord);
|
2019-08-30 17:27:59 +02:00
|
|
|
|
|
|
|
if (tipRecord.pickedUp) {
|
|
|
|
console.log("tip already picked up");
|
|
|
|
return;
|
2018-01-17 03:49:54 +01:00
|
|
|
}
|
2019-08-30 17:27:59 +02:00
|
|
|
await this.updateExchangeFromUrl(tipRecord.exchangeUrl);
|
2019-06-26 15:30:32 +02:00
|
|
|
const denomsForWithdraw = await this.getVerifiedWithdrawDenomList(
|
2019-08-30 17:27:59 +02:00
|
|
|
tipRecord.exchangeUrl,
|
|
|
|
tipRecord.amount,
|
2019-06-26 15:30:32 +02:00
|
|
|
);
|
2018-01-19 01:27:27 +01:00
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
if (!tipRecord.planchets) {
|
|
|
|
const planchets = await Promise.all(
|
|
|
|
denomsForWithdraw.map(d => this.cryptoApi.createTipPlanchet(d)),
|
|
|
|
);
|
|
|
|
const coinPubs: string[] = planchets.map(x => x.coinPub);
|
2018-02-01 07:19:03 +01:00
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await oneShotMutate(this.db, Stores.tips, [tipId, merchantOrigin], r => {
|
2019-08-30 17:27:59 +02:00
|
|
|
if (!r.planchets) {
|
|
|
|
r.planchets = planchets;
|
|
|
|
r.coinPubs = coinPubs;
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
});
|
|
|
|
|
|
|
|
this.notifier.notify();
|
|
|
|
}
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
tipRecord = await oneShotGet(this.db, Stores.tips, [tipId, merchantOrigin]);
|
2019-08-30 17:27:59 +02:00
|
|
|
if (!tipRecord) {
|
|
|
|
throw Error("tip not in database");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!tipRecord.planchets) {
|
|
|
|
throw Error("invariant violated");
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log("got planchets for tip!");
|
2018-02-01 07:19:03 +01:00
|
|
|
|
2018-01-19 01:27:27 +01:00
|
|
|
// Planchets in the form that the merchant expects
|
2019-06-26 15:30:32 +02:00
|
|
|
const planchetsDetail: TipPlanchetDetail[] = tipRecord.planchets.map(p => ({
|
2018-01-19 01:27:27 +01:00
|
|
|
coin_ev: p.coinEv,
|
|
|
|
denom_pub_hash: p.denomPubHash,
|
|
|
|
}));
|
2018-01-17 03:49:54 +01:00
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
let merchantResp;
|
|
|
|
|
2018-01-17 03:49:54 +01:00
|
|
|
try {
|
2019-08-30 17:27:59 +02:00
|
|
|
const req = { planchets: planchetsDetail, tip_id: tipId };
|
|
|
|
merchantResp = await this.http.postJson(tipRecord.pickupUrl, req);
|
|
|
|
console.log("got merchant resp:", merchantResp);
|
2018-01-17 03:49:54 +01:00
|
|
|
} catch (e) {
|
|
|
|
console.log("tipping failed", e);
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
2019-08-22 23:36:36 +02:00
|
|
|
const response = TipResponse.checked(merchantResp.responseJson);
|
2018-01-17 03:49:54 +01:00
|
|
|
|
2017-11-30 04:07:36 +01:00
|
|
|
if (response.reserve_sigs.length !== tipRecord.planchets.length) {
|
|
|
|
throw Error("number of tip responses does not match requested planchets");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = 0; i < tipRecord.planchets.length; i++) {
|
2018-01-03 14:42:06 +01:00
|
|
|
const planchet = tipRecord.planchets[i];
|
|
|
|
const preCoin = {
|
|
|
|
blindingKey: planchet.blindingKey,
|
2017-11-30 04:07:36 +01:00
|
|
|
coinEv: planchet.coinEv,
|
2018-01-03 14:42:06 +01:00
|
|
|
coinPriv: planchet.coinPriv,
|
|
|
|
coinPub: planchet.coinPub,
|
2017-11-30 04:07:36 +01:00
|
|
|
coinValue: planchet.coinValue,
|
|
|
|
denomPub: planchet.denomPub,
|
2019-05-08 07:01:17 +02:00
|
|
|
denomPubHash: planchet.denomPubHash,
|
2017-11-30 04:07:36 +01:00
|
|
|
exchangeBaseUrl: tipRecord.exchangeUrl,
|
|
|
|
isFromTip: true,
|
2018-01-03 14:42:06 +01:00
|
|
|
reservePub: response.reserve_pub,
|
|
|
|
withdrawSig: response.reserve_sigs[i].reserve_sig,
|
2017-11-30 04:07:36 +01:00
|
|
|
};
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.precoins, preCoin);
|
2019-08-30 17:27:59 +02:00
|
|
|
await this.processPreCoin(preCoin.coinPub);
|
2017-11-30 04:07:36 +01:00
|
|
|
}
|
2018-01-19 01:27:27 +01:00
|
|
|
|
|
|
|
tipRecord.pickedUp = true;
|
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.tips, tipRecord);
|
|
|
|
|
2018-04-09 00:20:54 +02:00
|
|
|
this.notifier.notify();
|
2019-08-30 17:27:59 +02:00
|
|
|
this.badge.showNotification();
|
|
|
|
return;
|
2017-11-30 04:07:36 +01:00
|
|
|
}
|
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
async getTipStatus(talerTipUri: string): Promise<TipStatus> {
|
|
|
|
const res = parseTipUri(talerTipUri);
|
|
|
|
if (!res) {
|
|
|
|
throw Error("invalid taler://tip URI");
|
2017-11-30 04:07:36 +01:00
|
|
|
}
|
|
|
|
|
2019-11-01 18:39:23 +01:00
|
|
|
const tipStatusUrl = new URI(res.tipPickupUrl).href();
|
2019-08-30 17:27:59 +02:00
|
|
|
console.log("checking tip status from", tipStatusUrl);
|
|
|
|
const merchantResp = await this.http.get(tipStatusUrl);
|
|
|
|
console.log("resp:", merchantResp.responseJson);
|
|
|
|
const tipPickupStatus = TipPickupGetResponse.checked(
|
|
|
|
merchantResp.responseJson,
|
|
|
|
);
|
2017-11-30 04:07:36 +01:00
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
console.log("status", tipPickupStatus);
|
2017-11-30 04:07:36 +01:00
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
let amount = Amounts.parseOrThrow(tipPickupStatus.amount);
|
2017-11-30 04:07:36 +01:00
|
|
|
|
2019-11-20 19:48:43 +01:00
|
|
|
let tipRecord = await oneShotGet(this.db, Stores.tips, [
|
2019-08-30 17:27:59 +02:00
|
|
|
res.tipId,
|
|
|
|
res.merchantOrigin,
|
2019-11-20 20:02:48 +01:00
|
|
|
]);
|
2019-11-20 19:48:43 +01:00
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
if (!tipRecord) {
|
|
|
|
const withdrawDetails = await this.getWithdrawDetailsForAmount(
|
|
|
|
tipPickupStatus.exchange_url,
|
|
|
|
amount,
|
|
|
|
);
|
2017-11-30 04:07:36 +01:00
|
|
|
|
2019-08-30 17:27:59 +02:00
|
|
|
tipRecord = {
|
|
|
|
accepted: false,
|
|
|
|
amount,
|
|
|
|
coinPubs: [],
|
|
|
|
deadline: getTalerStampSec(tipPickupStatus.stamp_expire)!,
|
|
|
|
exchangeUrl: tipPickupStatus.exchange_url,
|
|
|
|
merchantDomain: res.merchantOrigin,
|
|
|
|
nextUrl: undefined,
|
|
|
|
pickedUp: false,
|
|
|
|
planchets: undefined,
|
|
|
|
response: undefined,
|
2019-11-21 23:09:43 +01:00
|
|
|
timestamp: getTimestampNow(),
|
2019-08-30 17:27:59 +02:00
|
|
|
tipId: res.tipId,
|
|
|
|
pickupUrl: res.tipPickupUrl,
|
2019-08-31 13:27:12 +02:00
|
|
|
totalFees: Amounts.add(
|
|
|
|
withdrawDetails.overhead,
|
|
|
|
withdrawDetails.withdrawFee,
|
|
|
|
).amount,
|
2019-08-30 17:27:59 +02:00
|
|
|
};
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.tips, tipRecord);
|
2019-08-30 17:27:59 +02:00
|
|
|
}
|
2017-11-30 04:07:36 +01:00
|
|
|
|
|
|
|
const tipStatus: TipStatus = {
|
2018-02-01 07:19:03 +01:00
|
|
|
accepted: !!tipRecord && tipRecord.accepted,
|
2019-08-30 17:27:59 +02:00
|
|
|
amount: Amounts.parseOrThrow(tipPickupStatus.amount),
|
|
|
|
amountLeft: Amounts.parseOrThrow(tipPickupStatus.amount_left),
|
|
|
|
exchangeUrl: tipPickupStatus.exchange_url,
|
|
|
|
nextUrl: tipPickupStatus.extra.next_url,
|
|
|
|
merchantOrigin: res.merchantOrigin,
|
|
|
|
tipId: res.tipId,
|
|
|
|
expirationTimestamp: getTalerStampSec(tipPickupStatus.stamp_expire)!,
|
|
|
|
timestamp: getTalerStampSec(tipPickupStatus.stamp_created)!,
|
|
|
|
totalFees: tipRecord.totalFees,
|
2017-11-30 04:07:36 +01:00
|
|
|
};
|
2019-08-30 17:27:59 +02:00
|
|
|
|
2017-11-30 04:07:36 +01:00
|
|
|
return tipStatus;
|
|
|
|
}
|
2017-12-10 21:34:56 +01:00
|
|
|
|
2018-01-29 16:41:17 +01:00
|
|
|
async abortFailedPayment(contractTermsHash: string): Promise<void> {
|
2019-11-20 20:02:48 +01:00
|
|
|
const purchase = await oneShotGet(
|
|
|
|
this.db,
|
|
|
|
Stores.purchases,
|
|
|
|
contractTermsHash,
|
|
|
|
);
|
2018-01-29 16:41:17 +01:00
|
|
|
if (!purchase) {
|
|
|
|
throw Error("Purchase not found, unable to abort with refund");
|
|
|
|
}
|
|
|
|
if (purchase.finished) {
|
|
|
|
throw Error("Purchase already finished, not aborting");
|
|
|
|
}
|
|
|
|
if (purchase.abortDone) {
|
|
|
|
console.warn("abort requested on already aborted purchase");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
purchase.abortRequested = true;
|
|
|
|
|
|
|
|
// From now on, we can't retry payment anymore,
|
|
|
|
// so mark this in the DB in case the /pay abort
|
|
|
|
// does not complete on the first try.
|
2019-11-20 19:48:43 +01:00
|
|
|
await oneShotPut(this.db, Stores.purchases, purchase);
|
2018-01-29 16:41:17 +01:00
|
|
|
|
|
|
|
let resp;
|
|
|
|
|
|
|
|
const abortReq = { ...purchase.payReq, mode: "abort-refund" };
|
|
|
|
|
2019-11-02 00:22:55 +01:00
|
|
|
const payUrl = new URI("pay")
|
|
|
|
.absoluteTo(purchase.contractTerms.merchant_base_url)
|
|
|
|
.href();
|
2019-11-01 18:39:23 +01:00
|
|
|
|
2018-01-29 16:41:17 +01:00
|
|
|
try {
|
2019-11-01 18:39:23 +01:00
|
|
|
resp = await this.http.postJson(payUrl, abortReq);
|
2018-01-29 16:41:17 +01:00
|
|
|
} catch (e) {
|
|
|
|
// Gives the user the option to retry / abort and refresh
|
|
|
|
console.log("aborting payment failed", e);
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
2019-08-22 23:36:36 +02:00
|
|
|
const refundResponse = MerchantRefundResponse.checked(resp.responseJson);
|
2018-01-29 16:41:17 +01:00
|
|
|
await this.acceptRefundResponse(refundResponse);
|
|
|
|
|
2019-11-20 20:02:48 +01:00
|
|
|
await runWithWriteTransaction(this.db, [Stores.purchases], async tx => {
|
2019-11-20 19:48:43 +01:00
|
|
|
const p = await tx.get(Stores.purchases, purchase.contractTermsHash);
|
|
|
|
if (!p) {
|
|
|
|
return;
|
|
|
|
}
|
2018-01-29 16:41:17 +01:00
|
|
|
p.abortDone = true;
|
2019-11-20 19:48:43 +01:00
|
|
|
await tx.put(Stores.purchases, p);
|
|
|
|
});
|
2018-01-29 16:41:17 +01:00
|
|
|
}
|
|
|
|
|
2017-12-10 21:34:56 +01:00
|
|
|
/**
|
|
|
|
* Remove unreferenced / expired data from the wallet's database
|
|
|
|
* based on the current system time.
|
|
|
|
*/
|
|
|
|
async collectGarbage() {
|
2019-08-20 17:58:01 +02:00
|
|
|
// FIXME(#5845)
|
|
|
|
// We currently do not garbage-collect the wallet database. This might change
|
|
|
|
// after the feature has been properly re-designed, and we have come up with a
|
|
|
|
// strategy to test it.
|
2017-12-10 21:34:56 +01:00
|
|
|
}
|
2017-12-12 16:39:55 +01:00
|
|
|
|
2019-09-01 01:05:38 +02:00
|
|
|
async getWithdrawalInfo(
|
2019-08-28 02:49:27 +02:00
|
|
|
talerWithdrawUri: string,
|
|
|
|
): Promise<DownloadedWithdrawInfo> {
|
|
|
|
const uriResult = parseWithdrawUri(talerWithdrawUri);
|
|
|
|
if (!uriResult) {
|
|
|
|
throw Error("can't parse URL");
|
|
|
|
}
|
|
|
|
const resp = await this.http.get(uriResult.statusUrl);
|
|
|
|
console.log("resp:", resp.responseJson);
|
|
|
|
const status = WithdrawOperationStatusResponse.checked(resp.responseJson);
|
|
|
|
return {
|
|
|
|
amount: Amounts.parseOrThrow(status.amount),
|
|
|
|
confirmTransferUrl: status.confirm_transfer_url,
|
|
|
|
extractedStatusUrl: uriResult.statusUrl,
|
|
|
|
selectionDone: status.selection_done,
|
|
|
|
senderWire: status.sender_wire,
|
|
|
|
suggestedExchange: status.suggested_exchange,
|
|
|
|
transferDone: status.transfer_done,
|
|
|
|
wireTypes: status.wire_types,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-08-29 23:12:55 +02:00
|
|
|
async acceptWithdrawal(
|
2019-08-28 02:49:27 +02:00
|
|
|
talerWithdrawUri: string,
|
|
|
|
selectedExchange: string,
|
2019-08-29 23:12:55 +02:00
|
|
|
): Promise<AcceptWithdrawalResponse> {
|
2019-09-01 01:05:38 +02:00
|
|
|
const withdrawInfo = await this.getWithdrawalInfo(talerWithdrawUri);
|
2019-08-28 02:49:27 +02:00
|
|
|
const exchangeWire = await this.getExchangePaytoUri(
|
|
|
|
selectedExchange,
|
|
|
|
withdrawInfo.wireTypes,
|
|
|
|
);
|
|
|
|
const reserve = await this.createReserve({
|
|
|
|
amount: withdrawInfo.amount,
|
|
|
|
bankWithdrawStatusUrl: withdrawInfo.extractedStatusUrl,
|
|
|
|
exchange: selectedExchange,
|
|
|
|
senderWire: withdrawInfo.senderWire,
|
|
|
|
exchangeWire: exchangeWire,
|
|
|
|
});
|
|
|
|
return {
|
|
|
|
reservePub: reserve.reservePub,
|
|
|
|
confirmTransferUrl: withdrawInfo.confirmTransferUrl,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-08-31 13:27:12 +02:00
|
|
|
async getPurchaseDetails(hc: string): Promise<PurchaseDetails> {
|
2019-11-20 19:48:43 +01:00
|
|
|
const purchase = await oneShotGet(this.db, Stores.purchases, hc);
|
2019-08-31 13:27:12 +02:00
|
|
|
if (!purchase) {
|
|
|
|
throw Error("unknown purchase");
|
|
|
|
}
|
|
|
|
const refundsDoneAmounts = Object.values(purchase.refundsDone).map(x =>
|
|
|
|
Amounts.parseOrThrow(x.refund_amount),
|
|
|
|
);
|
2019-11-20 20:02:48 +01:00
|
|
|
const refundsPendingAmounts = Object.values(
|
|
|
|
purchase.refundsPending,
|
|
|
|
).map(x => Amounts.parseOrThrow(x.refund_amount));
|
2019-08-31 13:27:12 +02:00
|
|
|
const totalRefundAmount = Amounts.sum([
|
|
|
|
...refundsDoneAmounts,
|
|
|
|
...refundsPendingAmounts,
|
|
|
|
]).amount;
|
|
|
|
const refundsDoneFees = Object.values(purchase.refundsDone).map(x =>
|
|
|
|
Amounts.parseOrThrow(x.refund_amount),
|
|
|
|
);
|
2019-09-04 15:28:19 +02:00
|
|
|
const refundsPendingFees = Object.values(purchase.refundsPending).map(x =>
|
|
|
|
Amounts.parseOrThrow(x.refund_amount),
|
2019-08-31 13:27:12 +02:00
|
|
|
);
|
|
|
|
const totalRefundFees = Amounts.sum([
|
|
|
|
...refundsDoneFees,
|
|
|
|
...refundsPendingFees,
|
|
|
|
]).amount;
|
|
|
|
const totalFees = totalRefundFees;
|
|
|
|
return {
|
|
|
|
contractTerms: purchase.contractTerms,
|
2019-11-21 23:09:43 +01:00
|
|
|
hasRefund: purchase.timestamp_refund !== undefined,
|
2019-08-31 13:27:12 +02:00
|
|
|
totalRefundAmount: totalRefundAmount,
|
|
|
|
totalRefundAndRefreshFees: totalFees,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-08-28 02:49:27 +02:00
|
|
|
/**
|
|
|
|
* Reset the retry timeouts for ongoing operations.
|
|
|
|
*/
|
|
|
|
resetRetryTimeouts(): void {
|
|
|
|
// FIXME: implement
|
|
|
|
}
|
|
|
|
|
2017-12-12 16:39:55 +01:00
|
|
|
clearNotification(): void {
|
|
|
|
this.badge.clearNotification();
|
|
|
|
}
|
2018-09-20 02:56:13 +02:00
|
|
|
|
|
|
|
benchmarkCrypto(repetitions: number): Promise<BenchmarkResult> {
|
|
|
|
return this.cryptoApi.benchmark(repetitions);
|
|
|
|
}
|
2016-10-18 01:16:31 +02:00
|
|
|
}
|