@nomicfoundation/hardhat-ethers
Version:
Hardhat plugin for ethers
451 lines (382 loc) • 12.9 kB
text/typescript
import type { ethers as EthersT } from "ethers";
import type { HardhatEthersSigner } from "../signers";
import type {
DeployContractOptions,
FactoryOptions,
Libraries,
} from "../types";
import { Artifact, HardhatRuntimeEnvironment } from "hardhat/types";
import { HardhatEthersError } from "./errors";
interface Link {
sourceName: string;
libraryName: string;
address: string;
}
function isArtifact(artifact: any): artifact is Artifact {
const {
contractName,
sourceName,
abi,
bytecode,
deployedBytecode,
linkReferences,
deployedLinkReferences,
} = artifact;
return (
typeof contractName === "string" &&
typeof sourceName === "string" &&
Array.isArray(abi) &&
typeof bytecode === "string" &&
typeof deployedBytecode === "string" &&
linkReferences !== undefined &&
deployedLinkReferences !== undefined
);
}
export async function getSigners(
hre: HardhatRuntimeEnvironment
): Promise<HardhatEthersSigner[]> {
let accounts: string[];
try {
accounts = await hre.ethers.provider.send("eth_accounts", []);
} catch (error) {
if (
error instanceof Error &&
/the method has been deprecated: eth_accounts/.test(error.message)
) {
return [];
}
throw error;
}
const signersWithAddress = await Promise.all(
accounts.map((account) => getSigner(hre, account))
);
return signersWithAddress;
}
export async function getSigner(
hre: HardhatRuntimeEnvironment,
address: string
): Promise<HardhatEthersSigner> {
const { HardhatEthersSigner: SignerWithAddressImpl } = await import(
"../signers"
);
const signerWithAddress = await SignerWithAddressImpl.create(
hre.ethers.provider,
address
);
return signerWithAddress;
}
export async function getImpersonatedSigner(
hre: HardhatRuntimeEnvironment,
address: string
): Promise<HardhatEthersSigner> {
await hre.ethers.provider.send("hardhat_impersonateAccount", [address]);
return getSigner(hre, address);
}
export function getContractFactory<
A extends any[] = any[],
I = EthersT.Contract
>(
hre: HardhatRuntimeEnvironment,
name: string,
signerOrOptions?: EthersT.Signer | FactoryOptions
): Promise<EthersT.ContractFactory<A, I>>;
export function getContractFactory<
A extends any[] = any[],
I = EthersT.Contract
>(
hre: HardhatRuntimeEnvironment,
abi: any[],
bytecode: EthersT.BytesLike,
signer?: EthersT.Signer
): Promise<EthersT.ContractFactory<A, I>>;
export async function getContractFactory<
A extends any[] = any[],
I = EthersT.Contract
>(
hre: HardhatRuntimeEnvironment,
nameOrAbi: string | any[],
bytecodeOrFactoryOptions?:
| (EthersT.Signer | FactoryOptions)
| EthersT.BytesLike,
signer?: EthersT.Signer
): Promise<EthersT.ContractFactory<A, I>> {
if (typeof nameOrAbi === "string") {
const artifact = await hre.artifacts.readArtifact(nameOrAbi);
return getContractFactoryFromArtifact<A, I>(
hre,
artifact,
bytecodeOrFactoryOptions as EthersT.Signer | FactoryOptions | undefined
);
}
return getContractFactoryByAbiAndBytecode(
hre,
nameOrAbi,
bytecodeOrFactoryOptions as EthersT.BytesLike,
signer
);
}
function isFactoryOptions(
signerOrOptions?: EthersT.Signer | FactoryOptions
): signerOrOptions is FactoryOptions {
if (signerOrOptions === undefined || "provider" in signerOrOptions) {
return false;
}
return true;
}
export async function getContractFactoryFromArtifact<
A extends any[] = any[],
I = EthersT.Contract
>(
hre: HardhatRuntimeEnvironment,
artifact: Artifact,
signerOrOptions?: EthersT.Signer | FactoryOptions
): Promise<EthersT.ContractFactory<A, I>> {
let libraries: Libraries = {};
let signer: EthersT.Signer | undefined;
if (!isArtifact(artifact)) {
throw new HardhatEthersError(
`You are trying to create a contract factory from an artifact, but you have not passed a valid artifact parameter.`
);
}
if (isFactoryOptions(signerOrOptions)) {
signer = signerOrOptions.signer;
libraries = signerOrOptions.libraries ?? {};
} else {
signer = signerOrOptions;
}
if (artifact.bytecode === "0x") {
throw new HardhatEthersError(
`You are trying to create a contract factory for the contract ${artifact.contractName}, which is abstract and can't be deployed.
If you want to call a contract using ${artifact.contractName} as its interface use the "getContractAt" function instead.`
);
}
const linkedBytecode = await collectLibrariesAndLink(artifact, libraries);
return getContractFactoryByAbiAndBytecode(
hre,
artifact.abi,
linkedBytecode,
signer
);
}
async function collectLibrariesAndLink(
artifact: Artifact,
libraries: Libraries
) {
const ethers = require("ethers") as typeof EthersT;
const neededLibraries: Array<{
sourceName: string;
libName: string;
}> = [];
for (const [sourceName, sourceLibraries] of Object.entries(
artifact.linkReferences
)) {
for (const libName of Object.keys(sourceLibraries)) {
neededLibraries.push({ sourceName, libName });
}
}
const linksToApply: Map<string, Link> = new Map();
for (const [linkedLibraryName, linkedLibraryAddress] of Object.entries(
libraries
)) {
let resolvedAddress: string;
if (ethers.isAddressable(linkedLibraryAddress)) {
resolvedAddress = await linkedLibraryAddress.getAddress();
} else {
resolvedAddress = linkedLibraryAddress;
}
if (!ethers.isAddress(resolvedAddress)) {
throw new HardhatEthersError(
`You tried to link the contract ${
artifact.contractName
} with the library ${linkedLibraryName}, but provided this invalid address: ${
resolvedAddress as any
}`
);
}
const matchingNeededLibraries = neededLibraries.filter((lib) => {
return (
lib.libName === linkedLibraryName ||
`${lib.sourceName}:${lib.libName}` === linkedLibraryName
);
});
if (matchingNeededLibraries.length === 0) {
let detailedMessage: string;
if (neededLibraries.length > 0) {
const libraryFQNames = neededLibraries
.map((lib) => `${lib.sourceName}:${lib.libName}`)
.map((x) => `* ${x}`)
.join("\n");
detailedMessage = `The libraries needed are:
${libraryFQNames}`;
} else {
detailedMessage = "This contract doesn't need linking any libraries.";
}
throw new HardhatEthersError(
`You tried to link the contract ${artifact.contractName} with ${linkedLibraryName}, which is not one of its libraries.
${detailedMessage}`
);
}
if (matchingNeededLibraries.length > 1) {
const matchingNeededLibrariesFQNs = matchingNeededLibraries
.map(({ sourceName, libName }) => `${sourceName}:${libName}`)
.map((x) => `* ${x}`)
.join("\n");
throw new HardhatEthersError(
`The library name ${linkedLibraryName} is ambiguous for the contract ${artifact.contractName}.
It may resolve to one of the following libraries:
${matchingNeededLibrariesFQNs}
To fix this, choose one of these fully qualified library names and replace where appropriate.`
);
}
const [neededLibrary] = matchingNeededLibraries;
const neededLibraryFQN = `${neededLibrary.sourceName}:${neededLibrary.libName}`;
// The only way for this library to be already mapped is
// for it to be given twice in the libraries user input:
// once as a library name and another as a fully qualified library name.
if (linksToApply.has(neededLibraryFQN)) {
throw new HardhatEthersError(
`The library names ${neededLibrary.libName} and ${neededLibraryFQN} refer to the same library and were given as two separate library links.
Remove one of them and review your library links before proceeding.`
);
}
linksToApply.set(neededLibraryFQN, {
sourceName: neededLibrary.sourceName,
libraryName: neededLibrary.libName,
address: resolvedAddress,
});
}
if (linksToApply.size < neededLibraries.length) {
const missingLibraries = neededLibraries
.map((lib) => `${lib.sourceName}:${lib.libName}`)
.filter((libFQName) => !linksToApply.has(libFQName))
.map((x) => `* ${x}`)
.join("\n");
throw new HardhatEthersError(
`The contract ${artifact.contractName} is missing links for the following libraries:
${missingLibraries}
Learn more about linking contracts at https://hardhat.org/hardhat-runner/plugins/nomicfoundation-hardhat-ethers#library-linking
`
);
}
return linkBytecode(artifact, [...linksToApply.values()]);
}
async function getContractFactoryByAbiAndBytecode<
A extends any[] = any[],
I = EthersT.Contract
>(
hre: HardhatRuntimeEnvironment,
abi: any[],
bytecode: EthersT.BytesLike,
signer?: EthersT.Signer
): Promise<EthersT.ContractFactory<A, I>> {
const { ContractFactory } = require("ethers") as typeof EthersT;
if (signer === undefined) {
const signers = await hre.ethers.getSigners();
signer = signers[0];
}
return new ContractFactory(abi, bytecode, signer);
}
export async function getContractAt(
hre: HardhatRuntimeEnvironment,
nameOrAbi: string | any[],
address: string | EthersT.Addressable,
signer?: EthersT.Signer
) {
if (typeof nameOrAbi === "string") {
const artifact = await hre.artifacts.readArtifact(nameOrAbi);
return getContractAtFromArtifact(hre, artifact, address, signer);
}
const ethers = require("ethers") as typeof EthersT;
if (signer === undefined) {
const signers = await hre.ethers.getSigners();
signer = signers[0];
}
// If there's no signer, we want to put the provider for the selected network here.
// This allows read only operations on the contract interface.
const signerOrProvider: EthersT.Signer | EthersT.Provider =
signer !== undefined ? signer : hre.ethers.provider;
let resolvedAddress;
if (ethers.isAddressable(address)) {
resolvedAddress = await address.getAddress();
} else {
resolvedAddress = address;
}
return new ethers.Contract(resolvedAddress, nameOrAbi, signerOrProvider);
}
export async function deployContract(
hre: HardhatRuntimeEnvironment,
name: string,
args?: any[],
signerOrOptions?: EthersT.Signer | DeployContractOptions
): Promise<EthersT.Contract>;
export async function deployContract(
hre: HardhatRuntimeEnvironment,
name: string,
signerOrOptions?: EthersT.Signer | DeployContractOptions
): Promise<EthersT.Contract>;
export async function deployContract(
hre: HardhatRuntimeEnvironment,
name: string,
argsOrSignerOrOptions?: any[] | EthersT.Signer | DeployContractOptions,
signerOrOptions?: EthersT.Signer | DeployContractOptions
): Promise<EthersT.Contract> {
let args = [];
if (Array.isArray(argsOrSignerOrOptions)) {
args = argsOrSignerOrOptions;
} else {
signerOrOptions = argsOrSignerOrOptions;
}
let overrides: EthersT.Overrides = {};
if (signerOrOptions !== undefined && !("getAddress" in signerOrOptions)) {
const overridesAndFactoryOptions = { ...signerOrOptions };
// we delete the factory options properties in case ethers
// rejects unknown properties
delete overridesAndFactoryOptions.signer;
delete overridesAndFactoryOptions.libraries;
overrides = overridesAndFactoryOptions;
}
const factory = await getContractFactory(hre, name, signerOrOptions);
return factory.deploy(...args, overrides);
}
export async function getContractAtFromArtifact(
hre: HardhatRuntimeEnvironment,
artifact: Artifact,
address: string | EthersT.Addressable,
signer?: EthersT.Signer
) {
const ethers = require("ethers") as typeof EthersT;
if (!isArtifact(artifact)) {
throw new HardhatEthersError(
`You are trying to create a contract by artifact, but you have not passed a valid artifact parameter.`
);
}
if (signer === undefined) {
const signers = await hre.ethers.getSigners();
signer = signers[0];
}
let resolvedAddress;
if (ethers.isAddressable(address)) {
resolvedAddress = await address.getAddress();
} else {
resolvedAddress = address;
}
let contract = new ethers.Contract(resolvedAddress, artifact.abi, signer);
if (contract.runner === null) {
contract = contract.connect(hre.ethers.provider) as EthersT.Contract;
}
return contract;
}
function linkBytecode(artifact: Artifact, libraries: Link[]): string {
let bytecode = artifact.bytecode;
// TODO: measure performance impact
for (const { sourceName, libraryName, address } of libraries) {
const linkReferences = artifact.linkReferences[sourceName][libraryName];
for (const { start, length } of linkReferences) {
bytecode =
bytecode.substr(0, 2 + start * 2) +
address.substr(2) +
bytecode.substr(2 + (start + length) * 2);
}
}
return bytecode;
}