Skip to content

Commit 7708a4b

Browse files
feat: added delegateFirstStakeCredential method to TxBuilder
1 parent 5887d40 commit 7708a4b

File tree

3 files changed

+77
-1
lines changed

3 files changed

+77
-1
lines changed

packages/tx-construction/src/tx-builder/TxBuilder.ts

+65-1
Original file line numberDiff line numberDiff line change
@@ -103,6 +103,8 @@ export class GenericTxBuilder implements TxBuilder {
103103
#logger: Logger;
104104
#handleProvider?: HandleProvider;
105105
#handleResolutions: HandleResolution[];
106+
#delegateFirstStakeCredConfig: Cardano.PoolId | null | undefined = undefined;
107+
106108
#customizeCb: CustomizeCb;
107109

108110
constructor(dependencies: TxBuilderDependencies) {
@@ -153,6 +155,11 @@ export class GenericTxBuilder implements TxBuilder {
153155
});
154156
}
155157

158+
delegateFirstStakeCredential(poolId: Cardano.PoolId | null): TxBuilder {
159+
this.#delegateFirstStakeCredConfig = poolId;
160+
return this;
161+
}
162+
156163
delegatePortfolio(portfolio: Cardano.Cip17DelegationPortfolio | null): TxBuilder {
157164
if (!this.#dependencies.bip32Account) throw new Error('BIP32 account is required to delegate portfolio.');
158165

@@ -212,7 +219,7 @@ export class GenericTxBuilder implements TxBuilder {
212219
try {
213220
const rewardAccountsWithWeights = this.#dependencies.bip32Account
214221
? await this.#delegatePortfolio()
215-
: new Map();
222+
: await this.#delegateFirstStakeCredential();
216223

217224
await this.#validateOutputs();
218225
// Take a snapshot of returned properties,
@@ -346,6 +353,63 @@ export class GenericTxBuilder implements TxBuilder {
346353
}
347354
}
348355

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+
349413
async #delegatePortfolio(): Promise<RewardAccountsAndWeights> {
350414
const rewardAccountsWithWeights: RewardAccountsAndWeights = new Map();
351415
if (!this.#requestedPortfolio) {

packages/tx-construction/src/tx-builder/types.ts

+8
Original file line numberDiff line numberDiff line change
@@ -206,6 +206,14 @@ export interface TxBuilder {
206206
* @returns {OutputBuilder} {@link OutputBuilder} util for building transaction outputs.
207207
*/
208208
buildOutput(txOut?: PartialTxOut): OutputBuilder;
209+
210+
/**
211+
* Delegates the first stake credential controlled by the wallet.
212+
*
213+
* @param poolId The pool to delegate to, or null if the stake credential should be de-registered.
214+
*/
215+
delegateFirstStakeCredential(poolId: Cardano.PoolId | null): TxBuilder;
216+
209217
/**
210218
* Configures the transaction to include all the certificates needed to delegate to the pools from the portfolio.
211219
*

packages/web-extension/src/observableWallet/util.ts

+4
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,10 @@ export const txBuilderProperties: RemoteApiProperties<Omit<TxBuilder, 'customize
5454
getApiProperties: () => outputBuilderProperties,
5555
propType: RemoteApiPropertyType.ApiFactory
5656
},
57+
delegateFirstStakeCredential: {
58+
getApiProperties: () => txBuilderProperties,
59+
propType: RemoteApiPropertyType.ApiFactory
60+
},
5761
delegatePortfolio: {
5862
getApiProperties: () => txBuilderProperties,
5963
propType: RemoteApiPropertyType.ApiFactory

0 commit comments

Comments
 (0)