wallet: improve error handling and error codes
This commit is contained in:
parent
f8d12f7b0d
commit
5d23eb3635
@ -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 {
|
||||||
|
@ -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,
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -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 =
|
||||||
|
@ -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 {
|
||||||
|
@ -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;
|
||||||
},
|
},
|
||||||
|
@ -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");
|
||||||
|
@ -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);
|
||||||
|
@ -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,
|
||||||
|
@ -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,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -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,
|
||||||
});
|
});
|
||||||
|
@ -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();
|
||||||
|
@ -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;
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -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,
|
||||||
|
@ -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.
|
||||||
|
@ -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,
|
||||||
|
},
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
|
@ -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[];
|
||||||
|
@ -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),
|
||||||
|
@ -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(
|
||||||
|
@ -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);
|
||||||
|
@ -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),
|
||||||
|
@ -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),
|
||||||
|
@ -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),
|
||||||
|
@ -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",
|
|
||||||
{},
|
{},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
});
|
});
|
||||||
|
@ -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)`,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
|
@ -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}`,
|
|
||||||
{},
|
|
||||||
),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
|
@ -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 && (
|
||||||
|
@ -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>
|
||||||
|
@ -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 () => {
|
||||||
|
@ -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}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
@ -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,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
|
Loading…
Reference in New Issue
Block a user