Skip to content

Commit

Permalink
add UCH upgrade tests
Browse files Browse the repository at this point in the history
  • Loading branch information
RnkSngh committed Apr 25, 2024
1 parent a22db9a commit fe30152
Show file tree
Hide file tree
Showing 3 changed files with 6,071 additions and 3 deletions.
17 changes: 17 additions & 0 deletions test/upgradeableProxy/Dispatcher.upgrade.t.sol
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,10 @@ import {IProofVerifier} from "../../contracts/core/OptimisticProofVerifier.sol";
import {DummyLightClient} from "../../contracts/utils/DummyLightClient.sol";

import {IDispatcher} from "../../contracts/interfaces/IDispatcher.sol";
import {UniversalChannelHandler} from "../../contracts/core/UniversalChannelHandler.sol";
import {IUniversalChannelHandler} from "../../contracts/interfaces/IUniversalChannelHandler.sol";
import {DispatcherRc4} from "./upgrades/DispatcherRc4.sol";
import {UniversalChannelHandlerV2} from "./upgrades/UCHV2.sol";
import {DispatcherV2Initializable} from "./upgrades/DispatcherV2Initializable.sol";
import {DispatcherV2} from "./upgrades/DispatcherV2.sol";
import {ERC1967Proxy} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol";
Expand All @@ -48,6 +51,14 @@ abstract contract UpgradeTestUtils {
UUPSUpgradeable(dispatcherProxy).upgradeToAndCall(address(newDispatcherImplementation), initData);
}

function upgradeUch(address uchProxy, address dispatcherProxy)
public
returns (UniversalChannelHandler newUCHImplementation)
{
newUCHImplementation = new UniversalChannelHandler(); // Upgrade from v2 to v3
UUPSUpgradeable(address(uchProxy)).upgradeTo(address(newUCHImplementation));
}

function deployDispatcherRC4ProxyAndImpl(string memory initPortPrefix, ILightClient initLightClient)
public
returns (IDispatcher proxy)
Expand All @@ -57,6 +68,12 @@ abstract contract UpgradeTestUtils {
abi.encodeWithSelector(DispatcherRc4.initialize.selector, initPortPrefix, initLightClient);
proxy = IDispatcher(address(new ERC1967Proxy(address(dispatcherImplementation), initData)));
}

function deployUCHV2ProxyAndImpl(address dispatcherProxy) public returns (IUniversalChannelHandler proxy) {
UniversalChannelHandlerV2 uchImplementation = new UniversalChannelHandlerV2();
bytes memory initData = abi.encodeWithSelector(UniversalChannelHandlerV2.initialize.selector, dispatcherProxy);
proxy = IUniversalChannelHandler(address(new ERC1967Proxy(address(uchImplementation), initData)));
}
}

