import { SignerWithAddress } from '@nomiclabs/hardhat-ethers/signers'; import { expect } from 'chai'; import { ContractTransaction } from 'ethers'; import { ethers } from 'hardhat'; import { AbacusConnectionManager, AbacusConnectionManager__factory, InterchainGasPaymaster, InterchainGasPaymaster__factory, TestInbox, TestInbox__factory, TestOutbox, TestOutbox__factory, } from '../types'; const ONLY_OWNER_REVERT_MSG = 'Ownable: caller is not the owner'; const localDomain = 1000; const remoteDomain = 2000; describe('AbacusConnectionManager', async () => { let connectionManager: AbacusConnectionManager, enrolledInbox: TestInbox, signer: SignerWithAddress, nonOwner: SignerWithAddress; before(async () => { [signer, nonOwner] = await ethers.getSigners(); }); beforeEach(async () => { const outboxFactory = new TestOutbox__factory(signer); const outbox = await outboxFactory.deploy(localDomain); const inboxFactory = new TestInbox__factory(signer); enrolledInbox = await inboxFactory.deploy(localDomain); // The ValidatorManager is unused in these tests *but* needs to be a // contract. await enrolledInbox.initialize( remoteDomain, outbox.address, ethers.constants.HashZero, 0, ); const connectionManagerFactory = new AbacusConnectionManager__factory( signer, ); connectionManager = await connectionManagerFactory.deploy(); await connectionManager.setOutbox(outbox.address); await connectionManager.enrollInbox(remoteDomain, enrolledInbox.address); }); it('Returns the local domain', async () => { expect(await connectionManager!.localDomain()).to.equal(localDomain); }); // Used for testing `setOutbox` and `setOutboxAndInterchainGasPaymaster` to avoid // code duplication const setOutboxTests = ( setter: ( outbox: string, signer?: SignerWithAddress, ) => Promise, ) => { let newOutbox: TestOutbox; beforeEach(async () => { const outboxFactory = new TestOutbox__factory(signer); newOutbox = await outboxFactory.deploy(localDomain); }); it('Allows owner to set the outbox', async () => { await setter(newOutbox.address); expect(await connectionManager.outbox()).to.equal(newOutbox.address); }); it('Emits the NewOutbox event', async () => { await expect(setter(newOutbox.address)) .to.emit(connectionManager, 'NewOutbox') .withArgs(newOutbox.address); }); it('Reverts a call from non-owner', async () => { await expect(setter(newOutbox.address, nonOwner)).to.be.revertedWith( ONLY_OWNER_REVERT_MSG, ); }); }; // Used for testing `setInterchainGasPaymaster` and `setOutboxAndInterchainGasPaymaster` // to avoid code duplication const setInterchainGasPaymasterTests = ( setter: ( interchainGasPaymaster: string, signer?: SignerWithAddress, ) => Promise, ) => { let newPaymaster: InterchainGasPaymaster; beforeEach(async () => { const paymasterFactory = new InterchainGasPaymaster__factory(signer); newPaymaster = await paymasterFactory.deploy(); }); it('Allows owner to set the interchainGasPaymaster', async () => { await setter(newPaymaster.address); expect(await connectionManager.interchainGasPaymaster()).to.equal( newPaymaster.address, ); }); it('Emits the NewInterchainGasPaymaster event', async () => { await expect(setter(newPaymaster.address)) .to.emit(connectionManager, 'NewInterchainGasPaymaster') .withArgs(newPaymaster.address); }); it('Reverts a call from non-owner', async () => { await expect(setter(newPaymaster.address, nonOwner)).to.be.revertedWith( ONLY_OWNER_REVERT_MSG, ); }); }; describe('#setOutboxAndInterchainGasPaymaster', () => { const dummyAddress = '0xdEADBEeF00000000000000000000000000000000'; // Test the outbox setting works setOutboxTests((outbox: string, signer?: SignerWithAddress) => { const connManager = signer ? connectionManager.connect(signer) : connectionManager; return connManager.setOutboxAndInterchainGasPaymaster( outbox, dummyAddress, ); }); // And test the interchain gas paymaster setting works setInterchainGasPaymasterTests( (interchainGasPaymaster: string, signer?: SignerWithAddress) => { const connManager = signer ? connectionManager.connect(signer) : connectionManager; return connManager.setOutboxAndInterchainGasPaymaster( dummyAddress, interchainGasPaymaster, ); }, ); }); describe('#setOutbox', () => { setOutboxTests((outbox: string, signer?: SignerWithAddress) => { const connManager = signer ? connectionManager.connect(signer) : connectionManager; return connManager.setOutbox(outbox); }); }); describe('#setInterchainGasPaymaster', () => { setInterchainGasPaymasterTests( (interchainGasPaymaster: string, signer?: SignerWithAddress) => { const connManager = signer ? connectionManager.connect(signer) : connectionManager; return connManager.setInterchainGasPaymaster(interchainGasPaymaster); }, ); }); it('isInbox returns true for enrolledInbox and false for non-enrolled Inbox', async () => { const [nonEnrolledInbox] = await ethers.getSigners(); expect(await connectionManager.isInbox(enrolledInbox.address)).to.be.true; expect(await connectionManager.isInbox(nonEnrolledInbox.address)).to.be .false; }); it('Owner can enroll a inbox', async () => { const newRemoteDomain = 3000; const inboxFactory = new TestInbox__factory(signer); const newInbox = await inboxFactory.deploy(localDomain); // Assert new inbox not considered inbox before enrolled expect(await connectionManager.isInbox(newInbox.address)).to.be.false; await expect( connectionManager.enrollInbox(newRemoteDomain, newInbox.address), ).to.emit(connectionManager, 'InboxEnrolled'); expect(await connectionManager.getInboxes(newRemoteDomain)).to.eql([ newInbox.address, ]); expect(await connectionManager.inboxToDomain(newInbox.address)).to.equal( newRemoteDomain, ); expect(await connectionManager.isInbox(newInbox.address)).to.be.true; }); it('Owner can unenroll a inbox', async () => { expect(await connectionManager.getInboxes(remoteDomain)).to.eql([ enrolledInbox.address, ]); await expect( connectionManager.unenrollInbox(enrolledInbox.address), ).to.emit(connectionManager, 'InboxUnenrolled'); expect( await connectionManager.inboxToDomain(enrolledInbox.address), ).to.equal(0); expect(await connectionManager.getInboxes(remoteDomain)).to.eql([]); expect(await connectionManager.isInbox(enrolledInbox.address)).to.be.false; }); it('Owner can enroll multiple inboxes per domain', async () => { const newRemoteDomain = 3000; const inboxFactory = new TestInbox__factory(signer); const newInbox1 = await inboxFactory.deploy(localDomain); const newInbox2 = await inboxFactory.deploy(localDomain); // Assert new inbox not considered inbox before enrolled expect(await connectionManager.isInbox(newInbox1.address)).to.be.false; expect(await connectionManager.isInbox(newInbox2.address)).to.be.false; await expect( connectionManager.enrollInbox(newRemoteDomain, newInbox1.address), ).to.emit(connectionManager, 'InboxEnrolled'); await expect( connectionManager.enrollInbox(newRemoteDomain, newInbox2.address), ).to.emit(connectionManager, 'InboxEnrolled'); expect(await connectionManager.inboxToDomain(newInbox1.address)).to.equal( newRemoteDomain, ); expect(await connectionManager.inboxToDomain(newInbox2.address)).to.equal( newRemoteDomain, ); expect(await connectionManager.isInbox(newInbox1.address)).to.be.true; expect(await connectionManager.isInbox(newInbox2.address)).to.be.true; expect(await connectionManager.getInboxes(newRemoteDomain)).to.eql([ newInbox1.address, newInbox2.address, ]); }); it('Owner cannot enroll an inbox twice', async () => { const newRemoteDomain = 3000; await expect( connectionManager.enrollInbox(newRemoteDomain, enrolledInbox.address), ).to.be.revertedWith('already inbox'); }); });