"use strict" import { ethers } from "ethers"; import { version } from "./_version"; const logger = new ethers.utils.Logger(version); // @TODO: Keep a per-NonceManager pool of sent but unmined transactions for // rebroadcasting, in case we overrun the transaction pool export class NonceManager extends ethers.Signer { readonly signer: ethers.Signer; _initialPromise: Promise; _deltaCount: number; constructor(signer: ethers.Signer) { logger.checkNew(new.target, NonceManager); super(); this._deltaCount = 0; ethers.utils.defineReadOnly(this, "signer", signer); } get provider(): ethers.providers.Provider { return this.signer.provider; } connect(provider: ethers.providers.Provider): NonceManager { return new NonceManager(this.signer.connect(provider)); } getAddress(): Promise { return this.signer.getAddress(); } getTransactionCount(blockTag?: ethers.providers.BlockTag): Promise { if (blockTag === "pending") { if (!this._initialPromise) { this._initialPromise = this.signer.getTransactionCount("pending"); } const deltaCount = this._deltaCount; return this._initialPromise.then((initial) => (initial + deltaCount)); } return this.signer.getTransactionCount(blockTag); } setTransactionCount(transactionCount: ethers.BigNumberish | Promise): void { this._initialPromise = Promise.resolve(transactionCount).then((nonce) => { return ethers.BigNumber.from(nonce).toNumber(); }); this._deltaCount = 0; } incrementTransactionCount(count?: number): void { this._deltaCount += (count ? count: 1); } signMessage(message: ethers.Bytes | string): Promise { return this.signer.signMessage(message);; } signTransaction(transaction: ethers.utils.Deferrable): Promise { return this.signer.signTransaction(transaction); } sendTransaction(transaction: ethers.utils.Deferrable): Promise { if (transaction.nonce == null) { transaction = ethers.utils.shallowCopy(transaction); transaction.nonce = this.getTransactionCount("pending"); this.incrementTransactionCount(); } else { this.setTransactionCount(transaction.nonce); } return this.signer.sendTransaction(transaction).then((tx) => { return tx; }); } }