@@ -17,13 +17,13 @@ import {
17
17
WalletApiExtension ,
18
18
WithSenderContext
19
19
} from '@cardano-sdk/dapp-connector' ;
20
- import { Cardano , Serialization , coalesceValueQuantities } from '@cardano-sdk/core' ;
20
+ import { Cardano , Milliseconds , Serialization , coalesceValueQuantities } from '@cardano-sdk/core' ;
21
21
import { Ed25519KeyHashHex , Hash28ByteBase16 } from '@cardano-sdk/crypto' ;
22
22
import { HexBlob , ManagedFreeableScope } from '@cardano-sdk/util' ;
23
23
import { InputSelectionError , InputSelectionFailure } from '@cardano-sdk/input-selection' ;
24
24
import { Logger } from 'ts-log' ;
25
25
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' ;
27
27
import { ObservableWallet } from './types' ;
28
28
import { requiresForeignSignatures } from './services' ;
29
29
import uniq from 'lodash/uniq.js' ;
@@ -84,6 +84,16 @@ export type CallbackConfirmation = {
84
84
getCollateral ?: GetCollateralCallback ;
85
85
} ;
86
86
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
+
87
97
const mapCallbackFailure = ( err : unknown , logger : Logger ) : false => {
88
98
logger . error ( err ) ;
89
99
return false ;
@@ -294,6 +304,7 @@ const baseCip30WalletApi = (
294
304
logger . debug ( 'getting balance' ) ;
295
305
try {
296
306
const wallet = await firstValueFrom ( wallet$ ) ;
307
+ await waitForWalletStateSettle ( wallet ) ;
297
308
const value = await firstValueFrom ( wallet . balance . utxo . available$ ) ;
298
309
return Serialization . Value . fromCore ( value ) . toCbor ( ) ;
299
310
} catch ( error ) {
@@ -332,6 +343,7 @@ const baseCip30WalletApi = (
332
343
> => {
333
344
logger . debug ( 'getting collateral' ) ;
334
345
const wallet = await firstValueFrom ( wallet$ ) ;
346
+ await waitForWalletStateSettle ( wallet ) ;
335
347
let unspendables = await getSortedUtxos ( wallet . utxo . unspendable$ ) ;
336
348
const available = await getSortedUtxos ( wallet . utxo . available$ ) ;
337
349
// No available unspendable UTxO
@@ -441,6 +453,7 @@ const baseCip30WalletApi = (
441
453
const scope = new ManagedFreeableScope ( ) ;
442
454
try {
443
455
const wallet = await firstValueFrom ( wallet$ ) ;
456
+ await waitForWalletStateSettle ( wallet ) ;
444
457
let utxos = amount
445
458
? await selectUtxo ( wallet , parseValueCbor ( amount ) . toCore ( ) , ! ! paginate )
446
459
: await firstValueFrom ( wallet . utxo . available$ ) ;
@@ -583,7 +596,7 @@ const baseCip30WalletApi = (
583
596
584
597
const getPubStakeKeys = async (
585
598
wallet$ : Observable < ObservableWallet > ,
586
- filter : Cardano . StakeCredentialStatus . Registered | Cardano . StakeCredentialStatus . Unregistered
599
+ filterCredentialStatus : Cardano . StakeCredentialStatus . Registered | Cardano . StakeCredentialStatus . Unregistered
587
600
) => {
588
601
const wallet = await firstValueFrom ( wallet$ ) ;
589
602
return firstValueFrom (
@@ -595,7 +608,7 @@ const getPubStakeKeys = async (
595
608
credentialStatus === Cardano . StakeCredentialStatus . Registering
596
609
? Cardano . StakeCredentialStatus . Registered
597
610
: Cardano . StakeCredentialStatus . Unregistered ;
598
- return filter === status ;
611
+ return filterCredentialStatus === status ;
599
612
} )
600
613
) ,
601
614
map ( ( keys ) => keys . map ( ( { publicStakeKey } ) => publicStakeKey ) )
0 commit comments