const { expectRevert } = require('@openzeppelin/test-helpers'); const { getSlot, BeaconSlot } = require('../../helpers/erc1967'); const { expect } = require('chai'); const UpgradeableBeacon = artifacts.require('UpgradeableBeacon'); const BeaconProxy = artifacts.require('BeaconProxy'); const DummyImplementation = artifacts.require('DummyImplementation'); const DummyImplementationV2 = artifacts.require('DummyImplementationV2'); const BadBeaconNoImpl = artifacts.require('BadBeaconNoImpl'); const BadBeaconNotContract = artifacts.require('BadBeaconNotContract'); contract('BeaconProxy', function (accounts) { const [anotherAccount] = accounts; describe('bad beacon is not accepted', async function () { it('non-contract beacon', async function () { await expectRevert(BeaconProxy.new(anotherAccount, '0x'), 'ERC1967: new beacon is not a contract'); }); it('non-compliant beacon', async function () { const beacon = await BadBeaconNoImpl.new(); await expectRevert.unspecified(BeaconProxy.new(beacon.address, '0x')); }); it('non-contract implementation', async function () { const beacon = await BadBeaconNotContract.new(); await expectRevert(BeaconProxy.new(beacon.address, '0x'), 'ERC1967: beacon implementation is not a contract'); }); }); before('deploy implementation', async function () { this.implementationV0 = await DummyImplementation.new(); this.implementationV1 = await DummyImplementationV2.new(); }); describe('initialization', function () { before(function () { this.assertInitialized = async ({ value, balance }) => { const beaconSlot = await getSlot(this.proxy, BeaconSlot); const beaconAddress = web3.utils.toChecksumAddress(beaconSlot.substr(-40)); expect(beaconAddress).to.equal(this.beacon.address); const dummy = new DummyImplementation(this.proxy.address); expect(await dummy.value()).to.bignumber.eq(value); expect(await web3.eth.getBalance(this.proxy.address)).to.bignumber.eq(balance); }; }); beforeEach('deploy beacon', async function () { this.beacon = await UpgradeableBeacon.new(this.implementationV0.address); }); it('no initialization', async function () { const data = Buffer.from(''); const balance = '10'; this.proxy = await BeaconProxy.new(this.beacon.address, data, { value: balance }); await this.assertInitialized({ value: '0', balance }); }); it('non-payable initialization', async function () { const value = '55'; const data = this.implementationV0.contract.methods.initializeNonPayableWithValue(value).encodeABI(); this.proxy = await BeaconProxy.new(this.beacon.address, data); await this.assertInitialized({ value, balance: '0' }); }); it('payable initialization', async function () { const value = '55'; const data = this.implementationV0.contract.methods.initializePayableWithValue(value).encodeABI(); const balance = '100'; this.proxy = await BeaconProxy.new(this.beacon.address, data, { value: balance }); await this.assertInitialized({ value, balance }); }); it('reverting initialization', async function () { const data = this.implementationV0.contract.methods.reverts().encodeABI(); await expectRevert(BeaconProxy.new(this.beacon.address, data), 'DummyImplementation reverted'); }); }); it('upgrade a proxy by upgrading its beacon', async function () { const beacon = await UpgradeableBeacon.new(this.implementationV0.address); const value = '10'; const data = this.implementationV0.contract.methods.initializeNonPayableWithValue(value).encodeABI(); const proxy = await BeaconProxy.new(beacon.address, data); const dummy = new DummyImplementation(proxy.address); // test initial values expect(await dummy.value()).to.bignumber.eq(value); // test initial version expect(await dummy.version()).to.eq('V1'); // upgrade beacon await beacon.upgradeTo(this.implementationV1.address); // test upgraded version expect(await dummy.version()).to.eq('V2'); }); it('upgrade 2 proxies by upgrading shared beacon', async function () { const value1 = '10'; const value2 = '42'; const beacon = await UpgradeableBeacon.new(this.implementationV0.address); const proxy1InitializeData = this.implementationV0.contract.methods .initializeNonPayableWithValue(value1) .encodeABI(); const proxy1 = await BeaconProxy.new(beacon.address, proxy1InitializeData); const proxy2InitializeData = this.implementationV0.contract.methods .initializeNonPayableWithValue(value2) .encodeABI(); const proxy2 = await BeaconProxy.new(beacon.address, proxy2InitializeData); const dummy1 = new DummyImplementation(proxy1.address); const dummy2 = new DummyImplementation(proxy2.address); // test initial values expect(await dummy1.value()).to.bignumber.eq(value1); expect(await dummy2.value()).to.bignumber.eq(value2); // test initial version expect(await dummy1.version()).to.eq('V1'); expect(await dummy2.version()).to.eq('V1'); // upgrade beacon await beacon.upgradeTo(this.implementationV1.address); // test upgraded version expect(await dummy1.version()).to.eq('V2'); expect(await dummy2.version()).to.eq('V2'); }); });