Press n or j to go to the next uncovered block, b, p or k for the previous block.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 | 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x | import { defineProperties } from "../utils/index.js"; import { AbstractSigner } from "./abstract-signer.js"; import type { TypedDataDomain, TypedDataField } from "../hash/index.js"; import type { BlockTag, Provider, TransactionRequest, TransactionResponse } from "./provider.js"; import type { Signer } from "./signer.js"; /** * A **NonceManager** wraps another [[Signer]] and automatically manages * the nonce, ensuring serialized and sequential nonces are used during * transaction. */ export class NonceManager extends AbstractSigner { /** * The Signer being managed. */ signer!: Signer; #noncePromise: null | Promise<number>; #delta: number; /** * Creates a new **NonceManager** to manage %%signer%%. */ constructor(signer: Signer) { super(signer.provider); defineProperties<NonceManager>(this, { signer }); this.#noncePromise = null; this.#delta = 0; } async getAddress(): Promise<string> { return this.signer.getAddress(); } connect(provider: null | Provider): NonceManager { return new NonceManager(this.signer.connect(provider)); } async getNonce(blockTag?: BlockTag): Promise<number> { if (blockTag === "pending") { if (this.#noncePromise == null) { this.#noncePromise = super.getNonce("pending"); } const delta = this.#delta; return (await this.#noncePromise) + delta; } return super.getNonce(blockTag); } /** * Manually increment the nonce. This may be useful when managng * offline transactions. */ increment(): void { this.#delta++; } /** * Resets the nonce, causing the **NonceManager** to reload the current * nonce from the blockchain on the next transaction. */ reset(): void { this.#delta = 0; this.#noncePromise = null; } async sendTransaction(tx: TransactionRequest): Promise<TransactionResponse> { const noncePromise = this.getNonce("pending"); this.increment(); tx = await this.signer.populateTransaction(tx); tx.nonce = await noncePromise; // @TODO: Maybe handle interesting/recoverable errors? // Like don't increment if the tx was certainly not sent return await this.signer.sendTransaction(tx); } signTransaction(tx: TransactionRequest): Promise<string> { return this.signer.signTransaction(tx); } signMessage(message: string | Uint8Array): Promise<string> { return this.signer.signMessage(message); } signTypedData(domain: TypedDataDomain, types: Record<string, Array<TypedDataField>>, value: Record<string, any>): Promise<string> { return this.signer.signTypedData(domain, types, value); } } |