From f187c1e13f749dbf929a9644481fa93220068bad Mon Sep 17 00:00:00 2001 From: Florian Dold Date: Wed, 23 Mar 2022 16:37:40 +0100 Subject: [PATCH] missing files --- .../src/internal-wallet-state.ts | 257 ++++++++++++++++++ .../src/operations/common.ts | 51 ++++ 2 files changed, 308 insertions(+) create mode 100644 packages/taler-wallet-core/src/internal-wallet-state.ts create mode 100644 packages/taler-wallet-core/src/operations/common.ts diff --git a/packages/taler-wallet-core/src/internal-wallet-state.ts b/packages/taler-wallet-core/src/internal-wallet-state.ts new file mode 100644 index 000000000..904398722 --- /dev/null +++ b/packages/taler-wallet-core/src/internal-wallet-state.ts @@ -0,0 +1,257 @@ +/* + This file is part of GNU Taler + (C) 2019 GNUnet e.V. + + GNU Taler is free software; you can redistribute it and/or modify it under the + terms of the GNU General Public License as published by the Free Software + Foundation; either version 3, or (at your option) any later version. + + GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY + WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR + A PARTICULAR PURPOSE. See the GNU General Public License for more details. + + You should have received a copy of the GNU General Public License along with + GNU Taler; see the file COPYING. If not, see + */ + +/** + * Common interface of the internal wallet state. This object is passed + * to the various operations (exchange management, withdrawal, refresh, reserve + * management, etc.). + * + * Some operations can be accessed via this state object. This allows mutual + * recursion between operations, without having cyclic dependencies between + * the respective TypeScript files. + * + * (You can think of this as a "header file" for the wallet implementation.) + */ + +/** + * Imports. + */ +import { + WalletNotification, + BalancesResponse, + AmountJson, + DenominationPubKey, + TalerProtocolTimestamp, +} from "@gnu-taler/taler-util"; +import { CryptoApi } from "./crypto/workers/cryptoApi.js"; +import { ExchangeDetailsRecord, ExchangeRecord, WalletStoresV1 } from "./db.js"; +import { PendingOperationsResponse } from "./pending-types.js"; +import { AsyncOpMemoMap, AsyncOpMemoSingle } from "./util/asyncMemo.js"; +import { HttpRequestLibrary } from "./util/http.js"; +import { AsyncCondition } from "./util/promiseUtils.js"; +import { + DbAccess, + GetReadOnlyAccess, + GetReadWriteAccess, +} from "./util/query.js"; +import { TimerGroup } from "./util/timer.js"; + +export const EXCHANGE_COINS_LOCK = "exchange-coins-lock"; +export const EXCHANGE_RESERVES_LOCK = "exchange-reserves-lock"; + +export interface TrustInfo { + isTrusted: boolean; + isAudited: boolean; +} + +export interface MerchantInfo { + protocolVersionCurrent: number; +} + +/** + * Interface for merchant-related operations. + */ +export interface MerchantOperations { + getMerchantInfo( + ws: InternalWalletState, + merchantBaseUrl: string, + ): Promise; +} + +export interface ReserveOperations { + processReserve( + ws: InternalWalletState, + reservePub: string, + forceNow?: boolean, + ): Promise; +} + +/** + * Interface for exchange-related operations. + */ +export interface ExchangeOperations { + // FIXME: Should other operations maybe always use + // updateExchangeFromUrl? + getExchangeDetails( + tx: GetReadOnlyAccess<{ + exchanges: typeof WalletStoresV1.exchanges; + exchangeDetails: typeof WalletStoresV1.exchangeDetails; + }>, + exchangeBaseUrl: string, + ): Promise; + getExchangeTrust( + ws: InternalWalletState, + exchangeInfo: ExchangeRecord, + ): Promise; + updateExchangeFromUrl( + ws: InternalWalletState, + baseUrl: string, + acceptedFormat?: string[], + forceNow?: boolean, + ): Promise<{ + exchange: ExchangeRecord; + exchangeDetails: ExchangeDetailsRecord; + }>; +} + +export interface RecoupOperations { + createRecoupGroup( + ws: InternalWalletState, + tx: GetReadWriteAccess<{ + recoupGroups: typeof WalletStoresV1.recoupGroups; + denominations: typeof WalletStoresV1.denominations; + refreshGroups: typeof WalletStoresV1.refreshGroups; + coins: typeof WalletStoresV1.coins; + }>, + coinPubs: string[], + ): Promise; + processRecoupGroup( + ws: InternalWalletState, + recoupGroupId: string, + forceNow?: boolean, + ): Promise; +} + +export interface DenomInfo { + /** + * Value of one coin of the denomination. + */ + value: AmountJson; + + /** + * The denomination public key. + */ + denomPub: DenominationPubKey; + + /** + * Hash of the denomination public key. + * Stored in the database for faster lookups. + */ + denomPubHash: string; + + /** + * Fee for withdrawing. + */ + feeWithdraw: AmountJson; + + /** + * Fee for depositing. + */ + feeDeposit: AmountJson; + + /** + * Fee for refreshing. + */ + feeRefresh: AmountJson; + + /** + * Fee for refunding. + */ + feeRefund: AmountJson; + + /** + * Validity start date of the denomination. + */ + stampStart: TalerProtocolTimestamp; + + /** + * Date after which the currency can't be withdrawn anymore. + */ + stampExpireWithdraw: TalerProtocolTimestamp; + + /** + * Date after the denomination officially doesn't exist anymore. + */ + stampExpireLegal: TalerProtocolTimestamp; + + /** + * Data after which coins of this denomination can't be deposited anymore. + */ + stampExpireDeposit: TalerProtocolTimestamp; +} + +export type NotificationListener = (n: WalletNotification) => void; + +/** + * Internal, shard wallet state that is used by the implementation + * of wallet operations. + * + * FIXME: This should not be exported anywhere from the taler-wallet-core package, + * as it's an opaque implementation detail. + */ +export interface InternalWalletState { + memoProcessReserve: AsyncOpMemoMap; + memoMakePlanchet: AsyncOpMemoMap; + memoGetPending: AsyncOpMemoSingle; + memoGetBalance: AsyncOpMemoSingle; + memoProcessRefresh: AsyncOpMemoMap; + memoProcessRecoup: AsyncOpMemoMap; + memoProcessDeposit: AsyncOpMemoMap; + cryptoApi: CryptoApi; + + timerGroup: TimerGroup; + stopped: boolean; + + insecureTrustExchange: boolean; + + /** + * Asynchronous condition to interrupt the sleep of the + * retry loop. + * + * Used to allow processing of new work faster. + */ + latch: AsyncCondition; + + listeners: NotificationListener[]; + + initCalled: boolean; + + merchantInfoCache: Record; + + exchangeOps: ExchangeOperations; + recoupOps: RecoupOperations; + merchantOps: MerchantOperations; + reserveOps: ReserveOperations; + + getDenomInfo( + ws: InternalWalletState, + tx: GetReadOnlyAccess<{ + denominations: typeof WalletStoresV1.denominations; + }>, + exchangeBaseUrl: string, + denomPubHash: string, + ): Promise; + + db: DbAccess; + http: HttpRequestLibrary; + + notify(n: WalletNotification): void; + + addNotificationListener(f: (n: WalletNotification) => void): void; + + /** + * Stop ongoing processing. + */ + stop(): void; + + /** + * Run an async function after acquiring a list of locks, identified + * by string tokens. + */ + runSequentialized(tokens: string[], f: () => Promise): Promise; + + runUntilDone(req?: { maxRetries?: number }): Promise; +} diff --git a/packages/taler-wallet-core/src/operations/common.ts b/packages/taler-wallet-core/src/operations/common.ts new file mode 100644 index 000000000..5525b4deb --- /dev/null +++ b/packages/taler-wallet-core/src/operations/common.ts @@ -0,0 +1,51 @@ +/* + This file is part of GNU Taler + (C) 2022 GNUnet e.V. + + GNU Taler is free software; you can redistribute it and/or modify it under the + terms of the GNU General Public License as published by the Free Software + Foundation; either version 3, or (at your option) any later version. + + GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY + WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR + A PARTICULAR PURPOSE. See the GNU General Public License for more details. + + You should have received a copy of the GNU General Public License along with + GNU Taler; see the file COPYING. If not, see + */ + +import { TalerErrorDetail, TalerErrorCode } from "@gnu-taler/taler-util"; +import { CryptoApiStoppedError } from "../crypto/workers/cryptoApi.js"; +import { TalerError, getErrorDetailFromException } from "../errors.js"; + +/** + * Run an operation and call the onOpError callback + * when there was an exception or operation error that must be reported. + * The cause will be re-thrown to the caller. + */ +export async function guardOperationException( + op: () => Promise, + onOpError: (e: TalerErrorDetail) => Promise, +): Promise { + try { + return await op(); + } catch (e: any) { + if (e instanceof CryptoApiStoppedError) { + throw e; + } + if ( + e instanceof TalerError && + e.hasErrorCode(TalerErrorCode.WALLET_PENDING_OPERATION_FAILED) + ) { + throw e; + } + const opErr = getErrorDetailFromException(e); + await onOpError(opErr); + throw TalerError.fromDetail( + TalerErrorCode.WALLET_PENDING_OPERATION_FAILED, + { + innerError: e.errorDetail, + }, + ); + } +}