simplify task loop, test coin suspension
This commit is contained in:
parent
7383b89cab
commit
e35c2f581b
@ -32,7 +32,6 @@ import {
|
||||
Headers,
|
||||
WALLET_EXCHANGE_PROTOCOL_VERSION,
|
||||
WALLET_MERCHANT_PROTOCOL_VERSION,
|
||||
runRetryLoop,
|
||||
Wallet,
|
||||
} from "@gnu-taler/taler-wallet-core";
|
||||
|
||||
|
@ -33,7 +33,6 @@ import {
|
||||
codecForList,
|
||||
codecForString,
|
||||
Logger,
|
||||
WithdrawalType,
|
||||
} from "@gnu-taler/taler-util";
|
||||
import {
|
||||
NodeHttpLib,
|
||||
@ -45,10 +44,6 @@ import {
|
||||
NodeThreadCryptoWorkerFactory,
|
||||
CryptoApi,
|
||||
walletCoreDebugFlags,
|
||||
handleCoreApiRequest,
|
||||
runPending,
|
||||
runUntilDone,
|
||||
getClientFromWalletState,
|
||||
WalletApiOperation,
|
||||
WalletCoreApiClient,
|
||||
Wallet,
|
||||
@ -314,8 +309,9 @@ walletCli
|
||||
.maybeOption("maxRetries", ["--max-retries"], clk.INT)
|
||||
.action(async (args) => {
|
||||
await withWallet(args, async (wallet) => {
|
||||
await wallet.ws.runUntilDone({
|
||||
await wallet.ws.runTaskLoop({
|
||||
maxRetries: args.finishPendingOpt.maxRetries,
|
||||
stopWhenDone: true,
|
||||
});
|
||||
wallet.ws.stop();
|
||||
});
|
||||
|
@ -22,8 +22,9 @@
|
||||
/**
|
||||
* Imports.
|
||||
*/
|
||||
import { Amounts } from "@gnu-taler/taler-util";
|
||||
import { WalletApiOperation } from "@gnu-taler/taler-wallet-core";
|
||||
import { CoinConfig, defaultCoinConfig } from "./denomStructures";
|
||||
import { CoinConfig, defaultCoinConfig } from "./denomStructures.js";
|
||||
import {
|
||||
BankService,
|
||||
ExchangeService,
|
||||
@ -31,8 +32,8 @@ import {
|
||||
MerchantService,
|
||||
setupDb,
|
||||
WalletCli,
|
||||
} from "./harness";
|
||||
import { SimpleTestEnvironment } from "./helpers";
|
||||
} from "./harness.js";
|
||||
import { SimpleTestEnvironment } from "./helpers.js";
|
||||
|
||||
const merchantAuthToken = "secret-token:sandbox";
|
||||
|
||||
@ -162,6 +163,63 @@ export async function runWallettestingTest(t: GlobalTestState) {
|
||||
|
||||
t.assertDeepEqual(txTypes, ["withdrawal", "payment"]);
|
||||
|
||||
wallet.deleteDatabase();
|
||||
|
||||
await wallet.client.call(WalletApiOperation.WithdrawTestBalance, {
|
||||
amount: "TESTKUDOS:10",
|
||||
bankBaseUrl: bank.baseUrl,
|
||||
exchangeBaseUrl: exchange.baseUrl,
|
||||
});
|
||||
|
||||
await wallet.runUntilDone();
|
||||
|
||||
const coinDump = await wallet.client.call(WalletApiOperation.DumpCoins, {});
|
||||
|
||||
console.log("coin dump:", JSON.stringify(coinDump, undefined, 2));
|
||||
|
||||
let susp: string | undefined;
|
||||
{
|
||||
for (const c of coinDump.coins) {
|
||||
if (0 === Amounts.cmp(c.remaining_value, "TESTKUDOS:8")) {
|
||||
susp = c.coin_pub;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
t.assertTrue(susp !== undefined);
|
||||
|
||||
console.log("suspending coin");
|
||||
|
||||
await wallet.client.call(WalletApiOperation.SetCoinSuspended, {
|
||||
coinPub: susp,
|
||||
suspended: true,
|
||||
});
|
||||
|
||||
// This should fail, as we've suspended a coin that we need
|
||||
// to pay.
|
||||
await t.assertThrowsAsync(async () => {
|
||||
await wallet.client.call(WalletApiOperation.TestPay, {
|
||||
amount: "TESTKUDOS:5",
|
||||
merchantAuthToken: merchantAuthToken,
|
||||
merchantBaseUrl: merchant.makeInstanceBaseUrl(),
|
||||
summary: "foo",
|
||||
});
|
||||
});
|
||||
|
||||
console.log("unsuspending coin");
|
||||
|
||||
await wallet.client.call(WalletApiOperation.SetCoinSuspended, {
|
||||
coinPub: susp,
|
||||
suspended: false,
|
||||
});
|
||||
|
||||
await wallet.client.call(WalletApiOperation.TestPay, {
|
||||
amount: "TESTKUDOS:5",
|
||||
merchantAuthToken: merchantAuthToken,
|
||||
merchantBaseUrl: merchant.makeInstanceBaseUrl(),
|
||||
summary: "foo",
|
||||
});
|
||||
|
||||
await t.shutdown();
|
||||
}
|
||||
|
||||
|
@ -116,9 +116,15 @@ export interface InternalWalletState {
|
||||
cryptoApi: CryptoApi;
|
||||
|
||||
timerGroup: TimerGroup;
|
||||
latch: AsyncCondition;
|
||||
stopped: boolean;
|
||||
memoRunRetryLoop: AsyncOpMemoSingle<void>;
|
||||
|
||||
/**
|
||||
* Asynchronous condition to interrupt the sleep of the
|
||||
* retry loop.
|
||||
*
|
||||
* Used to allow processing of new work faster.
|
||||
*/
|
||||
latch: AsyncCondition;
|
||||
|
||||
listeners: NotificationListener[];
|
||||
|
||||
|
@ -205,10 +205,7 @@ export async function getEffectiveDepositAmount(
|
||||
return Amounts.sub(Amounts.sum(amt).amount, Amounts.sum(fees).amount).amount;
|
||||
}
|
||||
|
||||
export function isSpendableCoin(
|
||||
coin: CoinRecord,
|
||||
denom: DenominationRecord,
|
||||
): boolean {
|
||||
function isSpendableCoin(coin: CoinRecord, denom: DenominationRecord): boolean {
|
||||
if (coin.suspended) {
|
||||
return false;
|
||||
}
|
||||
@ -721,7 +718,9 @@ async function processDownloadProposalImpl(
|
||||
);
|
||||
|
||||
if (!isWellFormed) {
|
||||
logger.trace(`malformed contract terms: ${j2s(proposalResp.contract_terms)}`);
|
||||
logger.trace(
|
||||
`malformed contract terms: ${j2s(proposalResp.contract_terms)}`,
|
||||
);
|
||||
const err = makeErrorDetails(
|
||||
TalerErrorCode.WALLET_CONTRACT_TERMS_MALFORMED,
|
||||
"validation for well-formedness failed",
|
||||
|
@ -276,81 +276,31 @@ export async function runPending(
|
||||
}
|
||||
}
|
||||
|
||||
export interface RetryLoopOpts {
|
||||
/**
|
||||
* Stop when the number of retries is exceeded for any pending
|
||||
* operation.
|
||||
*/
|
||||
maxRetries?: number;
|
||||
|
||||
/**
|
||||
* Stop the retry loop when all lifeness-giving pending operations
|
||||
* are done.
|
||||
*
|
||||
* Defaults to false.
|
||||
*/
|
||||
stopWhenDone?: boolean;
|
||||
}
|
||||
|
||||
/**
|
||||
* Run the wallet until there are no more pending operations that give
|
||||
* liveness left. The wallet will be in a stopped state when this function
|
||||
* returns without resolving to an exception.
|
||||
* Main retry loop of the wallet.
|
||||
*
|
||||
* Looks up pending operations from the wallet, runs them, repeat.
|
||||
*/
|
||||
export async function runUntilDone(
|
||||
async function runTaskLoop(
|
||||
ws: InternalWalletState,
|
||||
req: {
|
||||
maxRetries?: number;
|
||||
} = {},
|
||||
opts: RetryLoopOpts = {},
|
||||
): Promise<void> {
|
||||
let done = false;
|
||||
const p = new Promise<void>((resolve, reject) => {
|
||||
// Monitor for conditions that means we're done or we
|
||||
// should quit with an error (due to exceeded retries).
|
||||
ws.addNotificationListener((n) => {
|
||||
if (done) {
|
||||
return;
|
||||
}
|
||||
if (
|
||||
n.type === NotificationType.WaitingForRetry &&
|
||||
n.numGivingLiveness == 0
|
||||
) {
|
||||
done = true;
|
||||
logger.trace("no liveness-giving operations left");
|
||||
resolve();
|
||||
}
|
||||
const maxRetries = req.maxRetries;
|
||||
if (!maxRetries) {
|
||||
return;
|
||||
}
|
||||
getPendingOperations(ws)
|
||||
.then((pending) => {
|
||||
for (const p of pending.pendingOperations) {
|
||||
if (p.retryInfo && p.retryInfo.retryCounter > maxRetries) {
|
||||
console.warn(
|
||||
`stopping, as ${maxRetries} retries are exceeded in an operation of type ${p.type}`,
|
||||
);
|
||||
ws.stop();
|
||||
done = true;
|
||||
resolve();
|
||||
}
|
||||
}
|
||||
})
|
||||
.catch((e) => {
|
||||
logger.error(e);
|
||||
reject(e);
|
||||
});
|
||||
});
|
||||
// Run this asynchronously
|
||||
runRetryLoop(ws).catch((e) => {
|
||||
logger.error("exception in wallet retry loop");
|
||||
reject(e);
|
||||
});
|
||||
});
|
||||
await p;
|
||||
}
|
||||
|
||||
/**
|
||||
* Process pending operations and wait for scheduled operations in
|
||||
* a loop until the wallet is stopped explicitly.
|
||||
*/
|
||||
export async function runRetryLoop(ws: InternalWalletState): Promise<void> {
|
||||
// Make sure we only run one main loop at a time.
|
||||
return ws.memoRunRetryLoop.memo(async () => {
|
||||
try {
|
||||
await runRetryLoopImpl(ws);
|
||||
} catch (e) {
|
||||
console.error("error during retry loop execution", e);
|
||||
throw e;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
async function runRetryLoopImpl(ws: InternalWalletState): Promise<void> {
|
||||
for (let iteration = 0; !ws.stopped; iteration++) {
|
||||
const pending = await getPendingOperations(ws);
|
||||
logger.trace(`pending operations: ${j2s(pending)}`);
|
||||
@ -365,7 +315,22 @@ async function runRetryLoopImpl(ws: InternalWalletState): Promise<void> {
|
||||
if (p.givesLifeness) {
|
||||
numGivingLiveness++;
|
||||
}
|
||||
|
||||
const maxRetries = opts.maxRetries;
|
||||
|
||||
if (maxRetries && p.retryInfo && p.retryInfo.retryCounter > maxRetries) {
|
||||
logger.warn(
|
||||
`stopping, as ${maxRetries} retries are exceeded in an operation of type ${p.type}`,
|
||||
);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (opts.stopWhenDone && numGivingLiveness === 0) {
|
||||
logger.warn(`stopping, as no pending operations have lifeness`);
|
||||
return;
|
||||
}
|
||||
|
||||
// Make sure that we run tasks that don't give lifeness at least
|
||||
// one time.
|
||||
if (iteration !== 0 && numDue === 0) {
|
||||
@ -993,19 +958,15 @@ export class Wallet {
|
||||
}
|
||||
|
||||
runRetryLoop(): Promise<void> {
|
||||
return runRetryLoop(this.ws);
|
||||
return runTaskLoop(this.ws);
|
||||
}
|
||||
|
||||
runPending(forceNow: boolean = false) {
|
||||
return runPending(this.ws, forceNow);
|
||||
}
|
||||
|
||||
runUntilDone(
|
||||
req: {
|
||||
maxRetries?: number;
|
||||
} = {},
|
||||
) {
|
||||
return runUntilDone(this.ws, req);
|
||||
runTaskLoop(opts: RetryLoopOpts) {
|
||||
return runTaskLoop(this.ws, opts);
|
||||
}
|
||||
|
||||
handleCoreApiRequest(
|
||||
@ -1035,7 +996,6 @@ class InternalWalletStateImpl implements InternalWalletState {
|
||||
timerGroup: TimerGroup = new TimerGroup();
|
||||
latch = new AsyncCondition();
|
||||
stopped = false;
|
||||
memoRunRetryLoop = new AsyncOpMemoSingle<void>();
|
||||
|
||||
listeners: NotificationListener[] = [];
|
||||
|
||||
@ -1102,7 +1062,7 @@ class InternalWalletStateImpl implements InternalWalletState {
|
||||
maxRetries?: number;
|
||||
} = {},
|
||||
): Promise<void> {
|
||||
runUntilDone(this, req);
|
||||
await runTaskLoop(this, { ...req, stopWhenDone: true });
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -33,12 +33,7 @@ import {
|
||||
deleteTalerDatabase,
|
||||
DbAccess,
|
||||
WalletStoresV1,
|
||||
handleCoreApiRequest,
|
||||
runRetryLoop,
|
||||
handleNotifyReserve,
|
||||
InternalWalletState,
|
||||
Wallet,
|
||||
WalletApiOperation,
|
||||
} from "@gnu-taler/taler-wallet-core";
|
||||
import {
|
||||
classifyTalerUri,
|
||||
|
Loading…
Reference in New Issue
Block a user