hardhat
Version:
Hardhat is an extensible developer tool that helps smart contract developers increase productivity by reliably bringing together the tools they want.
386 lines (321 loc) • 11.3 kB
text/typescript
import * as t from "io-ts";
import { signTypedData, SignTypedDataVersion } from "@metamask/eth-sig-util";
import { FeeMarketEIP1559Transaction } from "@nomicfoundation/ethereumjs-tx";
import { EIP1193Provider, RequestArguments } from "../../../types";
import { HardhatError } from "../errors";
import { ERRORS } from "../errors-list";
import {
rpcAddress,
rpcData,
rpcQuantityToBigInt,
} from "../jsonrpc/types/base-types";
import {
RpcTransactionRequest,
rpcTransactionRequest,
} from "../jsonrpc/types/input/transactionRequest";
import { validateParams } from "../jsonrpc/types/input/validation";
import { ProviderWrapperWithChainId } from "./chainId";
import { derivePrivateKeys } from "./util";
import { ProviderWrapper } from "./wrapper";
export interface JsonRpcTransactionData {
from?: string;
to?: string;
gas?: string | number;
gasPrice?: string | number;
value?: string | number;
data?: string;
nonce?: string | number;
}
export class LocalAccountsProvider extends ProviderWrapperWithChainId {
private _addressToPrivateKey: Map<string, Buffer> = new Map();
constructor(
provider: EIP1193Provider,
localAccountsHexPrivateKeys: string[]
) {
super(provider);
this._initializePrivateKeys(localAccountsHexPrivateKeys);
}
public async request(args: RequestArguments): Promise<unknown> {
const {
ecsign,
hashPersonalMessage,
toRpcSig,
toBytes,
bytesToHex: bufferToHex,
} = await import("@nomicfoundation/ethereumjs-util");
if (
args.method === "eth_accounts" ||
args.method === "eth_requestAccounts"
) {
return [...this._addressToPrivateKey.keys()];
}
const params = this._getParams(args);
if (args.method === "eth_sign") {
if (params.length > 0) {
const [address, data] = validateParams(params, rpcAddress, rpcData);
if (address !== undefined) {
if (data === undefined) {
throw new HardhatError(ERRORS.NETWORK.ETHSIGN_MISSING_DATA_PARAM);
}
const privateKey = this._getPrivateKeyForAddress(address);
const messageHash = hashPersonalMessage(toBytes(data));
const signature = ecsign(messageHash, privateKey);
return toRpcSig(signature.v, signature.r, signature.s);
}
}
}
if (args.method === "personal_sign") {
if (params.length > 0) {
const [data, address] = validateParams(params, rpcData, rpcAddress);
if (data !== undefined) {
if (address === undefined) {
throw new HardhatError(
ERRORS.NETWORK.PERSONALSIGN_MISSING_ADDRESS_PARAM
);
}
const privateKey = this._getPrivateKeyForAddress(address);
const messageHash = hashPersonalMessage(toBytes(data));
const signature = ecsign(messageHash, privateKey);
return toRpcSig(signature.v, signature.r, signature.s);
}
}
}
if (args.method === "eth_signTypedData_v4") {
const [address, data] = validateParams(params, rpcAddress, t.any);
if (data === undefined) {
throw new HardhatError(ERRORS.NETWORK.ETHSIGN_MISSING_DATA_PARAM);
}
let typedMessage = data;
if (typeof data === "string") {
try {
typedMessage = JSON.parse(data);
} catch {
throw new HardhatError(
ERRORS.NETWORK.ETHSIGN_TYPED_DATA_V4_INVALID_DATA_PARAM
);
}
}
// if we don't manage the address, the method is forwarded
const privateKey = this._getPrivateKeyForAddressOrNull(address);
if (privateKey !== null) {
return signTypedData({
privateKey,
version: SignTypedDataVersion.V4,
data: typedMessage,
});
}
}
if (args.method === "eth_sendTransaction" && params.length > 0) {
const [txRequest] = validateParams(params, rpcTransactionRequest);
if (txRequest.gas === undefined) {
throw new HardhatError(
ERRORS.NETWORK.MISSING_TX_PARAM_TO_SIGN_LOCALLY,
{ param: "gas" }
);
}
if (txRequest.from === undefined) {
throw new HardhatError(
ERRORS.NETWORK.MISSING_TX_PARAM_TO_SIGN_LOCALLY,
{ param: "from" }
);
}
const hasGasPrice = txRequest.gasPrice !== undefined;
const hasEip1559Fields =
txRequest.maxFeePerGas !== undefined ||
txRequest.maxPriorityFeePerGas !== undefined;
if (!hasGasPrice && !hasEip1559Fields) {
throw new HardhatError(ERRORS.NETWORK.MISSING_FEE_PRICE_FIELDS);
}
if (hasGasPrice && hasEip1559Fields) {
throw new HardhatError(ERRORS.NETWORK.INCOMPATIBLE_FEE_PRICE_FIELDS);
}
if (hasEip1559Fields && txRequest.maxFeePerGas === undefined) {
throw new HardhatError(
ERRORS.NETWORK.MISSING_TX_PARAM_TO_SIGN_LOCALLY,
{ param: "maxFeePerGas" }
);
}
if (hasEip1559Fields && txRequest.maxPriorityFeePerGas === undefined) {
throw new HardhatError(
ERRORS.NETWORK.MISSING_TX_PARAM_TO_SIGN_LOCALLY,
{ param: "maxPriorityFeePerGas" }
);
}
if (txRequest.nonce === undefined) {
txRequest.nonce = await this._getNonce(txRequest.from);
}
const privateKey = this._getPrivateKeyForAddress(txRequest.from!);
const chainId = await this._getChainId();
const rawTransaction = await this._getSignedTransaction(
txRequest,
chainId,
privateKey
);
return this._wrappedProvider.request({
method: "eth_sendRawTransaction",
params: [bufferToHex(rawTransaction)],
});
}
return this._wrappedProvider.request(args);
}
private _initializePrivateKeys(localAccountsHexPrivateKeys: string[]) {
const {
bytesToHex: bufferToHex,
toBytes,
privateToAddress,
} = require("@nomicfoundation/ethereumjs-util");
const privateKeys: Buffer[] = localAccountsHexPrivateKeys.map((h) =>
toBytes(h)
);
for (const pk of privateKeys) {
const address: string = bufferToHex(privateToAddress(pk)).toLowerCase();
this._addressToPrivateKey.set(address, pk);
}
}
private _getPrivateKeyForAddress(address: Buffer): Buffer {
const {
bytesToHex: bufferToHex,
} = require("@nomicfoundation/ethereumjs-util");
const pk = this._addressToPrivateKey.get(bufferToHex(address));
if (pk === undefined) {
throw new HardhatError(ERRORS.NETWORK.NOT_LOCAL_ACCOUNT, {
account: bufferToHex(address),
});
}
return pk;
}
private _getPrivateKeyForAddressOrNull(address: Buffer): Buffer | null {
try {
return this._getPrivateKeyForAddress(address);
} catch {
return null;
}
}
private async _getNonce(address: Buffer): Promise<bigint> {
const { bytesToHex: bufferToHex } = await import(
"@nomicfoundation/ethereumjs-util"
);
const response = (await this._wrappedProvider.request({
method: "eth_getTransactionCount",
params: [bufferToHex(address), "pending"],
})) as string;
return rpcQuantityToBigInt(response);
}
private async _getSignedTransaction(
transactionRequest: RpcTransactionRequest,
chainId: number,
privateKey: Buffer
): Promise<Uint8Array> {
const { AccessListEIP2930Transaction, LegacyTransaction } = await import(
"@nomicfoundation/ethereumjs-tx"
);
const { Common } = await import("@nomicfoundation/ethereumjs-common");
const txData = {
...transactionRequest,
gasLimit: transactionRequest.gas,
};
// We don't specify a hardfork here because the default hardfork should
// support all possible types of transactions.
// If the network doesn't support a given transaction type, then the
// transaction it will be rejected somewhere else.
const common = Common.custom({ chainId, networkId: chainId });
// we convert the access list to the type
// that AccessListEIP2930Transaction expects
const accessList = txData.accessList?.map(
({ address, storageKeys }) => [address, storageKeys] as [Buffer, Buffer[]]
);
let transaction;
if (txData.maxFeePerGas !== undefined) {
transaction = FeeMarketEIP1559Transaction.fromTxData(
{
...txData,
accessList,
gasPrice: undefined,
},
{ common }
);
} else if (accessList !== undefined) {
transaction = AccessListEIP2930Transaction.fromTxData(
{
...txData,
accessList,
},
{ common }
);
} else {
transaction = LegacyTransaction.fromTxData(txData, { common });
}
const signedTransaction = transaction.sign(privateKey);
return signedTransaction.serialize();
}
}
export class HDWalletProvider extends LocalAccountsProvider {
constructor(
provider: EIP1193Provider,
mnemonic: string,
hdpath: string = "m/44'/60'/0'/0/",
initialIndex: number = 0,
count: number = 10,
passphrase: string = ""
) {
// NOTE: If mnemonic has space or newline at the beginning or end, it will be trimmed.
// This is because mnemonic containing them may generate different private keys.
const trimmedMnemonic = mnemonic.trim();
const privateKeys = derivePrivateKeys(
trimmedMnemonic,
hdpath,
initialIndex,
count,
passphrase
);
const {
bytesToHex: bufferToHex,
} = require("@nomicfoundation/ethereumjs-util");
const privateKeysAsHex = privateKeys.map((pk) => bufferToHex(pk));
super(provider, privateKeysAsHex);
}
}
abstract class SenderProvider extends ProviderWrapper {
public async request(args: RequestArguments): Promise<unknown> {
const method = args.method;
const params = this._getParams(args);
if (
method === "eth_sendTransaction" ||
method === "eth_call" ||
method === "eth_estimateGas"
) {
// TODO: Should we validate this type?
const tx: JsonRpcTransactionData = params[0];
if (tx !== undefined && tx.from === undefined) {
const senderAccount = await this._getSender();
if (senderAccount !== undefined) {
tx.from = senderAccount;
} else if (method === "eth_sendTransaction") {
throw new HardhatError(ERRORS.NETWORK.NO_REMOTE_ACCOUNT_AVAILABLE);
}
}
}
return this._wrappedProvider.request(args);
}
protected abstract _getSender(): Promise<string | undefined>;
}
export class AutomaticSenderProvider extends SenderProvider {
private _firstAccount: string | undefined;
protected async _getSender(): Promise<string | undefined> {
if (this._firstAccount === undefined) {
const accounts = (await this._wrappedProvider.request({
method: "eth_accounts",
})) as string[];
this._firstAccount = accounts[0];
}
return this._firstAccount;
}
}
export class FixedSenderProvider extends SenderProvider {
constructor(provider: EIP1193Provider, private readonly _sender: string) {
super(provider);
}
protected async _getSender(): Promise<string | undefined> {
return this._sender;
}
}