| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223 | 
							- /**
 
-  *  Generally the [[Wallet]] and [[JsonRpcSigner]] and their sub-classes
 
-  *  are sufficent for most developers, but this is provided to
 
-  *  fascilitate more complex Signers.
 
-  *
 
-  *  @_section: api/providers/abstract-signer: Subclassing Signer [abstract-signer]
 
-  */
 
- import { resolveAddress } from "../address/index.js";
 
- import { Transaction } from "../transaction/index.js";
 
- import { defineProperties, getBigInt, resolveProperties, assert, assertArgument } from "../utils/index.js";
 
- import { copyRequest } from "./provider.js";
 
- function checkProvider(signer, operation) {
 
-     if (signer.provider) {
 
-         return signer.provider;
 
-     }
 
-     assert(false, "missing provider", "UNSUPPORTED_OPERATION", { operation });
 
- }
 
- async function populate(signer, tx) {
 
-     let pop = copyRequest(tx);
 
-     if (pop.to != null) {
 
-         pop.to = resolveAddress(pop.to, signer);
 
-     }
 
-     if (pop.from != null) {
 
-         const from = pop.from;
 
-         pop.from = Promise.all([
 
-             signer.getAddress(),
 
-             resolveAddress(from, signer)
 
-         ]).then(([address, from]) => {
 
-             assertArgument(address.toLowerCase() === from.toLowerCase(), "transaction from mismatch", "tx.from", from);
 
-             return address;
 
-         });
 
-     }
 
-     else {
 
-         pop.from = signer.getAddress();
 
-     }
 
-     return await resolveProperties(pop);
 
- }
 
- /**
 
-  *  An **AbstractSigner** includes most of teh functionality required
 
-  *  to get a [[Signer]] working as expected, but requires a few
 
-  *  Signer-specific methods be overridden.
 
-  *
 
-  */
 
