wallet-core/src/wallet.ts

2227 lines
67 KiB
TypeScript
Raw Normal View History

2015-12-25 22:42:14 +01:00
/*
This file is part of TALER
(C) 2015 GNUnet e.V.
TALER is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 3, or (at your option) any later version.
TALER is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
2016-07-07 17:59:29 +02:00
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
2015-12-25 22:42:14 +01:00
*/
2016-01-05 15:42:46 +01:00
/**
* High-level wallet operations that should be indepentent from the underlying
* browser extension interface.
*/
2017-05-24 16:52:00 +02:00
/**
* Imports.
*/
2016-05-24 01:53:56 +02:00
import {
AmountJson,
Amounts,
2016-11-15 15:07:17 +01:00
CoinRecord,
CoinPaySig,
Contract,
2016-05-24 01:53:56 +02:00
CreateReserveResponse,
Denomination,
ExchangeHandle,
2016-11-15 15:07:17 +01:00
ExchangeRecord,
2016-05-24 01:53:56 +02:00
Notifier,
PayCoinInfo,
2016-11-15 15:07:17 +01:00
PreCoinRecord,
RefreshSessionRecord,
ReserveCreationInfo,
ReserveRecord,
2017-03-24 17:54:22 +01:00
CurrencyRecord,
Auditor,
2017-03-24 17:54:22 +01:00
AuditorRecord,
WalletBalance,
WalletBalanceEntry,
WireFee,
ExchangeWireFeesRecord,
WireInfo,
DenominationRecord,
DenominationStatus,
CoinStatus,
PaybackConfirmation,
2016-05-24 01:53:56 +02:00
} from "./types";
import {
HttpRequestLibrary,
HttpResponse,
RequestException,
} from "./http";
import {
AbortTransaction,
Index,
JoinResult,
QueryRoot,
Store, JoinLeftResult,
} from "./query";
2016-10-13 02:23:24 +02:00
import {Checkable} from "./checkable";
import {
amountToPretty,
canonicalizeBaseUrl,
canonicalJson,
deepEquals,
flatMap,
getTalerStampSec,
} from "./helpers";
2016-10-13 02:23:24 +02:00
import {CryptoApi} from "./cryptoApi";
import URI = require("urijs");
2016-01-05 15:42:46 +01:00
2015-12-13 23:47:30 +01:00
2017-05-24 16:56:46 +02:00
/**
* Named tuple of coin and denomination.
*/
2016-02-10 02:03:31 +01:00
export interface CoinWithDenom {
2016-11-15 15:07:17 +01:00
coin: CoinRecord;
denom: DenominationRecord;
2016-02-10 02:03:31 +01:00
}
2017-05-24 16:56:46 +02:00
/**
* Element of the payback list that the
* exchange gives us in /keys.
*/
@Checkable.Class()
2017-04-13 16:14:47 +02:00
export class Payback {
@Checkable.String
h_denom_pub: string;
}
2016-02-18 22:50:17 +01:00
2017-05-24 16:56:46 +02:00
/**
* Structure that the exchange gives us in /keys.
*/
@Checkable.Class({extra: true})
2016-02-18 22:50:17 +01:00
export class KeysJson {
@Checkable.List(Checkable.Value(Denomination))
2016-02-10 02:03:31 +01:00
denoms: Denomination[];
2016-02-18 22:50:17 +01:00
@Checkable.String
master_public_key: string;
@Checkable.Any
auditors: any[];
@Checkable.String
list_issue_date: string;
2017-04-13 16:14:47 +02:00
@Checkable.List(Checkable.Value(Payback))
payback?: Payback[];
2016-02-18 22:50:17 +01:00
@Checkable.Any
signkeys: any;
@Checkable.String
eddsa_pub: string;
@Checkable.String
eddsa_sig: string;
static checked: (obj: any) => KeysJson;
2016-02-10 02:03:31 +01:00
}
2016-02-18 22:50:17 +01:00
@Checkable.Class()
class WireFeesJson {
@Checkable.Value(AmountJson)
wire_fee: AmountJson;
@Checkable.Value(AmountJson)
closing_fee: AmountJson;
@Checkable.String
sig: string;
@Checkable.String
start_date: string;
@Checkable.String
end_date: string;
static checked: (obj: any) => WireFeesJson;
}
@Checkable.Class({extra: true})
class WireDetailJson {
@Checkable.String
type: string;
@Checkable.List(Checkable.Value(WireFeesJson))
fees: WireFeesJson[];
static checked: (obj: any) => WireDetailJson;
}
@Checkable.Class()
2016-02-10 02:03:31 +01:00
export class CreateReserveRequest {
/**
* The initial amount for the reserve.
*/
@Checkable.Value(AmountJson)
amount: AmountJson;
/**
2016-03-01 19:39:17 +01:00
* Exchange URL where the bank should create the reserve.
2016-02-10 02:03:31 +01:00
*/
@Checkable.String
2016-03-01 19:39:17 +01:00
exchange: string;
2016-02-10 02:03:31 +01:00
static checked: (obj: any) => CreateReserveRequest;
}
@Checkable.Class()
2016-02-10 02:03:31 +01:00
export class ConfirmReserveRequest {
/**
* Public key of then reserve that should be marked
* as confirmed.
*/
@Checkable.String
reservePub: string;
static checked: (obj: any) => ConfirmReserveRequest;
}
@Checkable.Class()
2016-11-15 15:07:17 +01:00
export class OfferRecord {
2016-02-10 02:03:31 +01:00
@Checkable.Value(Contract)
contract: Contract;
@Checkable.String
merchant_sig: string;
@Checkable.String
H_contract: string;
2016-11-13 10:17:39 +01:00
@Checkable.Number
offer_time: number;
/**
* Serial ID when the offer is stored in the wallet DB.
*/
@Checkable.Optional(Checkable.Number)
id?: number;
2016-11-15 15:07:17 +01:00
static checked: (obj: any) => OfferRecord;
2016-02-10 02:03:31 +01:00
}
2016-09-28 23:41:34 +02:00
export interface HistoryRecord {
type: string;
timestamp: number;
subjectId?: string;
detail: any;
2016-09-29 01:40:29 +02:00
level: HistoryLevel;
2016-09-28 23:41:34 +02:00
}
2016-02-10 02:03:31 +01:00
2016-10-17 15:58:36 +02:00
interface PayReq {
coins: CoinPaySig[];
2017-02-13 00:44:44 +01:00
merchant_pub: string;
order_id: string;
2016-10-17 15:58:36 +02:00
exchange: string;
}
2016-11-15 15:07:17 +01:00
interface TransactionRecord {
contractHash: string;
2016-02-23 14:07:53 +01:00
contract: Contract;
2016-10-17 15:58:36 +02:00
payReq: PayReq;
2016-02-23 14:07:53 +01:00
merchantSig: string;
/**
* The transaction isn't active anymore, it's either successfully paid
* or refunded/aborted.
*/
finished: boolean;
}
2016-09-29 01:40:29 +02:00
export enum HistoryLevel {
Trace = 1,
Developer = 2,
Expert = 3,
User = 4,
}
export interface Badge {
setText(s: string): void;
setColor(c: string): void;
2016-09-12 17:41:12 +02:00
startBusy(): void;
stopBusy(): void;
}
export interface NonceRecord {
priv: string;
pub: string;
}
export interface ConfigRecord {
key: string;
value: any;
}
const builtinCurrencies: CurrencyRecord[] = [
{
name: "KUDOS",
fractionalDigits: 2,
auditors: [
{
baseUrl: "https://auditor.demo.taler.net/",
expirationStamp: (new Date(2027, 1)).getTime(),
auditorPub: "XN9KMN5G2KGPCAN0E89MM5HE8FV4WBWA9KDTMTDR817MWBCYA7H0",
},
2017-04-12 17:47:14 +02:00
],
exchanges: [],
},
{
name: "PUDOS",
fractionalDigits: 2,
auditors: [
],
exchanges: [
{ baseUrl: "https://exchange.test.taler.net/", priority: 0 },
],
},
];
2016-02-10 02:03:31 +01:00
2017-05-24 16:56:46 +02:00
// FIXME: these functions should be dependency-injected
// into the wallet, as this is chrome specific => bad
2016-09-12 17:41:12 +02:00
function setTimeout(f: any, t: number) {
return chrome.extension.getBackgroundPage().setTimeout(f, t);
}
2017-05-01 04:09:52 +02:00
function setInterval(f: any, t: number) {
return chrome.extension.getBackgroundPage().setInterval(f, t);
}
function isWithdrawableDenom(d: DenominationRecord) {
2016-02-10 02:03:31 +01:00
const now_sec = (new Date).getTime() / 1000;
const stamp_withdraw_sec = getTalerStampSec(d.stampExpireWithdraw);
if (stamp_withdraw_sec == null) {
return false;
}
const stamp_start_sec = getTalerStampSec(d.stampStart);
if (stamp_start_sec == null) {
return false;
}
2016-02-10 02:03:31 +01:00
// Withdraw if still possible to withdraw within a minute
if ((stamp_withdraw_sec + 60 > now_sec) && (now_sec >= stamp_start_sec)) {
2016-02-10 02:03:31 +01:00
return true;
}
return false;
}
2016-11-14 03:01:42 +01:00
export type CoinSelectionResult = {exchangeUrl: string, cds: CoinWithDenom[]}|undefined;
export function selectCoins(cds: CoinWithDenom[], paymentAmount: AmountJson,
depositFeeLimit: AmountJson): CoinWithDenom[]|undefined {
2016-11-14 03:01:42 +01:00
if (cds.length == 0) {
return undefined;
}
// Sort by ascending deposit fee
cds.sort((o1, o2) => Amounts.cmp(o1.denom.feeDeposit,
o2.denom.feeDeposit));
2016-11-14 03:01:42 +01:00
let currency = cds[0].denom.value.currency;
let cdsResult: CoinWithDenom[] = [];
let accFee: AmountJson = Amounts.getZero(currency);
let accAmount: AmountJson = Amounts.getZero(currency);
let isBelowFee = false;
let coversAmount = false;
let coversAmountWithFee = false;
for (let i = 0; i < cds.length; i++) {
let {coin, denom} = cds[i];
2016-11-18 00:09:43 +01:00
if (coin.suspended) {
continue;
}
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;
}
2016-11-18 00:09:43 +01:00
cdsResult.push(cds[i]);
accFee = Amounts.add(denom.feeDeposit, accFee).amount;
2016-11-14 03:01:42 +01:00
accAmount = Amounts.add(coin.currentAmount, accAmount).amount;
coversAmount = Amounts.cmp(accAmount, paymentAmount) >= 0;
coversAmountWithFee = Amounts.cmp(accAmount,
Amounts.add(paymentAmount,
denom.feeDeposit).amount) >= 0;
2016-11-14 03:01:42 +01:00
isBelowFee = Amounts.cmp(accFee, depositFeeLimit) <= 0;
if ((coversAmount && isBelowFee) || coversAmountWithFee) {
return cdsResult;
}
}
return undefined;
}
2015-12-17 22:56:24 +01:00
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.
*/
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;
for (let d of denoms) {
let 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;
}
2016-10-18 02:07:38 +02:00
export namespace Stores {
2016-11-15 15:07:17 +01:00
class ExchangeStore extends Store<ExchangeRecord> {
2016-10-18 01:36:47 +02:00
constructor() {
2016-10-18 02:07:38 +02:00
super("exchanges", {keyPath: "baseUrl"});
2016-10-18 01:36:47 +02:00
}
2016-10-18 02:07:38 +02:00
2016-11-15 15:07:17 +01:00
pubKeyIndex = new Index<string,ExchangeRecord>(this, "pubKey", "masterPublicKey");
2016-10-18 01:36:47 +02:00
}
class NonceStore extends Store<NonceRecord> {
constructor() {
super("nonces", {keyPath: "pub"});
}
}
2016-11-15 15:07:17 +01:00
class CoinsStore extends Store<CoinRecord> {
2016-10-18 01:36:47 +02:00
constructor() {
2016-10-18 02:07:38 +02:00
super("coins", {keyPath: "coinPub"});
2016-10-18 01:36:47 +02:00
}
2016-11-15 15:07:17 +01:00
exchangeBaseUrlIndex = new Index<string,CoinRecord>(this, "exchangeBaseUrl", "exchangeBaseUrl");
denomPubIndex = new Index<string,CoinRecord>(this, "denomPub", "denomPub");
2016-10-18 01:36:47 +02:00
}
class HistoryStore extends Store<HistoryRecord> {
constructor() {
2016-10-18 02:07:38 +02:00
super("history", {
keyPath: "id",
autoIncrement: true
});
2016-10-18 01:36:47 +02:00
}
2016-10-18 02:07:38 +02:00
timestampIndex = new Index<number,HistoryRecord>(this, "timestamp", "timestamp");
2016-10-18 01:36:47 +02:00
}
2016-11-15 15:07:17 +01:00
class OffersStore extends Store<OfferRecord> {
2016-11-13 10:17:39 +01:00
constructor() {
super("offers", {
keyPath: "id",
autoIncrement: true
});
}
}
2016-11-15 15:07:17 +01:00
class TransactionsStore extends Store<TransactionRecord> {
2016-10-18 01:47:40 +02:00
constructor() {
2016-10-18 02:07:38 +02:00
super("transactions", {keyPath: "contractHash"});
2016-10-18 01:47:40 +02:00
}
2017-02-13 00:44:44 +01:00
fulfillmentUrlIndex = new Index<string,TransactionRecord>(this, "fulfillment_url", "contract.fulfillment_url");
orderIdIndex = new Index<string,TransactionRecord>(this, "order_id", "contract.order_id");
2016-10-18 01:47:40 +02:00
}
class DenominationsStore extends Store<DenominationRecord> {
constructor() {
// cast needed because of bug in type annotations
super("denominations",
2016-11-16 23:54:22 +01:00
{keyPath: ["exchangeBaseUrl", "denomPub"] as any as IDBKeyPath});
}
denomPubHashIndex = new Index<string,DenominationRecord>(this, "denomPubHash", "denomPubHash");
exchangeBaseUrlIndex = new Index<string, DenominationRecord>(this, "exchangeBaseUrl", "exchangeBaseUrl");
denomPubIndex = new Index<string, DenominationRecord>(this, "denomPub", "denomPub");
}
class CurrenciesStore extends Store<CurrencyRecord> {
constructor() {
super("currencies", {keyPath: "name"});
}
}
class ConfigStore extends Store<ConfigRecord> {
constructor() {
super("config", {keyPath: "key"});
}
}
class ExchangeWireFeesStore extends Store<ExchangeWireFeesRecord> {
constructor() {
super("exchangeWireFees", {keyPath: "exchangeBaseUrl"});
}
}
export const exchanges: ExchangeStore = new ExchangeStore();
export const exchangeWireFees: ExchangeWireFeesStore = new ExchangeWireFeesStore();
export const nonces: NonceStore = new NonceStore();
export const transactions: TransactionsStore = new TransactionsStore();
export const reserves: Store<ReserveRecord> = new Store<ReserveRecord>("reserves", {keyPath: "reserve_pub"});
export const coins: CoinsStore = new CoinsStore();
export const refresh: Store<RefreshSessionRecord> = new Store<RefreshSessionRecord>("refresh", {keyPath: "meltCoinPub"});
export const history: HistoryStore = new HistoryStore();
export const offers: OffersStore = new OffersStore();
export const precoins: Store<PreCoinRecord> = new Store<PreCoinRecord>("precoins", {keyPath: "coinPub"});
export const denominations: DenominationsStore = new DenominationsStore();
export const currencies: CurrenciesStore = new CurrenciesStore();
export const config: ConfigStore = new ConfigStore();
2016-10-18 01:16:31 +02:00
}
export class Wallet {
private db: IDBDatabase;
private http: HttpRequestLibrary;
private badge: Badge;
2016-02-18 23:41:29 +01:00
private notifier: Notifier;
2016-02-22 19:21:06 +01:00
public cryptoApi: CryptoApi;
2016-01-05 14:20:13 +01:00
2016-11-17 01:23:53 +01:00
private processPreCoinConcurrent = 0;
2016-11-17 15:32:08 +01:00
private processPreCoinThrottle: {[url: string]: number} = {};
2016-11-17 01:23:53 +01:00
2016-05-24 02:05:19 +02:00
/**
* Set of identifiers for running operations.
*/
private runningOperations: Set<string> = new Set();
2016-02-11 18:17:02 +01:00
2016-10-13 02:23:24 +02:00
q(): QueryRoot {
return new QueryRoot(this.db);
}
2016-02-18 23:41:29 +01:00
constructor(db: IDBDatabase,
2016-10-13 02:23:24 +02:00
http: HttpRequestLibrary,
badge: Badge,
notifier: Notifier) {
this.db = db;
this.http = http;
this.badge = badge;
2016-02-18 23:41:29 +01:00
this.notifier = notifier;
2016-02-22 19:21:06 +01:00
this.cryptoApi = new CryptoApi();
this.fillDefaults();
this.resumePendingFromDb();
2017-05-01 04:09:52 +02:00
setInterval(() => this.updateExchanges(), 1000 * 60 * 15);
}
private async fillDefaults() {
let onTrue = (r: QueryRoot) => {
console.log("defaults already applied");
};
let onFalse = (r: QueryRoot) => {
console.log("applying defaults");
r.put(Stores.config, {key: "currencyDefaultsApplied", value: true})
.putAll(Stores.currencies, builtinCurrencies)
.finish();
};
await (
this.q()
.iter(Stores.config)
.filter(x => x.key == "currencyDefaultsApplied")
.first()
.cond((x) => x && x.value, onTrue, onFalse)
);
}
2016-05-24 02:05:19 +02:00
private startOperation(operationId: string) {
this.runningOperations.add(operationId);
this.badge.startBusy();
}
private stopOperation(operationId: string) {
this.runningOperations.delete(operationId);
if (this.runningOperations.size == 0) {
this.badge.stopBusy();
}
}
2016-10-20 01:37:00 +02:00
async updateExchanges(): Promise<void> {
2016-05-24 17:30:27 +02:00
console.log("updating exchanges");
2016-10-20 01:37:00 +02:00
let exchangesUrls = await this.q()
.iter(Stores.exchanges)
.map((e) => e.baseUrl)
.toArray();
2016-10-19 23:55:58 +02:00
for (let url of exchangesUrls) {
this.updateExchangeFromUrl(url)
.catch((e) => {
console.error("updating exchange failed", e);
});
}
2016-05-24 17:30:27 +02:00
}
/**
* Resume various pending operations that are pending
* by looking at the database.
*/
private resumePendingFromDb(): void {
console.log("resuming pending operations from db");
2016-10-13 02:23:24 +02:00
this.q()
2016-10-18 01:16:31 +02:00
.iter(Stores.reserves)
.reduce((reserve) => {
2016-10-13 02:23:24 +02:00
console.log("resuming reserve", reserve.reserve_pub);
this.processReserve(reserve);
});
this.q()
2016-10-18 01:16:31 +02:00
.iter(Stores.precoins)
.reduce((preCoin) => {
2016-10-13 02:23:24 +02:00
console.log("resuming precoin");
this.processPreCoin(preCoin);
});
2016-10-17 23:49:04 +02:00
this.q()
2016-10-18 01:16:31 +02:00
.iter(Stores.refresh)
2016-11-15 15:07:17 +01:00
.reduce((r: RefreshSessionRecord) => {
2016-10-17 23:49:04 +02:00
this.continueRefreshSession(r);
});
// FIXME: optimize via index
this.q()
2016-10-18 01:16:31 +02:00
.iter(Stores.coins)
2016-11-15 15:07:17 +01:00
.reduce((c: CoinRecord) => {
if (c.status == CoinStatus.Dirty) {
2017-04-13 15:05:38 +02:00
console.log("resuming pending refresh for coin", c);
2016-10-17 23:49:04 +02:00
this.refresh(c.coinPub);
}
});
2016-01-05 14:20:13 +01:00
}
2016-02-11 18:17:02 +01:00
/**
2016-03-01 19:39:17 +01:00
* Get exchanges and associated coins that are still spendable,
* but only if the sum the coins' remaining value exceeds the payment amount.
*/
2016-11-14 02:52:29 +01:00
private async getCoinsForPayment(paymentAmount: AmountJson,
2017-04-27 04:06:48 +02:00
wireMethod: string,
wireFeeTime: number,
2016-11-14 02:52:29 +01:00
depositFeeLimit: AmountJson,
2017-04-27 04:06:48 +02:00
wireFeeLimit: AmountJson,
wireFeeAmortization: number,
allowedExchanges: ExchangeHandle[],
allowedAuditors: Auditor[]): Promise<CoinSelectionResult> {
2016-11-14 02:52:29 +01:00
let exchanges = await this.q().iter(Stores.exchanges).toArray();
for (let exchange of exchanges) {
let isOkay: boolean = false;
// is the exchange explicitly allowed?
for (let allowedExchange of allowedExchanges) {
if (allowedExchange.master_pub == exchange.masterPublicKey) {
isOkay = true;
break;
}
}
// is the exchange allowed because of one of its auditors?
if (!isOkay) {
for (let allowedAuditor of allowedAuditors) {
for (let auditor of exchange.auditors) {
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
}
let coins: CoinRecord[] = await this.q()
.iterIndex(Stores.coins.exchangeBaseUrlIndex,
exchange.baseUrl)
.toArray();
2016-11-14 02:52:29 +01:00
if (!coins || coins.length == 0) {
continue;
}
2016-11-14 02:52:29 +01:00
// Denomination of the first coin, we assume that all other
// coins have the same currency
let firstDenom = await this.q().get(Stores.denominations,
[
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
}
2016-11-14 02:52:29 +01:00
let currency = firstDenom.value.currency;
let cds: CoinWithDenom[] = [];
for (let i = 0; i < coins.length; i++) {
let coin = coins[i];
let denom = await this.q().get(Stores.denominations,
[exchange.baseUrl, coin.denomPub]);
2016-11-14 02:52:29 +01:00
if (!denom) {
throw Error("db inconsistent");
}
if (denom.value.currency != currency) {
console.warn(`same pubkey for different currencies at exchange ${exchange.baseUrl}`);
continue;
}
if (coin.suspended) {
continue;
}
if (coin.status != CoinStatus.Fresh) {
2016-11-18 00:09:43 +01:00
continue;
}
2016-11-14 02:52:29 +01:00
cds.push({coin, denom});
}
2017-04-27 04:06:48 +02:00
let fees = await this.q().get(Stores.exchangeWireFees, exchange.baseUrl);
if (!fees) {
console.error("no fees found for exchange", exchange);
continue;
}
let wireFee: AmountJson|undefined = undefined;
for (let fee of (fees.feesForType[wireMethod] || [])) {
if (fee.startStamp >= wireFeeTime && fee.endStamp <= wireFeeTime) {
wireFee = fee.wireFee;
break;
}
}
if (wireFee) {
let amortizedWireFee = Amounts.divide(wireFee, wireFeeAmortization);
if (Amounts.cmp(wireFeeLimit, amortizedWireFee) < 0) {
paymentAmount = Amounts.add(amortizedWireFee, paymentAmount).amount;
}
}
2016-11-14 03:01:42 +01:00
let res = selectCoins(cds, paymentAmount, depositFeeLimit);
if (res) {
return {
exchangeUrl: exchange.baseUrl,
2016-11-14 03:01:42 +01:00
cds: res,
2016-11-14 02:52:29 +01: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
* pay for a contract in the wallet's database.
*/
2016-11-15 15:07:17 +01:00
private async recordConfirmPay(offer: OfferRecord,
2016-10-13 02:23:24 +02:00
payCoinInfo: PayCoinInfo,
chosenExchange: string): Promise<void> {
2016-10-17 15:58:36 +02:00
let payReq: PayReq = {
coins: payCoinInfo.map((x) => x.sig),
2017-02-13 00:44:44 +01:00
merchant_pub: offer.contract.merchant_pub,
order_id: offer.contract.order_id,
exchange: chosenExchange,
2016-10-17 15:58:36 +02:00
};
2016-11-15 15:07:17 +01:00
let t: TransactionRecord = {
contractHash: offer.H_contract,
contract: offer.contract,
2016-01-26 17:21:17 +01:00
payReq: payReq,
2016-02-23 14:07:53 +01:00
merchantSig: offer.merchant_sig,
finished: false,
};
2015-12-14 16:54:47 +01:00
2016-10-18 01:16:31 +02:00
let historyEntry: HistoryRecord = {
type: "pay",
timestamp: (new Date).getTime(),
2016-09-28 23:41:34 +02:00
subjectId: `contract-${offer.H_contract}`,
detail: {
merchantName: offer.contract.merchant.name,
amount: offer.contract.amount,
2016-02-01 15:10:20 +01:00
contractHash: offer.H_contract,
2016-10-10 03:16:12 +02:00
fulfillmentUrl: offer.contract.fulfillment_url,
2016-10-18 01:16:31 +02:00
},
level: HistoryLevel.User
};
2016-10-13 02:23:24 +02:00
await this.q()
2016-10-18 01:16:31 +02:00
.put(Stores.transactions, t)
.put(Stores.history, historyEntry)
.putAll(Stores.coins, payCoinInfo.map((pci) => pci.updatedCoin))
2016-10-13 02:23:24 +02:00
.finish();
2016-09-28 18:00:13 +02:00
this.notifier.notify();
}
2015-12-14 16:54:47 +01:00
2016-02-10 02:03:31 +01:00
2016-09-28 23:41:34 +02:00
async putHistory(historyEntry: HistoryRecord): Promise<void> {
2016-10-18 01:16:31 +02:00
await this.q().put(Stores.history, historyEntry).finish();
2016-09-29 01:40:29 +02:00
this.notifier.notify();
2016-09-28 23:41:34 +02:00
}
2016-11-15 15:07:17 +01:00
async saveOffer(offer: OfferRecord): Promise<number> {
2016-11-13 10:17:39 +01:00
console.log(`saving offer in wallet.ts`);
let id = await this.q().putWithResult(Stores.offers, offer);
this.notifier.notify();
console.log(`saved offer with id ${id}`);
if (typeof id !== "number") {
throw Error("db schema wrong");
}
return id;
}
2016-02-10 02:03:31 +01:00
/**
* Add a contract to the wallet and sign coins,
* but do not send them yet.
*/
2016-11-15 15:07:17 +01:00
async confirmPay(offer: OfferRecord): Promise<any> {
2016-02-15 15:53:59 +01:00
console.log("executing confirmPay");
2016-02-22 21:52:53 +01:00
2016-10-18 01:16:31 +02:00
let transaction = await this.q().get(Stores.transactions, offer.H_contract);
2016-05-24 01:53:56 +02:00
2016-09-28 18:54:48 +02:00
if (transaction) {
// Already payed ...
return {};
}
2016-11-14 02:52:29 +01:00
let res = await this.getCoinsForPayment(offer.contract.amount,
2017-04-27 04:06:48 +02:00
offer.contract.wire_method,
getTalerStampSec(offer.contract.timestamp) || 0,
2016-11-14 02:52:29 +01:00
offer.contract.max_fee,
2017-04-27 04:06:48 +02:00
offer.contract.max_wire_fee || Amounts.getZero(offer.contract.amount.currency),
offer.contract.wire_fee_amortization || 1,
offer.contract.exchanges,
offer.contract.auditors);
2016-09-28 18:54:48 +02:00
2016-11-20 04:15:49 +01:00
console.log("max_fee", offer.contract.max_fee);
console.log("coin selection result", res);
2016-11-14 02:52:29 +01:00
if (!res) {
2016-09-28 18:54:48 +02:00
console.log("not confirming payment, insufficient coins");
return {
error: "coins-insufficient",
};
}
2016-11-14 02:52:29 +01:00
let {exchangeUrl, cds} = res;
2016-09-28 18:54:48 +02:00
2016-11-14 02:52:29 +01:00
let ds = await this.cryptoApi.signDeposit(offer, cds);
2016-09-28 18:54:48 +02:00
await this.recordConfirmPay(offer,
2016-10-13 02:23:24 +02:00
ds,
exchangeUrl);
2016-09-28 18:54:48 +02:00
return {};
}
2015-12-16 00:38:36 +01:00
2016-02-10 02:03:31 +01:00
2016-04-27 06:03:04 +02:00
/**
* Add a contract to the wallet and sign coins,
* but do not send them yet.
*/
2016-11-15 15:07:17 +01:00
async checkPay(offer: OfferRecord): Promise<any> {
2016-05-24 01:53:56 +02:00
// First check if we already payed for it.
2016-10-18 01:16:31 +02:00
let transaction = await this.q().get(Stores.transactions, offer.H_contract);
2016-09-28 18:54:48 +02:00
if (transaction) {
2016-10-13 02:23:24 +02:00
return {isPayed: true};
2016-09-28 18:54:48 +02:00
}
2016-05-24 01:53:56 +02:00
2016-09-28 18:54:48 +02:00
// If not already payed, check if we could pay for it.
2016-11-14 02:52:29 +01:00
let res = await this.getCoinsForPayment(offer.contract.amount,
2017-04-27 04:06:48 +02:00
offer.contract.wire_method,
getTalerStampSec(offer.contract.timestamp) || 0,
2016-11-14 02:52:29 +01:00
offer.contract.max_fee,
2017-04-27 04:06:48 +02:00
offer.contract.max_wire_fee || Amounts.getZero(offer.contract.amount.currency),
offer.contract.wire_fee_amortization || 1,
offer.contract.exchanges,
offer.contract.auditors);
2016-05-24 01:53:56 +02:00
2016-11-14 02:52:29 +01:00
if (!res) {
2016-09-28 18:54:48 +02:00
console.log("not confirming payment, insufficient coins");
return {
error: "coins-insufficient",
};
}
2016-10-13 02:23:24 +02:00
return {isPayed: false};
2016-04-27 06:03:04 +02:00
}
2016-02-10 02:03:31 +01:00
/**
* Retrieve information required to pay for a contract, where the
* contract is identified via the fulfillment url.
2016-02-10 02:03:31 +01:00
*/
async queryPayment(url: string): Promise<any> {
console.log("query for payment", url);
const t = await this.q().getIndexed(Stores.transactions.fulfillmentUrlIndex, url);
2017-02-13 00:44:44 +01:00
2016-09-28 18:54:48 +02:00
if (!t) {
2017-02-13 00:44:44 +01:00
console.log("query for payment failed");
2016-09-28 18:54:48 +02:00
return {
success: false,
}
}
2017-02-13 00:44:44 +01:00
console.log("query for payment succeeded:", t);
2016-09-28 18:54:48 +02:00
let resp = {
success: true,
payReq: t.payReq,
2017-02-13 00:44:44 +01:00
H_contract: t.contractHash,
2016-09-28 18:54:48 +02:00
contract: t.contract,
};
return resp;
}
2015-12-16 00:38:36 +01:00
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.
*/
2016-09-28 23:41:34 +02:00
private async processReserve(reserveRecord: ReserveRecord,
2016-10-13 02:23:24 +02:00
retryDelayMs: number = 250): Promise<void> {
2016-05-24 02:05:19 +02:00
const opId = "reserve-" + reserveRecord.reserve_pub;
this.startOperation(opId);
2016-09-28 18:54:48 +02:00
try {
let exchange = await this.updateExchangeFromUrl(reserveRecord.exchange_base_url);
let reserve = await this.updateReserve(reserveRecord.reserve_pub);
let n = await this.depleteReserve(reserve);
2016-09-29 01:40:29 +02:00
if (n != 0) {
2016-10-18 01:16:31 +02:00
let depleted: HistoryRecord = {
2016-09-29 01:40:29 +02:00
type: "depleted-reserve",
subjectId: `reserve-progress-${reserveRecord.reserve_pub}`,
timestamp: (new Date).getTime(),
detail: {
2016-10-10 03:16:12 +02:00
exchangeBaseUrl: reserveRecord.exchange_base_url,
2016-09-29 01:40:29 +02:00
reservePub: reserveRecord.reserve_pub,
requestedAmount: reserveRecord.requested_amount,
currentAmount: reserveRecord.current_amount,
2016-10-18 01:16:31 +02:00
},
level: HistoryLevel.User
2016-09-29 01:40:29 +02:00
};
2016-10-18 01:16:31 +02:00
await this.q().put(Stores.history, depleted).finish();
2016-09-29 01:40:29 +02:00
}
2016-09-28 18:54:48 +02:00
} catch (e) {
// random, exponential backoff truncated at 3 minutes
let nextDelay = Math.min(2 * retryDelayMs + retryDelayMs * Math.random(),
2016-10-13 02:23:24 +02:00
3000 * 60);
2016-09-28 18:54:48 +02:00
console.warn(`Failed to deplete reserve, trying again in ${retryDelayMs} ms`);
setTimeout(() => this.processReserve(reserveRecord, nextDelay),
2016-10-13 02:23:24 +02:00
retryDelayMs);
2016-09-28 18:54:48 +02:00
} finally {
this.stopOperation(opId);
}
}
2016-11-15 15:07:17 +01:00
private async processPreCoin(preCoin: PreCoinRecord,
2016-11-17 01:23:53 +01:00
retryDelayMs = 200): Promise<void> {
2016-11-17 15:32:08 +01:00
if (this.processPreCoinConcurrent >= 4 || this.processPreCoinThrottle[preCoin.exchangeBaseUrl]) {
2016-11-17 01:23:53 +01:00
console.log("delaying processPreCoin");
2016-11-17 15:46:59 +01:00
setTimeout(() => this.processPreCoin(preCoin, Math.min(retryDelayMs * 2, 5 * 60 * 1000)),
2016-11-17 01:23:53 +01:00
retryDelayMs);
2016-10-19 22:59:24 +02:00
return;
}
2016-11-17 01:23:53 +01:00
console.log("executing processPreCoin");
this.processPreCoinConcurrent++;
2016-09-28 18:54:48 +02:00
try {
2016-11-17 01:23:53 +01:00
const exchange = await this.q().get(Stores.exchanges,
preCoin.exchangeBaseUrl);
if (!exchange) {
console.error("db inconsistend: exchange for precoin not found");
return;
}
const denom = await this.q().get(Stores.denominations,
[preCoin.exchangeBaseUrl, preCoin.denomPub]);
if (!denom) {
console.error("db inconsistent: denom for precoin not found");
return;
}
2016-09-28 18:54:48 +02:00
const coin = await this.withdrawExecute(preCoin);
2016-10-19 22:59:24 +02:00
const mutateReserve = (r: ReserveRecord) => {
2016-10-20 01:37:00 +02:00
console.log(`before committing coin: current ${amountToPretty(r.current_amount!)}, precoin: ${amountToPretty(
r.precoin_amount)})}`);
let x = Amounts.sub(r.precoin_amount,
2016-10-19 22:59:24 +02:00
preCoin.coinValue,
denom.feeWithdraw);
2016-10-19 22:59:24 +02:00
if (x.saturated) {
2016-10-20 01:37:00 +02:00
console.error("database inconsistent");
2016-10-19 22:59:24 +02:00
throw AbortTransaction;
}
2016-10-20 01:37:00 +02:00
r.precoin_amount = x.amount;
2016-10-19 22:59:24 +02:00
return r;
};
const historyEntry: HistoryRecord = {
2016-10-19 22:59:24 +02:00
type: "withdraw",
timestamp: (new Date).getTime(),
level: HistoryLevel.Expert,
detail: {
coinPub: coin.coinPub,
}
};
await this.q()
.mutate(Stores.reserves, preCoin.reservePub, mutateReserve)
.delete("precoins", coin.coinPub)
.add(Stores.coins, coin)
.add(Stores.history, historyEntry)
.finish();
this.notifier.notify();
2016-09-28 18:54:48 +02:00
} catch (e) {
console.error("Failed to withdraw coin from precoin, retrying in",
2016-10-13 02:23:24 +02:00
retryDelayMs,
"ms", e);
2016-09-28 18:54:48 +02:00
// exponential backoff truncated at one minute
2016-11-17 15:46:59 +01:00
let nextRetryDelayMs = Math.min(retryDelayMs * 2, 5 * 60 * 1000);
2016-09-28 18:54:48 +02:00
setTimeout(() => this.processPreCoin(preCoin, nextRetryDelayMs),
2016-10-13 02:23:24 +02:00
retryDelayMs);
2016-11-17 15:32:08 +01:00
this.processPreCoinThrottle[preCoin.exchangeBaseUrl] = (this.processPreCoinThrottle[preCoin.exchangeBaseUrl] || 0) + 1;
setTimeout(() => {this.processPreCoinThrottle[preCoin.exchangeBaseUrl]--; }, retryDelayMs);
2016-11-17 01:23:53 +01:00
} finally {
this.processPreCoinConcurrent--;
2016-09-28 18:54:48 +02:00
}
}
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
*/
2016-09-28 18:54:48 +02:00
async createReserve(req: CreateReserveRequest): Promise<CreateReserveResponse> {
let keypair = await this.cryptoApi.createEddsaKeypair();
const now = (new Date).getTime();
const canonExchange = canonicalizeBaseUrl(req.exchange);
2016-09-28 23:41:34 +02:00
const reserveRecord: ReserveRecord = {
hasPayback: false,
2016-09-28 18:54:48 +02:00
reserve_pub: keypair.pub,
reserve_priv: keypair.priv,
exchange_base_url: canonExchange,
created: now,
last_query: null,
current_amount: null,
requested_amount: req.amount,
confirmed: false,
2016-10-20 01:37:00 +02:00
precoin_amount: Amounts.getZero(req.amount.currency),
2016-09-28 18:54:48 +02:00
};
2016-02-09 21:56:06 +01:00
2016-09-28 18:54:48 +02:00
const historyEntry = {
type: "create-reserve",
2016-10-10 04:07:34 +02:00
level: HistoryLevel.Expert,
2016-09-28 18:54:48 +02:00
timestamp: now,
2016-09-28 23:41:34 +02:00
subjectId: `reserve-progress-${reserveRecord.reserve_pub}`,
2016-09-28 18:54:48 +02:00
detail: {
requestedAmount: req.amount,
reservePub: reserveRecord.reserve_pub,
}
};
let exchangeInfo = await this.updateExchangeFromUrl(req.exchange);
let {isAudited, isTrusted} = await this.getExchangeTrust(exchangeInfo);
let currencyRecord = await this.q().get(Stores.currencies, exchangeInfo.currency);
if (!currencyRecord) {
currencyRecord = {
name: exchangeInfo.currency,
fractionalDigits: 2,
exchanges: [],
auditors: [],
}
}
if (!isAudited && !isTrusted) {
currencyRecord.exchanges.push({baseUrl: req.exchange, priority: 0});
}
2016-10-13 02:23:24 +02:00
await this.q()
.put(Stores.currencies, currencyRecord)
2016-10-18 01:16:31 +02:00
.put(Stores.reserves, reserveRecord)
.put(Stores.history, historyEntry)
2016-10-13 02:23:24 +02:00
.finish();
2016-02-11 18:17:02 +01:00
2016-09-28 18:54:48 +02:00
let r: CreateReserveResponse = {
exchange: canonExchange,
reservePub: keypair.pub,
};
return r;
2016-02-09 21:56:06 +01:00
}
/**
* Mark an existing reserve as confirmed. The wallet will start trying
* to withdraw from that reserve. This may not immediately succeed,
2016-03-01 19:39:17 +01:00
* since the exchange might not know about the reserve yet, even though the
2016-02-09 21:56:06 +01:00
* bank confirmed its creation.
*
* A confirmed reserve should be shown to the user in the UI, while
* an unconfirmed reserve should be hidden.
*/
2016-09-28 18:54:48 +02:00
async confirmReserve(req: ConfirmReserveRequest): Promise<void> {
2016-02-09 21:56:06 +01:00
const now = (new Date).getTime();
2016-10-13 02:23:24 +02:00
let reserve: ReserveRecord|undefined = await (
2016-10-18 01:16:31 +02:00
this.q().get<ReserveRecord>(Stores.reserves,
2016-10-13 02:23:24 +02:00
req.reservePub));
if (!reserve) {
console.error("Unable to confirm reserve, not found in DB");
return;
}
2016-10-17 23:49:04 +02:00
console.log("reserve confirmed");
2016-10-18 01:16:31 +02:00
const historyEntry: HistoryRecord = {
2016-02-09 21:56:06 +01:00
type: "confirm-reserve",
timestamp: now,
2016-09-28 23:41:34 +02:00
subjectId: `reserve-progress-${reserve.reserve_pub}`,
2016-02-09 21:56:06 +01:00
detail: {
2016-10-10 03:16:12 +02:00
exchangeBaseUrl: reserve.exchange_base_url,
2016-02-09 21:56:06 +01:00
reservePub: req.reservePub,
2016-09-28 23:41:34 +02:00
requestedAmount: reserve.requested_amount,
2016-10-18 01:16:31 +02:00
},
level: HistoryLevel.User,
2016-02-09 21:56:06 +01:00
};
2016-09-28 23:41:34 +02:00
reserve.confirmed = true;
2016-10-13 02:23:24 +02:00
await this.q()
2016-10-18 01:16:31 +02:00
.put(Stores.reserves, reserve)
.put(Stores.history, historyEntry)
2016-10-13 02:23:24 +02:00
.finish();
this.notifier.notify();
2016-09-28 19:09:10 +02:00
2016-09-28 23:41:34 +02:00
this.processReserve(reserve);
2015-12-16 05:53:55 +01:00
}
2016-11-15 15:07:17 +01:00
private async withdrawExecute(pc: PreCoinRecord): Promise<CoinRecord> {
2016-10-18 01:16:31 +02:00
let reserve = await this.q().get<ReserveRecord>(Stores.reserves,
pc.reservePub);
2016-10-13 02:23:24 +02:00
if (!reserve) {
throw Error("db inconsistent");
}
2016-09-28 18:54:48 +02:00
let wd: any = {};
wd.denom_pub = pc.denomPub;
wd.reserve_pub = pc.reservePub;
wd.reserve_sig = pc.withdrawSig;
wd.coin_ev = pc.coinEv;
let reqUrl = (new URI("reserve/withdraw")).absoluteTo(reserve.exchange_base_url);
let resp = await this.http.postJson(reqUrl.href(), wd);
2016-09-28 18:54:48 +02:00
if (resp.status != 200) {
throw new RequestException({
hint: "Withdrawal failed",
status: resp.status
});
2016-09-28 18:54:48 +02:00
}
let r = JSON.parse(resp.responseText);
let denomSig = await this.cryptoApi.rsaUnblind(r.ev_sig,
2016-10-13 02:23:24 +02:00
pc.blindingKey,
pc.denomPub);
2016-11-15 15:07:17 +01:00
let coin: CoinRecord = {
reservePub: pc.reservePub,
2016-09-28 18:54:48 +02:00
coinPub: pc.coinPub,
coinPriv: pc.coinPriv,
denomPub: pc.denomPub,
denomSig: denomSig,
blindingKey: pc.blindingKey,
2016-09-28 18:54:48 +02:00
currentAmount: pc.coinValue,
exchangeBaseUrl: pc.exchangeBaseUrl,
status: CoinStatus.Fresh,
2016-09-28 18:54:48 +02:00
};
return coin;
}
2015-12-13 23:47:30 +01:00
/**
* Withdraw coins from a reserve until it is empty.
*/
private async depleteReserve(reserve: ReserveRecord): Promise<number> {
2016-11-18 00:09:43 +01:00
console.log("depleting reserve");
2016-10-20 01:37:00 +02:00
if (!reserve.current_amount) {
throw Error("can't withdraw when amount is unknown");
}
let currentAmount = reserve.current_amount;
if (!currentAmount) {
throw Error("can't withdraw when amount is unknown");
}
let denomsForWithdraw = await this.getVerifiedWithdrawDenomList(reserve.exchange_base_url,
currentAmount);
2016-02-11 18:17:02 +01:00
2016-11-18 00:09:43 +01:00
console.log(`withdrawing ${denomsForWithdraw.length} coins`);
2016-10-19 22:59:24 +02:00
let ps = denomsForWithdraw.map(async(denom) => {
2016-10-20 01:37:00 +02:00
function mutateReserve(r: ReserveRecord): ReserveRecord {
let currentAmount = r.current_amount;
if (!currentAmount) {
throw Error("can't withdraw when amount is unknown");
}
r.precoin_amount = Amounts.add(r.precoin_amount,
denom.value,
denom.feeWithdraw).amount;
2016-10-20 01:37:00 +02:00
let result = Amounts.sub(currentAmount,
denom.value,
denom.feeWithdraw);
2016-10-20 01:37:00 +02:00
if (result.saturated) {
console.error("can't create precoin, saturated");
throw AbortTransaction;
}
r.current_amount = result.amount;
console.log(`after creating precoin: current ${amountToPretty(r.current_amount)}, precoin: ${amountToPretty(
r.precoin_amount)})}`);
return r;
}
2016-10-19 22:59:24 +02:00
let preCoin = await this.cryptoApi
.createPreCoin(denom, reserve);
2016-10-20 01:37:00 +02:00
await this.q()
.put(Stores.precoins, preCoin)
.mutate(Stores.reserves, reserve.reserve_pub, mutateReserve);
2016-10-19 22:59:24 +02:00
await this.processPreCoin(preCoin);
});
2016-10-20 01:37:00 +02:00
2016-09-28 17:52:36 +02:00
await Promise.all(ps);
2016-09-29 01:40:29 +02:00
return ps.length;
}
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
*/
private async updateReserve(reservePub: string): Promise<ReserveRecord> {
2016-10-13 02:23:24 +02:00
let reserve = await this.q()
2016-10-18 01:16:31 +02:00
.get<ReserveRecord>(Stores.reserves, reservePub);
2016-10-13 02:23:24 +02:00
if (!reserve) {
throw Error("reserve not in db");
}
let reqUrl = new URI("reserve/status").absoluteTo(reserve.exchange_base_url);
2016-10-13 02:23:24 +02:00
reqUrl.query({'reserve_pub': reservePub});
let resp = await this.http.get(reqUrl.href());
2016-09-28 19:09:10 +02:00
if (resp.status != 200) {
throw Error();
}
let reserveInfo = JSON.parse(resp.responseText);
if (!reserveInfo) {
throw Error();
}
let oldAmount = reserve.current_amount;
let newAmount = reserveInfo.balance;
reserve.current_amount = reserveInfo.balance;
let historyEntry = {
type: "reserve-update",
timestamp: (new Date).getTime(),
2016-09-28 23:41:34 +02:00
subjectId: `reserve-progress-${reserve.reserve_pub}`,
2016-09-28 19:09:10 +02:00
detail: {
reservePub,
2016-09-28 23:41:34 +02:00
requestedAmount: reserve.requested_amount,
2016-09-28 19:09:10 +02:00
oldAmount,
newAmount
}
};
2016-10-13 02:23:24 +02:00
await this.q()
2016-10-18 01:16:31 +02:00
.put(Stores.reserves, reserve)
2016-10-13 02:23:24 +02:00
.finish();
this.notifier.notify();
2016-09-28 19:09:10 +02:00
return reserve;
}
2016-01-26 17:21:17 +01:00
2016-05-24 00:36:20 +02:00
/**
* Get the wire information for the exchange with the given base URL.
*/
2016-09-28 18:54:48 +02:00
async getWireInfo(exchangeBaseUrl: string): Promise<WireInfo> {
2016-05-24 00:36:20 +02:00
exchangeBaseUrl = canonicalizeBaseUrl(exchangeBaseUrl);
let reqUrl = new URI("wire").absoluteTo(exchangeBaseUrl);
let resp = await this.http.get(reqUrl.href());
2016-09-28 18:54:48 +02:00
if (resp.status != 200) {
throw Error("/wire request failed");
}
let wiJson = JSON.parse(resp.responseText);
if (!wiJson) {
throw Error("/wire response malformed")
}
return wiJson;
}
async getPossibleDenoms(exchangeBaseUrl: string) {
return (
this.q().iterIndex(Stores.denominations.exchangeBaseUrlIndex,
exchangeBaseUrl)
.filter((d) => d.status == DenominationStatus.Unverified || d.status == DenominationStatus.VerifiedGood)
.toArray()
);
}
/**
* 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.
*/
async getVerifiedWithdrawDenomList(exchangeBaseUrl: string,
amount: AmountJson): Promise<DenominationRecord[]> {
const exchange = await this.q().get(Stores.exchanges, exchangeBaseUrl);
if (!exchange) {
throw Error(`exchange ${exchangeBaseUrl} not found`);
}
const possibleDenoms = await (
this.q().iterIndex(Stores.denominations.exchangeBaseUrlIndex,
exchange.baseUrl)
.filter((d) => d.status == DenominationStatus.Unverified || d.status == DenominationStatus.VerifiedGood)
.toArray()
);
let allValid = false;
let currentPossibleDenoms = possibleDenoms;
let selectedDenoms: DenominationRecord[];
do {
allValid = true;
let nextPossibleDenoms = [];
selectedDenoms = getWithdrawDenomList(amount, possibleDenoms);
for (let denom of selectedDenoms || []) {
if (denom.status == DenominationStatus.Unverified) {
console.log(`verifying denom ${denom.denomPub.substr(0, 15)}`);
let valid = await this.cryptoApi.isValidDenom(denom,
exchange.masterPublicKey);
if (!valid) {
denom.status = DenominationStatus.VerifiedBad;
allValid = false;
} else {
denom.status = DenominationStatus.VerifiedGood;
nextPossibleDenoms.push(denom);
}
await this.q().put(Stores.denominations, denom).finish();
} else {
nextPossibleDenoms.push(denom);
}
}
currentPossibleDenoms = nextPossibleDenoms;
} while (selectedDenoms.length > 0 && !allValid);
return selectedDenoms;
}
/**
* Check if and how an exchange is trusted and/or audited.
*/
async getExchangeTrust(exchangeInfo: ExchangeRecord): Promise<{isTrusted: boolean, isAudited: boolean}> {
let isTrusted = false;
let isAudited = false;
let currencyRecord = await this.q().get(Stores.currencies, exchangeInfo.currency);
if (currencyRecord) {
for (let trustedExchange of currencyRecord.exchanges) {
if (trustedExchange.baseUrl == exchangeInfo.baseUrl) {
isTrusted = true;
break;
}
}
for (let trustedAuditor of currencyRecord.auditors) {
for (let exchangeAuditor of exchangeInfo.auditors) {
if (trustedAuditor.baseUrl == exchangeAuditor.url) {
isAudited = true;
break;
}
}
}
}
return {isTrusted, isAudited};
}
2016-09-28 18:54:48 +02:00
async getReserveCreationInfo(baseUrl: string,
2016-10-13 02:23:24 +02:00
amount: AmountJson): Promise<ReserveCreationInfo> {
2016-09-28 18:54:48 +02:00
let exchangeInfo = await this.updateExchangeFromUrl(baseUrl);
let selectedDenoms = await this.getVerifiedWithdrawDenomList(baseUrl,
amount);
2016-09-28 18:54:48 +02:00
let acc = Amounts.getZero(amount.currency);
for (let d of selectedDenoms) {
acc = Amounts.add(acc, d.feeWithdraw).amount;
2016-09-28 18:54:48 +02:00
}
let actualCoinCost = selectedDenoms
.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);
let wireInfo = await this.getWireInfo(baseUrl);
let wireFees = await this.q().get(Stores.exchangeWireFees, baseUrl);
if (!wireFees) {
// should never happen unless DB is inconsistent
throw Error(`no wire fees found for exchange ${baseUrl}`);
}
let {isTrusted, isAudited} = await this.getExchangeTrust(exchangeInfo);
let earliestDepositExpiration = Infinity;;
for (let denom of selectedDenoms) {
let expireDeposit = getTalerStampSec(denom.stampExpireDeposit)!;
if (expireDeposit < earliestDepositExpiration) {
earliestDepositExpiration = expireDeposit;
}
}
2016-09-28 18:54:48 +02:00
let ret: ReserveCreationInfo = {
exchangeInfo,
selectedDenoms,
wireInfo,
wireFees,
isAudited,
isTrusted,
2016-09-28 18:54:48 +02:00
withdrawFee: acc,
earliestDepositExpiration,
2016-09-28 18:54:48 +02:00
overhead: Amounts.sub(amount, actualCoinCost).amount,
};
return ret;
2016-02-18 22:50:17 +01:00
}
/**
2016-03-01 19:39:17 +01:00
* Update or add exchange DB entry by fetching the /keys information.
* Optionally link the reserve entry to the new or existing
2016-03-01 19:39:17 +01:00
* exchange entry in then DB.
*/
2016-11-15 15:07:17 +01:00
async updateExchangeFromUrl(baseUrl: string): Promise<ExchangeRecord> {
2016-02-18 22:50:17 +01:00
baseUrl = canonicalizeBaseUrl(baseUrl);
let keysUrl = new URI("keys").absoluteTo(baseUrl);
let wireUrl = new URI("wire").absoluteTo(baseUrl);
let keysResp = await this.http.get(keysUrl.href());
if (keysResp.status != 200) {
2016-09-28 18:54:48 +02:00
throw Error("/keys request failed");
}
let wireResp = await this.http.get(wireUrl.href());
if (wireResp.status != 200) {
throw Error("/wire request failed");
}
let exchangeKeysJson = KeysJson.checked(JSON.parse(keysResp.responseText));
let wireRespJson = JSON.parse(wireResp.responseText);
if (typeof wireRespJson !== "object") {
throw Error("/wire response is not an object");
}
console.log("exchange wire", wireRespJson);
let wireMethodDetails: WireDetailJson[] = [];
for (let methodName in wireRespJson) {
wireMethodDetails.push(WireDetailJson.checked(wireRespJson[methodName]));
}
return this.updateExchangeFromJson(baseUrl, exchangeKeysJson, wireMethodDetails);
2016-05-24 17:30:27 +02:00
}
2016-02-18 22:50:17 +01:00
2016-11-15 15:07:17 +01:00
private async suspendCoins(exchangeInfo: ExchangeRecord): Promise<void> {
2016-10-13 02:23:24 +02:00
let suspendedCoins = await (
this.q()
2016-10-18 01:36:47 +02:00
.iterIndex(Stores.coins.exchangeBaseUrlIndex, exchangeInfo.baseUrl)
.indexJoinLeft(Stores.denominations.exchangeBaseUrlIndex,
(e) => e.exchangeBaseUrl)
.reduce((cd: JoinLeftResult<CoinRecord,DenominationRecord>,
suspendedCoins: CoinRecord[]) => {
2016-11-20 04:15:49 +01:00
if ((!cd.right) || (!cd.right.isOffered)) {
return Array.prototype.concat(suspendedCoins, [cd.left]);
2016-10-13 02:23:24 +02:00
}
return Array.prototype.concat(suspendedCoins);
}, []));
let q = this.q();
2016-09-28 18:00:13 +02:00
suspendedCoins.map((c) => {
console.log("suspending coin", c);
c.suspended = true;
2016-10-18 01:16:31 +02:00
q.put(Stores.coins, c);
2016-09-28 18:00:13 +02:00
});
await q.finish();
}
2016-02-18 22:50:17 +01:00
2016-09-28 18:00:13 +02:00
private async updateExchangeFromJson(baseUrl: string,
exchangeKeysJson: KeysJson,
wireMethodDetails: WireDetailJson[]): Promise<ExchangeRecord> {
// FIXME: all this should probably be commited atomically
2016-09-12 17:41:12 +02:00
const updateTimeSec = getTalerStampSec(exchangeKeysJson.list_issue_date);
if (updateTimeSec === null) {
2016-05-24 17:30:27 +02:00
throw Error("invalid update time");
}
if (exchangeKeysJson.denoms.length == 0) {
throw Error("exchange doesn't offer any denominations");
}
const r = await this.q().get<ExchangeRecord>(Stores.exchanges, baseUrl);
2016-09-28 18:00:13 +02:00
2016-11-15 15:07:17 +01:00
let exchangeInfo: ExchangeRecord;
2016-09-28 18:00:13 +02:00
if (!r) {
exchangeInfo = {
baseUrl,
lastUpdateTime: updateTimeSec,
2016-09-28 18:00:13 +02:00
masterPublicKey: exchangeKeysJson.master_public_key,
auditors: exchangeKeysJson.auditors,
currency: exchangeKeysJson.denoms[0].value.currency,
2016-09-28 18:00:13 +02:00
};
console.log("making fresh exchange");
} else {
if (updateTimeSec < r.lastUpdateTime) {
2016-09-28 18:00:13 +02:00
console.log("outdated /keys, not updating");
return r
2016-05-24 17:30:27 +02:00
}
2016-09-28 18:00:13 +02:00
exchangeInfo = r;
exchangeInfo.lastUpdateTime = updateTimeSec;
2016-09-28 18:00:13 +02:00
console.log("updating old exchange");
}
2016-05-24 17:30:27 +02:00
2016-09-28 18:00:13 +02:00
let updatedExchangeInfo = await this.updateExchangeInfo(exchangeInfo,
2016-10-13 02:23:24 +02:00
exchangeKeysJson);
2016-09-28 18:00:13 +02:00
await this.suspendCoins(updatedExchangeInfo);
2016-10-13 02:23:24 +02:00
await this.q()
2016-10-18 01:16:31 +02:00
.put(Stores.exchanges, updatedExchangeInfo)
2016-10-13 02:23:24 +02:00
.finish();
2016-09-28 18:00:13 +02:00
let oldWireFees = await this.q().get(Stores.exchangeWireFees, baseUrl);
if (!oldWireFees) {
oldWireFees = {
exchangeBaseUrl: baseUrl,
feesForType: {},
};
}
for (let detail of wireMethodDetails) {
let latestFeeStamp = 0;
let fees = oldWireFees.feesForType[detail.type] || [];
oldWireFees.feesForType[detail.type] = fees;
for (let oldFee of fees) {
if (oldFee.endStamp > latestFeeStamp) {
latestFeeStamp = oldFee.endStamp;
}
}
for (let fee of detail.fees) {
let start = getTalerStampSec(fee.start_date);
if (start == null) {
console.error("invalid start stamp in fee", fee);
continue;
}
if (start < latestFeeStamp) {
continue;
}
let end = getTalerStampSec(fee.end_date);
if (end == null) {
console.error("invalid end stamp in fee", fee);
continue;
}
let wf: WireFee = {
wireFee: fee.wire_fee,
closingFee: fee.closing_fee,
sig: fee.sig,
startStamp: start,
endStamp: end,
}
let valid: boolean = await this.cryptoApi.isValidWireFee(detail.type, wf, exchangeInfo.masterPublicKey);
if (!valid) {
console.error("fee signature invalid", fee);
2017-04-27 04:06:48 +02:00
throw Error("fee signature invalid");
}
fees.push(wf);
}
}
await this.q().put(Stores.exchangeWireFees, oldWireFees);
if (exchangeKeysJson.payback) {
for (let payback of exchangeKeysJson.payback) {
let denom = await this.q().getIndexed(Stores.denominations.denomPubHashIndex, payback.h_denom_pub);
if (!denom) {
continue;
}
console.log(`cashing back denom`, denom);
let coins = await this.q().iterIndex(Stores.coins.denomPubIndex, denom.denomPub).toArray();
for (let coin of coins) {
this.payback(coin.coinPub);
}
}
}
2016-09-28 18:00:13 +02:00
return updatedExchangeInfo;
2016-05-24 17:30:27 +02:00
}
2016-02-18 22:50:17 +01:00
2016-02-19 00:49:22 +01:00
2016-11-15 15:07:17 +01:00
private async updateExchangeInfo(exchangeInfo: ExchangeRecord,
newKeys: KeysJson): Promise<ExchangeRecord> {
2016-05-24 17:30:27 +02:00
if (exchangeInfo.masterPublicKey != newKeys.master_public_key) {
throw Error("public keys do not match");
}
const existingDenoms: {[denomPub: string]: DenominationRecord} = await (
this.q().iterIndex(Stores.denominations.exchangeBaseUrlIndex,
exchangeInfo.baseUrl)
.reduce((x: DenominationRecord,
acc: typeof existingDenoms) => (acc[x.denomPub] = x, acc),
{})
);
2016-05-24 17:30:27 +02:00
const newDenoms: typeof existingDenoms = {};
2016-11-20 04:15:49 +01:00
const newAndUnseenDenoms: typeof existingDenoms = {};
2016-05-24 17:30:27 +02:00
for (let d of newKeys.denoms) {
let dr = await this.denominationRecordFromKeys(exchangeInfo.baseUrl, d);
if (!(d.denom_pub in existingDenoms)) {
2016-11-20 04:15:49 +01:00
newAndUnseenDenoms[dr.denomPub] = dr;
2016-09-28 19:09:10 +02:00
}
2016-11-20 04:15:49 +01:00
newDenoms[dr.denomPub] = dr;
}
2016-05-24 17:30:27 +02:00
for (let oldDenomPub in existingDenoms) {
if (!(oldDenomPub in newDenoms)) {
let d = existingDenoms[oldDenomPub];
d.isOffered = false;
}
}
2016-09-28 19:09:10 +02:00
await this.q()
.putAll(Stores.denominations,
2016-11-20 04:15:49 +01:00
Object.keys(newAndUnseenDenoms).map((d) => newAndUnseenDenoms[d]))
.putAll(Stores.denominations,
Object.keys(existingDenoms).map((d) => existingDenoms[d]))
.finish();
2016-09-28 19:09:10 +02:00
return exchangeInfo;
}
2016-02-11 18:17:02 +01:00
/**
* Retrieve a mapping from currency name to the amount
* that is currenctly available for spending in the wallet.
*/
2016-10-19 18:40:29 +02:00
async getBalances(): Promise<WalletBalance> {
function ensureEntry(balance: WalletBalance, currency: string) {
let entry: WalletBalanceEntry|undefined = balance[currency];
let z = Amounts.getZero(currency);
if (!entry) {
balance[currency] = entry = {
available: z,
pendingIncoming: z,
pendingPayment: z,
paybackAmount: z,
2016-10-19 18:40:29 +02:00
};
}
return entry;
}
2016-11-15 15:07:17 +01:00
function collectBalances(c: CoinRecord, balance: WalletBalance) {
2016-05-24 17:30:27 +02:00
if (c.suspended) {
2016-10-19 18:40:29 +02:00
return balance;
}
if (!(c.status == CoinStatus.Dirty || c.status == CoinStatus.Fresh)) {
return balance;
}
2016-10-19 18:40:29 +02:00
let currency = c.currentAmount.currency;
let entry = ensureEntry(balance, currency);
entry.available = Amounts.add(entry.available, c.currentAmount).amount;
return balance;
}
function collectPendingWithdraw(r: ReserveRecord, balance: WalletBalance) {
if (!r.confirmed) {
return balance;
2016-05-24 17:30:27 +02:00
}
2016-10-19 18:40:29 +02:00
let entry = ensureEntry(balance, r.requested_amount.currency);
let amount = r.current_amount;
if (!amount) {
amount = r.requested_amount;
}
2016-10-20 01:37:00 +02:00
amount = Amounts.add(amount, r.precoin_amount).amount;
2016-10-19 18:40:29 +02:00
if (Amounts.cmp(smallestWithdraw[r.exchange_base_url], amount) < 0) {
entry.pendingIncoming = Amounts.add(entry.pendingIncoming,
amount).amount;
}
return balance;
}
2015-12-13 23:47:30 +01:00
function collectPaybacks(r: ReserveRecord, balance: WalletBalance) {
if (!r.hasPayback) {
return balance;
}
let entry = ensureEntry(balance, r.requested_amount.currency);
if (Amounts.cmp(smallestWithdraw[r.exchange_base_url], r.current_amount!) < 0) {
entry.paybackAmount = Amounts.add(entry.paybackAmount, r.current_amount!).amount;
}
return balance;
}
function collectPendingRefresh(r: RefreshSessionRecord,
balance: WalletBalance) {
2017-03-10 15:27:36 +01:00
// Don't count finished refreshes, since the refresh already resulted
// in coins being added to the wallet.
2017-03-10 15:25:54 +01:00
if (r.finished) {
2016-10-19 18:40:29 +02:00
return balance;
}
let entry = ensureEntry(balance, r.valueWithFee.currency);
entry.pendingIncoming = Amounts.add(entry.pendingIncoming,
r.valueOutput).amount;
return balance;
}
2016-11-15 15:07:17 +01:00
function collectPayments(t: TransactionRecord, balance: WalletBalance) {
if (t.finished) {
return balance;
}
let entry = ensureEntry(balance, t.contract.amount.currency);
entry.pendingPayment = Amounts.add(entry.pendingPayment,
t.contract.amount).amount;
return balance;
}
function collectSmallestWithdraw(e: JoinResult<ExchangeRecord, DenominationRecord>,
sw: any) {
let min = sw[e.left.baseUrl];
let v = Amounts.add(e.right.value, e.right.feeWithdraw).amount;
if (!min) {
min = v;
} else if (Amounts.cmp(v, min) < 0) {
min = v;
2016-10-19 18:40:29 +02:00
}
sw[e.left.baseUrl] = min;
2016-10-19 18:40:29 +02:00
return sw;
}
let balance = {};
// Mapping from exchange pub to smallest
// possible amount we can withdraw
let smallestWithdraw: {[baseUrl: string]: AmountJson} = {};
smallestWithdraw = await (this.q()
.iter(Stores.exchanges)
.indexJoin(Stores.denominations.exchangeBaseUrlIndex,
(x) => x.baseUrl)
2016-10-19 18:40:29 +02:00
.reduce(collectSmallestWithdraw, {}));
let tx = this.q();
tx.iter(Stores.coins)
.reduce(collectBalances, balance);
tx.iter(Stores.refresh)
.reduce(collectPendingRefresh, balance);
tx.iter(Stores.reserves)
.reduce(collectPendingWithdraw, balance);
tx.iter(Stores.reserves)
.reduce(collectPaybacks, balance);
tx.iter(Stores.transactions)
.reduce(collectPayments, balance);
await tx.finish();
2016-10-19 18:40:29 +02:00
return balance;
2016-10-13 02:23:24 +02:00
}
2016-11-15 15:07:17 +01:00
async createRefreshSession(oldCoinPub: string): Promise<RefreshSessionRecord|undefined> {
let coin = await this.q().get<CoinRecord>(Stores.coins, oldCoinPub);
2016-10-13 02:23:24 +02:00
if (!coin) {
2016-10-17 23:49:04 +02:00
throw Error("coin not found");
2016-10-13 02:23:24 +02:00
}
2017-04-13 15:05:38 +02:00
if (coin.currentAmount.value == 0 && coin.currentAmount.fraction == 0) {
return undefined;
}
let exchange = await this.updateExchangeFromUrl(coin.exchangeBaseUrl);
2016-10-13 02:23:24 +02:00
if (!exchange) {
throw Error("db inconsistent");
}
let oldDenom = await this.q().get(Stores.denominations,
[exchange.baseUrl, coin.denomPub]);
2016-10-13 02:23:24 +02:00
if (!oldDenom) {
throw Error("db inconsistent");
}
let availableDenoms: DenominationRecord[] = await (
this.q()
.iterIndex(Stores.denominations.exchangeBaseUrlIndex,
exchange.baseUrl)
.toArray()
);
2016-10-13 02:23:24 +02:00
2016-10-14 02:13:06 +02:00
let availableAmount = Amounts.sub(coin.currentAmount,
oldDenom.feeRefresh).amount;
2016-10-14 02:13:06 +02:00
let newCoinDenoms = getWithdrawDenomList(availableAmount,
2016-10-13 02:23:24 +02:00
availableDenoms);
2017-04-13 15:05:38 +02:00
console.log("refreshing coin", coin);
2016-10-13 02:23:24 +02:00
console.log("refreshing into", newCoinDenoms);
2016-10-17 15:58:36 +02:00
if (newCoinDenoms.length == 0) {
2017-04-13 15:05:38 +02:00
console.log(`not refreshing, available amount ${amountToPretty(availableAmount)} too small`);
2016-10-17 23:49:04 +02:00
return undefined;
2016-10-17 15:58:36 +02:00
}
2016-10-13 02:23:24 +02:00
2016-11-15 15:07:17 +01:00
let refreshSession: RefreshSessionRecord = await (
2016-10-14 02:13:06 +02:00
this.cryptoApi.createRefreshSession(exchange.baseUrl,
2016-10-17 15:58:36 +02:00
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 {
2016-10-20 01:37:00 +02:00
let r = Amounts.sub(c.currentAmount,
2016-10-19 23:55:58 +02:00
refreshSession.valueWithFee);
if (r.saturated) {
// Something else must have written the coin value
throw AbortTransaction;
}
c.currentAmount = r.amount;
c.status = CoinStatus.Refreshed;
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.
2016-10-17 15:58:36 +02:00
await this.q()
2016-10-18 01:16:31 +02:00
.put(Stores.refresh, refreshSession)
2016-10-19 23:55:58 +02:00
.mutate(Stores.coins, coin.coinPub, mutateCoin)
2016-10-17 15:58:36 +02:00
.finish();
2016-10-17 23:49:04 +02:00
return refreshSession;
}
2016-10-13 02:23:24 +02:00
2016-10-17 23:49:04 +02:00
async refresh(oldCoinPub: string): Promise<void> {
2016-11-15 15:07:17 +01:00
let refreshSession: RefreshSessionRecord|undefined;
2016-10-18 01:16:31 +02:00
let oldSession = await this.q().get(Stores.refresh, oldCoinPub);
2016-10-17 23:49:04 +02:00
if (oldSession) {
2017-04-13 15:05:38 +02:00
console.log("got old session for", oldCoinPub);
console.log(oldSession);
2016-10-17 23:49:04 +02:00
refreshSession = oldSession;
} else {
2016-10-18 02:39:54 +02:00
refreshSession = await this.createRefreshSession(oldCoinPub);
2016-10-17 15:58:36 +02:00
}
2016-10-17 23:49:04 +02:00
if (!refreshSession) {
// refreshing not necessary
2017-04-13 15:05:38 +02:00
console.log("not refreshing", oldCoinPub);
2016-10-17 23:49:04 +02:00
return;
}
this.continueRefreshSession(refreshSession);
}
2016-11-15 15:07:17 +01:00
async continueRefreshSession(refreshSession: RefreshSessionRecord) {
2016-10-17 23:49:04 +02:00
if (refreshSession.finished) {
return;
}
if (typeof refreshSession.norevealIndex !== "number") {
let coinPub = refreshSession.meltCoinPub;
await this.refreshMelt(refreshSession);
2016-11-15 15:07:17 +01:00
let r = await this.q().get<RefreshSessionRecord>(Stores.refresh, coinPub);
2016-10-17 23:49:04 +02:00
if (!r) {
throw Error("refresh session does not exist anymore");
}
refreshSession = r;
}
await this.refreshReveal(refreshSession);
2016-10-17 15:58:36 +02:00
}
2016-11-15 15:07:17 +01:00
async refreshMelt(refreshSession: RefreshSessionRecord): Promise<void> {
2016-10-17 15:58:36 +02:00
if (refreshSession.norevealIndex != undefined) {
console.error("won't melt again");
return;
}
2016-11-15 15:07:17 +01:00
let coin = await this.q().get<CoinRecord>(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;
}
let reqUrl = new URI("refresh/melt").absoluteTo(refreshSession.exchangeBaseUrl);
2016-10-14 02:13:06 +02:00
let meltCoin = {
coin_pub: coin.coinPub,
denom_pub: coin.denomPub,
denom_sig: coin.denomSig,
confirm_sig: refreshSession.confirmSig,
value_with_fee: refreshSession.valueWithFee,
};
let coinEvs = refreshSession.preCoinsForGammas.map((x) => x.map((y) => y.coinEv));
let req = {
2016-10-17 15:58:36 +02:00
"new_denoms": refreshSession.newDenoms,
2016-10-14 02:13:06 +02:00
"melt_coin": meltCoin,
"transfer_pubs": refreshSession.transferPubs,
"coin_evs": coinEvs,
};
console.log("melt request:", req);
let resp = await this.http.postJson(reqUrl.href(), req);
2016-10-13 02:36:33 +02:00
2016-10-14 02:13:06 +02:00
console.log("melt request:", req);
2016-10-13 02:36:33 +02:00
console.log("melt response:", resp.responseText);
2016-09-28 18:00:13 +02:00
2016-10-14 02:13:06 +02:00
if (resp.status != 200) {
console.error(resp.responseText);
throw Error("refresh failed");
}
let respJson = JSON.parse(resp.responseText);
if (!respJson) {
throw Error("exchange responded with garbage");
}
let norevealIndex = respJson.noreveal_index;
if (typeof norevealIndex != "number") {
throw Error("invalid response");
}
refreshSession.norevealIndex = norevealIndex;
2016-10-18 01:16:31 +02:00
await this.q().put(Stores.refresh, refreshSession).finish();
}
2016-10-17 15:58:36 +02:00
2016-11-15 15:07:17 +01:00
async refreshReveal(refreshSession: RefreshSessionRecord): Promise<void> {
2016-10-14 02:13:06 +02:00
let norevealIndex = refreshSession.norevealIndex;
if (norevealIndex == undefined) {
throw Error("can't reveal without melting first");
}
let privs = Array.from(refreshSession.transferPrivs);
privs.splice(norevealIndex, 1);
let req = {
"session_hash": refreshSession.hash,
"transfer_privs": privs,
};
let reqUrl = new URI("refresh/reveal")
2016-10-17 15:58:36 +02:00
.absoluteTo(refreshSession.exchangeBaseUrl);
2016-10-14 02:13:06 +02:00
console.log("reveal request:", req);
let resp = await this.http.postJson(reqUrl.href(), req);
2016-10-14 02:13:06 +02:00
console.log("session:", refreshSession);
console.log("reveal response:", resp);
2016-10-17 15:58:36 +02:00
if (resp.status != 200) {
console.log("error: /refresh/reveal returned status " + resp.status);
return;
}
let respJson = JSON.parse(resp.responseText);
if (!respJson.ev_sigs || !Array.isArray(respJson.ev_sigs)) {
console.log("/refresh/reveal did not contain ev_sigs");
}
2016-11-15 15:07:17 +01:00
let exchange = await this.q().get<ExchangeRecord>(Stores.exchanges,
refreshSession.exchangeBaseUrl);
2016-10-17 15:58:36 +02:00
if (!exchange) {
console.error(`exchange ${refreshSession.exchangeBaseUrl} not found`);
return;
}
2016-11-15 15:07:17 +01:00
let 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++) {
let denom = await (
this.q()
.get(Stores.denominations,
[
refreshSession.exchangeBaseUrl,
refreshSession.newDenoms[i]
]));
2016-10-17 15:58:36 +02:00
if (!denom) {
console.error("denom not found");
continue;
}
let pc = refreshSession.preCoinsForGammas[refreshSession.norevealIndex!][i];
let denomSig = await this.cryptoApi.rsaUnblind(respJson.ev_sigs[i].ev_sig,
pc.blindingKey,
denom.denomPub);
2016-11-15 15:07:17 +01:00
let coin: CoinRecord = {
reservePub: undefined,
blindingKey: pc.blindingKey,
2016-10-17 15:58:36 +02:00
coinPub: pc.publicKey,
coinPriv: pc.privateKey,
denomPub: denom.denomPub,
2016-10-17 15:58:36 +02:00
denomSig: denomSig,
currentAmount: denom.value,
exchangeBaseUrl: refreshSession.exchangeBaseUrl,
status: CoinStatus.Fresh,
2016-10-17 15:58:36 +02:00
};
2016-10-17 23:49:04 +02:00
coins.push(coin);
2016-10-17 15:58:36 +02:00
}
2016-10-17 23:49:04 +02:00
refreshSession.finished = true;
await this.q()
2016-10-18 01:16:31 +02:00
.putAll(Stores.coins, coins)
.put(Stores.refresh, refreshSession)
2016-10-17 23:49:04 +02:00
.finish();
2016-10-14 02:13:06 +02:00
}
2016-01-26 17:21:17 +01:00
2016-02-11 18:17:02 +01:00
/**
* Retrive the full event history for this wallet.
*/
async getHistory(): Promise<{history: HistoryRecord[]}> {
2016-09-12 17:41:12 +02:00
function collect(x: any, acc: any) {
acc.push(x);
2016-03-02 05:23:16 +01:00
return acc;
}
2016-01-26 17:21:17 +01:00
2016-10-13 02:23:24 +02:00
let history = await (
this.q()
2016-10-18 01:36:47 +02:00
.iterIndex(Stores.history.timestampIndex)
2016-10-13 02:23:24 +02:00
.reduce(collect, []));
2016-09-28 17:52:36 +02:00
2016-10-13 02:23:24 +02:00
return {history};
2016-10-12 02:55:53 +02:00
}
async getDenoms(exchangeUrl: string): Promise<DenominationRecord[]> {
let denoms = await this.q().iterIndex(Stores.denominations.exchangeBaseUrlIndex, exchangeUrl).toArray();
return denoms;
}
2016-11-13 10:17:39 +01:00
async getOffer(offerId: number): Promise<any> {
let offer = await this.q() .get(Stores.offers, offerId);
return offer;
}
2016-11-15 15:07:17 +01:00
async getExchanges(): Promise<ExchangeRecord[]> {
2016-10-13 02:23:24 +02:00
return this.q()
2016-11-15 15:07:17 +01:00
.iter<ExchangeRecord>(Stores.exchanges)
2016-10-13 02:23:24 +02:00
.flatMap((e) => [e])
.toArray();
}
2016-02-23 14:07:53 +01:00
2017-03-24 17:54:22 +01:00
async getCurrencies(): Promise<CurrencyRecord[]> {
return this.q()
.iter<CurrencyRecord>(Stores.currencies)
.flatMap((e) => [e])
.toArray();
}
async updateCurrency(currencyRecord: CurrencyRecord): Promise<void> {
console.log("updating currency to", currencyRecord);
await this.q()
.put(Stores.currencies, currencyRecord)
.finish();
this.notifier.notify();
}
2016-10-13 02:23:24 +02:00
async getReserves(exchangeBaseUrl: string): Promise<ReserveRecord[]> {
return this.q()
2016-10-18 01:16:31 +02:00
.iter<ReserveRecord>(Stores.reserves)
2016-10-13 02:23:24 +02:00
.filter((r: ReserveRecord) => r.exchange_base_url === exchangeBaseUrl)
.toArray();
2016-10-12 02:55:53 +02:00
}
2016-11-15 15:07:17 +01:00
async getCoins(exchangeBaseUrl: string): Promise<CoinRecord[]> {
2016-10-13 02:23:24 +02:00
return this.q()
2016-11-15 15:07:17 +01:00
.iter<CoinRecord>(Stores.coins)
.filter((c: CoinRecord) => c.exchangeBaseUrl === exchangeBaseUrl)
2016-10-13 02:23:24 +02:00
.toArray();
2016-10-12 02:55:53 +02:00
}
2016-11-15 15:07:17 +01:00
async getPreCoins(exchangeBaseUrl: string): Promise<PreCoinRecord[]> {
2016-10-13 02:23:24 +02:00
return this.q()
2016-11-15 15:07:17 +01:00
.iter<PreCoinRecord>(Stores.precoins)
.filter((c: PreCoinRecord) => c.exchangeBaseUrl === exchangeBaseUrl)
2016-10-13 02:23:24 +02:00
.toArray();
2016-10-12 02:55:53 +02:00
}
async hashContract(contract: Contract): Promise<string> {
2016-09-28 23:41:34 +02:00
return this.cryptoApi.hashString(canonicalJson(contract));
}
2016-10-17 15:58:36 +02:00
/**
* Generate a nonce in form of an EdDSA public key.
* Store the private key in our DB, so we can prove ownership.
*/
async generateNonce(): Promise<string> {
let {priv, pub} = await this.cryptoApi.createEddsaKeypair();
await this.q()
.put(Stores.nonces, {priv, pub})
.finish();
return pub;
}
async getCurrencyRecord(currency: string): Promise<CurrencyRecord|undefined> {
return this.q().get(Stores.currencies, currency);
}
async paymentSucceeded(contractHash: string, merchantSig: string): Promise<any> {
2016-10-17 23:49:04 +02:00
const doPaymentSucceeded = async() => {
2016-11-15 15:07:17 +01:00
let t = await this.q().get<TransactionRecord>(Stores.transactions,
contractHash);
2016-10-17 15:58:36 +02:00
if (!t) {
console.error("contract not found");
return;
}
let merchantPub = t.contract.merchant_pub;
let valid = this.cryptoApi.isValidPaymentSignature(merchantSig, contractHash, merchantPub);
if (!valid) {
console.error("merchant payment signature invalid");
// FIXME: properly display error
return;
}
t.finished = true;
2016-11-15 15:07:17 +01:00
let modifiedCoins: CoinRecord[] = [];
2016-10-17 15:58:36 +02:00
for (let pc of t.payReq.coins) {
2016-11-15 15:07:17 +01:00
let c = await this.q().get<CoinRecord>(Stores.coins, pc.coin_pub);
2016-10-17 15:58:36 +02:00
if (!c) {
console.error("coin not found");
return;
}
c.status = CoinStatus.Dirty;
modifiedCoins.push(c);
2016-10-17 15:58:36 +02:00
}
await this.q()
.putAll(Stores.coins, modifiedCoins)
.put(Stores.transactions, t)
.finish();
2016-10-17 15:58:36 +02:00
for (let c of t.payReq.coins) {
this.refresh(c.coin_pub);
}
};
doPaymentSucceeded();
return;
}
async payback(coinPub: string): Promise<void> {
let coin = await this.q().get(Stores.coins, coinPub);
if (!coin) {
throw Error(`Coin ${coinPub} not found, can't request payback`);
}
let reservePub = coin.reservePub;
if (!reservePub) {
throw Error(`Can't request payback for a refreshed coin`);
}
let reserve = await this.q().get(Stores.reserves, reservePub);
if (!reserve) {
throw Error(`Reserve of coin ${coinPub} not found`);
}
switch (coin.status) {
case CoinStatus.Refreshed:
throw Error(`Can't do payback for coin ${coinPub} since it's refreshed`);
case CoinStatus.PaybackDone:
console.log(`Coin ${coinPub} already payed back`);
return;
}
coin.status = CoinStatus.PaybackPending;
// 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;
await this.q().put(Stores.coins, coin).put(Stores.reserves, reserve);
let paybackRequest = await this.cryptoApi.createPaybackRequest(coin);
let reqUrl = new URI("payback").absoluteTo(coin.exchangeBaseUrl);
let resp = await this.http.postJson(reqUrl.href(), paybackRequest);
if (resp.status != 200) {
throw Error();
}
let paybackConfirmation = PaybackConfirmation.checked(JSON.parse(resp.responseText));
if (paybackConfirmation.reserve_pub != coin.reservePub) {
throw Error(`Coin's reserve doesn't match reserve on payback`);
}
coin = await this.q().get(Stores.coins, coinPub);
if (!coin) {
throw Error(`Coin ${coinPub} not found, can't confirm payback`);
}
coin.status = CoinStatus.PaybackDone;
await this.q().put(Stores.coins, coin);
await this.updateReserve(reservePub!);
}
async denominationRecordFromKeys(exchangeBaseUrl: string, denomIn: Denomination): Promise<DenominationRecord> {
let denomPubHash = await this.cryptoApi.hashDenomPub(denomIn.denom_pub);
let d: DenominationRecord = {
denomPubHash,
denomPub: denomIn.denom_pub,
exchangeBaseUrl: exchangeBaseUrl,
feeDeposit: denomIn.fee_deposit,
masterSig: denomIn.master_sig,
feeRefund: denomIn.fee_refund,
feeRefresh: denomIn.fee_refresh,
feeWithdraw: denomIn.fee_withdraw,
stampExpireDeposit: denomIn.stamp_expire_deposit,
stampExpireLegal: denomIn.stamp_expire_legal,
stampExpireWithdraw: denomIn.stamp_expire_withdraw,
stampStart: denomIn.stamp_start,
status: DenominationStatus.Unverified,
isOffered: true,
value: denomIn.value,
};
return d;
}
async withdrawPaybackReserve(reservePub: string): Promise<void> {
let reserve = await this.q().get(Stores.reserves, reservePub);
if (!reserve) {
throw Error(`Reserve ${reservePub} does not exist`);
}
reserve.hasPayback = false;
await this.q().put(Stores.reserves, reserve);
this.depleteReserve(reserve);
}
async getPaybackReserves(): Promise<ReserveRecord[]> {
return await this.q().iter(Stores.reserves).filter(r => r.hasPayback).toArray()
}
2016-10-18 01:16:31 +02:00
}