123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184 |
- const { ethers } = require('hardhat');
- const { expect } = require('chai');
- const { getAddressInSlot, ImplementationSlot } = require('../helpers/storage');
- module.exports = function shouldBehaveLikeProxy() {
- it('cannot be initialized with a non-contract address', async function () {
- const initializeData = '0x';
- await expect(this.createProxy(this.nonContractAddress, initializeData))
- .to.be.revertedWithCustomError(await ethers.getContractFactory('ERC1967Proxy'), 'ERC1967InvalidImplementation')
- .withArgs(this.nonContractAddress);
- });
- const assertProxyInitialization = function ({ value, balance }) {
- it('sets the implementation address', async function () {
- expect(await getAddressInSlot(this.proxy, ImplementationSlot)).to.equal(this.implementation);
- });
- it('initializes the proxy', async function () {
- const dummy = this.implementation.attach(this.proxy);
- expect(await dummy.value()).to.equal(value);
- });
- it('has expected balance', async function () {
- expect(await ethers.provider.getBalance(this.proxy)).to.equal(balance);
- });
- };
- describe('without initialization', function () {
- const initializeData = '0x';
- describe('when not sending balance', function () {
- beforeEach('creating proxy', async function () {
- this.proxy = await this.createProxy(this.implementation, initializeData);
- });
- assertProxyInitialization({ value: 0n, balance: 0n });
- });
- describe('when sending some balance', function () {
- const value = 10n ** 5n;
- it('reverts', async function () {
- await expect(this.createProxy(this.implementation, initializeData, { value })).to.be.reverted;
- });
- });
- });
- describe('initialization without parameters', function () {
- describe('non payable', function () {
- const expectedInitializedValue = 10n;
- beforeEach(function () {
- this.initializeData = this.implementation.interface.encodeFunctionData('initializeNonPayable');
- });
- describe('when not sending balance', function () {
- beforeEach('creating proxy', async function () {
- this.proxy = await this.createProxy(this.implementation, this.initializeData);
- });
- assertProxyInitialization({
- value: expectedInitializedValue,
- balance: 0n,
- });
- });
- describe('when sending some balance', function () {
- const value = 10n ** 5n;
- it('reverts', async function () {
- await expect(this.createProxy(this.implementation, this.initializeData, { value })).to.be.reverted;
- });
- });
- });
- describe('payable', function () {
- const expectedInitializedValue = 100n;
- beforeEach(function () {
- this.initializeData = this.implementation.interface.encodeFunctionData('initializePayable');
- });
- describe('when not sending balance', function () {
- beforeEach('creating proxy', async function () {
- this.proxy = await this.createProxy(this.implementation, this.initializeData);
- });
- assertProxyInitialization({
- value: expectedInitializedValue,
- balance: 0n,
- });
- });
- describe('when sending some balance', function () {
- const value = 10e5;
- beforeEach('creating proxy', async function () {
- this.proxy = await this.createProxy(this.implementation, this.initializeData, { value });
- });
- assertProxyInitialization({
- value: expectedInitializedValue,
- balance: value,
- });
- });
- });
- });
- describe('initialization with parameters', function () {
- describe('non payable', function () {
- const expectedInitializedValue = 10n;
- beforeEach(function () {
- this.initializeData = this.implementation.interface.encodeFunctionData('initializeNonPayableWithValue', [
- expectedInitializedValue,
- ]);
- });
- describe('when not sending balance', function () {
- beforeEach('creating proxy', async function () {
- this.proxy = await this.createProxy(this.implementation, this.initializeData);
- });
- assertProxyInitialization({
- value: expectedInitializedValue,
- balance: 0,
- });
- });
- describe('when sending some balance', function () {
- const value = 10e5;
- it('reverts', async function () {
- await expect(this.createProxy(this.implementation, this.initializeData, { value })).to.be.reverted;
- });
- });
- });
- describe('payable', function () {
- const expectedInitializedValue = 42n;
- beforeEach(function () {
- this.initializeData = this.implementation.interface.encodeFunctionData('initializePayableWithValue', [
- expectedInitializedValue,
- ]);
- });
- describe('when not sending balance', function () {
- beforeEach('creating proxy', async function () {
- this.proxy = await this.createProxy(this.implementation, this.initializeData);
- });
- assertProxyInitialization({
- value: expectedInitializedValue,
- balance: 0n,
- });
- });
- describe('when sending some balance', function () {
- const value = 10n ** 5n;
- beforeEach('creating proxy', async function () {
- this.proxy = await this.createProxy(this.implementation, this.initializeData, { value });
- });
- assertProxyInitialization({
- value: expectedInitializedValue,
- balance: value,
- });
- });
- });
- describe('reverting initialization', function () {
- beforeEach(function () {
- this.initializeData = this.implementation.interface.encodeFunctionData('reverts');
- });
- it('reverts', async function () {
- await expect(this.createProxy(this.implementation, this.initializeData)).to.be.reverted;
- });
- });
- });
- };
|