"use strict" import { ethers } from "ethers"; // @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) { super(); this._deltaCount = 0; ethers.utils.defineReadOnly(this, "signer", signer); ethers.utils.defineReadOnly(this, "provider", signer.provider || null); } 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 == null) ? 1: count); } 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); this._deltaCount++; } return this.signer.sendTransaction(transaction).then((tx) => { return tx; }); } }