contract ChannelHandShakeUpgradeUtil is ChannelHandshakeUtils {
Expand Down
92 changes: 89 additions & 3 deletions test/upgradeableProxy/DispatcherRC4.upgrade.t.sol
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import "forge-std/console2.sol";
import {DispatcherUpdateClientTestSuite} from "../Dispatcher/Dispatcher.client.t.sol";
import {DispatcherIbcWithRealProofsSuite} from "../Dispatcher/Dispatcher.proof.t.sol";
import {Mars} from "../../contracts/examples/Mars.sol";
import {Earth} from "../../contracts/examples/Earth.sol";
import "../../contracts/core/OptimisticLightClient.sol";
import {ChannelHandshakeTestSuite, ChannelHandshakeTest, ChannelHandshakeUtils} from "../Dispatcher/Dispatcher.t.sol";
import {LocalEnd} from "../utils/Dispatcher.base.t.sol";
Expand All @@ -17,7 +18,8 @@ import {
IbcPacket,
AckPacket,
Ibc,
Height
Height,
UniversalPacket
} from "../../contracts/libs/Ibc.sol";
import {IbcUtils} from "../../contracts/libs/IbcUtils.sol";
import {IbcReceiver} from "../../contracts/interfaces/IbcReceiver.sol";
Expand All @@ -30,11 +32,17 @@ import {IDispatcher} from "../../contracts/interfaces/IDispatcher.sol";

import {ChannelHandShakeUpgradeUtil, UpgradeTestUtils} from "./Dispatcher.upgrade.t.sol";
import {IDispatcherRc4, DispatcherRc4} from "./upgrades/DispatcherRc4.sol";
import {IUniversalChannelHandler} from "../../contracts/interfaces/IUniversalChannelHandler.sol";
import {VirtualChain} from "../VirtualChain.sol";
import {GeneralMiddleware} from "../../contracts/base/GeneralMiddleware.sol";

// Tests to ensure that the upgrade between rc4 -> preaudit fixes doesn't break the state of the contract
contract DispatcherRC4UpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeTestUtils {
Mars receivingMars;
Mars dummyMars;
IUniversalChannelHandler uch;
LocalEnd _localUch;
Earth earth;

function setUp() public override {
// In Rc4 version, there can only be one dispatcher per light client so we deploy multiple clients
Expand All @@ -44,13 +52,17 @@ contract DispatcherRC4UpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeTestUti
// Deploy op old dispatcher
opLightClient = new OptimisticLightClient(1, opProofVerifier, l1BlockProvider);
dispatcherProxy = deployDispatcherRC4ProxyAndImpl(portPrefix, opLightClient);
uch = deployUCHV2ProxyAndImpl(address(dispatcherProxy));
earth = new Earth(address(uch));

// Set up dispatcher with non-trivial state
mars = new Mars(dispatcherProxy);
receivingMars = new Mars(dispatcherProxy);
string memory sendingPortId = IbcUtils.addressToPortId(portPrefix, address(mars));
string memory uchSendingPortId = IbcUtils.addressToPortId(portPrefix, address(uch));
string memory receivingPortId = IbcUtils.addressToPortId(portPrefix, address(receivingMars));
_local = LocalEnd(mars, sendingPortId, "channel-1", connectionHops1, "1.0", "1.0");
_localUch = LocalEnd(uch, uchSendingPortId, "uch-channel", connectionHops1, "1.0", "1.0");
_remote = ChannelEnd(receivingPortId, "channel-2", "1.0");

// Should now be able to able to open a connection without proofs on the upgraded dispatcherproxy now
Expand All @@ -62,8 +74,14 @@ contract DispatcherRC4UpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeTestUti
doProofChannelHandshake(_local, _remote, receivingMars);
sendPacket(_local.channelId);

// Upgrade dispatcherProxy for tests
// Do channel handshake via uch
mockHandShakeProofs(_localUch, _remote);
doProofChannelHandshake(_localUch, _remote, receivingMars);
earth.greet(address(mars), _localUch.channelId, bytes("hello mars"), UINT64_MAX);

// Upgrade dispatcherProxy and uch for tests
upgradeDispatcher(portPrefix, address(dispatcherProxy));
upgradeUch(address(uch), address(dispatcherProxy));
dispatcherProxy.setClientForConnection(connectionHops0[0], dummyLightClient);
dispatcherProxy.setClientForConnection(connectionHops1[0], opLightClient);
}
Expand All @@ -76,10 +94,13 @@ contract DispatcherRC4UpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeTestUti

assertEq(4, nextSequenceSendValue);

// Packets
// Validate packets from previous send
assert(vm.load(address(dispatcherProxy), findSendPacketCommitmentSlot(address(mars), _local.channelId, 1)) > 0);
assert(vm.load(address(dispatcherProxy), findSendPacketCommitmentSlot(address(mars), _local.channelId, 2)) > 0);
assert(vm.load(address(dispatcherProxy), findSendPacketCommitmentSlot(address(mars), _local.channelId, 3)) > 0);
assert(
vm.load(address(dispatcherProxy), findSendPacketCommitmentSlot(address(uch), _localUch.channelId, 1)) > 0
);

// Test sending packet with the updated contract
sendOnePacket(_local.channelId, 4, mars);
Expand All @@ -88,6 +109,14 @@ contract DispatcherRC4UpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeTestUti
uint256(vm.load(address(dispatcherProxy), findNextSequenceSendSlot(address(mars), _local.channelId)))
);
assertEq(5, nextSequenceSendAfterSending);
earth.greet(address(receivingMars), _localUch.channelId, bytes("hello from upgrade mars!"), UINT64_MAX);
assert(
vm.load(address(dispatcherProxy), findSendPacketCommitmentSlot(address(uch), _localUch.channelId, 2)) > 0
);
uint64 nextUCHSequenceSendAfterSending = uint64(
uint256(vm.load(address(dispatcherProxy), findNextSequenceSendSlot(address(uch), _localUch.channelId)))
);
assertEq(3, nextUCHSequenceSendAfterSending);
}

