2021-08-12 09:58:13 +00:00
|
|
|
//
|
|
|
|
// Copyright 2021 Vulcanize, Inc.
|
|
|
|
//
|
|
|
|
|
2021-06-23 04:03:42 +00:00
|
|
|
/* eslint-disable no-unused-expressions */
|
|
|
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
2021-06-02 05:53:33 +00:00
|
|
|
import { ContractInterface } from '@ethersproject/contracts';
|
2021-06-18 12:39:50 +00:00
|
|
|
import '@nomiclabs/hardhat-ethers';
|
2021-06-02 05:53:33 +00:00
|
|
|
import { artifacts, ethers } from 'hardhat';
|
2021-05-31 05:37:11 +00:00
|
|
|
import { CompilerOutput, CompilerOutputBytecode } from 'hardhat/types';
|
2021-06-23 04:03:42 +00:00
|
|
|
import { expect } from 'chai';
|
|
|
|
import isArray from 'lodash/isArray';
|
2021-05-31 05:37:11 +00:00
|
|
|
|
|
|
|
import { StorageLayout, GetStorageAt } from '../src';
|
|
|
|
|
|
|
|
// storageLayout doesnt exist in type CompilerOutput doesnt.
|
|
|
|
// Extending CompilerOutput type to include storageLayout property.
|
|
|
|
interface StorageCompilerOutput extends CompilerOutput {
|
|
|
|
contracts: {
|
|
|
|
[sourceName: string]: {
|
|
|
|
[contractName: string]: {
|
2021-06-02 05:53:33 +00:00
|
|
|
abi: ContractInterface;
|
2021-05-31 05:37:11 +00:00
|
|
|
evm: {
|
|
|
|
bytecode: CompilerOutputBytecode;
|
|
|
|
deployedBytecode: CompilerOutputBytecode;
|
|
|
|
methodIdentifiers: {
|
|
|
|
[methodSignature: string]: string;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
storageLayout?: StorageLayout;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-06-23 04:03:42 +00:00
|
|
|
interface ProofData {
|
|
|
|
blockHash: string;
|
|
|
|
account: {
|
|
|
|
address: string;
|
|
|
|
storage: {
|
|
|
|
ipldBlock: string;
|
|
|
|
cid: string;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-31 05:37:11 +00:00
|
|
|
/**
|
|
|
|
* Get storage layout of specified contract.
|
|
|
|
* @param contractName
|
|
|
|
*/
|
2021-06-02 05:53:33 +00:00
|
|
|
export const getStorageLayout = async (contractName: string): Promise<StorageLayout> => {
|
2021-05-31 05:37:11 +00:00
|
|
|
const artifact = await artifacts.readArtifact(contractName);
|
2021-06-02 05:53:33 +00:00
|
|
|
const buildInfo = await artifacts.getBuildInfo(`${artifact.sourceName}:${artifact.contractName}`);
|
2021-05-31 05:37:11 +00:00
|
|
|
|
|
|
|
if (!buildInfo) {
|
|
|
|
throw new Error('storageLayout not present in compiler output.');
|
|
|
|
}
|
|
|
|
|
2021-06-02 05:53:33 +00:00
|
|
|
const output: StorageCompilerOutput = buildInfo.output;
|
2021-05-31 05:37:11 +00:00
|
|
|
const { storageLayout } = output.contracts[artifact.sourceName][artifact.contractName];
|
|
|
|
|
|
|
|
if (!storageLayout) {
|
2021-06-02 05:53:33 +00:00
|
|
|
throw new Error('Contract hasn\'t been compiled.');
|
2021-05-31 05:37:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return storageLayout;
|
2021-06-02 05:53:33 +00:00
|
|
|
};
|
2021-05-31 05:37:11 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get storage value in hardhat environment using ethers.
|
|
|
|
* @param address
|
|
|
|
* @param position
|
|
|
|
*/
|
2021-06-03 06:22:23 +00:00
|
|
|
export const getStorageAt: GetStorageAt = async ({ blockHash, contract, slot }) => {
|
|
|
|
// TODO: Fix use of blockHash as hex string in getStorageAt.
|
|
|
|
// Using blockHash in getStorageAt throws error.
|
|
|
|
// https://github.com/ethers-io/ethers.js/pull/1550#issuecomment-841746994
|
2021-06-03 09:33:39 +00:00
|
|
|
// Using latest tag for temporary fix in test scenario.
|
2021-06-03 06:22:23 +00:00
|
|
|
blockHash = 'latest';
|
|
|
|
const value = await ethers.provider.getStorageAt(contract, slot, blockHash);
|
2021-05-31 05:37:11 +00:00
|
|
|
|
2021-06-03 06:22:23 +00:00
|
|
|
return {
|
|
|
|
value,
|
|
|
|
proof: {
|
2021-06-14 10:42:42 +00:00
|
|
|
// Returning null value as proof, since ethers library getStorageAt method doesnt return proof.
|
|
|
|
// This function is used in tests to mock the getStorageAt method of ipld-eth-client which returns proof along with value.
|
2021-06-03 06:22:23 +00:00
|
|
|
data: JSON.stringify(null)
|
|
|
|
}
|
|
|
|
};
|
2021-06-02 05:53:33 +00:00
|
|
|
};
|
2021-06-18 12:39:50 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate array of dummy addresses of specified length.
|
|
|
|
* @param length
|
|
|
|
*/
|
|
|
|
export const generateDummyAddresses = (length: number): Array<string> => {
|
|
|
|
return Array.from({ length }, () => {
|
2021-07-16 13:04:51 +00:00
|
|
|
return ethers.utils.getAddress(ethers.utils.hexlify(ethers.utils.randomBytes(20)));
|
2021-06-18 12:39:50 +00:00
|
|
|
});
|
|
|
|
};
|
2021-06-23 04:03:42 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get latest blockHash.
|
|
|
|
*/
|
|
|
|
export const getBlockHash = async (): Promise<string> => {
|
|
|
|
const blockNumber = await ethers.provider.getBlockNumber();
|
|
|
|
const { hash } = await ethers.provider.getBlock(blockNumber);
|
|
|
|
return hash;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assert proof data returned from ipld graphql server.
|
|
|
|
* @param blockHash
|
|
|
|
* @param address
|
|
|
|
* @param proofData
|
|
|
|
*/
|
|
|
|
export const assertProofData = (blockHash: string, address: string, proofData: ProofData): void => {
|
|
|
|
const {
|
|
|
|
blockHash: proofBlockHash,
|
|
|
|
account: {
|
|
|
|
address: proofAddress,
|
|
|
|
storage
|
|
|
|
}
|
|
|
|
} = proofData;
|
|
|
|
|
|
|
|
expect(proofBlockHash).to.equal(blockHash);
|
|
|
|
expect(proofAddress).to.equal(address);
|
|
|
|
expect(storage.cid).to.not.be.empty;
|
|
|
|
expect(storage.ipldBlock).to.not.be.empty;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assert array of proof data.
|
|
|
|
* @param blockHash
|
|
|
|
* @param address
|
|
|
|
* @param proofArray
|
|
|
|
*/
|
|
|
|
export const assertProofArray = (blockHash: string, address: string, proofArray: Array<any>): void => {
|
|
|
|
proofArray.forEach(proofData => {
|
|
|
|
if (isArray(proofData)) {
|
|
|
|
assertProofArray(blockHash, address, proofData);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (['blockHash', 'account'].every(key => key in proofData)) {
|
|
|
|
assertProofData(blockHash, address, proofData);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assertProofStruct(blockHash, address, proofData);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assert array of proof data from structure type.
|
|
|
|
* @param blockHash
|
|
|
|
* @param address
|
|
|
|
* @param proofStruct
|
|
|
|
*/
|
|
|
|
export const assertProofStruct = (blockHash: string, address: string, proofStruct: {[key: string]: any}): void => {
|
|
|
|
Object.values(proofStruct).forEach(proofData => {
|
|
|
|
if (isArray(proofData)) {
|
|
|
|
assertProofArray(blockHash, address, proofData);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (['blockHash', 'account'].every(key => key in proofData)) {
|
|
|
|
assertProofData(blockHash, address, proofData);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assertProofStruct(blockHash, address, proofData);
|
|
|
|
});
|
|
|
|
};
|