@@ -103,6 +103,8 @@ export class GenericTxBuilder implements TxBuilder {
103
103
#logger: Logger ;
104
104
#handleProvider?: HandleProvider ;
105
105
#handleResolutions: HandleResolution [ ] ;
106
+ #delegateFirstStakeCredConfig: Cardano . PoolId | null | undefined = undefined ;
107
+
106
108
#customizeCb: CustomizeCb ;
107
109
108
110
constructor ( dependencies : TxBuilderDependencies ) {
@@ -153,6 +155,11 @@ export class GenericTxBuilder implements TxBuilder {
153
155
} ) ;
154
156
}
155
157
158
+ delegateFirstStakeCredential ( poolId : Cardano . PoolId | null ) : TxBuilder {
159
+ this . #delegateFirstStakeCredConfig = poolId ;
160
+ return this ;
161
+ }
162
+
156
163
delegatePortfolio ( portfolio : Cardano . Cip17DelegationPortfolio | null ) : TxBuilder {
157
164
if ( ! this . #dependencies. bip32Account ) throw new Error ( 'BIP32 account is required to delegate portfolio.' ) ;
158
165
@@ -212,7 +219,7 @@ export class GenericTxBuilder implements TxBuilder {
212
219
try {
213
220
const rewardAccountsWithWeights = this . #dependencies. bip32Account
214
221
? await this . #delegatePortfolio( )
215
- : new Map ( ) ;
222
+ : await this . #delegateFirstStakeCredential ( ) ;
216
223
217
224
await this . #validateOutputs( ) ;
218
225
// Take a snapshot of returned properties,
@@ -346,6 +353,63 @@ export class GenericTxBuilder implements TxBuilder {
346
353
}
347
354
}
348
355
356
+ async #delegateFirstStakeCredential( ) : Promise < RewardAccountsAndWeights > {
357
+ const rewardAccountsWithWeights : RewardAccountsAndWeights = new Map ( ) ;
358
+ if ( this . #delegateFirstStakeCredConfig === undefined ) {
359
+ // Delegation was not configured by user
360
+ return Promise . resolve ( rewardAccountsWithWeights ) ;
361
+ }
362
+
363
+ const rewardAccounts = await this . #dependencies. txBuilderProviders . rewardAccounts ( ) ;
364
+
365
+ if ( ! rewardAccounts ?. length ) {
366
+ // This shouldn't happen
367
+ throw new Error ( 'Could not find any rewardAccount.' ) ;
368
+ }
369
+
370
+ const rewardAccount = rewardAccounts [ 0 ] ;
371
+
372
+ this . partialTxBody = { ...this . partialTxBody , certificates : [ ] } ;
373
+
374
+ const stakeCredential = Cardano . Address . fromBech32 ( rewardAccount . address ) . asReward ( ) ?. getPaymentCredential ( ) ;
375
+
376
+ if ( ! stakeCredential ) {
377
+ // This shouldn't happen
378
+ throw new Error ( `Invalid credential ${ stakeCredential } .` ) ;
379
+ }
380
+
381
+ if ( this . #delegateFirstStakeCredConfig === null ) {
382
+ // Deregister scenario
383
+ if ( rewardAccount . credentialStatus === Cardano . StakeCredentialStatus . Unregistered ) {
384
+ this . #logger. warn ( 'Stake key not registered.' , rewardAccount . address , rewardAccount . credentialStatus ) ;
385
+ } else {
386
+ this . partialTxBody . certificates ! . push ( {
387
+ __typename : Cardano . CertificateType . StakeDeregistration ,
388
+ stakeCredential
389
+ } ) ;
390
+ }
391
+ } else {
392
+ // Register and delegate scenario
393
+ if ( rewardAccount . credentialStatus !== Cardano . StakeCredentialStatus . Unregistered ) {
394
+ this . #logger. debug ( 'Stake key already registered' , rewardAccount . address , rewardAccount . credentialStatus ) ;
395
+ } else {
396
+ this . partialTxBody . certificates ! . push ( {
397
+ __typename : Cardano . CertificateType . StakeRegistration ,
398
+ stakeCredential
399
+ } ) ;
400
+ }
401
+
402
+ rewardAccountsWithWeights . set ( rewardAccount . address , 1 ) ;
403
+ this . partialTxBody . certificates ! . push ( {
404
+ __typename : Cardano . CertificateType . StakeDelegation ,
405
+ poolId : this . #delegateFirstStakeCredConfig,
406
+ stakeCredential
407
+ } ) ;
408
+ }
409
+
410
+ return rewardAccountsWithWeights ;
411
+ }
412
+
349
413
async #delegatePortfolio( ) : Promise < RewardAccountsAndWeights > {
350
414
const rewardAccountsWithWeights : RewardAccountsAndWeights = new Map ( ) ;
351
415
if ( ! this . #requestedPortfolio) {
0 commit comments