function test_OpenChannelState_Conserved_RC4Upgrade() public {
Expand Down Expand Up @@ -141,6 +170,22 @@ contract DispatcherRC4MidwayUpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeT
ChannelEnd _le;
string sendingPortId;
string receivingPortId;
IUniversalChannelHandler uch;
LocalEnd _localUch;
Earth earth;
uint256 mwBitmap;

GeneralMiddleware mw1;

event RecvMWPacket(
bytes32 indexed channelId,
bytes32 indexed srcPortAddr,
bytes32 indexed destPortAddr,
// middleware UID
uint256 mwId,
bytes appData,
bytes mwData
);

function setUp() public override {
ChannelHandshakeSetting memory setting = ChannelHandshakeSetting(ChannelOrder.ORDERED, false, true, validProof);
Expand All @@ -150,22 +195,37 @@ contract DispatcherRC4MidwayUpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeT
dispatcherProxy = IDispatcher(address(new ERC1967Proxy(address(oldOpDispatcherImplementation), initData)));

mars = new Mars(dispatcherProxy);
uch = deployUCHV2ProxyAndImpl(address(dispatcherProxy));
earth = new Earth(address(uch));
// Register mw stack and test it isn't corrupted after the upgrade
mw1 = new GeneralMiddleware(1 << 1, address(uch));
mwBitmap = uch.MW_ID() | mw1.MW_ID();
address[] memory mwAddrs = new address[](1);
mwAddrs[0] = address(mw1);
uch.registerMwStack(mwBitmap, mwAddrs);
receivingMars = new Mars(dispatcherProxy);

sendingPortId = IbcUtils.addressToPortId(portPrefix, address(mars));
string memory uchSendingPortId = IbcUtils.addressToPortId(portPrefix, address(uch));
receivingPortId = IbcUtils.addressToPortId(portPrefix, address(receivingMars));
// LocalEnd version of _remote
_re = LocalEnd(receivingMars, receivingPortId, "channel-2", connectionHops1, "1.0", "1.0");
// ChannelEnd version of _local
_le = ChannelEnd(sendingPortId, "channel-1", "1.0");
_local = LocalEnd(mars, sendingPortId, "channel-1", connectionHops1, "1.0", "1.0");
_localUch = LocalEnd(uch, uchSendingPortId, "uch-channel", connectionHops1, "1.0", "1.0");
_remote = ChannelEnd(receivingPortId, "channel-2", "1.0");

// Do only half a handshake to test upgrading between handshakes
channelOpenInit(_local, _remote, setting, true);

mockProofs(ChannelState.TRY_PENDING, _re.portId, _le.portId, _re.channelId, _le.channelId);
channelOpenTry(_re, _le, setting, true);

// Do channel handshake via uch using mw set before the upgrade, to ensure that mw addrs state is still useful
mockHandShakeProofs(_localUch, _remote);
doProofChannelHandshake(_localUch, _remote, receivingMars);
earth.greet(address(mars), _localUch.channelId, bytes("hello mars"), UINT64_MAX);
}

// Test that channel handshake can be finished even if done during an upgrade
Expand Down Expand Up @@ -196,6 +256,7 @@ contract DispatcherRC4MidwayUpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeT

// Do upgrade before finishing packet handshake
upgradeDispatcher(portPrefix, address(dispatcherProxy));
upgradeUch(address(uch), address(dispatcherProxy));
dispatcherProxy.setClientForConnection(connectionHops1[0], dummyLightClient);

// Now recv and ack packet
Expand All @@ -221,5 +282,30 @@ contract DispatcherRC4MidwayUpgradeTest is ChannelHandShakeUpgradeUtil, UpgradeT

assertEq(2, nextSequenceAckValue);
assertEq(2, nextSequenceSendValue);

// Now recv uch packet

bytes memory appData = abi.encodePacked("hello using mw stack");
UniversalPacket memory ucPacket =
UniversalPacket(IbcUtils.toBytes32(address(mars)), mwBitmap, IbcUtils.toBytes32(address(earth)), appData);
bytes memory packetData = IbcUtils.toUniversalPacketBytes(ucPacket);
IbcPacket memory uchPacket = IbcPacket(
IbcEndpoint(sendingPortId, _local.channelId),
IbcEndpoint(_localUch.portId, _localUch.channelId),
1,
packetData,
Height(0, 0),
maxTimeout
);
vm.expectEmit(true, true, true, true);
emit RecvMWPacket(
_localUch.channelId,
ucPacket.srcPortAddr,
ucPacket.destPortAddr,
mw1.MW_ID(),
ucPacket.appData,
abi.encodePacked(mw1.MW_ID())
);
dispatcherProxy.recvPacket(uchPacket, validProof);
}
}
Loading

0 comments on commit fe30152

Please sign in to comment.