- export class AbstractSigner {
 
-     /**
 
-      *  The provider this signer is connected to.
 
-      */
 
-     provider;
 
-     /**
 
-      *  Creates a new Signer connected to %%provider%%.
 
-      */
 
-     constructor(provider) {
 
-         defineProperties(this, { provider: (provider || null) });
 
-     }
 
-     async getNonce(blockTag) {
 
-         return checkProvider(this, "getTransactionCount").getTransactionCount(await this.getAddress(), blockTag);
 
-     }
 
-     async populateCall(tx) {
 
-         const pop = await populate(this, tx);
 
-         return pop;
 
-     }
 
-     async populateTransaction(tx) {
 
-         const provider = checkProvider(this, "populateTransaction");
 
-         const pop = await populate(this, tx);
 
-         if (pop.nonce == null) {
 
-             pop.nonce = await this.getNonce("pending");
 
-         }
 
-         if (pop.gasLimit == null) {
 
-             pop.gasLimit = await this.estimateGas(pop);
 
-         }
 
-         // Populate the chain ID
 
-         const network = await (this.provider).getNetwork();
 
-         if (pop.chainId != null) {
 
-             const chainId = getBigInt(pop.chainId);
 
-             assertArgument(chainId === network.chainId, "transaction chainId mismatch", "tx.chainId", tx.chainId);
 
-         }
 
-         else {
 
-             pop.chainId = network.chainId;
 
-         }
 
-         // Do not allow mixing pre-eip-1559 and eip-1559 properties
 
-         const hasEip1559 = (pop.maxFeePerGas != null || pop.maxPriorityFeePerGas != null);
 
-         if (pop.gasPrice != null && (pop.type === 2 || hasEip1559)) {
 
-             assertArgument(false, "eip-1559 transaction do not support gasPrice", "tx", tx);
 
-         }
 
-         else if ((pop.type === 0 || pop.type === 1) && hasEip1559) {
 
-             assertArgument(false, "pre-eip-1559 transaction do not support maxFeePerGas/maxPriorityFeePerGas", "tx", tx);
 
-         }
 
-         if ((pop.type === 2 || pop.type == null) && (pop.maxFeePerGas != null && pop.maxPriorityFeePerGas != null)) {
 
-             // Fully-formed EIP-1559 transaction (skip getFeeData)
 
-             pop.type = 2;
 
-         }
 
-         else if (pop.type === 0 || pop.type === 1) {
 
-             // Explicit Legacy or EIP-2930 transaction
 
-             // We need to get fee data to determine things
 
-             const feeData = await provider.getFeeData();
 
-             assert(feeData.gasPrice != null, "network does not support gasPrice", "UNSUPPORTED_OPERATION", {
 
-                 operation: "getGasPrice"
 
-             });
 
-             // Populate missing gasPrice
 
-             if (pop.gasPrice == null) {
 
-                 pop.gasPrice = feeData.gasPrice;
 
-             }
 
-         }
 
-         else {
 
-             // We need to get fee data to determine things
 
-             const feeData = await provider.getFeeData();
 
-             if (pop.type == null) {
 
-                 // We need to auto-detect the intended type of this transaction...
 
-                 if (feeData.maxFeePerGas != null && feeData.maxPriorityFeePerGas != null) {
 
-                     // The network supports EIP-1559!
 
-                     // Upgrade transaction from null to eip-1559
 
-                     pop.type = 2;
 
-                     if (pop.gasPrice != null) {
 
-                         // Using legacy gasPrice property on an eip-1559 network,
 
-                         // so use gasPrice as both fee properties
 
-                         const gasPrice = pop.gasPrice;
 
-                         delete pop.gasPrice;
 
-                         pop.maxFeePerGas = gasPrice;
 
-                         pop.maxPriorityFeePerGas = gasPrice;
 
-                     }
 
-                     else {
 
-                         // Populate missing fee data
 
-                         if (pop.maxFeePerGas == null) {
 
-                             pop.maxFeePerGas = feeData.maxFeePerGas;
 
-                         }
 
-                         if (pop.maxPriorityFeePerGas == null) {
 
-                             pop.maxPriorityFeePerGas = feeData.maxPriorityFeePerGas;
 
-                         }
 
-                     }
 
-                 }
 
-                 else if (feeData.gasPrice != null) {
 
-                     // Network doesn't support EIP-1559...
 
-                     // ...but they are trying to use EIP-1559 properties
 
-                     assert(!hasEip1559, "network does not support EIP-1559", "UNSUPPORTED_OPERATION", {
 
-                         operation: "populateTransaction"
 
-                     });
 
-                     // Populate missing fee data
 
-                     if (pop.gasPrice == null) {
 
-                         pop.gasPrice = feeData.gasPrice;
 
-                     }
 
-                     // Explicitly set untyped transaction to legacy
 
-                     // @TODO: Maybe this shold allow type 1?
 
-                     pop.type = 0;
 
-                 }
 
-                 else {
 
-                     // getFeeData has failed us.
 
-                     assert(false, "failed to get consistent fee data", "UNSUPPORTED_OPERATION", {
 
-                         operation: "signer.getFeeData"
 
-                     });
 
-                 }
 
-             }
 
-             else if (pop.type === 2 || pop.type === 3) {
 
-                 // Explicitly using EIP-1559 or EIP-4844
 
-                 // Populate missing fee data
 
-                 if (pop.maxFeePerGas == null) {
 
-                     pop.maxFeePerGas = feeData.maxFeePerGas;
 
-                 }
 
-                 if (pop.maxPriorityFeePerGas == null) {
 
-                     pop.maxPriorityFeePerGas = feeData.maxPriorityFeePerGas;
 
-                 }
 
-             }
 
-         }
 
-         //@TOOD: Don't await all over the place; save them up for
 
-         // the end for better batching
 
-         return await resolveProperties(pop);
 
-     }
 
-     async estimateGas(tx) {
 
-         return checkProvider(this, "estimateGas").estimateGas(await this.populateCall(tx));
 
-     }
 
-     async call(tx) {
 
-         return checkProvider(this, "call").call(await this.populateCall(tx));
 
-     }
 
-     async resolveName(name) {
 
-         const provider = checkProvider(this, "resolveName");
 
-         return await provider.resolveName(name);
 
-     }
 
-     async sendTransaction(tx) {
 
-         const provider = checkProvider(this, "sendTransaction");
 
-         const pop = await this.populateTransaction(tx);
 
-         delete pop.from;
 
-         const txObj = Transaction.from(pop);
 
-         return await provider.broadcastTransaction(await this.signTransaction(txObj));
 
-     }
 
- }
 
- /**
 
-  *  A **VoidSigner** is a class deisgned to allow an address to be used
 
-  *  in any API which accepts a Signer, but for which there are no
 
-  *  credentials available to perform any actual signing.
 
-  *
 
-  *  This for example allow impersonating an account for the purpose of
 
-  *  static calls or estimating gas, but does not allow sending transactions.
 
-  */
 
- export class VoidSigner extends AbstractSigner {
 
-     /**
 
-      *  The signer address.
 
-      */
 
-     address;
 
-     /**
 
-      *  Creates a new **VoidSigner** with %%address%% attached to
 
-      *  %%provider%%.
 
-      */
 
-     constructor(address, provider) {
 
-         super(provider);
 
-         defineProperties(this, { address });
 
-     }
 
-     async getAddress() { return this.address; }
 
-     connect(provider) {
 
-         return new VoidSigner(this.address, provider);
 
-     }
 
-     #throwUnsupported(suffix, operation) {
 
-         assert(false, `VoidSigner cannot sign ${suffix}`, "UNSUPPORTED_OPERATION", { operation });
 
-     }
 
-     async signTransaction(tx) {
 
-         this.#throwUnsupported("transactions", "signTransaction");
 
-     }
 
-     async signMessage(message) {
 
-         this.#throwUnsupported("messages", "signMessage");
 
-     }
 
-     async signTypedData(domain, types, value) {
 
-         this.#throwUnsupported("typed-data", "signTypedData");
 
-     }
 
- }
 
- //# sourceMappingURL=abstract-signer.js.map
 
 
  |