wallet: improve error handling and error codes

This commit is contained in:
Florian Dold 2022-03-22 21:16:38 +01:00
parent f8d12f7b0d
commit 5d23eb3635
No known key found for this signature in database
GPG Key ID: D2E4F00F29D02A4B
41 changed files with 672 additions and 508 deletions

View File

@ -22,7 +22,7 @@
/** /**
* Imports. * Imports.
*/ */
import { TalerErrorDetails } from "./walletTypes.js"; import { TalerErrorDetail } from "./walletTypes.js";
export enum NotificationType { export enum NotificationType {
CoinWithdrawn = "coin-withdrawn", CoinWithdrawn = "coin-withdrawn",
@ -157,62 +157,62 @@ export interface ExchangeAddedNotification {
export interface ExchangeOperationErrorNotification { export interface ExchangeOperationErrorNotification {
type: NotificationType.ExchangeOperationError; type: NotificationType.ExchangeOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface RefreshOperationErrorNotification { export interface RefreshOperationErrorNotification {
type: NotificationType.RefreshOperationError; type: NotificationType.RefreshOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface BackupOperationErrorNotification { export interface BackupOperationErrorNotification {
type: NotificationType.BackupOperationError; type: NotificationType.BackupOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface RefundStatusOperationErrorNotification { export interface RefundStatusOperationErrorNotification {
type: NotificationType.RefundStatusOperationError; type: NotificationType.RefundStatusOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface RefundApplyOperationErrorNotification { export interface RefundApplyOperationErrorNotification {
type: NotificationType.RefundApplyOperationError; type: NotificationType.RefundApplyOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface PayOperationErrorNotification { export interface PayOperationErrorNotification {
type: NotificationType.PayOperationError; type: NotificationType.PayOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface ProposalOperationErrorNotification { export interface ProposalOperationErrorNotification {
type: NotificationType.ProposalOperationError; type: NotificationType.ProposalOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface TipOperationErrorNotification { export interface TipOperationErrorNotification {
type: NotificationType.TipOperationError; type: NotificationType.TipOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface WithdrawOperationErrorNotification { export interface WithdrawOperationErrorNotification {
type: NotificationType.WithdrawOperationError; type: NotificationType.WithdrawOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface RecoupOperationErrorNotification { export interface RecoupOperationErrorNotification {
type: NotificationType.RecoupOperationError; type: NotificationType.RecoupOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface DepositOperationErrorNotification { export interface DepositOperationErrorNotification {
type: NotificationType.DepositOperationError; type: NotificationType.DepositOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface ReserveOperationErrorNotification { export interface ReserveOperationErrorNotification {
type: NotificationType.ReserveOperationError; type: NotificationType.ReserveOperationError;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface ReserveCreatedNotification { export interface ReserveCreatedNotification {

View File

@ -22,6 +22,8 @@
*/ */
export enum TalerErrorCode { export enum TalerErrorCode {
/** /**
* Special code to indicate success (no error). * Special code to indicate success (no error).
* Returned with an HTTP status code of #MHD_HTTP_UNINITIALIZED (0). * Returned with an HTTP status code of #MHD_HTTP_UNINITIALIZED (0).
@ -78,6 +80,13 @@ export enum TalerErrorCode {
*/ */
GENERIC_CONFIGURATION_INVALID = 14, GENERIC_CONFIGURATION_INVALID = 14,
/**
* The client made a request to a service, but received an error response it does not know how to handle.
* Returned with an HTTP status code of #MHD_HTTP_UNINITIALIZED (0).
* (A value of 0 indicates that the error is generated client-side).
*/
GENERIC_UNEXPECTED_REQUEST_ERROR = 15,
/** /**
* The HTTP method used is invalid for this endpoint. * The HTTP method used is invalid for this endpoint.
* Returned with an HTTP status code of #MHD_HTTP_METHOD_NOT_ALLOWED (405). * Returned with an HTTP status code of #MHD_HTTP_METHOD_NOT_ALLOWED (405).
@ -372,6 +381,20 @@ export enum TalerErrorCode {
*/ */
EXCHANGE_GENERIC_NEW_DENOMS_ARRAY_SIZE_EXCESSIVE = 1018, EXCHANGE_GENERIC_NEW_DENOMS_ARRAY_SIZE_EXCESSIVE = 1018,
/**
* The reserve public key was malformed.
* Returned with an HTTP status code of #MHD_HTTP_BAD_REQUEST (400).
* (A value of 0 indicates that the error is generated client-side).
*/
EXCHANGE_GENERIC_RESERVE_PUB_MALFORMED = 1019,
/**
* The time at the server is too far off from the time specified in the request. Most likely the client system time is wrong.
* Returned with an HTTP status code of #MHD_HTTP_BAD_REQUEST (400).
* (A value of 0 indicates that the error is generated client-side).
*/
EXCHANGE_GENERIC_CLOCK_SKEW = 1020,
/** /**
* The exchange did not find information about the specified transaction in the database. * The exchange did not find information about the specified transaction in the database.
* Returned with an HTTP status code of #MHD_HTTP_NOT_FOUND (404). * Returned with an HTTP status code of #MHD_HTTP_NOT_FOUND (404).
@ -541,11 +564,25 @@ export enum TalerErrorCode {
EXCHANGE_DEPOSIT_FEE_ABOVE_AMOUNT = 1222, EXCHANGE_DEPOSIT_FEE_ABOVE_AMOUNT = 1222,
/** /**
* The reserve status was requested using a unknown key. * The reserve balance, status or history was requested for a reserve which is not known to the exchange.
* Returned with an HTTP status code of #MHD_HTTP_NOT_FOUND (404). * Returned with an HTTP status code of #MHD_HTTP_NOT_FOUND (404).
* (A value of 0 indicates that the error is generated client-side). * (A value of 0 indicates that the error is generated client-side).
*/ */
EXCHANGE_RESERVES_GET_STATUS_UNKNOWN = 1250, EXCHANGE_RESERVES_STATUS_UNKNOWN = 1250,
/**
* The reserve status was requested with a bad signature.
* Returned with an HTTP status code of #MHD_HTTP_FORBIDDEN (403).
* (A value of 0 indicates that the error is generated client-side).
*/
EXCHANGE_RESERVES_STATUS_BAD_SIGNATURE = 1251,
/**
* The reserve history was requested with a bad signature.
* Returned with an HTTP status code of #MHD_HTTP_FORBIDDEN (403).
* (A value of 0 indicates that the error is generated client-side).
*/
EXCHANGE_RESERVES_HISTORY_BAD_SIGNATURE = 1252,
/** /**
* The exchange encountered melt fees exceeding the melted coin's contribution. * The exchange encountered melt fees exceeding the melted coin's contribution.
@ -1394,6 +1431,27 @@ export enum TalerErrorCode {
*/ */
MERCHANT_POST_ORDERS_ID_PAY_EXCHANGE_FAILED = 2170, MERCHANT_POST_ORDERS_ID_PAY_EXCHANGE_FAILED = 2170,
/**
* The payment required a minimum age but one of the coins (of a denomination with support for age restriction) did not provide any age_commitment.
* Returned with an HTTP status code of #MHD_HTTP_BAD_REQUEST (400).
* (A value of 0 indicates that the error is generated client-side).
*/
MERCHANT_POST_ORDERS_ID_PAY_AGE_COMMITMENT_MISSING = 2171,
/**
* The payment required a minimum age but one of the coins provided an age_commitment that contained a wrong number of public keys compared to the number of age groups defined in the denomination of the coin.
* Returned with an HTTP status code of #MHD_HTTP_BAD_REQUEST (400).
* (A value of 0 indicates that the error is generated client-side).
*/
MERCHANT_POST_ORDERS_ID_PAY_AGE_COMMITMENT_SIZE_MISMATCH = 2172,
/**
* The payment required a minimum age but one of the coins provided a minimum_age_sig that couldn't be verified with the given age_commitment for that particular minimum age.
* Returned with an HTTP status code of #MHD_HTTP_BAD_REQUEST (400).
* (A value of 0 indicates that the error is generated client-side).
*/
MERCHANT_POST_ORDERS_ID_PAY_AGE_VERIFICATION_FAILED = 2173,
/** /**
* The contract hash does not match the given order ID. * The contract hash does not match the given order ID.
* Returned with an HTTP status code of #MHD_HTTP_BAD_REQUEST (400). * Returned with an HTTP status code of #MHD_HTTP_BAD_REQUEST (400).
@ -2150,6 +2208,13 @@ export enum TalerErrorCode {
*/ */
WALLET_CONTRACT_TERMS_MALFORMED = 7020, WALLET_CONTRACT_TERMS_MALFORMED = 7020,
/**
* A pending operation failed, and thus the request can't be completed.
* Returned with an HTTP status code of #MHD_HTTP_UNINITIALIZED (0).
* (A value of 0 indicates that the error is generated client-side).
*/
WALLET_PENDING_OPERATION_FAILED = 7021,
/** /**
* We encountered a timeout with our payment backend. * We encountered a timeout with our payment backend.
* Returned with an HTTP status code of #MHD_HTTP_GATEWAY_TIMEOUT (504). * Returned with an HTTP status code of #MHD_HTTP_GATEWAY_TIMEOUT (504).
@ -2646,4 +2711,5 @@ export enum TalerErrorCode {
* (A value of 0 indicates that the error is generated client-side). * (A value of 0 indicates that the error is generated client-side).
*/ */
END = 9999, END = 9999,
} }

View File

@ -42,7 +42,7 @@ import {
codecForList, codecForList,
codecForAny, codecForAny,
} from "./codec.js"; } from "./codec.js";
import { TalerErrorDetails } from "./walletTypes.js"; import { TalerErrorDetail } from "./walletTypes.js";
export interface TransactionsRequest { export interface TransactionsRequest {
/** /**
@ -92,7 +92,7 @@ export interface TransactionCommon {
// Amount added or removed from the wallet's balance (including all fees and other costs) // Amount added or removed from the wallet's balance (including all fees and other costs)
amountEffective: AmountString; amountEffective: AmountString;
error?: TalerErrorDetails; error?: TalerErrorDetail;
} }
export type Transaction = export type Transaction =

View File

@ -60,6 +60,7 @@ import {
import { OrderShortInfo, codecForOrderShortInfo } from "./transactionsTypes.js"; import { OrderShortInfo, codecForOrderShortInfo } from "./transactionsTypes.js";
import { BackupRecovery } from "./backupTypes.js"; import { BackupRecovery } from "./backupTypes.js";
import { PaytoUri } from "./payto.js"; import { PaytoUri } from "./payto.js";
import { TalerErrorCode } from "./taler-error-codes.js";
/** /**
* Response for the create reserve request to the wallet. * Response for the create reserve request to the wallet.
@ -136,7 +137,7 @@ export interface ConfirmPayResultDone {
export interface ConfirmPayResultPending { export interface ConfirmPayResultPending {
type: ConfirmPayResultType.Pending; type: ConfirmPayResultType.Pending;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
export type ConfirmPayResult = ConfirmPayResultDone | ConfirmPayResultPending; export type ConfirmPayResult = ConfirmPayResultDone | ConfirmPayResultPending;
@ -455,11 +456,10 @@ export interface WalletDiagnostics {
dbOutdated: boolean; dbOutdated: boolean;
} }
export interface TalerErrorDetails { export interface TalerErrorDetail {
code: number; code: TalerErrorCode;
hint: string; hint?: string;
message: string; [x: string]: unknown;
details: unknown;
} }
/** /**
@ -850,7 +850,7 @@ export interface CoreApiResponseError {
type: "error"; type: "error";
operation: string; operation: string;
id: string; id: string;
error: TalerErrorDetails; error: TalerErrorDetail;
} }
export interface WithdrawTestBalanceRequest { export interface WithdrawTestBalanceRequest {

View File

@ -49,7 +49,7 @@ import {
HarnessExchangeBankAccount, HarnessExchangeBankAccount,
NodeHttpLib, NodeHttpLib,
openPromise, openPromise,
OperationFailedError, TalerError,
WalletCoreApiClient, WalletCoreApiClient,
} from "@gnu-taler/taler-wallet-core"; } from "@gnu-taler/taler-wallet-core";
import { import {
@ -227,19 +227,19 @@ export class GlobalTestState {
this.servers = []; this.servers = [];
} }
async assertThrowsOperationErrorAsync( async assertThrowsTalerErrorAsync(
block: () => Promise<void>, block: () => Promise<void>,
): Promise<OperationFailedError> { ): Promise<TalerError> {
try { try {
await block(); await block();
} catch (e) { } catch (e) {
if (e instanceof OperationFailedError) { if (e instanceof TalerError) {
return e; return e;
} }
throw Error(`expected OperationFailedError to be thrown, but got ${e}`); throw Error(`expected TalerError to be thrown, but got ${e}`);
} }
throw Error( throw Error(
`expected OperationFailedError to be thrown, but block finished without throwing`, `expected TalerError to be thrown, but block finished without throwing`,
); );
} }
@ -1904,7 +1904,7 @@ export class WalletCli {
throw new Error("wallet CLI did not return a proper JSON response"); throw new Error("wallet CLI did not return a proper JSON response");
} }
if (ar.type === "error") { if (ar.type === "error") {
throw new OperationFailedError(ar.error); throw TalerError.fromUncheckedDetail(ar.error);
} }
return ar.result; return ar.result;
}, },

View File

@ -49,14 +49,13 @@ import {
import { import {
NodeHttpLib, NodeHttpLib,
getDefaultNodeWallet, getDefaultNodeWallet,
OperationFailedAndReportedError,
OperationFailedError,
NodeThreadCryptoWorkerFactory, NodeThreadCryptoWorkerFactory,
CryptoApi, CryptoApi,
walletCoreDebugFlags, walletCoreDebugFlags,
WalletApiOperation, WalletApiOperation,
WalletCoreApiClient, WalletCoreApiClient,
Wallet, Wallet,
getErrorDetailFromException,
} from "@gnu-taler/taler-wallet-core"; } from "@gnu-taler/taler-wallet-core";
import { lintExchangeDeployment } from "./lint.js"; import { lintExchangeDeployment } from "./lint.js";
import { runBench1 } from "./bench1.js"; import { runBench1 } from "./bench1.js";
@ -206,18 +205,12 @@ async function withWallet<T>(
const ret = await f(w); const ret = await f(w);
return ret; return ret;
} catch (e) { } catch (e) {
if ( const ed = getErrorDetailFromException(e);
e instanceof OperationFailedAndReportedError || console.error("Operation failed: " + ed.message);
e instanceof OperationFailedError
) {
console.error("Operation failed: " + e.message);
console.error( console.error(
"Error details:", "Error details:",
JSON.stringify(e.operationError, undefined, 2), JSON.stringify(ed.operationError, undefined, 2),
); );
} else {
console.error("caught unhandled exception (bug?):", e);
}
process.exit(1); process.exit(1);
} finally { } finally {
logger.info("operation with wallet finished, stopping"); logger.info("operation with wallet finished, stopping");

View File

@ -32,7 +32,6 @@ import {
BankApi, BankApi,
BankAccessApi, BankAccessApi,
CreditDebitIndicator, CreditDebitIndicator,
OperationFailedError,
} from "@gnu-taler/taler-wallet-core"; } from "@gnu-taler/taler-wallet-core";
/** /**
@ -104,10 +103,10 @@ export async function runBankApiTest(t: GlobalTestState) {
// Make sure that registering twice results in a 409 Conflict // Make sure that registering twice results in a 409 Conflict
{ {
const e = await t.assertThrowsAsync(async () => { const e = await t.assertThrowsTalerErrorAsync(async () => {
await BankApi.registerAccount(bank, "user1", "pw1"); await BankApi.registerAccount(bank, "user1", "pw1");
}); });
t.assertTrue(e.details.httpStatusCode === 409); t.assertTrue(e.errorDetail.httpStatusCode === 409);
} }
let balResp = await BankAccessApi.getAccountBalance(bank, bankUser); let balResp = await BankAccessApi.getAccountBalance(bank, bankUser);

View File

@ -20,25 +20,21 @@
import { import {
PreparePayResultType, PreparePayResultType,
TalerErrorCode, TalerErrorCode,
TalerErrorDetails, TalerErrorDetail,
TransactionType,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { import { WalletApiOperation } from "@gnu-taler/taler-wallet-core";
WalletApiOperation,
} from "@gnu-taler/taler-wallet-core";
import { makeEventId } from "@gnu-taler/taler-wallet-core"; import { makeEventId } from "@gnu-taler/taler-wallet-core";
import { GlobalTestState, MerchantPrivateApi } from "../harness/harness.js"; import { GlobalTestState, MerchantPrivateApi } from "../harness/harness.js";
import { createSimpleTestkudosEnvironment, withdrawViaBank } from "../harness/helpers.js"; import {
createSimpleTestkudosEnvironment,
withdrawViaBank,
} from "../harness/helpers.js";
export async function runDenomUnofferedTest(t: GlobalTestState) { export async function runDenomUnofferedTest(t: GlobalTestState) {
// Set up test environment // Set up test environment
const { const { wallet, bank, exchange, merchant } =
wallet, await createSimpleTestkudosEnvironment(t);
bank,
exchange,
merchant,
} = await createSimpleTestkudosEnvironment(t);
// Withdraw digital cash into the wallet. // Withdraw digital cash into the wallet.
@ -95,19 +91,23 @@ export async function runDenomUnofferedTest(t: GlobalTestState) {
preparePayResult.status === PreparePayResultType.PaymentPossible, preparePayResult.status === PreparePayResultType.PaymentPossible,
); );
const exc = await t.assertThrowsAsync(async () => { const exc = await t.assertThrowsTalerErrorAsync(async () => {
await wallet.client.call(WalletApiOperation.ConfirmPay, { await wallet.client.call(WalletApiOperation.ConfirmPay, {
proposalId: preparePayResult.proposalId, proposalId: preparePayResult.proposalId,
}); });
}); });
const errorDetails: TalerErrorDetails = exc.operationError; t.assertTrue(
exc.hasErrorCode(TalerErrorCode.WALLET_PENDING_OPERATION_FAILED),
);
// FIXME: We might want a more specific error code here! // FIXME: We might want a more specific error code here!
t.assertDeepEqual( t.assertDeepEqual(
errorDetails.code, exc.errorDetail.innerError.code,
TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR, TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR,
); );
const merchantErrorCode = (errorDetails.details as any).errorResponse.code; const merchantErrorCode = (exc.errorDetail.innerError.errorResponse as any)
.code;
t.assertDeepEqual( t.assertDeepEqual(
merchantErrorCode, merchantErrorCode,
TalerErrorCode.MERCHANT_POST_ORDERS_ID_PAY_DENOMINATION_KEY_NOT_FOUND, TalerErrorCode.MERCHANT_POST_ORDERS_ID_PAY_DENOMINATION_KEY_NOT_FOUND,

View File

@ -181,16 +181,22 @@ export async function runExchangeManagementTest(t: GlobalTestState) {
}, },
}); });
const err1 = await t.assertThrowsOperationErrorAsync(async () => { const err1 = await t.assertThrowsTalerErrorAsync(async () => {
await wallet.client.call(WalletApiOperation.AddExchange, { await wallet.client.call(WalletApiOperation.AddExchange, {
exchangeBaseUrl: faultyExchange.baseUrl, exchangeBaseUrl: faultyExchange.baseUrl,
}); });
}); });
// Updating the exchange from the base URL is technically a pending operation
// and it will be retried later.
t.assertTrue(
err1.hasErrorCode(TalerErrorCode.WALLET_PENDING_OPERATION_FAILED),
);
// Response is malformed, since it didn't even contain a version code // Response is malformed, since it didn't even contain a version code
// in a format the wallet can understand. // in a format the wallet can understand.
t.assertTrue( t.assertTrue(
err1.operationError.code === err1.errorDetail.innerError.code ===
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
); );
@ -223,14 +229,18 @@ export async function runExchangeManagementTest(t: GlobalTestState) {
}, },
}); });
const err2 = await t.assertThrowsOperationErrorAsync(async () => { const err2 = await t.assertThrowsTalerErrorAsync(async () => {
await wallet.client.call(WalletApiOperation.AddExchange, { await wallet.client.call(WalletApiOperation.AddExchange, {
exchangeBaseUrl: faultyExchange.baseUrl, exchangeBaseUrl: faultyExchange.baseUrl,
}); });
}); });
t.assertTrue( t.assertTrue(
err2.operationError.code === err2.hasErrorCode(TalerErrorCode.WALLET_PENDING_OPERATION_FAILED),
);
t.assertTrue(
err2.errorDetail.innerError.code ===
TalerErrorCode.WALLET_EXCHANGE_PROTOCOL_VERSION_INCOMPATIBLE, TalerErrorCode.WALLET_EXCHANGE_PROTOCOL_VERSION_INCOMPATIBLE,
); );

View File

@ -123,7 +123,7 @@ export async function runPayAbortTest(t: GlobalTestState) {
}, },
}); });
await t.assertThrowsOperationErrorAsync(async () => { await t.assertThrowsTalerErrorAsync(async () => {
await wallet.client.call(WalletApiOperation.ConfirmPay, { await wallet.client.call(WalletApiOperation.ConfirmPay, {
proposalId: preparePayResult.proposalId, proposalId: preparePayResult.proposalId,
}); });

View File

@ -17,8 +17,15 @@
/** /**
* Imports. * Imports.
*/ */
import { GlobalTestState, MerchantPrivateApi, WalletCli } from "../harness/harness.js"; import {
import { createSimpleTestkudosEnvironment, withdrawViaBank } from "../harness/helpers.js"; GlobalTestState,
MerchantPrivateApi,
WalletCli,
} from "../harness/harness.js";
import {
createSimpleTestkudosEnvironment,
withdrawViaBank,
} from "../harness/helpers.js";
import { PreparePayResultType } from "@gnu-taler/taler-util"; import { PreparePayResultType } from "@gnu-taler/taler-util";
import { TalerErrorCode } from "@gnu-taler/taler-util"; import { TalerErrorCode } from "@gnu-taler/taler-util";
import { WalletApiOperation } from "@gnu-taler/taler-wallet-core"; import { WalletApiOperation } from "@gnu-taler/taler-wallet-core";
@ -29,12 +36,8 @@ import { WalletApiOperation } from "@gnu-taler/taler-wallet-core";
export async function runPaymentClaimTest(t: GlobalTestState) { export async function runPaymentClaimTest(t: GlobalTestState) {
// Set up test environment // Set up test environment
const { const { wallet, bank, exchange, merchant } =
wallet, await createSimpleTestkudosEnvironment(t);
bank,
exchange,
merchant,
} = await createSimpleTestkudosEnvironment(t);
const walletTwo = new WalletCli(t, "two"); const walletTwo = new WalletCli(t, "two");
@ -73,7 +76,7 @@ export async function runPaymentClaimTest(t: GlobalTestState) {
preparePayResult.status === PreparePayResultType.PaymentPossible, preparePayResult.status === PreparePayResultType.PaymentPossible,
); );
t.assertThrowsOperationErrorAsync(async () => { t.assertThrowsTalerErrorAsync(async () => {
await walletTwo.client.call(WalletApiOperation.PreparePayForUri, { await walletTwo.client.call(WalletApiOperation.PreparePayForUri, {
talerPayUri, talerPayUri,
}); });
@ -93,14 +96,19 @@ export async function runPaymentClaimTest(t: GlobalTestState) {
walletTwo.deleteDatabase(); walletTwo.deleteDatabase();
const err = await t.assertThrowsOperationErrorAsync(async () => { const err = await t.assertThrowsTalerErrorAsync(async () => {
await walletTwo.client.call(WalletApiOperation.PreparePayForUri, { await walletTwo.client.call(WalletApiOperation.PreparePayForUri, {
talerPayUri, talerPayUri,
}); });
}); });
t.assertTrue( t.assertTrue(
err.operationError.code === TalerErrorCode.WALLET_ORDER_ALREADY_CLAIMED, err.hasErrorCode(TalerErrorCode.WALLET_PENDING_OPERATION_FAILED),
);
t.assertTrue(
err.errorDetail.innerError.code ===
TalerErrorCode.WALLET_ORDER_ALREADY_CLAIMED,
); );
await t.shutdown(); await t.shutdown();

View File

@ -32,7 +32,7 @@ import {
ConfirmPayResultType, ConfirmPayResultType,
PreparePayResultType, PreparePayResultType,
TalerErrorCode, TalerErrorCode,
TalerErrorDetails, TalerErrorDetail,
URL, URL,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { WalletApiOperation } from "@gnu-taler/taler-wallet-core"; import { WalletApiOperation } from "@gnu-taler/taler-wallet-core";
@ -135,11 +135,9 @@ export async function runPaymentTransientTest(t: GlobalTestState) {
} }
faultInjected = true; faultInjected = true;
console.log("injecting pay fault"); console.log("injecting pay fault");
const err: TalerErrorDetails = { const err: TalerErrorDetail = {
code: TalerErrorCode.GENERIC_DB_COMMIT_FAILED, code: TalerErrorCode.GENERIC_DB_COMMIT_FAILED,
details: {}, hint: "something went wrong",
hint: "huh",
message: "something went wrong",
}; };
ctx.responseBody = Buffer.from(JSON.stringify(err)); ctx.responseBody = Buffer.from(JSON.stringify(err));
ctx.statusCode = 500; ctx.statusCode = 500;

View File

@ -26,9 +26,9 @@ import {
findDenomOrThrow, findDenomOrThrow,
generateReserveKeypair, generateReserveKeypair,
NodeHttpLib, NodeHttpLib,
OperationFailedError,
refreshCoin, refreshCoin,
SynchronousCryptoWorkerFactory, SynchronousCryptoWorkerFactory,
TalerError,
topupReserveWithDemobank, topupReserveWithDemobank,
withdrawCoin, withdrawCoin,
} from "@gnu-taler/taler-wallet-core"; } from "@gnu-taler/taler-wallet-core";
@ -95,9 +95,9 @@ export async function runWalletDblessTest(t: GlobalTestState) {
newDenoms: refreshDenoms, newDenoms: refreshDenoms,
}); });
} catch (e) { } catch (e) {
if (e instanceof OperationFailedError) { if (e instanceof TalerError) {
console.log(e); console.log(e);
console.log(j2s(e.operationError)); console.log(j2s(e.errorDetail));
} else { } else {
console.log(e); console.log(e);
} }

View File

@ -63,7 +63,7 @@ export async function runWithdrawalAbortBankTest(t: GlobalTestState) {
// //
// WHY ?! // WHY ?!
// //
const e = await t.assertThrowsOperationErrorAsync(async () => { const e = await t.assertThrowsTalerErrorAsync(async () => {
await wallet.client.call( await wallet.client.call(
WalletApiOperation.AcceptBankIntegratedWithdrawal, WalletApiOperation.AcceptBankIntegratedWithdrawal,
{ {
@ -73,7 +73,7 @@ export async function runWithdrawalAbortBankTest(t: GlobalTestState) {
); );
}); });
t.assertDeepEqual( t.assertDeepEqual(
e.operationError.code, e.errorDetail.code,
TalerErrorCode.WALLET_WITHDRAWAL_OPERATION_ABORTED_BY_BANK, TalerErrorCode.WALLET_WITHDRAWAL_OPERATION_ABORTED_BY_BANK,
); );

View File

@ -31,7 +31,9 @@ import {
getRandomBytes, getRandomBytes,
j2s, j2s,
Logger, Logger,
TalerErrorCode,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { TalerError } from "./errors.js";
import { import {
HttpRequestLibrary, HttpRequestLibrary,
readSuccessResponseJsonOrErrorCode, readSuccessResponseJsonOrErrorCode,
@ -104,15 +106,20 @@ export namespace BankApi {
let paytoUri = `payto://x-taler-bank/localhost/${username}`; let paytoUri = `payto://x-taler-bank/localhost/${username}`;
if (resp.status !== 200 && resp.status !== 202) { if (resp.status !== 200 && resp.status !== 202) {
logger.error(`${j2s(await resp.json())}`); logger.error(`${j2s(await resp.json())}`);
throw new Error(); throw TalerError.fromDetail(
TalerErrorCode.GENERIC_UNEXPECTED_REQUEST_ERROR,
{
httpStatusCode: resp.status,
},
);
} }
const respJson = await readSuccessResponseJsonOrThrow(resp, codecForAny()); try {
// Pybank has no body, thus this might throw.
const respJson = await resp.json();
// LibEuFin demobank returns payto URI in response // LibEuFin demobank returns payto URI in response
if (respJson.paytoUri) { if (respJson.paytoUri) {
paytoUri = respJson.paytoUri; paytoUri = respJson.paytoUri;
} }
try {
const respJson = await resp.json();
} catch (e) {} } catch (e) {}
return { return {
password, password,

View File

@ -35,7 +35,7 @@ import {
MerchantInfo, MerchantInfo,
Product, Product,
RefreshReason, RefreshReason,
TalerErrorDetails, TalerErrorDetail,
UnblindedSignature, UnblindedSignature,
CoinEnvelope, CoinEnvelope,
TalerProtocolTimestamp, TalerProtocolTimestamp,
@ -229,7 +229,7 @@ export interface ReserveRecord {
* Last error that happened in a reserve operation * Last error that happened in a reserve operation
* (either talking to the bank or the exchange). * (either talking to the bank or the exchange).
*/ */
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
/** /**
@ -545,7 +545,7 @@ export interface ExchangeRecord {
* Last error (if any) for fetching updated information about the * Last error (if any) for fetching updated information about the
* exchange. * exchange.
*/ */
lastError?: TalerErrorDetails; lastError?: TalerErrorDetail;
/** /**
* Retry status for fetching updated information about the exchange. * Retry status for fetching updated information about the exchange.
@ -580,7 +580,7 @@ export interface PlanchetRecord {
withdrawalDone: boolean; withdrawalDone: boolean;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
/** /**
* Public key of the reserve that this planchet * Public key of the reserve that this planchet
@ -820,14 +820,14 @@ export interface ProposalRecord {
*/ */
retryInfo?: RetryInfo; retryInfo?: RetryInfo;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
/** /**
* Status of a tip we got from a merchant. * Status of a tip we got from a merchant.
*/ */
export interface TipRecord { export interface TipRecord {
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
/** /**
* Has the user accepted the tip? Only after the tip has been accepted coins * Has the user accepted the tip? Only after the tip has been accepted coins
@ -922,9 +922,9 @@ export interface RefreshGroupRecord {
*/ */
retryInfo: RetryInfo; retryInfo: RetryInfo;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
lastErrorPerCoin: { [coinIndex: number]: TalerErrorDetails }; lastErrorPerCoin: { [coinIndex: number]: TalerErrorDetail };
/** /**
* Unique, randomly generated identifier for this group of * Unique, randomly generated identifier for this group of
@ -1256,7 +1256,7 @@ export interface PurchaseRecord {
payRetryInfo?: RetryInfo; payRetryInfo?: RetryInfo;
lastPayError: TalerErrorDetails | undefined; lastPayError: TalerErrorDetail | undefined;
/** /**
* Retry information for querying the refund status with the merchant. * Retry information for querying the refund status with the merchant.
@ -1266,7 +1266,7 @@ export interface PurchaseRecord {
/** /**
* Last error (or undefined) for querying the refund status with the merchant. * Last error (or undefined) for querying the refund status with the merchant.
*/ */
lastRefundStatusError: TalerErrorDetails | undefined; lastRefundStatusError: TalerErrorDetail | undefined;
/** /**
* Continue querying the refund status until this deadline has expired. * Continue querying the refund status until this deadline has expired.
@ -1400,7 +1400,7 @@ export interface WithdrawalGroupRecord {
*/ */
retryInfo: RetryInfo; retryInfo: RetryInfo;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
export interface BankWithdrawUriRecord { export interface BankWithdrawUriRecord {
@ -1465,7 +1465,7 @@ export interface RecoupGroupRecord {
/** /**
* Last error that occurred, if any. * Last error that occurred, if any.
*/ */
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
export enum BackupProviderStateTag { export enum BackupProviderStateTag {
@ -1485,7 +1485,7 @@ export type BackupProviderState =
| { | {
tag: BackupProviderStateTag.Retrying; tag: BackupProviderStateTag.Retrying;
retryInfo: RetryInfo; retryInfo: RetryInfo;
lastError?: TalerErrorDetails; lastError?: TalerErrorDetail;
}; };
export interface BackupProviderTerms { export interface BackupProviderTerms {
@ -1598,7 +1598,7 @@ export interface DepositGroupRecord {
operationStatus: OperationStatus; operationStatus: OperationStatus;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
/** /**
* Retry info. * Retry info.

View File

@ -23,94 +23,127 @@
/** /**
* Imports. * Imports.
*/ */
import { TalerErrorCode, TalerErrorDetails } from "@gnu-taler/taler-util"; import {
TalerErrorCode,
TalerErrorDetail,
TransactionType,
} from "@gnu-taler/taler-util";
/** export interface DetailsMap {
* This exception is there to let the caller know that an error happened, [TalerErrorCode.WALLET_PENDING_OPERATION_FAILED]: {
* but the error has already been reported by writing it to the database. innerError: TalerErrorDetail;
*/ transactionId?: string;
export class OperationFailedAndReportedError extends Error {
static fromCode(
ec: TalerErrorCode,
message: string,
details: Record<string, unknown>,
): OperationFailedAndReportedError {
return new OperationFailedAndReportedError(
makeErrorDetails(ec, message, details),
);
}
constructor(public operationError: TalerErrorDetails) {
super(operationError.message);
// Set the prototype explicitly.
Object.setPrototypeOf(this, OperationFailedAndReportedError.prototype);
}
}
/**
* This exception is thrown when an error occurred and the caller is
* responsible for recording the failure in the database.
*/
export class OperationFailedError extends Error {
static fromCode(
ec: TalerErrorCode,
message: string,
details: Record<string, unknown>,
): OperationFailedError {
return new OperationFailedError(makeErrorDetails(ec, message, details));
}
constructor(public operationError: TalerErrorDetails) {
super(operationError.message);
// Set the prototype explicitly.
Object.setPrototypeOf(this, OperationFailedError.prototype);
}
}
export function makeErrorDetails(
ec: TalerErrorCode,
message: string,
details: Record<string, unknown>,
): TalerErrorDetails {
return {
code: ec,
hint: `Error: ${TalerErrorCode[ec]}`,
details: details,
message,
}; };
[TalerErrorCode.WALLET_EXCHANGE_DENOMINATIONS_INSUFFICIENT]: {
exchangeBaseUrl: string;
};
[TalerErrorCode.WALLET_EXCHANGE_PROTOCOL_VERSION_INCOMPATIBLE]: {
exchangeProtocolVersion: string;
walletProtocolVersion: string;
};
[TalerErrorCode.WALLET_WITHDRAWAL_OPERATION_ABORTED_BY_BANK]: {};
[TalerErrorCode.WALLET_TIPPING_COIN_SIGNATURE_INVALID]: {};
[TalerErrorCode.WALLET_ORDER_ALREADY_CLAIMED]: {
orderId: string;
claimUrl: string;
};
[TalerErrorCode.WALLET_CONTRACT_TERMS_MALFORMED]: {};
[TalerErrorCode.WALLET_CONTRACT_TERMS_SIGNATURE_INVALID]: {
merchantPub: string;
orderId: string;
};
[TalerErrorCode.WALLET_CONTRACT_TERMS_BASE_URL_MISMATCH]: {
baseUrlForDownload: string;
baseUrlFromContractTerms: string;
};
[TalerErrorCode.WALLET_INVALID_TALER_PAY_URI]: {
talerPayUri: string;
};
[TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR]: {};
[TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION]: {};
[TalerErrorCode.WALLET_BANK_INTEGRATION_PROTOCOL_VERSION_INCOMPATIBLE]: {};
[TalerErrorCode.WALLET_CORE_API_OPERATION_UNKNOWN]: {};
[TalerErrorCode.WALLET_HTTP_REQUEST_THROTTLED]: {};
[TalerErrorCode.WALLET_NETWORK_ERROR]: {};
[TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE]: {};
[TalerErrorCode.WALLET_EXCHANGE_COIN_SIGNATURE_INVALID]: {};
[TalerErrorCode.WALLET_WITHDRAWAL_GROUP_INCOMPLETE]: {};
[TalerErrorCode.WALLET_CORE_NOT_AVAILABLE]: {};
[TalerErrorCode.GENERIC_UNEXPECTED_REQUEST_ERROR]: {};
}
type ErrBody<Y> = Y extends keyof DetailsMap ? DetailsMap[Y] : never;
export function makeErrorDetail<C extends TalerErrorCode>(
code: C,
detail: ErrBody<C>,
hint?: string,
): TalerErrorDetail {
// FIXME: include default hint?
return { code, hint, ...detail };
}
export function makePendingOperationFailedError(
innerError: TalerErrorDetail,
tag: TransactionType,
uid: string,
): TalerError {
return TalerError.fromDetail(TalerErrorCode.WALLET_PENDING_OPERATION_FAILED, {
innerError,
transactionId: `${tag}:${uid}`,
});
}
export class TalerError<T = any> extends Error {
errorDetail: TalerErrorDetail & T;
private constructor(d: TalerErrorDetail & T) {
super();
this.errorDetail = d;
Object.setPrototypeOf(this, TalerError.prototype);
}
static fromDetail<C extends TalerErrorCode>(
code: C,
detail: ErrBody<C>,
hint?: string,
): TalerError {
// FIXME: include default hint?
return new TalerError<unknown>({ code, hint, ...detail });
}
static fromUncheckedDetail(d: TalerErrorDetail): TalerError {
return new TalerError<unknown>({ ...d });
}
static fromException(e: any): TalerError {
const errDetail = getErrorDetailFromException(e);
return new TalerError(errDetail);
}
hasErrorCode<C extends keyof DetailsMap>(
code: C,
): this is TalerError<DetailsMap[C]> {
return this.errorDetail.code === code;
}
} }
/** /**
* Run an operation and call the onOpError callback * Convert an exception (or anything that was thrown) into
* when there was an exception or operation error that must be reported. * a TalerErrorDetail object.
* The cause will be re-thrown to the caller.
*/ */
export async function guardOperationException<T>( export function getErrorDetailFromException(e: any): TalerErrorDetail {
op: () => Promise<T>, if (e instanceof TalerError) {
onOpError: (e: TalerErrorDetails) => Promise<void>, return e.errorDetail;
): Promise<T> {
try {
return await op();
} catch (e: any) {
if (e instanceof OperationFailedAndReportedError) {
throw e;
}
if (e instanceof OperationFailedError) {
await onOpError(e.operationError);
throw new OperationFailedAndReportedError(e.operationError);
} }
if (e instanceof Error) { if (e instanceof Error) {
const opErr = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION, TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION,
`unexpected exception (message: ${e.message})`,
{ {
stack: e.stack, stack: e.stack,
}, },
`unexpected exception (message: ${e.message})`,
); );
await onOpError(opErr); return err;
throw new OperationFailedAndReportedError(opErr);
} }
// Something was thrown that is not even an exception! // Something was thrown that is not even an exception!
// Try to stringify it. // Try to stringify it.
@ -121,12 +154,39 @@ export async function guardOperationException<T>(
// Something went horribly wrong. // Something went horribly wrong.
excString = "can't stringify exception"; excString = "can't stringify exception";
} }
const opErr = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION, TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION,
`unexpected exception (not an exception, ${excString})`,
{}, {},
`unexpected exception (not an exception, ${excString})`,
); );
return err;
}
/**
* 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<T>(
op: () => Promise<T>,
onOpError: (e: TalerErrorDetail) => Promise<void>,
): Promise<T> {
try {
return await op();
} catch (e: any) {
if (
e instanceof TalerError &&
e.hasErrorCode(TalerErrorCode.WALLET_PENDING_OPERATION_FAILED)
) {
throw e;
}
const opErr = getErrorDetailFromException(e);
await onOpError(opErr); await onOpError(opErr);
throw new OperationFailedAndReportedError(opErr); throw TalerError.fromDetail(
TalerErrorCode.WALLET_PENDING_OPERATION_FAILED,
{
innerError: e.errorDetail,
},
);
} }
} }

View File

@ -27,7 +27,7 @@ import {
} from "../util/http.js"; } from "../util/http.js";
import { RequestThrottler } from "@gnu-taler/taler-util"; import { RequestThrottler } from "@gnu-taler/taler-util";
import Axios, { AxiosResponse } from "axios"; import Axios, { AxiosResponse } from "axios";
import { OperationFailedError, makeErrorDetails } from "../errors.js"; import { TalerError } from "../errors.js";
import { Logger, bytesToString } from "@gnu-taler/taler-util"; import { Logger, bytesToString } from "@gnu-taler/taler-util";
import { TalerErrorCode, URL } from "@gnu-taler/taler-util"; import { TalerErrorCode, URL } from "@gnu-taler/taler-util";
@ -55,14 +55,14 @@ export class NodeHttpLib implements HttpRequestLibrary {
const parsedUrl = new URL(url); const parsedUrl = new URL(url);
if (this.throttlingEnabled && this.throttle.applyThrottle(url)) { if (this.throttlingEnabled && this.throttle.applyThrottle(url)) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_HTTP_REQUEST_THROTTLED, TalerErrorCode.WALLET_HTTP_REQUEST_THROTTLED,
`request to origin ${parsedUrl.origin} was throttled`,
{ {
requestMethod: method, requestMethod: method,
requestUrl: url, requestUrl: url,
throttleStats: this.throttle.getThrottleStats(url), throttleStats: this.throttle.getThrottleStats(url),
}, },
`request to origin ${parsedUrl.origin} was throttled`,
); );
} }
let timeout: number | undefined; let timeout: number | undefined;
@ -83,13 +83,13 @@ export class NodeHttpLib implements HttpRequestLibrary {
maxRedirects: 0, maxRedirects: 0,
}); });
} catch (e: any) { } catch (e: any) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_NETWORK_ERROR, TalerErrorCode.WALLET_NETWORK_ERROR,
`${e.message}`,
{ {
requestUrl: url, requestUrl: url,
requestMethod: method, requestMethod: method,
}, },
`${e.message}`,
); );
} }
@ -105,30 +105,26 @@ export class NodeHttpLib implements HttpRequestLibrary {
responseJson = JSON.parse(respText); responseJson = JSON.parse(respText);
} catch (e) { } catch (e) {
logger.trace(`invalid json: '${resp.data}'`); logger.trace(`invalid json: '${resp.data}'`);
throw new OperationFailedError( throw TalerError.fromDetail(
makeErrorDetails(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"invalid JSON",
{ {
httpStatusCode: resp.status, httpStatusCode: resp.status,
requestUrl: url, requestUrl: url,
requestMethod: method, requestMethod: method,
}, },
), "Could not parse response body as JSON",
); );
} }
if (responseJson === null || typeof responseJson !== "object") { if (responseJson === null || typeof responseJson !== "object") {
logger.trace(`invalid json (not an object): '${respText}'`); logger.trace(`invalid json (not an object): '${respText}'`);
throw new OperationFailedError( throw TalerError.fromDetail(
makeErrorDetails(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"invalid JSON",
{ {
httpStatusCode: resp.status, httpStatusCode: resp.status,
requestUrl: url, requestUrl: url,
requestMethod: method, requestMethod: method,
}, },
), `invalid JSON`,
); );
} }
return responseJson; return responseJson;

View File

@ -48,7 +48,7 @@ import {
PreparePayResultType, PreparePayResultType,
RecoveryLoadRequest, RecoveryLoadRequest,
RecoveryMergeStrategy, RecoveryMergeStrategy,
TalerErrorDetails, TalerErrorDetail,
AbsoluteTime, AbsoluteTime,
URL, URL,
WalletBackupContentV1, WalletBackupContentV1,
@ -464,7 +464,7 @@ async function incrementBackupRetryInTx(
backupProviders: typeof WalletStoresV1.backupProviders; backupProviders: typeof WalletStoresV1.backupProviders;
}>, }>,
backupProviderBaseUrl: string, backupProviderBaseUrl: string,
err: TalerErrorDetails | undefined, err: TalerErrorDetail | undefined,
): Promise<void> { ): Promise<void> {
const pr = await tx.backupProviders.get(backupProviderBaseUrl); const pr = await tx.backupProviders.get(backupProviderBaseUrl);
if (!pr) { if (!pr) {
@ -487,7 +487,7 @@ async function incrementBackupRetryInTx(
async function incrementBackupRetry( async function incrementBackupRetry(
ws: InternalWalletState, ws: InternalWalletState,
backupProviderBaseUrl: string, backupProviderBaseUrl: string,
err: TalerErrorDetails | undefined, err: TalerErrorDetail | undefined,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ backupProviders: x.backupProviders })) .mktx((x) => ({ backupProviders: x.backupProviders }))
@ -509,7 +509,7 @@ export async function processBackupForProvider(
throw Error("unknown backup provider"); throw Error("unknown backup provider");
} }
const onOpErr = (err: TalerErrorDetails): Promise<void> => const onOpErr = (err: TalerErrorDetail): Promise<void> =>
incrementBackupRetry(ws, backupProviderBaseUrl, err); incrementBackupRetry(ws, backupProviderBaseUrl, err);
const run = async () => { const run = async () => {
@ -700,7 +700,7 @@ export interface ProviderInfo {
/** /**
* Last communication issue with the provider. * Last communication issue with the provider.
*/ */
lastError?: TalerErrorDetails; lastError?: TalerErrorDetail;
lastSuccessfulBackupTimestamp?: TalerProtocolTimestamp; lastSuccessfulBackupTimestamp?: TalerProtocolTimestamp;
lastAttemptedBackupTimestamp?: TalerProtocolTimestamp; lastAttemptedBackupTimestamp?: TalerProtocolTimestamp;
paymentProposalIds: string[]; paymentProposalIds: string[];

View File

@ -36,7 +36,7 @@ import {
Logger, Logger,
NotificationType, NotificationType,
parsePaytoUri, parsePaytoUri,
TalerErrorDetails, TalerErrorDetail,
TalerProtocolTimestamp, TalerProtocolTimestamp,
TrackDepositGroupRequest, TrackDepositGroupRequest,
TrackDepositGroupResponse, TrackDepositGroupResponse,
@ -83,7 +83,7 @@ async function resetDepositGroupRetry(
async function incrementDepositRetry( async function incrementDepositRetry(
ws: InternalWalletState, ws: InternalWalletState,
depositGroupId: string, depositGroupId: string,
err: TalerErrorDetails | undefined, err: TalerErrorDetail | undefined,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ depositGroups: x.depositGroups })) .mktx((x) => ({ depositGroups: x.depositGroups }))
@ -111,7 +111,7 @@ export async function processDepositGroup(
forceNow = false, forceNow = false,
): Promise<void> { ): Promise<void> {
await ws.memoProcessDeposit.memo(depositGroupId, async () => { await ws.memoProcessDeposit.memo(depositGroupId, async () => {
const onOpErr = (e: TalerErrorDetails): Promise<void> => const onOpErr = (e: TalerErrorDetail): Promise<void> =>
incrementDepositRetry(ws, depositGroupId, e); incrementDepositRetry(ws, depositGroupId, e);
return await guardOperationException( return await guardOperationException(
async () => await processDepositGroupImpl(ws, depositGroupId, forceNow), async () => await processDepositGroupImpl(ws, depositGroupId, forceNow),

View File

@ -34,7 +34,7 @@ import {
Recoup, Recoup,
TalerErrorCode, TalerErrorCode,
URL, URL,
TalerErrorDetails, TalerErrorDetail,
AbsoluteTime, AbsoluteTime,
hashDenomPub, hashDenomPub,
LibtoolVersion, LibtoolVersion,
@ -64,11 +64,7 @@ import {
} from "../util/http.js"; } from "../util/http.js";
import { DbAccess, GetReadOnlyAccess } from "../util/query.js"; import { DbAccess, GetReadOnlyAccess } from "../util/query.js";
import { initRetryInfo, updateRetryInfoTimeout } from "../util/retries.js"; import { initRetryInfo, updateRetryInfoTimeout } from "../util/retries.js";
import { import { guardOperationException, TalerError } from "../errors.js";
guardOperationException,
makeErrorDetails,
OperationFailedError,
} from "../errors.js";
import { InternalWalletState, TrustInfo } from "../common.js"; import { InternalWalletState, TrustInfo } from "../common.js";
import { import {
WALLET_CACHE_BREAKER_CLIENT_VERSION, WALLET_CACHE_BREAKER_CLIENT_VERSION,
@ -112,7 +108,7 @@ function denominationRecordFromKeys(
async function handleExchangeUpdateError( async function handleExchangeUpdateError(
ws: InternalWalletState, ws: InternalWalletState,
baseUrl: string, baseUrl: string,
err: TalerErrorDetails, err: TalerErrorDetail,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ exchanges: x.exchanges })) .mktx((x) => ({ exchanges: x.exchanges }))
@ -353,7 +349,7 @@ export async function updateExchangeFromUrl(
exchange: ExchangeRecord; exchange: ExchangeRecord;
exchangeDetails: ExchangeDetailsRecord; exchangeDetails: ExchangeDetailsRecord;
}> { }> {
const onOpErr = (e: TalerErrorDetails): Promise<void> => const onOpErr = (e: TalerErrorDetail): Promise<void> =>
handleExchangeUpdateError(ws, baseUrl, e); handleExchangeUpdateError(ws, baseUrl, e);
return await guardOperationException( return await guardOperationException(
() => updateExchangeFromUrlImpl(ws, baseUrl, acceptedFormat, forceNow), () => updateExchangeFromUrlImpl(ws, baseUrl, acceptedFormat, forceNow),
@ -429,14 +425,13 @@ async function downloadExchangeKeysInfo(
logger.info("received /keys response"); logger.info("received /keys response");
if (exchangeKeysJsonUnchecked.denoms.length === 0) { if (exchangeKeysJsonUnchecked.denoms.length === 0) {
const opErr = makeErrorDetails( throw TalerError.fromDetail(
TalerErrorCode.WALLET_EXCHANGE_DENOMINATIONS_INSUFFICIENT, TalerErrorCode.WALLET_EXCHANGE_DENOMINATIONS_INSUFFICIENT,
"exchange doesn't offer any denominations",
{ {
exchangeBaseUrl: baseUrl, exchangeBaseUrl: baseUrl,
}, },
"exchange doesn't offer any denominations",
); );
throw new OperationFailedError(opErr);
} }
const protocolVersion = exchangeKeysJsonUnchecked.version; const protocolVersion = exchangeKeysJsonUnchecked.version;
@ -446,15 +441,14 @@ async function downloadExchangeKeysInfo(
protocolVersion, protocolVersion,
); );
if (versionRes?.compatible != true) { if (versionRes?.compatible != true) {
const opErr = makeErrorDetails( throw TalerError.fromDetail(
TalerErrorCode.WALLET_EXCHANGE_PROTOCOL_VERSION_INCOMPATIBLE, TalerErrorCode.WALLET_EXCHANGE_PROTOCOL_VERSION_INCOMPATIBLE,
"exchange protocol version not compatible with wallet",
{ {
exchangeProtocolVersion: protocolVersion, exchangeProtocolVersion: protocolVersion,
walletProtocolVersion: WALLET_EXCHANGE_PROTOCOL_VERSION, walletProtocolVersion: WALLET_EXCHANGE_PROTOCOL_VERSION,
}, },
"exchange protocol version not compatible with wallet",
); );
throw new OperationFailedError(opErr);
} }
const currency = Amounts.parseOrThrow( const currency = Amounts.parseOrThrow(

View File

@ -25,6 +25,7 @@
* Imports. * Imports.
*/ */
import { import {
AbsoluteTime,
AmountJson, AmountJson,
Amounts, Amounts,
codecForContractTerms, codecForContractTerms,
@ -34,7 +35,6 @@ import {
ConfirmPayResult, ConfirmPayResult,
ConfirmPayResultType, ConfirmPayResultType,
ContractTerms, ContractTerms,
decodeCrock,
Duration, Duration,
durationMax, durationMax,
durationMin, durationMin,
@ -43,19 +43,17 @@ import {
getRandomBytes, getRandomBytes,
HttpStatusCode, HttpStatusCode,
j2s, j2s,
kdf,
Logger, Logger,
NotificationType, NotificationType,
parsePayUri, parsePayUri,
PreparePayResult, PreparePayResult,
PreparePayResultType, PreparePayResultType,
RefreshReason, RefreshReason,
stringToBytes,
TalerErrorCode, TalerErrorCode,
TalerErrorDetails, TalerErrorDetail,
AbsoluteTime,
URL,
TalerProtocolTimestamp, TalerProtocolTimestamp,
TransactionType,
URL,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { EXCHANGE_COINS_LOCK, InternalWalletState } from "../common.js"; import { EXCHANGE_COINS_LOCK, InternalWalletState } from "../common.js";
import { import {
@ -74,9 +72,9 @@ import {
} from "../db.js"; } from "../db.js";
import { import {
guardOperationException, guardOperationException,
makeErrorDetails, makeErrorDetail,
OperationFailedAndReportedError, makePendingOperationFailedError,
OperationFailedError, TalerError,
} from "../errors.js"; } from "../errors.js";
import { import {
AvailableCoinInfo, AvailableCoinInfo,
@ -467,7 +465,7 @@ async function recordConfirmPay(
async function reportProposalError( async function reportProposalError(
ws: InternalWalletState, ws: InternalWalletState,
proposalId: string, proposalId: string,
err: TalerErrorDetails, err: TalerErrorDetail,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ proposals: x.proposals })) .mktx((x) => ({ proposals: x.proposals }))
@ -550,7 +548,7 @@ async function incrementPurchasePayRetry(
async function reportPurchasePayError( async function reportPurchasePayError(
ws: InternalWalletState, ws: InternalWalletState,
proposalId: string, proposalId: string,
err: TalerErrorDetails, err: TalerErrorDetail,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ purchases: x.purchases })) .mktx((x) => ({ purchases: x.purchases }))
@ -575,7 +573,7 @@ export async function processDownloadProposal(
proposalId: string, proposalId: string,
forceNow = false, forceNow = false,
): Promise<void> { ): Promise<void> {
const onOpErr = (err: TalerErrorDetails): Promise<void> => const onOpErr = (err: TalerErrorDetail): Promise<void> =>
reportProposalError(ws, proposalId, err); reportProposalError(ws, proposalId, err);
await guardOperationException( await guardOperationException(
() => processDownloadProposalImpl(ws, proposalId, forceNow), () => processDownloadProposalImpl(ws, proposalId, forceNow),
@ -602,7 +600,7 @@ async function resetDownloadProposalRetry(
async function failProposalPermanently( async function failProposalPermanently(
ws: InternalWalletState, ws: InternalWalletState,
proposalId: string, proposalId: string,
err: TalerErrorDetails, err: TalerErrorDetail,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ proposals: x.proposals })) .mktx((x) => ({ proposals: x.proposals }))
@ -727,13 +725,13 @@ async function processDownloadProposalImpl(
if (r.isError) { if (r.isError) {
switch (r.talerErrorResponse.code) { switch (r.talerErrorResponse.code) {
case TalerErrorCode.MERCHANT_POST_ORDERS_ID_CLAIM_ALREADY_CLAIMED: case TalerErrorCode.MERCHANT_POST_ORDERS_ID_CLAIM_ALREADY_CLAIMED:
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_ORDER_ALREADY_CLAIMED, TalerErrorCode.WALLET_ORDER_ALREADY_CLAIMED,
"order already claimed (likely by other wallet)",
{ {
orderId: proposal.orderId, orderId: proposal.orderId,
claimUrl: orderClaimUrl, claimUrl: orderClaimUrl,
}, },
"order already claimed (likely by other wallet)",
); );
default: default:
throwUnexpectedRequestError(httpResponse, r.talerErrorResponse); throwUnexpectedRequestError(httpResponse, r.talerErrorResponse);
@ -758,13 +756,17 @@ async function processDownloadProposalImpl(
logger.trace( logger.trace(
`malformed contract terms: ${j2s(proposalResp.contract_terms)}`, `malformed contract terms: ${j2s(proposalResp.contract_terms)}`,
); );
const err = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_CONTRACT_TERMS_MALFORMED, TalerErrorCode.WALLET_CONTRACT_TERMS_MALFORMED,
"validation for well-formedness failed",
{}, {},
"validation for well-formedness failed",
); );
await failProposalPermanently(ws, proposalId, err); await failProposalPermanently(ws, proposalId, err);
throw new OperationFailedAndReportedError(err); throw makePendingOperationFailedError(
err,
TransactionType.Payment,
proposalId,
);
} }
const contractTermsHash = ContractTermsUtil.hashContractTerms( const contractTermsHash = ContractTermsUtil.hashContractTerms(
@ -780,13 +782,17 @@ async function processDownloadProposalImpl(
proposalResp.contract_terms, proposalResp.contract_terms,
); );
} catch (e) { } catch (e) {
const err = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_CONTRACT_TERMS_MALFORMED, TalerErrorCode.WALLET_CONTRACT_TERMS_MALFORMED,
`schema validation failed: ${e}`,
{}, {},
`schema validation failed: ${e}`,
); );
await failProposalPermanently(ws, proposalId, err); await failProposalPermanently(ws, proposalId, err);
throw new OperationFailedAndReportedError(err); throw makePendingOperationFailedError(
err,
TransactionType.Payment,
proposalId,
);
} }
const sigValid = await ws.cryptoApi.isValidContractTermsSignature( const sigValid = await ws.cryptoApi.isValidContractTermsSignature(
@ -796,16 +802,20 @@ async function processDownloadProposalImpl(
); );
if (!sigValid) { if (!sigValid) {
const err = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_CONTRACT_TERMS_SIGNATURE_INVALID, TalerErrorCode.WALLET_CONTRACT_TERMS_SIGNATURE_INVALID,
"merchant's signature on contract terms is invalid",
{ {
merchantPub: parsedContractTerms.merchant_pub, merchantPub: parsedContractTerms.merchant_pub,
orderId: parsedContractTerms.order_id, orderId: parsedContractTerms.order_id,
}, },
"merchant's signature on contract terms is invalid",
); );
await failProposalPermanently(ws, proposalId, err); await failProposalPermanently(ws, proposalId, err);
throw new OperationFailedAndReportedError(err); throw makePendingOperationFailedError(
err,
TransactionType.Payment,
proposalId,
);
} }
const fulfillmentUrl = parsedContractTerms.fulfillment_url; const fulfillmentUrl = parsedContractTerms.fulfillment_url;
@ -814,16 +824,20 @@ async function processDownloadProposalImpl(
const baseUrlFromContractTerms = parsedContractTerms.merchant_base_url; const baseUrlFromContractTerms = parsedContractTerms.merchant_base_url;
if (baseUrlForDownload !== baseUrlFromContractTerms) { if (baseUrlForDownload !== baseUrlFromContractTerms) {
const err = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_CONTRACT_TERMS_BASE_URL_MISMATCH, TalerErrorCode.WALLET_CONTRACT_TERMS_BASE_URL_MISMATCH,
"merchant base URL mismatch",
{ {
baseUrlForDownload, baseUrlForDownload,
baseUrlFromContractTerms, baseUrlFromContractTerms,
}, },
"merchant base URL mismatch",
); );
await failProposalPermanently(ws, proposalId, err); await failProposalPermanently(ws, proposalId, err);
throw new OperationFailedAndReportedError(err); throw makePendingOperationFailedError(
err,
TransactionType.Payment,
proposalId,
);
} }
const contractData = extractContractData( const contractData = extractContractData(
@ -895,10 +909,8 @@ async function startDownloadProposal(
]); ]);
}); });
/** /* If we have already claimed this proposal with the same sessionId
* If we have already claimed this proposal with the same sessionId * nonce and claim token, reuse it. */
* nonce and claim token, reuse it.
*/
if ( if (
oldProposal && oldProposal &&
oldProposal.downloadSessionId === sessionId && oldProposal.downloadSessionId === sessionId &&
@ -1029,7 +1041,7 @@ async function storePayReplaySuccess(
async function handleInsufficientFunds( async function handleInsufficientFunds(
ws: InternalWalletState, ws: InternalWalletState,
proposalId: string, proposalId: string,
err: TalerErrorDetails, err: TalerErrorDetail,
): Promise<void> { ): Promise<void> {
logger.trace("handling insufficient funds, trying to re-select coins"); logger.trace("handling insufficient funds, trying to re-select coins");
@ -1319,12 +1331,12 @@ export async function preparePayForUri(
const uriResult = parsePayUri(talerPayUri); const uriResult = parsePayUri(talerPayUri);
if (!uriResult) { if (!uriResult) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_INVALID_TALER_PAY_URI, TalerErrorCode.WALLET_INVALID_TALER_PAY_URI,
`invalid taler://pay URI (${talerPayUri})`,
{ {
talerPayUri, talerPayUri,
}, },
`invalid taler://pay URI (${talerPayUri})`,
); );
} }
@ -1503,7 +1515,7 @@ export async function processPurchasePay(
proposalId: string, proposalId: string,
forceNow = false, forceNow = false,
): Promise<ConfirmPayResult> { ): Promise<ConfirmPayResult> {
const onOpErr = (e: TalerErrorDetails): Promise<void> => const onOpErr = (e: TalerErrorDetail): Promise<void> =>
reportPurchasePayError(ws, proposalId, e); reportPurchasePayError(ws, proposalId, e);
return await guardOperationException( return await guardOperationException(
() => processPurchasePayImpl(ws, proposalId, forceNow), () => processPurchasePayImpl(ws, proposalId, forceNow),
@ -1527,9 +1539,8 @@ async function processPurchasePayImpl(
lastError: { lastError: {
// FIXME: allocate more specific error code // FIXME: allocate more specific error code
code: TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION, code: TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION,
message: `trying to pay for purchase that is not in the database`, hint: `trying to pay for purchase that is not in the database`,
hint: `proposal ID is ${proposalId}`, proposalId: proposalId,
details: {},
}, },
}; };
} }
@ -1594,10 +1605,10 @@ async function processPurchasePayImpl(
resp.status <= 599 resp.status <= 599
) { ) {
logger.trace("treating /pay error as transient"); logger.trace("treating /pay error as transient");
const err = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR, TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR,
"/pay failed",
getHttpResponseErrorDetails(resp), getHttpResponseErrorDetails(resp),
"/pay failed",
); );
return { return {
type: ConfirmPayResultType.Pending, type: ConfirmPayResultType.Pending,
@ -1621,8 +1632,11 @@ async function processPurchasePayImpl(
delete purch.payRetryInfo; delete purch.payRetryInfo;
await tx.purchases.put(purch); await tx.purchases.put(purch);
}); });
// FIXME: Maybe introduce a new return type for this instead of throwing? throw makePendingOperationFailedError(
throw new OperationFailedAndReportedError(errDetails); errDetails,
TransactionType.Payment,
proposalId,
);
} }
if (resp.status === HttpStatusCode.Conflict) { if (resp.status === HttpStatusCode.Conflict) {
@ -1692,10 +1706,10 @@ async function processPurchasePayImpl(
resp.status >= 500 && resp.status >= 500 &&
resp.status <= 599 resp.status <= 599
) { ) {
const err = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR, TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR,
"/paid failed",
getHttpResponseErrorDetails(resp), getHttpResponseErrorDetails(resp),
"/paid failed",
); );
return { return {
type: ConfirmPayResultType.Pending, type: ConfirmPayResultType.Pending,
@ -1703,10 +1717,10 @@ async function processPurchasePayImpl(
}; };
} }
if (resp.status !== 204) { if (resp.status !== 204) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR, TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR,
"/paid failed",
getHttpResponseErrorDetails(resp), getHttpResponseErrorDetails(resp),
"/paid failed",
); );
} }
await storePayReplaySuccess(ws, proposalId, sessionId); await storePayReplaySuccess(ws, proposalId, sessionId);

View File

@ -30,7 +30,7 @@ import {
j2s, j2s,
NotificationType, NotificationType,
RefreshReason, RefreshReason,
TalerErrorDetails, TalerErrorDetail,
TalerProtocolTimestamp, TalerProtocolTimestamp,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { encodeCrock, getRandomBytes } from "@gnu-taler/taler-util"; import { encodeCrock, getRandomBytes } from "@gnu-taler/taler-util";
@ -60,7 +60,7 @@ const logger = new Logger("operations/recoup.ts");
async function incrementRecoupRetry( async function incrementRecoupRetry(
ws: InternalWalletState, ws: InternalWalletState,
recoupGroupId: string, recoupGroupId: string,
err: TalerErrorDetails | undefined, err: TalerErrorDetail | undefined,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ .mktx((x) => ({
@ -384,7 +384,7 @@ export async function processRecoupGroup(
forceNow = false, forceNow = false,
): Promise<void> { ): Promise<void> {
await ws.memoProcessRecoup.memo(recoupGroupId, async () => { await ws.memoProcessRecoup.memo(recoupGroupId, async () => {
const onOpErr = (e: TalerErrorDetails): Promise<void> => const onOpErr = (e: TalerErrorDetail): Promise<void> =>
incrementRecoupRetry(ws, recoupGroupId, e); incrementRecoupRetry(ws, recoupGroupId, e);
return await guardOperationException( return await guardOperationException(
async () => await processRecoupGroupImpl(ws, recoupGroupId, forceNow), async () => await processRecoupGroupImpl(ws, recoupGroupId, forceNow),

View File

@ -43,7 +43,7 @@ import {
NotificationType, NotificationType,
RefreshGroupId, RefreshGroupId,
RefreshReason, RefreshReason,
TalerErrorDetails, TalerErrorDetail,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { AmountJson, Amounts } from "@gnu-taler/taler-util"; import { AmountJson, Amounts } from "@gnu-taler/taler-util";
import { amountToPretty } from "@gnu-taler/taler-util"; import { amountToPretty } from "@gnu-taler/taler-util";
@ -714,7 +714,7 @@ async function refreshReveal(
async function incrementRefreshRetry( async function incrementRefreshRetry(
ws: InternalWalletState, ws: InternalWalletState,
refreshGroupId: string, refreshGroupId: string,
err: TalerErrorDetails | undefined, err: TalerErrorDetail | undefined,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ .mktx((x) => ({
@ -747,7 +747,7 @@ export async function processRefreshGroup(
forceNow = false, forceNow = false,
): Promise<void> { ): Promise<void> {
await ws.memoProcessRefresh.memo(refreshGroupId, async () => { await ws.memoProcessRefresh.memo(refreshGroupId, async () => {
const onOpErr = (e: TalerErrorDetails): Promise<void> => const onOpErr = (e: TalerErrorDetail): Promise<void> =>
incrementRefreshRetry(ws, refreshGroupId, e); incrementRefreshRetry(ws, refreshGroupId, e);
return await guardOperationException( return await guardOperationException(
async () => await processRefreshGroupImpl(ws, refreshGroupId, forceNow), async () => await processRefreshGroupImpl(ws, refreshGroupId, forceNow),

View File

@ -40,7 +40,7 @@ import {
parseRefundUri, parseRefundUri,
RefreshReason, RefreshReason,
TalerErrorCode, TalerErrorCode,
TalerErrorDetails, TalerErrorDetail,
URL, URL,
codecForMerchantOrderStatusPaid, codecForMerchantOrderStatusPaid,
AbsoluteTime, AbsoluteTime,
@ -88,7 +88,7 @@ async function resetPurchaseQueryRefundRetry(
async function incrementPurchaseQueryRefundRetry( async function incrementPurchaseQueryRefundRetry(
ws: InternalWalletState, ws: InternalWalletState,
proposalId: string, proposalId: string,
err: TalerErrorDetails | undefined, err: TalerErrorDetail | undefined,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ .mktx((x) => ({
@ -592,7 +592,7 @@ export async function processPurchaseQueryRefund(
proposalId: string, proposalId: string,
forceNow = false, forceNow = false,
): Promise<void> { ): Promise<void> {
const onOpErr = (e: TalerErrorDetails): Promise<void> => const onOpErr = (e: TalerErrorDetail): Promise<void> =>
incrementPurchaseQueryRefundRetry(ws, proposalId, e); incrementPurchaseQueryRefundRetry(ws, proposalId, e);
await guardOperationException( await guardOperationException(
() => processPurchaseQueryRefundImpl(ws, proposalId, forceNow, true), () => processPurchaseQueryRefundImpl(ws, proposalId, forceNow, true),

View File

@ -34,7 +34,7 @@ import {
NotificationType, NotificationType,
randomBytes, randomBytes,
TalerErrorCode, TalerErrorCode,
TalerErrorDetails, TalerErrorDetail,
AbsoluteTime, AbsoluteTime,
URL, URL,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
@ -47,7 +47,7 @@ import {
WalletStoresV1, WalletStoresV1,
WithdrawalGroupRecord, WithdrawalGroupRecord,
} from "../db.js"; } from "../db.js";
import { guardOperationException, OperationFailedError } from "../errors.js"; import { guardOperationException, TalerError } from "../errors.js";
import { assertUnreachable } from "../util/assertUnreachable.js"; import { assertUnreachable } from "../util/assertUnreachable.js";
import { import {
readSuccessResponseJsonOrErrorCode, readSuccessResponseJsonOrErrorCode,
@ -135,7 +135,7 @@ async function incrementReserveRetry(
async function reportReserveError( async function reportReserveError(
ws: InternalWalletState, ws: InternalWalletState,
reservePub: string, reservePub: string,
err: TalerErrorDetails, err: TalerErrorDetail,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ .mktx((x) => ({
@ -338,7 +338,7 @@ export async function processReserve(
forceNow = false, forceNow = false,
): Promise<void> { ): Promise<void> {
return ws.memoProcessReserve.memo(reservePub, async () => { return ws.memoProcessReserve.memo(reservePub, async () => {
const onOpError = (err: TalerErrorDetails): Promise<void> => const onOpError = (err: TalerErrorDetail): Promise<void> =>
reportReserveError(ws, reservePub, err); reportReserveError(ws, reservePub, err);
await guardOperationException( await guardOperationException(
() => processReserveImpl(ws, reservePub, forceNow), () => processReserveImpl(ws, reservePub, forceNow),
@ -571,7 +571,7 @@ async function updateReserve(
if ( if (
resp.status === 404 && resp.status === 404 &&
result.talerErrorResponse.code === result.talerErrorResponse.code ===
TalerErrorCode.EXCHANGE_RESERVES_GET_STATUS_UNKNOWN TalerErrorCode.EXCHANGE_RESERVES_STATUS_UNKNOWN
) { ) {
ws.notify({ ws.notify({
type: NotificationType.ReserveNotYetFound, type: NotificationType.ReserveNotYetFound,
@ -803,9 +803,8 @@ export async function createTalerWithdrawReserve(
return tx.reserves.get(reserve.reservePub); return tx.reserves.get(reserve.reservePub);
}); });
if (processedReserve?.reserveStatus === ReserveRecordStatus.BankAborted) { if (processedReserve?.reserveStatus === ReserveRecordStatus.BankAborted) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_WITHDRAWAL_OPERATION_ABORTED_BY_BANK, TalerErrorCode.WALLET_WITHDRAWAL_OPERATION_ABORTED_BY_BANK,
"withdrawal aborted by bank",
{}, {},
); );
} }

View File

@ -22,7 +22,7 @@ import {
parseTipUri, parseTipUri,
codecForTipPickupGetResponse, codecForTipPickupGetResponse,
Amounts, Amounts,
TalerErrorDetails, TalerErrorDetail,
NotificationType, NotificationType,
TipPlanchetDetail, TipPlanchetDetail,
TalerErrorCode, TalerErrorCode,
@ -44,7 +44,7 @@ import {
import { j2s } from "@gnu-taler/taler-util"; import { j2s } from "@gnu-taler/taler-util";
import { checkDbInvariant, checkLogicInvariant } from "../util/invariants.js"; import { checkDbInvariant, checkLogicInvariant } from "../util/invariants.js";
import { initRetryInfo, updateRetryInfoTimeout } from "../util/retries.js"; import { initRetryInfo, updateRetryInfoTimeout } from "../util/retries.js";
import { guardOperationException, makeErrorDetails } from "../errors.js"; import { guardOperationException, makeErrorDetail } from "../errors.js";
import { updateExchangeFromUrl } from "./exchanges.js"; import { updateExchangeFromUrl } from "./exchanges.js";
import { InternalWalletState } from "../common.js"; import { InternalWalletState } from "../common.js";
import { import {
@ -163,7 +163,7 @@ export async function prepareTip(
async function incrementTipRetry( async function incrementTipRetry(
ws: InternalWalletState, ws: InternalWalletState,
walletTipId: string, walletTipId: string,
err: TalerErrorDetails | undefined, err: TalerErrorDetail | undefined,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ .mktx((x) => ({
@ -192,7 +192,7 @@ export async function processTip(
tipId: string, tipId: string,
forceNow = false, forceNow = false,
): Promise<void> { ): Promise<void> {
const onOpErr = (e: TalerErrorDetails): Promise<void> => const onOpErr = (e: TalerErrorDetail): Promise<void> =>
incrementTipRetry(ws, tipId, e); incrementTipRetry(ws, tipId, e);
await guardOperationException( await guardOperationException(
() => processTipImpl(ws, tipId, forceNow), () => processTipImpl(ws, tipId, forceNow),
@ -296,10 +296,10 @@ async function processTipImpl(
merchantResp.status === 424) merchantResp.status === 424)
) { ) {
logger.trace(`got transient tip error`); logger.trace(`got transient tip error`);
const err = makeErrorDetails( const err = makeErrorDetail(
TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR, TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR,
"tip pickup failed (transient)",
getHttpResponseErrorDetails(merchantResp), getHttpResponseErrorDetails(merchantResp),
"tip pickup failed (transient)",
); );
await incrementTipRetry(ws, tipRecord.walletTipId, err); await incrementTipRetry(ws, tipRecord.walletTipId, err);
// FIXME: Maybe we want to signal to the caller that the transient error happened? // FIXME: Maybe we want to signal to the caller that the transient error happened?
@ -355,10 +355,10 @@ async function processTipImpl(
if (!tipRecord) { if (!tipRecord) {
return; return;
} }
tipRecord.lastError = makeErrorDetails( tipRecord.lastError = makeErrorDetail(
TalerErrorCode.WALLET_TIPPING_COIN_SIGNATURE_INVALID, TalerErrorCode.WALLET_TIPPING_COIN_SIGNATURE_INVALID,
"invalid signature from the exchange (via merchant tip) after unblinding",
{}, {},
"invalid signature from the exchange (via merchant tip) after unblinding",
); );
await tx.tips.put(tipRecord); await tx.tips.put(tipRecord);
}); });

View File

@ -30,7 +30,7 @@ import {
NotificationType, NotificationType,
parseWithdrawUri, parseWithdrawUri,
TalerErrorCode, TalerErrorCode,
TalerErrorDetails, TalerErrorDetail,
AbsoluteTime, AbsoluteTime,
WithdrawResponse, WithdrawResponse,
URL, URL,
@ -42,6 +42,7 @@ import {
ExchangeWithdrawRequest, ExchangeWithdrawRequest,
Duration, Duration,
TalerProtocolTimestamp, TalerProtocolTimestamp,
TransactionType,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { import {
CoinRecord, CoinRecord,
@ -63,9 +64,11 @@ import {
} from "../util/http.js"; } from "../util/http.js";
import { initRetryInfo, updateRetryInfoTimeout } from "../util/retries.js"; import { initRetryInfo, updateRetryInfoTimeout } from "../util/retries.js";
import { import {
getErrorDetailFromException,
guardOperationException, guardOperationException,
makeErrorDetails, makeErrorDetail,
OperationFailedError, makePendingOperationFailedError,
TalerError,
} from "../errors.js"; } from "../errors.js";
import { InternalWalletState } from "../common.js"; import { InternalWalletState } from "../common.js";
import { import {
@ -299,15 +302,14 @@ export async function getBankWithdrawalInfo(
config.version, config.version,
); );
if (versionRes?.compatible != true) { if (versionRes?.compatible != true) {
const opErr = makeErrorDetails( throw TalerError.fromDetail(
TalerErrorCode.WALLET_BANK_INTEGRATION_PROTOCOL_VERSION_INCOMPATIBLE, TalerErrorCode.WALLET_BANK_INTEGRATION_PROTOCOL_VERSION_INCOMPATIBLE,
"bank integration protocol version not compatible with wallet",
{ {
exchangeProtocolVersion: config.version, exchangeProtocolVersion: config.version,
walletProtocolVersion: WALLET_BANK_INTEGRATION_PROTOCOL_VERSION, walletProtocolVersion: WALLET_BANK_INTEGRATION_PROTOCOL_VERSION,
}, },
"bank integration protocol version not compatible with wallet",
); );
throw new OperationFailedError(opErr);
} }
const reqUrl = new URL( const reqUrl = new URL(
@ -526,12 +528,9 @@ async function processPlanchetExchangeRequest(
); );
return r; return r;
} catch (e) { } catch (e) {
const errDetail = getErrorDetailFromException(e);
logger.trace("withdrawal request failed", e); logger.trace("withdrawal request failed", e);
logger.trace(e); logger.trace(e);
if (!(e instanceof OperationFailedError)) {
throw e;
}
const errDetails = e.operationError;
await ws.db await ws.db
.mktx((x) => ({ planchets: x.planchets })) .mktx((x) => ({ planchets: x.planchets }))
.runReadWrite(async (tx) => { .runReadWrite(async (tx) => {
@ -542,7 +541,7 @@ async function processPlanchetExchangeRequest(
if (!planchet) { if (!planchet) {
return; return;
} }
planchet.lastError = errDetails; planchet.lastError = errDetail;
await tx.planchets.put(planchet); await tx.planchets.put(planchet);
}); });
return; return;
@ -628,10 +627,10 @@ async function processPlanchetVerifyAndStoreCoin(
if (!planchet) { if (!planchet) {
return; return;
} }
planchet.lastError = makeErrorDetails( planchet.lastError = makeErrorDetail(
TalerErrorCode.WALLET_EXCHANGE_COIN_SIGNATURE_INVALID, TalerErrorCode.WALLET_EXCHANGE_COIN_SIGNATURE_INVALID,
"invalid signature from the exchange after unblinding",
{}, {},
"invalid signature from the exchange after unblinding",
); );
await tx.planchets.put(planchet); await tx.planchets.put(planchet);
}); });
@ -797,7 +796,7 @@ export async function updateWithdrawalDenoms(
async function incrementWithdrawalRetry( async function incrementWithdrawalRetry(
ws: InternalWalletState, ws: InternalWalletState,
withdrawalGroupId: string, withdrawalGroupId: string,
err: TalerErrorDetails | undefined, err: TalerErrorDetail | undefined,
): Promise<void> { ): Promise<void> {
await ws.db await ws.db
.mktx((x) => ({ withdrawalGroups: x.withdrawalGroups })) .mktx((x) => ({ withdrawalGroups: x.withdrawalGroups }))
@ -821,7 +820,7 @@ export async function processWithdrawGroup(
withdrawalGroupId: string, withdrawalGroupId: string,
forceNow = false, forceNow = false,
): Promise<void> { ): Promise<void> {
const onOpErr = (e: TalerErrorDetails): Promise<void> => const onOpErr = (e: TalerErrorDetail): Promise<void> =>
incrementWithdrawalRetry(ws, withdrawalGroupId, e); incrementWithdrawalRetry(ws, withdrawalGroupId, e);
await guardOperationException( await guardOperationException(
() => processWithdrawGroupImpl(ws, withdrawalGroupId, forceNow), () => processWithdrawGroupImpl(ws, withdrawalGroupId, forceNow),
@ -919,7 +918,7 @@ async function processWithdrawGroupImpl(
let numFinished = 0; let numFinished = 0;
let finishedForFirstTime = false; let finishedForFirstTime = false;
let errorsPerCoin: Record<number, TalerErrorDetails> = {}; let errorsPerCoin: Record<number, TalerErrorDetail> = {};
await ws.db await ws.db
.mktx((x) => ({ .mktx((x) => ({
@ -957,12 +956,12 @@ async function processWithdrawGroupImpl(
}); });
if (numFinished != numTotalCoins) { if (numFinished != numTotalCoins) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_WITHDRAWAL_GROUP_INCOMPLETE, TalerErrorCode.WALLET_WITHDRAWAL_GROUP_INCOMPLETE,
`withdrawal did not finish (${numFinished} / ${numTotalCoins} coins withdrawn)`,
{ {
errorsPerCoin, errorsPerCoin,
}, },
`withdrawal did not finish (${numFinished} / ${numTotalCoins} coins withdrawn)`,
); );
} }

View File

@ -25,7 +25,7 @@
* Imports. * Imports.
*/ */
import { import {
TalerErrorDetails, TalerErrorDetail,
BalancesResponse, BalancesResponse,
AbsoluteTime, AbsoluteTime,
TalerProtocolTimestamp, TalerProtocolTimestamp,
@ -71,7 +71,7 @@ export type PendingTaskInfo = PendingTaskInfoCommon &
export interface PendingBackupTask { export interface PendingBackupTask {
type: PendingTaskType.Backup; type: PendingTaskType.Backup;
backupProviderBaseUrl: string; backupProviderBaseUrl: string;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
/** /**
@ -80,7 +80,7 @@ export interface PendingBackupTask {
export interface PendingExchangeUpdateTask { export interface PendingExchangeUpdateTask {
type: PendingTaskType.ExchangeUpdate; type: PendingTaskType.ExchangeUpdate;
exchangeBaseUrl: string; exchangeBaseUrl: string;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
/** /**
@ -124,7 +124,7 @@ export interface PendingReserveTask {
*/ */
export interface PendingRefreshTask { export interface PendingRefreshTask {
type: PendingTaskType.Refresh; type: PendingTaskType.Refresh;
lastError?: TalerErrorDetails; lastError?: TalerErrorDetail;
refreshGroupId: string; refreshGroupId: string;
finishedPerCoin: boolean[]; finishedPerCoin: boolean[];
retryInfo: RetryInfo; retryInfo: RetryInfo;
@ -139,7 +139,7 @@ export interface PendingProposalDownloadTask {
proposalTimestamp: TalerProtocolTimestamp; proposalTimestamp: TalerProtocolTimestamp;
proposalId: string; proposalId: string;
orderId: string; orderId: string;
lastError?: TalerErrorDetails; lastError?: TalerErrorDetail;
retryInfo?: RetryInfo; retryInfo?: RetryInfo;
} }
@ -173,7 +173,7 @@ export interface PendingPayTask {
proposalId: string; proposalId: string;
isReplay: boolean; isReplay: boolean;
retryInfo?: RetryInfo; retryInfo?: RetryInfo;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
/** /**
@ -184,14 +184,14 @@ export interface PendingRefundQueryTask {
type: PendingTaskType.RefundQuery; type: PendingTaskType.RefundQuery;
proposalId: string; proposalId: string;
retryInfo: RetryInfo; retryInfo: RetryInfo;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
export interface PendingRecoupTask { export interface PendingRecoupTask {
type: PendingTaskType.Recoup; type: PendingTaskType.Recoup;
recoupGroupId: string; recoupGroupId: string;
retryInfo: RetryInfo; retryInfo: RetryInfo;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
} }
/** /**
@ -199,7 +199,7 @@ export interface PendingRecoupTask {
*/ */
export interface PendingWithdrawTask { export interface PendingWithdrawTask {
type: PendingTaskType.Withdraw; type: PendingTaskType.Withdraw;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
retryInfo: RetryInfo; retryInfo: RetryInfo;
withdrawalGroupId: string; withdrawalGroupId: string;
} }
@ -209,7 +209,7 @@ export interface PendingWithdrawTask {
*/ */
export interface PendingDepositTask { export interface PendingDepositTask {
type: PendingTaskType.Deposit; type: PendingTaskType.Deposit;
lastError: TalerErrorDetails | undefined; lastError: TalerErrorDetail | undefined;
retryInfo: RetryInfo | undefined; retryInfo: RetryInfo | undefined;
depositGroupId: string; depositGroupId: string;
} }

View File

@ -24,16 +24,16 @@
/** /**
* Imports * Imports
*/ */
import { OperationFailedError, makeErrorDetails } from "../errors.js";
import { import {
Logger, Logger,
Duration, Duration,
AbsoluteTime, AbsoluteTime,
TalerErrorDetails, TalerErrorDetail,
Codec, Codec,
j2s, j2s,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { TalerErrorCode } from "@gnu-taler/taler-util"; import { TalerErrorCode } from "@gnu-taler/taler-util";
import { makeErrorDetail, TalerError } from "../errors.js";
const logger = new Logger("http.ts"); const logger = new Logger("http.ts");
@ -125,7 +125,7 @@ type ResponseOrError<T> =
export async function readTalerErrorResponse( export async function readTalerErrorResponse(
httpResponse: HttpResponse, httpResponse: HttpResponse,
): Promise<TalerErrorDetails> { ): Promise<TalerErrorDetail> {
const errJson = await httpResponse.json(); const errJson = await httpResponse.json();
const talerErrorCode = errJson.code; const talerErrorCode = errJson.code;
if (typeof talerErrorCode !== "number") { if (typeof talerErrorCode !== "number") {
@ -134,16 +134,14 @@ export async function readTalerErrorResponse(
errJson, errJson,
)}`, )}`,
); );
throw new OperationFailedError( throw TalerError.fromDetail(
makeErrorDetails(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Error response did not contain error code",
{ {
requestUrl: httpResponse.requestUrl, requestUrl: httpResponse.requestUrl,
requestMethod: httpResponse.requestMethod, requestMethod: httpResponse.requestMethod,
httpStatusCode: httpResponse.status, httpStatusCode: httpResponse.status,
}, },
), "Error response did not contain error code",
); );
} }
return errJson; return errJson;
@ -151,28 +149,28 @@ export async function readTalerErrorResponse(
export async function readUnexpectedResponseDetails( export async function readUnexpectedResponseDetails(
httpResponse: HttpResponse, httpResponse: HttpResponse,
): Promise<TalerErrorDetails> { ): Promise<TalerErrorDetail> {
const errJson = await httpResponse.json(); const errJson = await httpResponse.json();
const talerErrorCode = errJson.code; const talerErrorCode = errJson.code;
if (typeof talerErrorCode !== "number") { if (typeof talerErrorCode !== "number") {
return makeErrorDetails( return makeErrorDetail(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Error response did not contain error code",
{ {
requestUrl: httpResponse.requestUrl, requestUrl: httpResponse.requestUrl,
requestMethod: httpResponse.requestMethod, requestMethod: httpResponse.requestMethod,
httpStatusCode: httpResponse.status, httpStatusCode: httpResponse.status,
}, },
"Error response did not contain error code",
); );
} }
return makeErrorDetails( return makeErrorDetail(
TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR, TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR,
`Unexpected HTTP status (${httpResponse.status}) in response`,
{ {
requestUrl: httpResponse.requestUrl, requestUrl: httpResponse.requestUrl,
httpStatusCode: httpResponse.status, httpStatusCode: httpResponse.status,
errorResponse: errJson, errorResponse: errJson,
}, },
`Unexpected HTTP status (${httpResponse.status}) in response`,
); );
} }
@ -191,14 +189,14 @@ export async function readSuccessResponseJsonOrErrorCode<T>(
try { try {
parsedResponse = codec.decode(respJson); parsedResponse = codec.decode(respJson);
} catch (e: any) { } catch (e: any) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Response invalid",
{ {
requestUrl: httpResponse.requestUrl, requestUrl: httpResponse.requestUrl,
httpStatusCode: httpResponse.status, httpStatusCode: httpResponse.status,
validationError: e.toString(), validationError: e.toString(),
}, },
"Response invalid",
); );
} }
return { return {
@ -220,16 +218,14 @@ export function throwUnexpectedRequestError(
httpResponse: HttpResponse, httpResponse: HttpResponse,
talerErrorResponse: TalerErrorResponse, talerErrorResponse: TalerErrorResponse,
): never { ): never {
throw new OperationFailedError( throw TalerError.fromDetail(
makeErrorDetails(
TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR, TalerErrorCode.WALLET_UNEXPECTED_REQUEST_ERROR,
`Unexpected HTTP status ${httpResponse.status} in response`,
{ {
requestUrl: httpResponse.requestUrl, requestUrl: httpResponse.requestUrl,
httpStatusCode: httpResponse.status, httpStatusCode: httpResponse.status,
errorResponse: talerErrorResponse, errorResponse: talerErrorResponse,
}, },
), `Unexpected HTTP status ${httpResponse.status} in response`,
); );
} }
@ -251,16 +247,14 @@ export async function readSuccessResponseTextOrErrorCode<T>(
const errJson = await httpResponse.json(); const errJson = await httpResponse.json();
const talerErrorCode = errJson.code; const talerErrorCode = errJson.code;
if (typeof talerErrorCode !== "number") { if (typeof talerErrorCode !== "number") {
throw new OperationFailedError( throw TalerError.fromDetail(
makeErrorDetails(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Error response did not contain error code",
{ {
httpStatusCode: httpResponse.status, httpStatusCode: httpResponse.status,
requestUrl: httpResponse.requestUrl, requestUrl: httpResponse.requestUrl,
requestMethod: httpResponse.requestMethod, requestMethod: httpResponse.requestMethod,
}, },
), "Error response did not contain error code",
); );
} }
return { return {
@ -282,16 +276,14 @@ export async function checkSuccessResponseOrThrow(
const errJson = await httpResponse.json(); const errJson = await httpResponse.json();
const talerErrorCode = errJson.code; const talerErrorCode = errJson.code;
if (typeof talerErrorCode !== "number") { if (typeof talerErrorCode !== "number") {
throw new OperationFailedError( throw TalerError.fromDetail(
makeErrorDetails(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Error response did not contain error code",
{ {
httpStatusCode: httpResponse.status, httpStatusCode: httpResponse.status,
requestUrl: httpResponse.requestUrl, requestUrl: httpResponse.requestUrl,
requestMethod: httpResponse.requestMethod, requestMethod: httpResponse.requestMethod,
}, },
), "Error response did not contain error code",
); );
} }
throwUnexpectedRequestError(httpResponse, errJson); throwUnexpectedRequestError(httpResponse, errJson);

View File

@ -100,9 +100,8 @@ import {
WalletStoresV1, WalletStoresV1,
} from "./db.js"; } from "./db.js";
import { import {
makeErrorDetails, getErrorDetailFromException,
OperationFailedAndReportedError, TalerError,
OperationFailedError,
} from "./errors.js"; } from "./errors.js";
import { exportBackup } from "./operations/backup/export.js"; import { exportBackup } from "./operations/backup/export.js";
import { import {
@ -297,10 +296,10 @@ export async function runPending(
try { try {
await processOnePendingOperation(ws, p, forceNow); await processOnePendingOperation(ws, p, forceNow);
} catch (e) { } catch (e) {
if (e instanceof OperationFailedAndReportedError) { if (e instanceof TalerError) {
console.error( console.error(
"Operation failed:", "Operation failed:",
JSON.stringify(e.operationError, undefined, 2), JSON.stringify(e.errorDetail, undefined, 2),
); );
} else { } else {
console.error(e); console.error(e);
@ -399,10 +398,16 @@ async function runTaskLoop(
try { try {
await processOnePendingOperation(ws, p); await processOnePendingOperation(ws, p);
} catch (e) { } catch (e) {
if (e instanceof OperationFailedAndReportedError) { if (
logger.warn("operation processed resulted in reported error"); e instanceof TalerError &&
logger.warn(`reported error was: ${j2s(e.operationError)}`); e.hasErrorCode(TalerErrorCode.WALLET_PENDING_OPERATION_FAILED)
) {
logger.warn("operation processed resulted in error");
logger.warn(`error was: ${j2s(e.errorDetail)}`);
} else { } else {
// This is a bug, as we expect pending operations to always
// do their own error handling and only throw WALLET_PENDING_OPERATION_FAILED
// or return something.
logger.error("Uncaught exception", e); logger.error("Uncaught exception", e);
ws.notify({ ws.notify({
type: NotificationType.InternalError, type: NotificationType.InternalError,
@ -722,7 +727,7 @@ export async function getClientFromWalletState(
const res = await handleCoreApiRequest(ws, op, `${id++}`, payload); const res = await handleCoreApiRequest(ws, op, `${id++}`, payload);
switch (res.type) { switch (res.type) {
case "error": case "error":
throw new OperationFailedError(res.error); throw TalerError.fromUncheckedDetail(res.error);
case "response": case "response":
return res.result; return res.result;
} }
@ -1040,12 +1045,12 @@ async function dispatchRequestInternal(
return []; return [];
} }
} }
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_CORE_API_OPERATION_UNKNOWN, TalerErrorCode.WALLET_CORE_API_OPERATION_UNKNOWN,
"unknown operation",
{ {
operation, operation,
}, },
"unknown operation",
); );
} }
@ -1067,34 +1072,13 @@ export async function handleCoreApiRequest(
result, result,
}; };
} catch (e: any) { } catch (e: any) {
if ( const err = getErrorDetailFromException(e);
e instanceof OperationFailedError ||
e instanceof OperationFailedAndReportedError
) {
logger.error("Caught operation failed error");
logger.trace((e as any).stack);
return { return {
type: "error", type: "error",
operation, operation,
id, id,
error: e.operationError, error: err,
}; };
} else {
try {
logger.error("Caught unexpected exception:");
logger.error(e.stack);
} catch (e) {}
return {
type: "error",
operation,
id,
error: makeErrorDetails(
TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION,
`unexpected exception: ${e}`,
{},
),
};
}
} }
} }

View File

@ -21,7 +21,6 @@ import {
getDefaultNodeWallet, getDefaultNodeWallet,
DefaultNodeWalletArgs, DefaultNodeWalletArgs,
NodeHttpLib, NodeHttpLib,
makeErrorDetails,
handleWorkerError, handleWorkerError,
handleWorkerMessage, handleWorkerMessage,
HttpRequestLibrary, HttpRequestLibrary,
@ -33,6 +32,7 @@ import {
WALLET_EXCHANGE_PROTOCOL_VERSION, WALLET_EXCHANGE_PROTOCOL_VERSION,
WALLET_MERCHANT_PROTOCOL_VERSION, WALLET_MERCHANT_PROTOCOL_VERSION,
Wallet, Wallet,
getErrorDetailFromException,
} from "@gnu-taler/taler-wallet-core"; } from "@gnu-taler/taler-wallet-core";
import fs from "fs"; import fs from "fs";
@ -270,11 +270,7 @@ export function installNativeWalletListener(): void {
type: "error", type: "error",
id, id,
operation, operation,
error: makeErrorDetails( error: getErrorDetailFromException(e),
TalerErrorCode.WALLET_UNEXPECTED_EXCEPTION,
"unexpected exception",
{},
),
}; };
sendNativeMessage(respMsg); sendNativeMessage(respMsg);
return; return;

View File

@ -18,11 +18,11 @@
* Imports. * Imports.
*/ */
import { import {
OperationFailedError,
HttpRequestLibrary, HttpRequestLibrary,
HttpRequestOptions, HttpRequestOptions,
HttpResponse, HttpResponse,
Headers, Headers,
TalerError,
} from "@gnu-taler/taler-wallet-core"; } from "@gnu-taler/taler-wallet-core";
import { import {
Logger, Logger,
@ -49,14 +49,14 @@ export class BrowserHttpLib implements HttpRequestLibrary {
if (this.throttlingEnabled && this.throttle.applyThrottle(requestUrl)) { if (this.throttlingEnabled && this.throttle.applyThrottle(requestUrl)) {
const parsedUrl = new URL(requestUrl); const parsedUrl = new URL(requestUrl);
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_HTTP_REQUEST_THROTTLED, TalerErrorCode.WALLET_HTTP_REQUEST_THROTTLED,
`request to origin ${parsedUrl.origin} was throttled`,
{ {
requestMethod, requestMethod,
requestUrl, requestUrl,
throttleStats: this.throttle.getThrottleStats(requestUrl), throttleStats: this.throttle.getThrottleStats(requestUrl),
}, },
`request to origin ${parsedUrl.origin} was throttled`,
); );
} }
@ -78,12 +78,12 @@ export class BrowserHttpLib implements HttpRequestLibrary {
myRequest.onerror = (e) => { myRequest.onerror = (e) => {
logger.error("http request error"); logger.error("http request error");
reject( reject(
OperationFailedError.fromCode( TalerError.fromDetail(
TalerErrorCode.WALLET_NETWORK_ERROR, TalerErrorCode.WALLET_NETWORK_ERROR,
"Could not make request",
{ {
requestUrl: requestUrl, requestUrl: requestUrl,
}, },
"Could not make request",
), ),
); );
}; };
@ -91,12 +91,12 @@ export class BrowserHttpLib implements HttpRequestLibrary {
myRequest.addEventListener("readystatechange", (e) => { myRequest.addEventListener("readystatechange", (e) => {
if (myRequest.readyState === XMLHttpRequest.DONE) { if (myRequest.readyState === XMLHttpRequest.DONE) {
if (myRequest.status === 0) { if (myRequest.status === 0) {
const exc = OperationFailedError.fromCode( const exc = TalerError.fromDetail(
TalerErrorCode.WALLET_NETWORK_ERROR, TalerErrorCode.WALLET_NETWORK_ERROR,
"HTTP request failed (status 0, maybe URI scheme was wrong?)",
{ {
requestUrl: requestUrl, requestUrl: requestUrl,
}, },
"HTTP request failed (status 0, maybe URI scheme was wrong?)",
); );
reject(exc); reject(exc);
return; return;
@ -112,23 +112,23 @@ export class BrowserHttpLib implements HttpRequestLibrary {
const responseString = td.decode(myRequest.response); const responseString = td.decode(myRequest.response);
responseJson = JSON.parse(responseString); responseJson = JSON.parse(responseString);
} catch (e) { } catch (e) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Invalid JSON from HTTP response",
{ {
requestUrl: requestUrl, requestUrl: requestUrl,
httpStatusCode: myRequest.status, httpStatusCode: myRequest.status,
}, },
"Invalid JSON from HTTP response",
); );
} }
if (responseJson === null || typeof responseJson !== "object") { if (responseJson === null || typeof responseJson !== "object") {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Invalid JSON from HTTP response",
{ {
requestUrl: requestUrl, requestUrl: requestUrl,
httpStatusCode: myRequest.status, httpStatusCode: myRequest.status,
}, },
"Invalid JSON from HTTP response",
); );
} }
return responseJson; return responseJson;

View File

@ -13,7 +13,7 @@
You should have received a copy of the GNU General Public License along with You should have received a copy of the GNU General Public License along with
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/> GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
*/ */
import { TalerErrorDetails } from "@gnu-taler/taler-util"; import { TalerErrorDetail } from "@gnu-taler/taler-util";
import { Fragment, h, VNode } from "preact"; import { Fragment, h, VNode } from "preact";
import { useState } from "preact/hooks"; import { useState } from "preact/hooks";
import arrowDown from "../../static/img/chevron-down.svg"; import arrowDown from "../../static/img/chevron-down.svg";
@ -25,7 +25,7 @@ export function ErrorTalerOperation({
error, error,
}: { }: {
title?: VNode; title?: VNode;
error?: TalerErrorDetails; error?: TalerErrorDetail;
}): VNode | null { }): VNode | null {
const { devMode } = useDevContext(); const { devMode } = useDevContext();
const [showErrorDetail, setShowErrorDetail] = useState(false); const [showErrorDetail, setShowErrorDetail] = useState(false);
@ -59,7 +59,7 @@ export function ErrorTalerOperation({
<Fragment> <Fragment>
<div style={{ padding: 5, textAlign: "left" }}> <div style={{ padding: 5, textAlign: "left" }}>
<div> <div>
<b>{error.message}</b> {!errorHint ? "" : `: ${errorHint}`}{" "} <b>{error.hint}</b> {!errorHint ? "" : `: ${errorHint}`}{" "}
</div> </div>
</div> </div>
{devMode && ( {devMode && (

View File

@ -35,7 +35,7 @@ import {
PreparePayResultType, PreparePayResultType,
Translate, Translate,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { OperationFailedError } from "@gnu-taler/taler-wallet-core"; import { TalerError } from "@gnu-taler/taler-wallet-core";
import { Fragment, h, VNode } from "preact"; import { Fragment, h, VNode } from "preact";
import { useEffect, useState } from "preact/hooks"; import { useEffect, useState } from "preact/hooks";
import { ErrorTalerOperation } from "../components/ErrorTalerOperation"; import { ErrorTalerOperation } from "../components/ErrorTalerOperation";
@ -64,9 +64,9 @@ export function DepositPage({ talerPayUri, goBack }: Props): VNode {
const [payResult, setPayResult] = useState<ConfirmPayResult | undefined>( const [payResult, setPayResult] = useState<ConfirmPayResult | undefined>(
undefined, undefined,
); );
const [payErrMsg, setPayErrMsg] = useState< const [payErrMsg, setPayErrMsg] = useState<TalerError | string | undefined>(
OperationFailedError | string | undefined undefined,
>(undefined); );
const balance = useAsyncAsHook(wxApi.getBalance, [ const balance = useAsyncAsHook(wxApi.getBalance, [
NotificationType.CoinWithdrawn, NotificationType.CoinWithdrawn,
@ -97,7 +97,7 @@ export function DepositPage({ talerPayUri, goBack }: Props): VNode {
setPayStatus(p); setPayStatus(p);
} catch (e) { } catch (e) {
console.log("Got error while trying to pay", e); console.log("Got error while trying to pay", e);
if (e instanceof OperationFailedError) { if (e instanceof TalerError) {
setPayErrMsg(e); setPayErrMsg(e);
} }
if (e instanceof Error) { if (e instanceof Error) {
@ -117,7 +117,7 @@ export function DepositPage({ talerPayUri, goBack }: Props): VNode {
} }
if (!payStatus) { if (!payStatus) {
if (payErrMsg instanceof OperationFailedError) { if (payErrMsg instanceof TalerError) {
return ( return (
<WalletAction> <WalletAction>
<LogoHeader /> <LogoHeader />
@ -131,7 +131,7 @@ export function DepositPage({ talerPayUri, goBack }: Props): VNode {
Could not get the payment information for this order Could not get the payment information for this order
</i18n.Translate> </i18n.Translate>
} }
error={payErrMsg?.operationError} error={payErrMsg?.errorDetail}
/> />
</section> </section>
</WalletAction> </WalletAction>

View File

@ -36,7 +36,7 @@ import {
PreparePayResultType, PreparePayResultType,
Product, Product,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { OperationFailedError } from "@gnu-taler/taler-wallet-core"; import { TalerError } from "@gnu-taler/taler-wallet-core";
import { Fragment, h, VNode } from "preact"; import { Fragment, h, VNode } from "preact";
import { useEffect, useState } from "preact/hooks"; import { useEffect, useState } from "preact/hooks";
import { ErrorMessage } from "../components/ErrorMessage"; import { ErrorMessage } from "../components/ErrorMessage";
@ -93,7 +93,7 @@ export function PayPage({
undefined, undefined,
); );
const [payErrMsg, setPayErrMsg] = useState< const [payErrMsg, setPayErrMsg] = useState<
OperationFailedError | string | undefined TalerError | string | undefined
>(undefined); >(undefined);
const hook = useAsyncAsHook(async () => { const hook = useAsyncAsHook(async () => {

View File

@ -25,10 +25,8 @@ import {
AmountJson, AmountJson,
Amounts, Amounts,
ExchangeListItem, ExchangeListItem,
Translate,
WithdrawUriInfoResponse, WithdrawUriInfoResponse,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { OperationFailedError } from "@gnu-taler/taler-wallet-core";
import { Fragment, h, VNode } from "preact"; import { Fragment, h, VNode } from "preact";
import { useState } from "preact/hooks"; import { useState } from "preact/hooks";
import { Loading } from "../components/Loading"; import { Loading } from "../components/Loading";
@ -52,6 +50,7 @@ import {
import * as wxApi from "../wxApi"; import * as wxApi from "../wxApi";
import { TermsOfServiceSection } from "./TermsOfServiceSection"; import { TermsOfServiceSection } from "./TermsOfServiceSection";
import { useTranslationContext } from "../context/translation"; import { useTranslationContext } from "../context/translation";
import { TalerError } from "@gnu-taler/taler-wallet-core";
interface Props { interface Props {
talerWithdrawUri?: string; talerWithdrawUri?: string;
@ -85,9 +84,9 @@ export function View({
reviewed, reviewed,
}: ViewProps): VNode { }: ViewProps): VNode {
const { i18n } = useTranslationContext(); const { i18n } = useTranslationContext();
const [withdrawError, setWithdrawError] = useState< const [withdrawError, setWithdrawError] = useState<TalerError | undefined>(
OperationFailedError | undefined undefined,
>(undefined); );
const [confirmDisabled, setConfirmDisabled] = useState<boolean>(false); const [confirmDisabled, setConfirmDisabled] = useState<boolean>(false);
const needsReview = terms.status === "changed" || terms.status === "new"; const needsReview = terms.status === "changed" || terms.status === "new";
@ -109,7 +108,7 @@ export function View({
setConfirmDisabled(true); setConfirmDisabled(true);
await onWithdraw(); await onWithdraw();
} catch (e) { } catch (e) {
if (e instanceof OperationFailedError) { if (e instanceof TalerError) {
setWithdrawError(e); setWithdrawError(e);
} }
setConfirmDisabled(false); setConfirmDisabled(false);
@ -130,7 +129,7 @@ export function View({
Could not finish the withdrawal operation Could not finish the withdrawal operation
</i18n.Translate> </i18n.Translate>
} }
error={withdrawError.operationError} error={withdrawError.errorDetail}
/> />
)} )}

View File

@ -13,28 +13,32 @@
You should have received a copy of the GNU General Public License along with You should have received a copy of the GNU General Public License along with
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/> TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
*/ */
import { NotificationType, TalerErrorDetails } from "@gnu-taler/taler-util"; import {
NotificationType,
TalerErrorCode,
TalerErrorDetail,
} from "@gnu-taler/taler-util";
import { TalerError } from "@gnu-taler/taler-wallet-core";
import { useEffect, useState } from "preact/hooks"; import { useEffect, useState } from "preact/hooks";
import * as wxApi from "../wxApi"; import * as wxApi from "../wxApi";
import { OperationFailedError } from "@gnu-taler/taler-wallet-core";
interface HookOk<T> { interface HookOk<T> {
hasError: false; hasError: false;
response: T; response: T;
} }
export type HookError = HookGenericError | HookOperationalError export type HookError = HookGenericError | HookOperationalError;
export interface HookGenericError { export interface HookGenericError {
hasError: true; hasError: true;
operational: false, operational: false;
message: string; message: string;
} }
export interface HookOperationalError { export interface HookOperationalError {
hasError: true; hasError: true;
operational: true, operational: true;
details: TalerErrorDetails; details: TalerErrorDetail;
} }
export type HookResponse<T> = HookOk<T> | HookError | undefined; export type HookResponse<T> = HookOk<T> | HookError | undefined;
@ -51,10 +55,18 @@ export function useAsyncAsHook<T>(
const response = await fn(); const response = await fn();
setHookResponse({ hasError: false, response }); setHookResponse({ hasError: false, response });
} catch (e) { } catch (e) {
if (e instanceof OperationFailedError) { if (e instanceof TalerError) {
setHookResponse({ hasError: true, operational: true, details: e.operationError }); setHookResponse({
hasError: true,
operational: true,
details: e.errorDetail,
});
} else if (e instanceof Error) { } else if (e instanceof Error) {
setHookResponse({ hasError: true, operational: false, message: e.message }); setHookResponse({
hasError: true,
operational: false,
message: e.message,
});
} }
} }
} }

View File

@ -23,7 +23,7 @@ import {
HttpRequestLibrary, HttpRequestLibrary,
HttpRequestOptions, HttpRequestOptions,
HttpResponse, HttpResponse,
OperationFailedError, TalerError,
} from "@gnu-taler/taler-wallet-core"; } from "@gnu-taler/taler-wallet-core";
/** /**
@ -44,14 +44,14 @@ export class ServiceWorkerHttpLib implements HttpRequestLibrary {
if (this.throttlingEnabled && this.throttle.applyThrottle(requestUrl)) { if (this.throttlingEnabled && this.throttle.applyThrottle(requestUrl)) {
const parsedUrl = new URL(requestUrl); const parsedUrl = new URL(requestUrl);
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_HTTP_REQUEST_THROTTLED, TalerErrorCode.WALLET_HTTP_REQUEST_THROTTLED,
`request to origin ${parsedUrl.origin} was throttled`,
{ {
requestMethod, requestMethod,
requestUrl, requestUrl,
throttleStats: this.throttle.getThrottleStats(requestUrl), throttleStats: this.throttle.getThrottleStats(requestUrl),
}, },
`request to origin ${parsedUrl.origin} was throttled`,
); );
} }
@ -107,13 +107,13 @@ function makeTextHandler(response: Response, requestUrl: string) {
try { try {
respText = await response.text(); respText = await response.text();
} catch (e) { } catch (e) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Invalid JSON from HTTP response",
{ {
requestUrl, requestUrl,
httpStatusCode: response.status, httpStatusCode: response.status,
}, },
"Invalid JSON from HTTP response",
); );
} }
return respText; return respText;
@ -126,23 +126,23 @@ function makeJsonHandler(response: Response, requestUrl: string) {
try { try {
responseJson = await response.json(); responseJson = await response.json();
} catch (e) { } catch (e) {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Invalid JSON from HTTP response",
{ {
requestUrl, requestUrl,
httpStatusCode: response.status, httpStatusCode: response.status,
}, },
"Invalid JSON from HTTP response",
); );
} }
if (responseJson === null || typeof responseJson !== "object") { if (responseJson === null || typeof responseJson !== "object") {
throw OperationFailedError.fromCode( throw TalerError.fromDetail(
TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE, TalerErrorCode.WALLET_RECEIVED_MALFORMED_RESPONSE,
"Invalid JSON from HTTP response",
{ {
requestUrl, requestUrl,
httpStatusCode: response.status, httpStatusCode: response.status,
}, },
"Invalid JSON from HTTP response",
); );
} }
return responseJson; return responseJson;

View File

@ -23,19 +23,41 @@
*/ */
import { import {
AcceptExchangeTosRequest, AcceptExchangeTosRequest,
AcceptManualWithdrawalResult, AcceptTipRequest, AcceptWithdrawalResponse, AcceptManualWithdrawalResult,
AddExchangeRequest, AmountString, ApplyRefundResponse, BalancesResponse, CoinDumpJson, ConfirmPayResult, AcceptTipRequest,
CoreApiResponse, CreateDepositGroupRequest, CreateDepositGroupResponse, DeleteTransactionRequest, ExchangesListRespose, AcceptWithdrawalResponse,
GetExchangeTosResult, GetExchangeWithdrawalInfo, AddExchangeRequest,
AmountString,
ApplyRefundResponse,
BalancesResponse,
CoinDumpJson,
ConfirmPayResult,
CoreApiResponse,
CreateDepositGroupRequest,
CreateDepositGroupResponse,
DeleteTransactionRequest,
ExchangesListRespose,
GetExchangeTosResult,
GetExchangeWithdrawalInfo,
GetFeeForDepositRequest, GetFeeForDepositRequest,
GetWithdrawalDetailsForUriRequest, KnownBankAccounts, NotificationType, PreparePayResult, PrepareTipRequest, GetWithdrawalDetailsForUriRequest,
PrepareTipResult, RetryTransactionRequest, KnownBankAccounts,
SetWalletDeviceIdRequest, TransactionsResponse, WalletDiagnostics, WithdrawUriInfoResponse NotificationType,
PreparePayResult,
PrepareTipRequest,
PrepareTipResult,
RetryTransactionRequest,
SetWalletDeviceIdRequest,
TransactionsResponse,
WalletDiagnostics,
WithdrawUriInfoResponse,
} from "@gnu-taler/taler-util"; } from "@gnu-taler/taler-util";
import { import {
AddBackupProviderRequest, BackupInfo, OperationFailedError, AddBackupProviderRequest,
BackupInfo,
PendingOperationsResponse, PendingOperationsResponse,
RemoveBackupProviderRequest RemoveBackupProviderRequest,
TalerError,
} from "@gnu-taler/taler-wallet-core"; } from "@gnu-taler/taler-wallet-core";
import { DepositFee } from "@gnu-taler/taler-wallet-core/src/operations/deposits"; import { DepositFee } from "@gnu-taler/taler-wallet-core/src/operations/deposits";
import type { ExchangeWithdrawDetails } from "@gnu-taler/taler-wallet-core/src/operations/withdraw"; import type { ExchangeWithdrawDetails } from "@gnu-taler/taler-wallet-core/src/operations/withdraw";
@ -88,7 +110,7 @@ async function callBackend(operation: string, payload: any): Promise<any> {
console.log("got response", resp); console.log("got response", resp);
const r = resp as CoreApiResponse; const r = resp as CoreApiResponse;
if (r.type === "error") { if (r.type === "error") {
reject(new OperationFailedError(r.error)); reject(TalerError.fromUncheckedDetail(r.error));
return; return;
} }
resolve(r.result); resolve(r.result);
@ -127,15 +149,23 @@ export function resetDb(): Promise<void> {
return callBackend("reset-db", {}); return callBackend("reset-db", {});
} }
export function getFeeForDeposit(depositPaytoUri: string, amount: AmountString): Promise<DepositFee> { export function getFeeForDeposit(
depositPaytoUri: string,
amount: AmountString,
): Promise<DepositFee> {
return callBackend("getFeeForDeposit", { return callBackend("getFeeForDeposit", {
depositPaytoUri, amount depositPaytoUri,
amount,
} as GetFeeForDepositRequest); } as GetFeeForDepositRequest);
} }
export function createDepositGroup(depositPaytoUri: string, amount: AmountString): Promise<CreateDepositGroupResponse> { export function createDepositGroup(
depositPaytoUri: string,
amount: AmountString,
): Promise<CreateDepositGroupResponse> {
return callBackend("createDepositGroup", { return callBackend("createDepositGroup", {
depositPaytoUri, amount depositPaytoUri,
amount,
} as CreateDepositGroupRequest); } as CreateDepositGroupRequest);
} }
@ -190,7 +220,9 @@ export function listKnownCurrencies(): Promise<ListOfKnownCurrencies> {
export function listExchanges(): Promise<ExchangesListRespose> { export function listExchanges(): Promise<ExchangesListRespose> {
return callBackend("listExchanges", {}); return callBackend("listExchanges", {});
} }
export function listKnownBankAccounts(currency?: string): Promise<KnownBankAccounts> { export function listKnownBankAccounts(
currency?: string,
): Promise<KnownBankAccounts> {
return callBackend("listKnownBankAccounts", { currency }); return callBackend("listKnownBankAccounts", { currency });
} }
@ -387,14 +419,17 @@ export function exportDB(): Promise<any> {
} }
export function importDB(dump: any): Promise<void> { export function importDB(dump: any): Promise<void> {
return callBackend("importDb", { dump }) return callBackend("importDb", { dump });
} }
export function onUpdateNotification(messageTypes: Array<NotificationType>, doCallback: () => void): () => void { export function onUpdateNotification(
messageTypes: Array<NotificationType>,
doCallback: () => void,
): () => void {
// eslint-disable-next-line no-undef // eslint-disable-next-line no-undef
const port = chrome.runtime.connect({ name: "notifications" }); const port = chrome.runtime.connect({ name: "notifications" });
const listener = (message: MessageFromBackend): void => { const listener = (message: MessageFromBackend): void => {
const shouldNotify = messageTypes.includes(message.type) const shouldNotify = messageTypes.includes(message.type);
if (shouldNotify) { if (shouldNotify) {
doCallback(); doCallback();
} }

View File

@ -35,7 +35,7 @@ import {
import { import {
DbAccess, DbAccess,
deleteTalerDatabase, deleteTalerDatabase,
makeErrorDetails, makeErrorDetail,
OpenedPromise, OpenedPromise,
openPromise, openPromise,
openTalerDatabase, openTalerDatabase,
@ -167,10 +167,10 @@ async function dispatch(
type: "error", type: "error",
id: req.id, id: req.id,
operation: req.operation, operation: req.operation,
error: makeErrorDetails( error: makeErrorDetail(
TalerErrorCode.WALLET_CORE_NOT_AVAILABLE, TalerErrorCode.WALLET_CORE_NOT_AVAILABLE,
"wallet core not available",
{}, {},
"wallet core not available",
), ),
}; };
break; break;
@ -233,7 +233,10 @@ function makeSyncWalletRedirect(
const tab = await getTab(tabId); const tab = await getTab(tabId);
if (tab.url === oldUrl) { if (tab.url === oldUrl) {
console.log("redirecting to", innerUrl.href); console.log("redirecting to", innerUrl.href);
chrome.tabs.update(tabId, { url: innerUrl.href, loadReplace: true } as any); chrome.tabs.update(tabId, {
url: innerUrl.href,
loadReplace: true,
} as any);
} }
}; };
doit(); doit();