wallet-core/src/wallet.ts

4218 lines
122 KiB
TypeScript
Raw Normal View History

2015-12-25 22:42:14 +01:00
/*
This file is part of TALER
2019-11-30 00:36:20 +01:00
(C) 2015-2019 GNUnet e.V.
2015-12-25 22:42:14 +01:00
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,
strcmp,
extractTalerStamp,
extractTalerStampOrThrow,
2017-05-28 01:10:54 +02:00
} from "./helpers";
2019-11-21 23:09:43 +01:00
import { HttpRequestLibrary } from "./http";
import * as LibtoolVersion from "./libtoolVersion";
2017-05-28 01:10:54 +02:00
import {
2019-11-21 23:09:43 +01:00
TransactionAbort,
oneShotPut,
oneShotGet,
runWithWriteTransaction,
oneShotIter,
oneShotIterIndex,
oneShotGetIndexed,
oneShotMutate,
2017-05-28 01:10:54 +02:00
} from "./query";
import { AmountJson } from "./amounts";
import * as Amounts from "./amounts";
import URI = require("urijs");
2016-05-24 01:53:56 +02:00
import {
2017-05-28 01:10:54 +02:00
CoinRecord,
CoinStatus,
CoinsReturnRecord,
2017-05-28 01:10:54 +02:00
CurrencyRecord,
DenominationRecord,
DenominationStatus,
2016-11-15 15:07:17 +01:00
ExchangeRecord,
2019-11-30 00:36:20 +01:00
PlanchetRecord,
ProposalRecord,
PurchaseRecord,
2019-11-30 00:36:20 +01:00
RefreshPlanchetRecord,
2016-11-15 15:07:17 +01:00
RefreshSessionRecord,
ReserveRecord,
Stores,
TipRecord,
WireFee,
2019-11-30 00:36:20 +01:00
WithdrawalSessionRecord,
ExchangeUpdateStatus,
2019-11-21 23:09:43 +01:00
ReserveRecordStatus,
2019-11-30 00:36:20 +01:00
ProposalStatus,
} from "./dbTypes";
import {
Auditor,
ContractTerms,
Denomination,
ExchangeHandle,
2019-05-08 04:53:26 +02:00
ExchangeWireJson,
KeysJson,
MerchantRefundPermission,
MerchantRefundResponse,
PayReq,
PaybackConfirmation,
Proposal,
RefundRequest,
ReserveStatus,
TipPlanchetDetail,
TipResponse,
2019-08-28 02:49:27 +02:00
WithdrawOperationStatusResponse,
2019-08-30 17:27:59 +02:00
TipPickupGetResponse,
} from "./talerTypes";
import {
Badge,
2018-09-20 02:56:13 +02:00
BenchmarkResult,
CoinSelectionResult,
CoinWithDenom,
ConfirmPayResult,
ConfirmReserveRequest,
CreateReserveRequest,
CreateReserveResponse,
2019-11-21 23:09:43 +01:00
HistoryEvent,
2018-01-18 02:50:18 +01:00
NextUrlResult,
Notifier,
PayCoinInfo,
ReserveCreationInfo,
ReturnCoinsRequest,
SenderWireInfos,
2017-11-30 04:07:36 +01:00
TipStatus,
WalletBalance,
WalletBalanceEntry,
PreparePayResult,
2019-08-28 02:49:27 +02:00
DownloadedWithdrawInfo,
WithdrawDetails,
AcceptWithdrawalResponse,
2019-08-31 13:27:12 +02:00
PurchaseDetails,
2019-11-19 16:16:12 +01:00
PendingOperationInfo,
PendingOperationsResponse,
HistoryQuery,
getTimestampNow,
OperationError,
2019-11-21 23:09:43 +01:00
Timestamp,
} from "./walletTypes";
2019-08-31 13:27:12 +02:00
import {
parsePayUri,
parseWithdrawUri,
parseTipUri,
parseRefundUri,
} from "./taleruri";
2019-11-21 23:09:43 +01:00
import { Logger } from "./logging";
2019-11-30 00:36:20 +01:00
import { randomBytes } from "./crypto/primitives/nacl-fast";
import { encodeCrock, getRandomBytes } from "./crypto/talerCrypto";
2015-12-13 23:47:30 +01:00
interface SpeculativePayData {
payCoinInfo: PayCoinInfo;
exchangeUrl: string;
2019-11-30 00:36:20 +01:00
orderDownloadId: string;
proposal: ProposalRecord;
}
/**
* 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
const WALLET_CACHE_BREAKER_CLIENT_VERSION = "2";
2019-08-31 22:07:16 +02:00
const builtinCurrencies: CurrencyRecord[] = [
{
auditors: [
{
2017-06-04 19:41:43 +02:00
auditorPub: "BW9DC48PHQY4NH011SHHX36DZZ3Q22Y6X7FZ1VD1CMZ2PTFZ6PN0",
baseUrl: "https://auditor.demo.taler.net/",
2019-06-26 15:30:32 +02:00
expirationStamp: new Date(2027, 1).getTime(),
},
2017-04-12 17:47:14 +02:00
],
exchanges: [],
2017-05-28 01:10:54 +02:00
fractionalDigits: 2,
name: "KUDOS",
},
];
function isWithdrawableDenom(d: DenominationRecord) {
const now = getTimestampNow();
const started = now.t_ms >= d.stampStart.t_ms;
2019-11-30 00:36:20 +01:00
const stillOkay = d.stampExpireWithdraw.t_ms + 60 * 1000 > now.t_ms;
return started && stillOkay;
2016-02-10 02:03:31 +01:00
}
interface SelectPayCoinsResult {
cds: CoinWithDenom[];
totalFees: AmountJson;
}
2019-11-30 00:36:20 +01:00
function assertUnreachable(x: never): never {
throw new Error("Didn't expect to get here");
}
/**
* 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.
*/
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;
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;
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),
);
return totalCost;
}
/**
* Select coins for a payment under the merchant's constraints.
*
* @param denoms all available denoms, used to compute refresh fees
*/
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;
}
// 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[] = [];
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;
}
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 });
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;
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;
const isBelowFee = Amounts.cmp(accDepositFee, depositFeeLimit) <= 0;
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;
leftAmount = Amounts.sub(leftAmount, depositFeeToCover).amount;
2019-08-28 02:49:27 +02:00
Wallet.enableTracing &&
console.log("deposit fee to cover", amountToPretty(depositFeeToCover));
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;
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[] {
let remaining = Amounts.copy(amountAvailable);
const ds: DenominationRecord[] = [];
2016-02-11 18:17:02 +01:00
denoms = denoms.filter(isWithdrawableDenom);
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;
if (Amounts.cmp(remaining, cost) < 0) {
2016-02-11 18:17:02 +01:00
continue;
}
found = true;
remaining = Amounts.sub(remaining, cost).amount;
2016-02-11 18:17:02 +01:00
ds.push(d);
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
}
/**
* This error is thrown when an
*/
2019-11-21 23:09:43 +01:00
export class OperationFailedAndReportedError extends Error {
constructor(message: string) {
super(message);
// Set the prototype explicitly.
Object.setPrototypeOf(this, OperationFailedAndReportedError.prototype);
}
}
2019-11-21 23:09:43 +01:00
const logger = new Logger("wallet.ts");
2019-11-30 00:36:20 +01:00
interface MemoEntry<T> {
p: Promise<T>;
t: number;
n: number;
}
class AsyncOpMemo<T> {
n = 0;
memo: { [k: string]: MemoEntry<T> } = {};
put(key: string, p: Promise<T>): Promise<T> {
const n = this.n++;
this.memo[key] = {
p,
n,
t: new Date().getTime(),
};
p.finally(() => {
const r = this.memo[key];
if (r && r.n === n) {
delete this.memo[key];
}
});
return p;
}
find(key: string): Promise<T> | undefined {
const res = this.memo[key];
const tNow = new Date().getTime();
if (res && res.t < tNow - 10 * 1000) {
delete this.memo[key];
return;
} else if (res) {
return res.p;
}
return;
}
}
/**
* The platform-independent wallet implementation.
*/
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;
private http: HttpRequestLibrary;
private badge: Badge;
2016-02-18 23:41:29 +01:00
private notifier: Notifier;
private cryptoApi: CryptoApi;
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-11-30 00:36:20 +01:00
private memoProcessReserve = new AsyncOpMemo<void>();
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
) {
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);
}
2017-06-05 02:00:03 +02:00
2019-11-21 23:09:43 +01:00
/**
2019-11-30 00:36:20 +01:00
* Execute one operation based on the pending operation info record.
2019-11-21 23:09:43 +01:00
*/
2019-11-30 00:36:20 +01:00
async processOnePendingOperation(
pending: PendingOperationInfo,
): Promise<void> {
switch (pending.type) {
case "bug":
return;
case "dirty-coin":
await this.refresh(pending.coinPub);
break;
case "exchange-update":
await this.updateExchangeFromUrl(pending.exchangeBaseUrl);
break;
case "planchet":
await this.processPlanchet(pending.coinPub);
break;
case "refresh":
await this.processRefreshSession(pending.refreshSessionId);
break;
case "reserve":
await this.processReserve(pending.reservePub);
break;
case "withdraw":
await this.processWithdrawSession(pending.withdrawSessionId);
break;
case "proposal":
// Nothing to do, user needs to accept/reject
break;
default:
assertUnreachable(pending);
2019-11-21 23:09:43 +01:00
}
2019-11-30 00:36:20 +01:00
}
2019-11-21 23:09:43 +01:00
2019-11-30 00:36:20 +01:00
/**
* Process pending operations.
*/
public async runPending(): Promise<void> {
const pendingOpsResponse = await this.getPendingOperations();
for (const p of pendingOpsResponse.pendingOperations) {
try {
await this.processOnePendingOperation(p);
} catch (e) {
console.error(e);
}
2019-11-21 23:09: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-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) {
2019-11-30 00:36:20 +01:00
const r = await this.getPendingOperations();
const allPending = r.pendingOperations;
const relevantPending = allPending.filter(x => {
switch (x.type) {
case "planchet":
case "reserve":
return x.reservePub === reservePub;
default:
return false;
2019-11-21 23:09:43 +01:00
}
2019-11-30 00:36:20 +01:00
});
if (relevantPending.length === 0) {
return;
}
for (const p of relevantPending) {
await this.processOnePendingOperation(p);
2019-11-21 23:09: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 => {
let applied = false;
2019-11-20 20:02:48 +01:00
await tx.iter(Stores.config).forEach(x => {
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
},
);
}
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,
);
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();
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();
// Denomination of the first coin, we assume that all other
// coins have the same currency
const firstDenom = await oneShotGet(this.db, Stores.denominations, [
2019-06-26 15:30:32 +02:00
exchange.baseUrl,
coins[0].denomPub,
]);
if (!firstDenom) {
throw Error("db inconsistent");
}
const currency = firstDenom.value.currency;
const cds: CoinWithDenom[] = [];
for (const coin of coins) {
const denom = await oneShotGet(this.db, Stores.denominations, [
2019-06-26 15:30:32 +02:00
exchange.baseUrl,
coin.denomPub,
]);
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
);
continue;
}
if (coin.suspended) {
continue;
}
if (coin.status !== CoinStatus.Fresh) {
continue;
}
2019-06-26 15:30:32 +02:00
cds.push({ coin, denom });
}
const res = selectPayCoins(denoms, cds, amount, amount);
if (res) {
return res.cds;
}
2017-10-15 19:28:35 +02:00
return undefined;
}
/**
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.
*/
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;
const exchanges = await oneShotIter(this.db, Stores.exchanges).toArray();
2017-05-28 01:10:54 +02:00
for (const exchange of exchanges) {
let isOkay: boolean = false;
const exchangeDetails = exchange.details;
if (!exchangeDetails) {
continue;
}
const exchangeFees = exchange.wireInfo;
if (!exchangeFees) {
continue;
}
// is the exchange explicitly allowed?
2017-05-28 01:10:54 +02:00
for (const allowedExchange of allowedExchanges) {
if (allowedExchange.master_pub === exchangeDetails.masterPublicKey) {
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) {
for (const auditor of exchangeDetails.auditors) {
2017-05-28 01:10:54 +02:00
if (auditor.auditor_pub === allowedAuditor.auditor_pub) {
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
}
2019-11-20 20:02:48 +01:00
const coins = await oneShotIterIndex(
this.db,
Stores.coins.exchangeBaseUrlIndex,
exchange.baseUrl,
).toArray();
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-11-14 02:52:29 +01:00
// Denomination of the first coin, we assume that all other
// coins have the same currency
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) {
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 });
}
let totalFees = Amounts.getZero(currency);
2019-06-26 15:30:32 +02:00
let wireFee: AmountJson | undefined;
for (const fee of exchangeFees.feesForType[wireMethod] || []) {
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) {
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
}
}
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) {
totalFees = Amounts.add(totalFees, res.totalFees).amount;
2016-11-14 03:01:42 +01:00
return {
cds: res.cds,
2017-05-28 01:10:54 +02:00
exchangeUrl: exchange.baseUrl,
totalAmount: remainingAmount,
totalFees,
2017-05-28 01:10:54 +02:00
};
}
2016-11-14 02:52:29 +01:00
}
return undefined;
}
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(
2019-11-30 00:36:20 +01:00
proposal: ProposalRecord,
2019-06-26 15:30:32 +02:00
payCoinInfo: PayCoinInfo,
chosenExchange: string,
): Promise<PurchaseRecord> {
2017-05-28 01:10:54 +02:00
const payReq: PayReq = {
coins: payCoinInfo.sigs,
merchant_pub: proposal.contractTerms.merchant_pub,
2018-01-04 13:22:23 +01:00
mode: "pay",
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 = {
abortDone: false,
abortRequested: false,
2017-06-01 18:46:07 +02:00
contractTerms: proposal.contractTerms,
contractTermsHash: proposal.contractTermsHash,
finished: false,
2018-01-18 02:50:18 +01:00
lastSessionId: undefined,
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,
};
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);
}
},
);
this.badge.showNotification();
2016-09-28 18:00:13 +02:00
this.notifier.notify();
return t;
}
2015-12-14 16:54:47 +01: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 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-28 02:49:27 +02:00
2019-11-30 00:36:20 +01:00
let proposalId: string;
try {
2019-08-28 02:49:27 +02:00
proposalId = await this.downloadProposal(
uriResult.downloadUrl,
uriResult.sessionId,
);
} catch (e) {
return {
status: "error",
error: e.toString(),
};
}
const proposal = await this.getProposal(proposalId);
if (!proposal) {
2019-11-30 00:36:20 +01:00
throw Error(`could not get proposal ${proposalId}`);
}
2019-08-28 02:49:27 +02:00
console.log("proposal", proposal);
const differentPurchase = await oneShotGetIndexed(
this.db,
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) {
// 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 {
if (uriResult.sessionId) {
await this.submitPay(
differentPurchase.contractTermsHash,
uriResult.sessionId,
);
}
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) {
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,
2019-11-30 00:36:20 +01:00
proposalId: proposal.proposalId,
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 &&
2019-11-30 00:36:20 +01:00
this.speculativePayData.orderDownloadId !== proposalId)
2019-09-06 09:48:00 +02:00
) {
const { exchangeUrl, cds, totalAmount } = res;
const payCoinInfo = await this.cryptoApi.signDeposit(
proposal.contractTerms,
cds,
totalAmount,
);
this.speculativePayData = {
exchangeUrl,
payCoinInfo,
proposal,
2019-11-30 00:36:20 +01:00
orderDownloadId: proposalId,
2019-09-06 09:48:00 +02:00
};
Wallet.enableTracing &&
console.log("created speculative pay data for payment");
}
return {
status: "payment-possible",
contractTerms: proposal.contractTerms,
2019-11-30 00:36:20 +01:00
proposalId: proposal.proposalId,
2019-09-06 09:48:00 +02:00
totalFees: res.totalFees,
};
}
2019-09-06 09:48:00 +02:00
if (uriResult.sessionId) {
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),
};
}
/**
* Download a proposal and store it in the database.
* Returns an id for it to retrieve it later.
*
* @param sessionId Current session ID, if the proposal is being
* downloaded in the context of a session ID.
*/
2019-11-30 00:36:20 +01:00
async downloadProposal(url: string, sessionId?: string): Promise<string> {
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) {
2019-11-30 00:36:20 +01:00
return oldProposal.proposalId;
2018-01-18 01:37:30 +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 + "'");
let resp;
try {
2019-08-22 23:36:36 +02:00
resp = await this.http.get(urlWithNonce);
} catch (e) {
console.log("contract download failed", e);
throw e;
}
2019-08-22 23:36:36 +02:00
const proposal = Proposal.checked(resp.responseJson);
const contractTermsHash = await this.hashContract(proposal.contract_terms);
2019-11-30 00:36:20 +01:00
const proposalId = encodeCrock(getRandomBytes(32));
const proposalRecord: ProposalRecord = {
contractTerms: proposal.contract_terms,
contractTermsHash,
merchantSig: proposal.sig,
noncePriv: priv,
2019-11-21 23:09:43 +01:00
timestamp: getTimestampNow(),
url,
downloadSessionId: sessionId,
2019-11-30 00:36:20 +01:00
proposalId: proposalId,
proposalStatus: ProposalStatus.PROPOSED,
};
2019-11-30 00:36:20 +01:00
await oneShotPut(this.db, Stores.proposals, proposalRecord);
2016-11-13 10:17:39 +01:00
this.notifier.notify();
2019-11-30 00:36:20 +01:00
return proposalId;
2016-11-13 10:17:39 +01:00
}
async refundFailedPay(proposalId: number) {
console.log(`refunding failed payment with proposal id ${proposalId}`);
const proposal = await oneShotGet(this.db, Stores.proposals, proposalId);
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,
);
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,
);
if (!purchase) {
throw Error("Purchase not found: " + contractTermsHash);
}
if (purchase.abortRequested) {
throw Error("not submitting payment for aborted purchase");
}
let resp;
const payReq = { ...purchase.payReq, session_id: sessionId };
const payUrl = new URI("pay")
.absoluteTo(purchase.contractTerms.merchant_base_url)
.href();
2019-11-01 18:39:23 +01:00
try {
2019-11-01 18:39:23 +01:00
resp = await this.http.postJson(payUrl, payReq);
} 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) {
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);
},
);
2018-01-23 16:19:03 +01:00
for (const c of purchase.payReq.coins) {
this.refresh(c.coin_pub);
}
const nextUrl = this.getNextUrl(purchase.contractTerms);
2019-06-26 15:30:32 +02:00
this.cachedNextUrl[purchase.contractTerms.fulfillment_url] = {
nextUrl,
lastSessionId: sessionId,
};
return { nextUrl };
}
/**
* Refresh all dirty coins.
* The returned promise resolves only after all refresh
* operations have completed.
*/
async refreshDirtyCoins(): Promise<{ numRefreshed: number }> {
let n = 0;
const coins = await oneShotIter(this.db, Stores.coins).toArray();
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
/**
* 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(
2019-11-30 00:36:20 +01:00
proposalId: string,
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}`,
);
const proposal = await oneShotGet(this.db, Stores.proposals, proposalId);
if (!proposal) {
throw Error(`proposal with id ${proposalId} not found`);
}
2016-02-22 21:52:53 +01: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) {
return this.submitPay(purchase.contractTermsHash, sessionId);
2016-09-28 18:54:48 +02: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({
allowedAuditors: proposal.contractTerms.auditors,
allowedExchanges: proposal.contractTerms.exchanges,
depositFeeLimit: Amounts.parseOrThrow(proposal.contractTerms.max_fee),
paymentAmount: Amounts.parseOrThrow(proposal.contractTerms.amount),
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,
},
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) {
// Should not happen, since checkPay should be called first
2016-09-28 18:54:48 +02:00
console.log("not confirming payment, insufficient coins");
throw Error("insufficient balance");
2016-09-28 18:54:48 +02:00
}
const sd = await this.getSpeculativePayData(proposalId);
if (!sd) {
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,
);
} else {
2019-06-26 15:30:32 +02:00
purchase = await this.recordConfirmPay(
sd.proposal,
sd.payCoinInfo,
sd.exchangeUrl,
);
}
return this.submitPay(purchase.contractTermsHash, sessionId);
}
2015-12-16 00:38:36 +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(
2019-11-30 00:36:20 +01:00
proposalId: string,
2019-06-26 15:30:32 +02:00
): Promise<SpeculativePayData | undefined> {
const sp = this.speculativePayData;
if (!sp) {
return;
}
2019-11-30 00:36:20 +01:00
if (sp.orderDownloadId !== proposalId) {
return;
}
const coinKeys = sp.payCoinInfo.updatedCoins.map(x => x.coinPub);
const coins: CoinRecord[] = [];
for (let coinKey of coinKeys) {
const cc = await oneShotGet(this.db, Stores.coins, coinKey);
if (cc) {
coins.push(cc);
}
}
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
) {
return;
}
}
return sp;
}
2016-02-10 02:03:31 +01:00
2019-11-30 00:36:20 +01:00
private async processReserveBankStatus(reservePub: string): Promise<void> {
let reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
switch (reserve?.reserveStatus) {
case ReserveRecordStatus.WAIT_CONFIRM_BANK:
case ReserveRecordStatus.REGISTERING_BANK:
break;
default:
return;
2019-11-21 23:09:43 +01:00
}
const bankStatusUrl = reserve.bankWithdrawStatusUrl;
if (!bankStatusUrl) {
2019-11-30 00:36:20 +01:00
return;
}
2019-11-30 00:36:20 +01:00
let status: WithdrawOperationStatusResponse;
try {
const statusResp = await this.http.get(bankStatusUrl);
status = WithdrawOperationStatusResponse.checked(statusResp.responseJson);
} catch (e) {
throw e;
}
2019-11-30 00:36:20 +01:00
if (status.selection_done) {
if (reserve.reserveStatus === ReserveRecordStatus.REGISTERING_BANK) {
await this.registerReserveWithBank(reservePub);
return await this.processReserveBankStatus(reservePub);
}
} else {
await this.registerReserveWithBank(reservePub);
return await this.processReserveBankStatus(reservePub);
}
if (status.transfer_done) {
2019-11-20 20:02:48 +01:00
await oneShotMutate(this.db, Stores.reserves, reservePub, r => {
2019-11-30 00:36:20 +01:00
switch (r.reserveStatus) {
case ReserveRecordStatus.REGISTERING_BANK:
case ReserveRecordStatus.WAIT_CONFIRM_BANK:
break;
default:
return;
}
const now = getTimestampNow();
2019-11-21 23:09:43 +01:00
r.timestampConfirmed = now;
2019-11-30 00:36:20 +01:00
r.reserveStatus = ReserveRecordStatus.QUERYING_STATUS;
return r;
});
2019-11-30 00:36:20 +01:00
await this.processReserveImpl(reservePub);
} else {
2019-11-20 20:02:48 +01:00
await oneShotMutate(this.db, Stores.reserves, reservePub, r => {
2019-11-30 00:36:20 +01:00
switch (r.reserveStatus) {
case ReserveRecordStatus.WAIT_CONFIRM_BANK:
break;
default:
return;
}
r.bankWithdrawConfirmUrl = status.confirm_transfer_url;
return r;
});
}
}
2019-11-30 00:36:20 +01:00
async registerReserveWithBank(reservePub: string) {
let reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
switch (reserve?.reserveStatus) {
case ReserveRecordStatus.WAIT_CONFIRM_BANK:
case ReserveRecordStatus.REGISTERING_BANK:
break;
default:
return;
}
const bankStatusUrl = reserve.bankWithdrawStatusUrl;
if (!bankStatusUrl) {
return;
}
console.log("making selection");
if (reserve.timestampReserveInfoPosted) {
throw Error("bank claims that reserve info selection is not done");
}
const bankResp = await this.http.postJson(bankStatusUrl, {
reserve_pub: reservePub,
selected_exchange: reserve.exchangeWire,
});
console.log("got response", bankResp);
await oneShotMutate(this.db, Stores.reserves, reservePub, r => {
switch (r.reserveStatus) {
case ReserveRecordStatus.REGISTERING_BANK:
case ReserveRecordStatus.WAIT_CONFIRM_BANK:
break;
default:
return;
}
r.timestampReserveInfoPosted = getTimestampNow();
r.reserveStatus = ReserveRecordStatus.WAIT_CONFIRM_BANK;
return r;
});
return this.processReserveBankStatus(reservePub);
}
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
*/
async processReserve(reservePub: string): Promise<void> {
2019-11-30 00:36:20 +01:00
const p = this.memoProcessReserve.find(reservePub);
if (p) {
return p;
} else {
return this.memoProcessReserve.put(
reservePub,
this.processReserveImpl(reservePub),
);
}
}
private async processReserveImpl(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-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:
2019-11-30 00:36:20 +01:00
await this.processReserveBankStatus(reservePub);
return this.processReserveImpl(reservePub);
2019-11-21 23:09:43 +01:00
case ReserveRecordStatus.QUERYING_STATUS:
await this.updateReserve(reservePub);
2019-11-30 00:36:20 +01:00
return this.processReserveImpl(reservePub);
2019-11-21 23:09:43 +01:00
case ReserveRecordStatus.WITHDRAWING:
await this.depleteReserve(reservePub);
break;
case ReserveRecordStatus.DORMANT:
// nothing to do
break;
2019-11-30 00:36:20 +01:00
case ReserveRecordStatus.WAIT_CONFIRM_BANK:
await this.processReserveBankStatus(reservePub);
break;
2019-11-21 23:09:43 +01:00
default:
console.warn("unknown reserve record status:", reserve.reserveStatus);
2019-11-30 00:36:20 +01:00
assertUnreachable(reserve.reserveStatus);
2019-11-21 23:09:43 +01:00
break;
2016-09-28 18:54:48 +02:00
}
}
/**
* Given a planchet, withdraw a coin from the exchange.
*/
2019-11-30 00:36:20 +01:00
private async processPlanchet(coinPub: string): Promise<void> {
logger.trace("process planchet", coinPub);
const planchet = await oneShotGet(this.db, Stores.planchets, coinPub);
if (!planchet) {
console.log("processPlanchet: planchet not found");
2019-11-21 23:09:43 +01:00
return;
}
const exchange = await oneShotGet(
this.db,
Stores.exchanges,
2019-11-30 00:36:20 +01:00
planchet.exchangeBaseUrl,
2019-11-21 23:09:43 +01:00
);
if (!exchange) {
2019-11-30 00:36:20 +01:00
console.error("db inconsistent: exchange for planchet not found");
2019-11-21 23:09:43 +01:00
return;
2016-10-19 22:59:24 +02:00
}
2019-11-21 23:09:43 +01:00
const denom = await oneShotGet(this.db, Stores.denominations, [
2019-11-30 00:36:20 +01:00
planchet.exchangeBaseUrl,
planchet.denomPub,
2019-11-21 23:09:43 +01:00
]);
2016-10-19 22:59:24 +02:00
2019-11-21 23:09:43 +01:00
if (!denom) {
2019-11-30 00:36:20 +01:00
console.error("db inconsistent: denom for planchet not found");
2019-11-21 23:09:43 +01:00
return;
}
2016-10-19 22:59:24 +02:00
2019-11-21 23:09:43 +01:00
const wd: any = {};
2019-11-30 00:36:20 +01:00
wd.denom_pub_hash = planchet.denomPubHash;
wd.reserve_pub = planchet.reservePub;
wd.reserve_sig = planchet.withdrawSig;
wd.coin_ev = planchet.coinEv;
2019-11-21 23:09:43 +01:00
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-11-21 23:09:43 +01:00
const denomSig = await this.cryptoApi.rsaUnblind(
r.ev_sig,
2019-11-30 00:36:20 +01:00
planchet.blindingKey,
planchet.denomPub,
2019-11-21 23:09:43 +01:00
);
2017-11-30 04:07:36 +01:00
2019-11-21 23:09:43 +01:00
const coin: CoinRecord = {
2019-11-30 00:36:20 +01:00
blindingKey: planchet.blindingKey,
coinPriv: planchet.coinPriv,
coinPub: planchet.coinPub,
currentAmount: planchet.coinValue,
denomPub: planchet.denomPub,
denomPubHash: planchet.denomPubHash,
2019-11-21 23:09:43 +01:00
denomSig,
2019-11-30 00:36:20 +01:00
exchangeBaseUrl: planchet.exchangeBaseUrl,
reservePub: planchet.reservePub,
2019-11-21 23:09:43 +01:00
status: CoinStatus.Fresh,
2019-11-30 00:36:20 +01:00
coinIndex: planchet.coinIndex,
withdrawSessionId: planchet.withdrawSessionId,
};
2019-11-21 23:09:43 +01:00
await runWithWriteTransaction(
this.db,
2019-11-30 00:36:20 +01:00
[Stores.planchets, Stores.coins, Stores.withdrawalSession, Stores.reserves],
2019-11-21 23:09:43 +01:00
async tx => {
2019-11-30 00:36:20 +01:00
const currentPc = await tx.get(Stores.planchets, coin.coinPub);
2019-11-21 23:09:43 +01:00
if (!currentPc) {
return;
}
2019-11-30 00:36:20 +01:00
const ws = await tx.get(
Stores.withdrawalSession,
planchet.withdrawSessionId,
);
if (!ws) {
return;
}
if (ws.withdrawn[planchet.coinIndex]) {
// Already withdrawn
return;
}
ws.withdrawn[planchet.coinIndex] = true;
await tx.put(Stores.withdrawalSession, ws);
const r = await tx.get(Stores.reserves, planchet.reservePub);
if (!r) {
return;
}
r.withdrawCompletedAmount = Amounts.add(
r.withdrawCompletedAmount,
Amounts.add(denom.value, denom.feeWithdraw).amount,
).amount;
tx.put(Stores.reserves, r);
await tx.delete(Stores.planchets, coin.coinPub);
2019-11-21 23:09:43 +01:00
await tx.add(Stores.coins, coin);
},
);
2019-11-30 00:36:20 +01:00
this.notifier.notify();
2019-11-21 23:09:43 +01:00
logger.trace(`withdraw of one coin ${coin.coinPub} finished`);
}
2016-02-09 21:56:06 +01:00
/**
* Create a reserve, but do not flag it as confirmed yet.
*
* 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;
}
2019-11-30 00:36:20 +01:00
const currency = req.amount.currency;
2016-09-28 23:41:34 +02:00
const reserveRecord: ReserveRecord = {
2016-09-28 18:54:48 +02:00
created: now,
2019-11-30 00:36:20 +01:00
withdrawAllocatedAmount: Amounts.getZero(currency),
withdrawCompletedAmount: Amounts.getZero(currency),
withdrawRemainingAmount: Amounts.getZero(currency),
2019-11-21 23:09:43 +01:00
exchangeBaseUrl: canonExchange,
2017-05-28 01:10:54 +02:00
hasPayback: false,
2019-11-30 00:36:20 +01:00
initiallyRequestedAmount: req.amount,
2019-11-21 23:09:43 +01:00
reservePriv: keypair.priv,
reservePub: keypair.pub,
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,
2019-11-30 00:36:20 +01:00
lastStatusQuery: undefined,
2016-09-28 18:54:48 +02:00
};
const senderWire = req.senderWire;
2019-05-08 04:53:26 +02:00
if (senderWire) {
const rec = {
2019-05-08 04:53:26 +02:00
paytoUri: senderWire,
};
await oneShotPut(this.db, Stores.senderWires, rec);
}
2017-05-28 01:10:54 +02:00
const exchangeInfo = await this.updateExchangeFromUrl(req.exchange);
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,
);
if (!currencyRecord) {
currencyRecord = {
auditors: [],
2017-05-28 01:10:54 +02:00
exchanges: [],
fractionalDigits: 2,
name: exchangeDetails.currency,
2017-05-28 01:10:54 +02:00
};
}
if (!isAudited && !isTrusted) {
2019-06-26 15:30:32 +02:00
currencyRecord.exchanges.push({
baseUrl: req.exchange,
exchangePub: exchangeDetails.masterPublicKey,
2019-06-26 15:30:32 +02:00
});
}
const cr: CurrencyRecord = currencyRecord;
2019-11-30 00:36:20 +01:00
const resp = await runWithWriteTransaction(
2019-11-20 20:02:48 +01:00
this.db,
2019-11-30 00:36:20 +01:00
[Stores.currencies, Stores.reserves, Stores.bankWithdrawUris],
2019-11-20 20:02:48 +01:00
async tx => {
2019-11-30 00:36:20 +01:00
// Check if we have already created a reserve for that bankWithdrawStatusUrl
if (reserveRecord.bankWithdrawStatusUrl) {
const bwi = await tx.get(
Stores.bankWithdrawUris,
reserveRecord.bankWithdrawStatusUrl,
);
if (bwi) {
const otherReserve = await tx.get(Stores.reserves, bwi.reservePub);
if (otherReserve) {
logger.trace(
"returning existing reserve for bankWithdrawStatusUri",
);
return {
exchange: otherReserve.exchangeBaseUrl,
reservePub: otherReserve.reservePub,
};
}
}
await tx.put(Stores.bankWithdrawUris, {
reservePub: reserveRecord.reservePub,
talerWithdrawUri: reserveRecord.bankWithdrawStatusUrl,
});
}
2019-11-20 20:02:48 +01:00
await tx.put(Stores.currencies, cr);
await tx.put(Stores.reserves, reserveRecord);
2019-11-30 00:36:20 +01:00
const r: CreateReserveResponse = {
exchange: canonExchange,
reservePub: keypair.pub,
};
return r;
2019-11-20 20:02:48 +01:00
},
);
2016-02-11 18:17:02 +01:00
2019-11-30 00:36:20 +01:00
// Asynchronously process the reserve, but return
// to the caller already.
this.processReserve(resp.reservePub).catch(e => {
2019-11-21 23:09:43 +01:00
console.error("Processing reserve failed:", e);
});
2019-08-28 02:49:27 +02:00
2019-11-30 00:36:20 +01:00
return resp;
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;
});
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);
});
}
2015-12-13 23:47:30 +01:00
/**
* Withdraw coins from a reserve until it is empty.
*
* When finished, marks the reserve as depleted by setting
* the depleted timestamp.
*/
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}`);
2019-11-30 00:36:20 +01:00
const withdrawAmount = reserve.withdrawRemainingAmount;
logger.trace(`getting denom list`);
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-30 00:36:20 +01:00
logger.trace(`got denom list`);
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: {},
});
2019-11-30 00:36:20 +01:00
console.log(m);
2019-11-21 23:09:43 +01:00
throw new OperationFailedAndReportedError(m);
}
2016-10-20 01:37:00 +02:00
2019-11-30 00:36:20 +01:00
logger.trace("selected denominations");
const withdrawalSessionId = encodeCrock(randomBytes(32));
const withdrawalRecord: WithdrawalSessionRecord = {
withdrawSessionId: withdrawalSessionId,
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(),
2019-11-30 00:36:20 +01:00
denoms: denomsForWithdraw.map(x => x.denomPub),
withdrawn: denomsForWithdraw.map(x => false),
planchetCreated: denomsForWithdraw.map(x => false),
};
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-30 00:36:20 +01:00
const remaining = Amounts.sub(
r.withdrawRemainingAmount,
totalWithdrawAmount,
);
if (remaining.saturated) {
console.error("can't create planchets, saturated");
throw TransactionAbort;
}
2019-11-30 00:36:20 +01:00
const allocated = Amounts.add(
r.withdrawAllocatedAmount,
totalWithdrawAmount,
2019-11-30 00:36:20 +01:00
);
if (allocated.saturated) {
console.error("can't create planchets, saturated");
2019-11-21 23:09:43 +01:00
throw TransactionAbort;
2019-11-19 16:16:12 +01:00
}
2019-11-30 00:36:20 +01:00
r.withdrawRemainingAmount = remaining.amount;
r.withdrawAllocatedAmount = allocated.amount;
2019-11-21 23:09:43 +01:00
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,
2019-11-30 00:36:20 +01:00
[Stores.planchets, Stores.withdrawalSession, Stores.reserves],
2019-11-21 23:09:43 +01:00
async tx => {
const myReserve = await tx.get(Stores.reserves, reservePub);
if (!myReserve) {
return false;
}
if (myReserve.reserveStatus !== ReserveRecordStatus.WITHDRAWING) {
return false;
}
await tx.mutate(Stores.reserves, reserve.reservePub, mutateReserve);
2019-11-30 00:36:20 +01:00
await tx.put(Stores.withdrawalSession, withdrawalRecord);
2019-11-21 23:09:43 +01:00
return true;
},
);
2019-11-19 16:16:12 +01:00
2019-11-21 23:09:43 +01:00
if (success) {
2019-11-30 00:36:20 +01:00
console.log("processing new withdraw session");
await this.processWithdrawSession(withdrawalSessionId);
} else {
console.trace("withdraw session already existed");
}
}
private async processWithdrawSession(withdrawalSessionId: string): Promise<void> {
logger.trace("processing withdraw session", withdrawalSessionId);
const ws = await oneShotGet(
this.db,
Stores.withdrawalSession,
withdrawalSessionId,
);
if (!ws) {
logger.trace("withdraw session doesn't exist");
return;
}
const ps = ws.denoms.map((d, i) =>
this.processWithdrawCoin(withdrawalSessionId, i),
);
await Promise.all(ps);
this.badge.showNotification();
return;
}
private async processWithdrawCoin(
withdrawalSessionId: string,
coinIndex: number,
) {
logger.info("starting withdraw for coin");
const ws = await oneShotGet(
this.db,
Stores.withdrawalSession,
withdrawalSessionId,
);
if (!ws) {
console.log("ws doesn't exist");
return;
}
const coin = await oneShotGetIndexed(
this.db,
Stores.coins.byWithdrawalWithIdx,
[withdrawalSessionId, coinIndex],
);
if (coin) {
console.log("coin already exists");
return;
}
const pc = await oneShotGetIndexed(
this.db,
Stores.planchets.byWithdrawalWithIdx,
[withdrawalSessionId, coinIndex],
);
if (pc) {
return this.processPlanchet(pc.coinPub);
} else {
const reserve = await oneShotGet(this.db, Stores.reserves, ws.reservePub);
if (!reserve) {
return;
2019-11-21 23:09:43 +01:00
}
2019-11-30 00:36:20 +01:00
const denom = await oneShotGet(this.db, Stores.denominations, [
reserve.exchangeBaseUrl,
ws.denoms[coinIndex],
]);
if (!denom) {
return;
}
const r = await this.cryptoApi.createPlanchet(denom, reserve);
const newPlanchet: PlanchetRecord = {
blindingKey: r.blindingKey,
coinEv: r.coinEv,
coinIndex,
coinPriv: r.coinPriv,
coinPub: r.coinPub,
coinValue: r.coinValue,
denomPub: r.denomPub,
denomPubHash: r.denomPubHash,
exchangeBaseUrl: r.exchangeBaseUrl,
isFromTip: false,
reservePub: r.reservePub,
withdrawSessionId: withdrawalSessionId,
withdrawSig: r.withdrawSig,
};
await runWithWriteTransaction(this.db, [Stores.planchets, Stores.withdrawalSession], async (tx) => {
const myWs = await tx.get(Stores.withdrawalSession, withdrawalSessionId);
if (!myWs) {
return;
}
if (myWs.planchetCreated[coinIndex]) {
return;
}
await tx.put(Stores.planchets, newPlanchet);
});
await this.processPlanchet(newPlanchet.coinPub);
2019-11-19 16:16:12 +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> {
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) {
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-30 00:36:20 +01:00
const balance = Amounts.parseOrThrow(reserveInfo.balance);
2019-11-21 23:09:43 +01:00
await oneShotMutate(this.db, Stores.reserves, reserve.reservePub, r => {
if (r.reserveStatus !== ReserveRecordStatus.QUERYING_STATUS) {
return;
}
2019-11-30 00:36:20 +01:00
// FIXME: check / compare history!
if (!r.lastStatusQuery) {
// FIXME: check if this matches initial expectations
r.withdrawRemainingAmount = balance;
} else {
const expectedBalance = Amounts.sub(
r.withdrawAllocatedAmount,
r.withdrawCompletedAmount,
);
const cmp = Amounts.cmp(balance, expectedBalance.amount);
if (cmp == 0) {
// Nothing changed.
return;
}
if (cmp > 0) {
const extra = Amounts.sub(balance, expectedBalance.amount).amount;
r.withdrawRemainingAmount = Amounts.add(
r.withdrawRemainingAmount,
extra,
).amount;
} else {
// We're missing some money.
}
}
r.lastStatusQuery = getTimestampNow();
r.reserveStatus = ReserveRecordStatus.WITHDRAWING;
2019-11-21 23:09:43 +01:00
return r;
});
this.notifier.notify();
}
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
);
});
}
/**
* 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,
);
if (!exchange) {
throw Error(`exchange ${exchangeBaseUrl} not found`);
}
const exchangeDetails = exchange.details;
if (!exchangeDetails) {
throw Error(`exchange ${exchangeBaseUrl} details not available`);
}
const possibleDenoms = await this.getPossibleDenoms(exchange.baseUrl);
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;
return Amounts.cmp(a1, a2);
});
2017-10-15 19:28:35 +02:00
for (const denom of possibleDenoms) {
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,
exchangeDetails.masterPublicKey,
2019-06-26 15:30:32 +02:00
);
if (!valid) {
denom.status = DenominationStatus.VerifiedBad;
} else {
denom.status = DenominationStatus.VerifiedGood;
}
await oneShotPut(this.db, Stores.denominations, denom);
if (valid) {
return Amounts.add(denom.feeWithdraw, denom.value).amount;
}
}
return Amounts.getZero(exchangeDetails.currency);
}
/**
* 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,
);
if (!exchange) {
2019-11-30 00:36:20 +01:00
console.log("exchange not found");
throw Error(`exchange ${exchangeBaseUrl} not found`);
}
const exchangeDetails = exchange.details;
if (!exchangeDetails) {
2019-11-30 00:36:20 +01:00
console.log("exchange details not available");
throw Error(`exchange ${exchangeBaseUrl} details not available`);
}
2019-11-30 00:36:20 +01:00
console.log("getting possible denoms");
const possibleDenoms = await this.getPossibleDenoms(exchange.baseUrl);
2019-11-30 00:36:20 +01:00
console.log("got possible denoms");
let allValid = false;
let selectedDenoms: DenominationRecord[];
do {
allValid = true;
2017-05-28 01:10:54 +02:00
const nextPossibleDenoms = [];
selectedDenoms = getWithdrawDenomList(amount, possibleDenoms);
2019-11-30 00:36:20 +01:00
console.log("got withdraw denom list");
2017-05-28 01:10:54 +02:00
for (const denom of selectedDenoms || []) {
if (denom.status === DenominationStatus.Unverified) {
2019-11-30 00:36:20 +01:00
console.log("checking validity", denom, exchangeDetails.masterPublicKey);
2019-06-26 15:30:32 +02:00
const valid = await this.cryptoApi.isValidDenom(
denom,
exchangeDetails.masterPublicKey,
2019-06-26 15:30:32 +02:00
);
2019-11-30 00:36:20 +01:00
console.log("done checking validity");
if (!valid) {
denom.status = DenominationStatus.VerifiedBad;
allValid = false;
} else {
denom.status = DenominationStatus.VerifiedGood;
nextPossibleDenoms.push(denom);
}
await oneShotPut(this.db, Stores.denominations, denom);
} else {
nextPossibleDenoms.push(denom);
}
}
} while (selectedDenoms.length > 0 && !allValid);
2019-11-30 00:36:20 +01:00
console.log("returning denoms");
return selectedDenoms;
}
/**
* 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 }> {
let isTrusted = false;
let isAudited = false;
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,
);
if (currencyRecord) {
2017-05-28 01:10:54 +02:00
for (const trustedExchange of currencyRecord.exchanges) {
if (trustedExchange.exchangePub === exchangeDetails.masterPublicKey) {
isTrusted = true;
break;
}
}
2017-05-28 01:10:54 +02:00
for (const trustedAuditor of currencyRecord.auditors) {
for (const exchangeAuditor of exchangeDetails.auditors) {
2017-06-04 20:25:28 +02:00
if (trustedAuditor.auditorPub === exchangeAuditor.auditor_pub) {
isAudited = true;
break;
}
}
}
}
2019-06-26 15:30:32 +02:00
return { isTrusted, isAudited };
}
2019-08-30 17:27:59 +02:00
async getWithdrawDetailsForUri(
talerWithdrawUri: string,
maybeSelectedExchange?: string,
): Promise<WithdrawDetails> {
2019-09-01 01:05:38 +02:00
const info = await this.getWithdrawalInfo(talerWithdrawUri);
let rci: ReserveCreationInfo | undefined = undefined;
if (maybeSelectedExchange) {
2019-08-30 17:27:59 +02:00
rci = await this.getWithdrawDetailsForAmount(
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);
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`,
);
}
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) {
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[] = [];
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);
let earliestDepositExpiration = selectedDenoms[0].stampExpireDeposit;
for (let i = 1; i < selectedDenoms.length; i++) {
const expireDeposit = selectedDenoms[i].stampExpireDeposit;
if (expireDeposit.t_ms < earliestDepositExpiration.t_ms) {
earliestDepositExpiration = expireDeposit;
}
}
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
}
let versionMatch;
if (exchangeDetails.protocolVersion) {
2019-06-26 15:30:32 +02:00
versionMatch = LibtoolVersion.compare(
WALLET_PROTOCOL_VERSION,
exchangeDetails.protocolVersion,
2019-06-26 15:30:32 +02:00
);
if (
versionMatch &&
!versionMatch.compatible &&
versionMatch.currentCmp === -1
) {
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:24:18 +01:00
if (isFirefox()) {
console.log("skipping update check on Firefox");
} else {
chrome.runtime.requestUpdateCheck((status, details) => {
console.log("update check status:", status);
});
}
}
}
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,
exchangeVersion: exchangeDetails.protocolVersion || "unknown",
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,
walletVersion: WALLET_PROTOCOL_VERSION,
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
}
async getExchangePaytoUri(
exchangeBaseUrl: string,
supportedTargetTypes: string[],
): Promise<string> {
2019-11-30 00:36:20 +01:00
// We do the update here, since the exchange might not even exist
// yet in our database.
const exchangeRecord = await this.updateExchangeFromUrl(exchangeBaseUrl);
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");
}
/**
* Update or add exchange DB entry by fetching the /keys and /wire information.
* Optionally link the reserve entry to the new or existing
2016-03-01 19:39:17 +01:00
* exchange entry in then DB.
*/
async updateExchangeFromUrl(
2019-06-26 15:30:32 +02:00
baseUrl: string,
force: boolean = false,
2019-06-26 15:30:32 +02:00
): Promise<ExchangeRecord> {
const now = getTimestampNow();
baseUrl = canonicalizeBaseUrl(baseUrl);
2016-09-28 18:00:13 +02:00
const r = await oneShotGet(this.db, Stores.exchanges, baseUrl);
2016-09-28 18:00:13 +02:00
if (!r) {
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
};
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 => {
const rec = await t.get(Stores.exchanges, baseUrl);
if (!rec) {
return;
}
2019-11-21 23:09:43 +01:00
if (rec.updateStatus != ExchangeUpdateStatus.FETCH_KEYS && !force) {
return;
}
2019-11-21 23:09:43 +01:00
if (rec.updateStatus != ExchangeUpdateStatus.FETCH_KEYS && force) {
rec.updateReason = "forced";
}
rec.updateStarted = now;
rec.updateStatus = ExchangeUpdateStatus.FETCH_KEYS;
2019-11-21 23:09:43 +01:00
rec.lastError = undefined;
t.put(Stores.exchanges, rec);
});
}
await this.updateExchangeWithKeys(baseUrl);
await this.updateExchangeWithWireInfo(baseUrl);
2019-11-20 20:02:48 +01:00
const updatedExchange = await oneShotGet(
this.db,
Stores.exchanges,
baseUrl,
);
if (!updatedExchange) {
// This should practically never happen
throw Error("exchange not found");
}
return updatedExchange;
}
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);
}
/**
* 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 20:02:48 +01:00
if (
existingExchangeRecord?.updateStatus != ExchangeUpdateStatus.FETCH_KEYS
) {
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}`;
await this.setExchangeError(baseUrl, {
type: "network",
2019-11-21 23:09:43 +01:00
details: {
requestUrl: e.config?.url,
},
message: m,
});
2019-11-21 23:09:43 +01:00
throw new OperationFailedAndReportedError(m);
}
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}`;
await this.setExchangeError(baseUrl, {
type: "protocol-violation",
details: {},
2019-11-21 23:09:43 +01:00
message: m,
});
2019-11-21 23:09:43 +01:00
throw new OperationFailedAndReportedError(m);
2016-05-24 17:30:27 +02: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);
}
2016-05-24 17:30:27 +02: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);
}
2016-05-24 17:30:27 +02: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-05-24 17:30:27 +02: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);
}
}
},
);
}
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.
*/
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;
}
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-02-11 18:17:02 +01: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> {
/**
* 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 {
const z = Amounts.getZero(amount.currency);
2019-06-26 15:30:32 +02:00
const balanceIdentity = {
available: z,
paybackAmount: z,
pendingIncoming: z,
pendingPayment: z,
pendingIncomingDirty: z,
pendingIncomingRefresh: z,
pendingIncomingWithdraw: z,
2019-06-26 15:30:32 +02:00
};
let entryCurr = balance.byCurrency[amount.currency];
if (!entryCurr) {
2019-06-26 15:30:32 +02:00
balance.byCurrency[amount.currency] = entryCurr = {
...balanceIdentity,
};
}
let entryEx = balance.byExchange[exchange];
if (!entryEx) {
balance.byExchange[exchange] = entryEx = { ...balanceIdentity };
2016-10-19 18:40:29 +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
}
const balanceStore = {
byCurrency: {},
byExchange: {},
};
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;
}
addTo(
balanceStore,
2019-11-20 20:02:48 +01:00
"pendingIncoming",
r.valueOutput,
r.exchangeBaseUrl,
);
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.purchases).forEach(t => {
if (t.finished) {
return;
}
for (const c of t.payReq.coins) {
addTo(
balanceStore,
"pendingPayment",
Amounts.parseOrThrow(c.contribution),
c.exchange_url,
);
}
});
},
);
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> {
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
}
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;
});
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}`);
this.notifier.notify();
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,
);
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,
);
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,
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);
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;
});
this.notifier.notify();
}
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);
2019-11-30 00:36:20 +01:00
const planchets = refreshSession.planchetsForGammas[norevealIndex];
if (!planchets) {
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");
}
2019-11-30 00:36:20 +01:00
const evs = planchets.map((x: RefreshPlanchetRecord) => 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],
2019-11-30 00:36:20 +01:00
planchets[i].coinEv,
2019-06-26 15:30:32 +02:00
);
linkSigs.push(linkSig);
}
2017-05-28 01:10:54 +02:00
const req = {
coin_evs: evs,
new_denoms_h: refreshSession.newDenomHashes,
rc: refreshSession.hash,
2017-05-28 01:10:54 +02:00
transfer_privs: privs,
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");
console.error(e);
2019-08-28 02:49:27 +02:00
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
}
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++) {
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 =
2019-11-30 00:36:20 +01:00
refreshSession.planchetsForGammas[refreshSession.norevealIndex!][i];
2019-06-26 15:30:32 +02:00
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 = {
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,
status: CoinStatus.Fresh,
2019-11-30 00:36:20 +01:00
coinIndex: -1,
withdrawSessionId: "",
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);
},
);
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> {
return await oneShotGet(this.db, Stores.exchanges, exchangeBaseUrl);
}
2016-02-11 18:17:02 +01:00
/**
* Retrive the full event history for this wallet.
*/
async getHistory(
historyQuery?: HistoryQuery,
2019-11-21 23:09:43 +01:00
): Promise<{ history: HistoryEvent[] }> {
const history: HistoryEvent[] = [];
// 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
// increasing even
2019-08-22 23:36:36 +02:00
const proposals = await oneShotIter(this.db, Stores.proposals).toArray();
2017-10-15 19:28:35 +02:00
for (const p of proposals) {
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,
2019-11-30 00:36:20 +01:00
Stores.withdrawalSession,
2019-11-20 20:02:48 +01:00
).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,
});
}
2016-01-26 17:21:17 +01:00
const purchases = await oneShotIter(this.db, Stores.purchases).toArray();
2017-10-15 19:28:35 +02:00
for (const p of purchases) {
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,
});
if (p.timestamp_refund) {
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),
);
2019-06-26 15:30:32 +02:00
const amountsDone = Object.keys(p.refundsDone).map(x =>
Amounts.parseOrThrow(p.refundsDone[x].refund_amount),
);
const amounts: AmountJson[] = amountsPending.concat(amountsDone);
2019-06-26 15:30:32 +02:00
const amount = Amounts.add(
Amounts.getZero(contractAmount.currency),
...amounts,
).amount;
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 19:28:35 +02:00
timestamp: p.timestamp_refund,
type: "refund",
2019-11-21 23:09:43 +01:00
explicit: false,
});
}
}
2016-09-28 17:52:36 +02: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";
history.push({
detail: {
2019-11-21 23:09:43 +01:00
exchangeBaseUrl: r.exchangeBaseUrl,
2019-11-30 00:36:20 +01:00
requestedAmount: Amounts.toString(r.initiallyRequestedAmount),
2019-11-21 23:09:43 +01:00
reservePub: r.reservePub,
reserveType,
bankWithdrawStatusUrl: r.bankWithdrawStatusUrl,
},
2017-10-15 19:28:35 +02:00
timestamp: r.created,
2019-11-21 23:09:43 +01:00
type: "reserve-created",
explicit: false,
});
2019-11-21 23:09:43 +01:00
if (r.timestampConfirmed) {
history.push({
detail: {
2019-11-21 23:09:43 +01:00
exchangeBaseUrl: r.exchangeBaseUrl,
2019-11-30 00:36:20 +01:00
requestedAmount: Amounts.toString(r.initiallyRequestedAmount),
2019-11-21 23:09:43 +01:00
reservePub: r.reservePub,
reserveType,
bankWithdrawStatusUrl: r.bankWithdrawStatusUrl,
},
2019-11-21 23:09:43 +01:00
timestamp: r.created,
type: "reserve-confirmed",
explicit: false,
});
}
}
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,
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> {
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,
},
});
}
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,
},
});
}
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",
});
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,
},
});
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:
2019-11-30 00:36:20 +01:00
case ReserveRecordStatus.REGISTERING_BANK:
2019-11-21 23:09:43 +01:00
pendingOperations.push({
type: "reserve",
stage: reserve.reserveStatus,
timestampCreated: reserve.created,
reserveType,
2019-11-30 00:36:20 +01:00
reservePub: reserve.reservePub,
});
break;
case ReserveRecordStatus.WAIT_CONFIRM_BANK:
pendingOperations.push({
type: "reserve",
stage: reserve.reserveStatus,
timestampCreated: reserve.created,
reserveType,
reservePub: reserve.reservePub,
bankWithdrawConfirmUrl: reserve.bankWithdrawConfirmUrl,
2019-11-21 23:09:43 +01:00
});
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,
2019-11-30 00:36:20 +01:00
refreshSessionId: r.refreshSessionId,
2019-11-21 23:09:43 +01:00
});
});
2019-11-30 00:36:20 +01:00
await oneShotIter(this.db, Stores.planchets).forEach(pc => {
2019-11-21 23:09:43 +01:00
pendingOperations.push({
2019-11-30 00:36:20 +01:00
type: "planchet",
coinPub: pc.coinPub,
2019-11-21 23:09:43 +01:00
reservePub: pc.reservePub,
});
});
2019-11-30 00:36:20 +01:00
await oneShotIter(this.db, Stores.coins).forEach(coin => {
if (coin.status == CoinStatus.Dirty) {
pendingOperations.push({
type: "dirty-coin",
coinPub: coin.coinPub,
});
}
});
await oneShotIter(this.db, Stores.withdrawalSession).forEach(ws => {
const numCoinsWithdrawn = ws.withdrawn.reduce(
(a, x) => a + (x ? 1 : 0),
0,
);
const numCoinsTotal = ws.withdrawn.length;
if (numCoinsWithdrawn < numCoinsTotal) {
pendingOperations.push({
type: "withdraw",
numCoinsTotal,
numCoinsWithdrawn,
reservePub: ws.reservePub,
withdrawSessionId: ws.withdrawSessionId,
});
}
});
await oneShotIter(this.db, Stores.proposals).forEach(proposal => {
if (proposal.proposalStatus == ProposalStatus.PROPOSED) {
pendingOperations.push({
type: "proposal",
merchantBaseUrl: proposal.contractTerms.merchant_base_url,
proposalId: proposal.proposalId,
proposalTimestamp: proposal.timestamp,
});
}
});
2019-11-19 16:16:12 +01:00
return {
pendingOperations,
2019-11-19 16:16:12 +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();
return denoms;
}
2016-11-13 10:17:39 +01:00
2019-11-30 00:36:20 +01:00
async getProposal(proposalId: string): Promise<ProposalRecord | undefined> {
const proposal = await oneShotGet(this.db, Stores.proposals, proposalId);
return proposal;
2016-11-13 10:17:39 +01:00
}
2016-11-15 15:07:17 +01:00
async getExchanges(): Promise<ExchangeRecord[]> {
return await oneShotIter(this.db, Stores.exchanges).toArray();
}
2016-02-23 14:07:53 +01:00
2017-03-24 17:54:22 +01:00
async getCurrencies(): Promise<CurrencyRecord[]> {
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);
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();
}
2019-11-30 00:36:20 +01:00
async getPlanchets(exchangeBaseUrl: string): Promise<PlanchetRecord[]> {
return await oneShotIter(this.db, Stores.planchets).filter(
2019-11-20 20:02:48 +01:00
c => c.exchangeBaseUrl === exchangeBaseUrl,
);
2016-10-12 02:55:53 +02:00
}
private async hashContract(contract: ContractTerms): Promise<string> {
2016-09-28 23:41:34 +02:00
return this.cryptoApi.hashString(canonicalJson(contract));
}
async payback(coinPub: string): Promise<void> {
let coin = await oneShotGet(this.db, Stores.coins, coinPub);
if (!coin) {
throw Error(`Coin ${coinPub} not found, can't request payback`);
}
2017-05-28 01:10:54 +02:00
const reservePub = coin.reservePub;
if (!reservePub) {
throw Error(`Can't request payback for a refreshed coin`);
}
const reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
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`);
}
2019-11-21 23:09:43 +01:00
coin.status = CoinStatus.Dormant;
// 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);
},
);
this.notifier.notify();
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) {
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) {
throw Error(`Coin's reserve doesn't match reserve on payback`);
}
coin = await oneShotGet(this.db, Stores.coins, coinPub);
if (!coin) {
throw Error(`Coin ${coinPub} not found, can't confirm payback`);
}
2019-11-21 23:09:43 +01:00
coin.status = CoinStatus.Dormant;
await oneShotPut(this.db, Stores.coins, coin);
this.notifier.notify();
await this.updateReserve(reservePub!);
}
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 = {
denomPub: denomIn.denom_pub,
2017-05-28 01:10:54 +02:00
denomPubHash,
exchangeBaseUrl,
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,
2019-11-30 00:36:20 +01:00
stampExpireDeposit: extractTalerStampOrThrow(
denomIn.stamp_expire_deposit,
),
stampExpireLegal: extractTalerStampOrThrow(denomIn.stamp_expire_legal),
2019-11-30 00:36:20 +01:00
stampExpireWithdraw: extractTalerStampOrThrow(
denomIn.stamp_expire_withdraw,
),
stampStart: extractTalerStampOrThrow(denomIn.stamp_start),
status: DenominationStatus.Unverified,
value: Amounts.parseOrThrow(denomIn.value),
};
return d;
}
async withdrawPaybackReserve(reservePub: string): Promise<void> {
const reserve = await oneShotGet(this.db, Stores.reserves, reservePub);
if (!reserve) {
throw Error(`Reserve ${reservePub} does not exist`);
}
reserve.hasPayback = false;
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);
});
}
async getPaybackReserves(): Promise<ReserveRecord[]> {
2019-11-20 20:02:48 +01:00
return await oneShotIter(this.db, Stores.reserves).filter(
r => r.hasPayback,
);
}
2017-06-05 02:00:03 +02:00
/**
* Stop ongoing processing.
*/
stop() {
2019-11-21 23:09:43 +01:00
//this.timerGroup.stopCurrentAndFutureTimers();
this.cryptoApi.stop();
2017-06-05 02:00:03 +02:00
}
async getSenderWireInfos(): Promise<SenderWireInfos> {
const m: { [url: string]: Set<string> } = {};
2019-11-20 20:02:48 +01:00
await oneShotIter(this.db, Stores.exchanges).forEach(x => {
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);
const exchangeWireTypes: { [url: string]: string[] } = {};
2019-06-26 15:30:32 +02:00
Object.keys(m).map(e => {
exchangeWireTypes[e] = Array.from(m[e]);
});
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 => {
senderWiresSet.add(x.paytoUri);
});
2019-07-21 23:50:10 +02:00
const senderWires: string[] = Array.from(senderWiresSet);
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);
const wireType = (req.senderWire as any).type;
2019-08-26 01:39:13 +02:00
Wallet.enableTracing && console.log("wireType", wireType);
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);
const exchange = await this.findExchange(req.exchange);
if (!exchange) {
console.error(`Exchange ${req.exchange} not known to the wallet`);
return;
}
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);
const cds = await this.getCoinsForReturn(req.exchange, req.amount);
2019-08-26 01:39:13 +02:00
Wallet.enableTracing && console.log(cds);
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),
);
const contractTerms: ContractTerms = {
H_wire: wireHash,
amount: Amounts.toString(req.amount),
auditors: [],
2019-06-26 15:30:32 +02:00
exchanges: [
{ 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: "",
locations: [],
max_fee: Amounts.toString(req.amount),
merchant: {},
merchant_pub: pub,
2017-10-15 19:28:35 +02:00
order_id: "none",
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",
products: [],
refund_deadline: `/Date(${stampSecNow + 60 * 5})/`,
timestamp: `/Date(${stampSecNow})/`,
2017-10-15 19:28:35 +02:00
wire_method: wireType,
};
2019-06-26 15:30:32 +02:00
const contractTermsHash = await this.cryptoApi.hashString(
canonicalJson(contractTerms),
);
2019-06-26 15:30:32 +02:00
const payCoinInfo = await this.cryptoApi.signDeposit(
contractTerms,
cds,
Amounts.parseOrThrow(contractTerms.amount),
);
2019-08-26 01:39:13 +02:00
Wallet.enableTracing && console.log("pci", payCoinInfo);
2019-06-26 15:30:32 +02:00
const coins = payCoinInfo.sigs.map(s => ({ coinPaySig: s }));
const coinsReturnRecord: CoinsReturnRecord = {
coins,
contractTerms,
contractTermsHash,
2017-10-15 19:28:35 +02:00
exchange: exchange.baseUrl,
merchantPriv: priv,
wire: req.senderWire,
2017-10-15 19:28:35 +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);
}
},
);
this.badge.showNotification();
this.notifier.notify();
this.depositReturnedCoins(coinsReturnRecord);
}
2019-06-26 15:30:32 +02:00
async depositReturnedCoins(
coinsReturnRecord: CoinsReturnRecord,
): Promise<void> {
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,
contribution: c.coinPaySig.contribution,
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,
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,
};
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);
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;
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,
);
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;
}
}
await oneShotPut(this.db, Stores.coinsReturns, currentCrr);
this.notifier.notify();
}
}
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> {
const refundPermissions = refundResponse.refund_permissions;
2017-08-27 03:56:19 +02:00
if (!refundPermissions.length) {
console.warn("got empty refund list");
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-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;
}
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
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);
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");
let resp;
try {
2019-08-22 23:36:36 +02:00
resp = await this.http.get(refundUrl);
} catch (e) {
2019-08-26 01:39:13 +02:00
console.error("error downloading refund permission", e);
throw e;
}
2019-08-22 23:36:36 +02:00
const refundResponse = MerchantRefundResponse.checked(resp.responseJson);
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];
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);
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;
}
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;
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);
}
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> {
return oneShotGet(this.db, Stores.purchases, contractTermsHash);
2017-08-27 03:56:19 +02:00
}
2019-06-26 15:30:32 +02:00
async getFullRefundFees(
refundPermissions: MerchantRefundPermission[],
): Promise<AmountJson> {
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,
);
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,
);
2019-11-20 20:02:48 +01:00
const denoms = await oneShotIterIndex(
this.db,
Stores.denominations.exchangeBaseUrlIndex,
coin0.exchangeBaseUrl,
).toArray();
for (const rp of refundPermissions) {
const coin = await oneShotGet(this.db, Stores.coins, rp.coin_pub);
if (!coin) {
throw Error("coin not found");
}
const denom = await oneShotGet(this.db, Stores.denominations, [
2019-06-26 15:30:32 +02:00
coin0.exchangeBaseUrl,
coin.denomPub,
]);
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?)
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,
);
feeAcc = Amounts.add(feeAcc, refreshCost, refundFee).amount;
}
return feeAcc;
}
2017-11-30 04:07:36 +01:00
2019-11-30 00:36:20 +01:00
async acceptTip(talerTipUri: string): 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");
}
2019-08-30 17:27:59 +02:00
tipRecord.accepted = true;
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;
}
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
);
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);
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();
}
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!");
// Planchets in the form that the merchant expects
2019-06-26 15:30:32 +02:00
const planchetsDetail: TipPlanchetDetail[] = tipRecord.planchets.map(p => ({
coin_ev: p.coinEv,
denom_pub_hash: p.denomPubHash,
}));
2019-08-30 17:27:59 +02:00
let merchantResp;
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);
} catch (e) {
console.log("tipping failed", e);
throw e;
}
2019-08-22 23:36:36 +02:00
const response = TipResponse.checked(merchantResp.responseJson);
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++) {
2019-11-30 00:36:20 +01:00
const tipPlanchet = tipRecord.planchets[i];
const planchet: PlanchetRecord = {
blindingKey: tipPlanchet.blindingKey,
coinEv: tipPlanchet.coinEv,
coinPriv: tipPlanchet.coinPriv,
coinPub: tipPlanchet.coinPub,
coinValue: tipPlanchet.coinValue,
denomPub: tipPlanchet.denomPub,
denomPubHash: tipPlanchet.denomPubHash,
2017-11-30 04:07:36 +01:00
exchangeBaseUrl: tipRecord.exchangeUrl,
isFromTip: true,
reservePub: response.reserve_pub,
withdrawSig: response.reserve_sigs[i].reserve_sig,
2019-11-30 00:36:20 +01:00
coinIndex: -1,
withdrawSessionId: "",
2017-11-30 04:07:36 +01:00
};
2019-11-30 00:36:20 +01:00
await oneShotPut(this.db, Stores.planchets, planchet);
await this.processPlanchet(planchet.coinPub);
2017-11-30 04:07:36 +01:00
}
tipRecord.pickedUp = true;
await oneShotPut(this.db, Stores.tips, tipRecord);
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
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-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
};
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 = {
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;
}
async abortFailedPayment(contractTermsHash: string): Promise<void> {
2019-11-20 20:02:48 +01:00
const purchase = await oneShotGet(
this.db,
Stores.purchases,
contractTermsHash,
);
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.
await oneShotPut(this.db, Stores.purchases, purchase);
let resp;
const abortReq = { ...purchase.payReq, mode: "abort-refund" };
const payUrl = new URI("pay")
.absoluteTo(purchase.contractTerms.merchant_base_url)
.href();
2019-11-01 18:39:23 +01:00
try {
2019-11-01 18:39:23 +01:00
resp = await this.http.postJson(payUrl, abortReq);
} 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);
await this.acceptRefundResponse(refundResponse);
2019-11-20 20:02:48 +01:00
await runWithWriteTransaction(this.db, [Stores.purchases], async tx => {
const p = await tx.get(Stores.purchases, purchase.contractTermsHash);
if (!p) {
return;
}
p.abortDone = true;
await tx.put(Stores.purchases, p);
});
}
2019-11-30 00:36:20 +01:00
public async handleNotifyReserve() {
const reserves = await oneShotIter(this.db, Stores.reserves).toArray();
for (const r of reserves) {
if (r.reserveStatus === ReserveRecordStatus.WAIT_CONFIRM_BANK) {
try {
this.processReserveBankStatus(r.reservePub);
} catch (e) {
console.error(e);
}
}
}
}
/**
* Remove unreferenced / expired data from the wallet's database
* based on the current system time.
*/
async collectGarbage() {
// 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.
}
2019-11-30 00:36:20 +01:00
/**
* Get information about a withdrawal from
* a taler://withdraw URI.
*/
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,
};
}
async acceptWithdrawal(
2019-08-28 02:49:27 +02:00
talerWithdrawUri: string,
selectedExchange: string,
): 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,
});
2019-11-30 00:36:20 +01:00
// We do this here, as the reserve should be registered before we return,
// so that we can redirect the user to the bank's status page.
await this.processReserveBankStatus(reserve.reservePub);
console.log("acceptWithdrawal: returning");
2019-08-28 02:49:27 +02:00
return {
reservePub: reserve.reservePub,
confirmTransferUrl: withdrawInfo.confirmTransferUrl,
};
}
2019-08-31 13:27:12 +02:00
async getPurchaseDetails(hc: string): Promise<PurchaseDetails> {
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),
);
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,
};
}
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
}