Skip to content

Commit 02543e8

Browse files
authored
Merge pull request #1578 from input-output-hk/fix/settle-before-resolving-cip30-utxo
fix(wallet): await for wallet settle before resolving cip30 utxo LW-12197
2 parents fb3ca11 + c478116 commit 02543e8

File tree

2 files changed

+35
-5
lines changed

2 files changed

+35
-5
lines changed

packages/wallet/src/cip30.ts

+17-4
Original file line numberDiff line numberDiff line change
@@ -17,13 +17,13 @@ import {
1717
WalletApiExtension,
1818
WithSenderContext
1919
} from '@cardano-sdk/dapp-connector';
20-
import { Cardano, Serialization, coalesceValueQuantities } from '@cardano-sdk/core';
20+
import { Cardano, Milliseconds, Serialization, coalesceValueQuantities } from '@cardano-sdk/core';
2121
import { Ed25519KeyHashHex, Hash28ByteBase16 } from '@cardano-sdk/crypto';
2222
import { HexBlob, ManagedFreeableScope } from '@cardano-sdk/util';
2323
import { InputSelectionError, InputSelectionFailure } from '@cardano-sdk/input-selection';
2424
import { Logger } from 'ts-log';
2525
import { MessageSender } from '@cardano-sdk/key-management';
26-
import { Observable, firstValueFrom, from, map, mergeMap, race, throwError } from 'rxjs';
26+
import { Observable, filter, firstValueFrom, from, map, mergeMap, race, throwError, timeout } from 'rxjs';
2727
import { ObservableWallet } from './types';
2828
import { requiresForeignSignatures } from './services';
2929
import uniq from 'lodash/uniq.js';
@@ -84,6 +84,16 @@ export type CallbackConfirmation = {
8484
getCollateral?: GetCollateralCallback;
8585
};
8686

87+
const firstValueFromTimed = <T>(observable$: Observable<T>, timeoutAfter: Milliseconds) =>
88+
firstValueFrom(
89+
observable$.pipe(
90+
timeout({ each: timeoutAfter, with: () => throwError(() => new ApiError(APIErrorCode.InternalError, 'Timeout')) })
91+
)
92+
);
93+
94+
const waitForWalletStateSettle = (wallet: ObservableWallet, syncTimeout = Milliseconds(120_000)) =>
95+
firstValueFromTimed(wallet.syncStatus.isSettled$.pipe(filter((isSettled) => isSettled)), syncTimeout);
96+
8797
const mapCallbackFailure = (err: unknown, logger: Logger): false => {
8898
logger.error(err);
8999
return false;
@@ -294,6 +304,7 @@ const baseCip30WalletApi = (
294304
logger.debug('getting balance');
295305
try {
296306
const wallet = await firstValueFrom(wallet$);
307+
await waitForWalletStateSettle(wallet);
297308
const value = await firstValueFrom(wallet.balance.utxo.available$);
298309
return Serialization.Value.fromCore(value).toCbor();
299310
} catch (error) {
@@ -332,6 +343,7 @@ const baseCip30WalletApi = (
332343
> => {
333344
logger.debug('getting collateral');
334345
const wallet = await firstValueFrom(wallet$);
346+
await waitForWalletStateSettle(wallet);
335347
let unspendables = await getSortedUtxos(wallet.utxo.unspendable$);
336348
const available = await getSortedUtxos(wallet.utxo.available$);
337349
// No available unspendable UTxO
@@ -441,6 +453,7 @@ const baseCip30WalletApi = (
441453
const scope = new ManagedFreeableScope();
442454
try {
443455
const wallet = await firstValueFrom(wallet$);
456+
await waitForWalletStateSettle(wallet);
444457
let utxos = amount
445458
? await selectUtxo(wallet, parseValueCbor(amount).toCore(), !!paginate)
446459
: await firstValueFrom(wallet.utxo.available$);
@@ -583,7 +596,7 @@ const baseCip30WalletApi = (
583596

584597
const getPubStakeKeys = async (
585598
wallet$: Observable<ObservableWallet>,
586-
filter: Cardano.StakeCredentialStatus.Registered | Cardano.StakeCredentialStatus.Unregistered
599+
filterCredentialStatus: Cardano.StakeCredentialStatus.Registered | Cardano.StakeCredentialStatus.Unregistered
587600
) => {
588601
const wallet = await firstValueFrom(wallet$);
589602
return firstValueFrom(
@@ -595,7 +608,7 @@ const getPubStakeKeys = async (
595608
credentialStatus === Cardano.StakeCredentialStatus.Registering
596609
? Cardano.StakeCredentialStatus.Registered
597610
: Cardano.StakeCredentialStatus.Unregistered;
598-
return filter === status;
611+
return filterCredentialStatus === status;
599612
})
600613
),
601614
map((keys) => keys.map(({ publicStakeKey }) => publicStakeKey))

packages/wallet/test/integration/cip30mapping.test.ts

+18-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ import { Ed25519KeyHashHex } from '@cardano-sdk/crypto';
3131
import { HexBlob, ManagedFreeableScope } from '@cardano-sdk/util';
3232
import { InMemoryUnspendableUtxoStore, createInMemoryWalletStores } from '../../src/persistence';
3333
import { InitializeTxProps, InitializeTxResult } from '@cardano-sdk/tx-construction';
34-
import { NEVER, firstValueFrom, of } from 'rxjs';
34+
import { NEVER, Observable, delay, firstValueFrom, of } from 'rxjs';
3535
import { Providers, createWallet } from './util';
3636
import { address_0_0, address_1_0, rewardAccount_0, rewardAccount_1 } from '../services/ChangeAddress/testData';
3737
import { buildDRepAddressFromDRepKey, signTx, waitForWalletStateSettle } from '../util';
@@ -190,6 +190,23 @@ describe('cip30', () => {
190190
).not.toThrow();
191191
});
192192

193+
it('subscribes to syncStatus.isSettled$ before utxo.available$', async () => {
194+
const isSettled$ = wallet.syncStatus.isSettled$;
195+
const available$ = wallet.utxo.available$;
196+
let isSettledSubscribedAt: number;
197+
let availableSubscribedAt: number;
198+
wallet.syncStatus.isSettled$ = new Observable((observer) => {
199+
if (!isSettledSubscribedAt) isSettledSubscribedAt = Date.now();
200+
return isSettled$.pipe(delay(1)).subscribe(observer);
201+
});
202+
wallet.utxo.available$ = new Observable((observer) => {
203+
if (!availableSubscribedAt) availableSubscribedAt = Date.now();
204+
return available$.subscribe(observer);
205+
});
206+
await api.getUtxos(context);
207+
expect(isSettledSubscribedAt!).toBeLessThan(availableSubscribedAt!);
208+
});
209+
193210
describe('with "amount" argument', () => {
194211
const getUtxoFiltered = async (coins: Cardano.Lovelace, tslaQuantity?: bigint, paginate?: Paginate) => {
195212
const filterAmountValue = new Serialization.Value(coins);

0 commit comments

Comments
 (0)