Skip to content

Commit

Permalink
add feeVault contract + tests
Browse files Browse the repository at this point in the history
  • Loading branch information
RnkSngh committed Jun 25, 2024
1 parent 0c95970 commit 42fcfb0
Show file tree
Hide file tree
Showing 28 changed files with 542 additions and 49 deletions.
21 changes: 15 additions & 6 deletions contracts/base/GeneralMiddleware.sol
Original file line number Diff line number Diff line change
Expand Up @@ -61,20 +61,29 @@ contract GeneralMiddleware is IbcMwUser, IbcMiddleware, IbcMwEventsEmitter, IbcM
bytes32 destPortAddr,
bytes calldata appData,
uint64 timeoutTimestamp
) external override {
) external override returns (uint64 sequence) {
emit UCHPacketSent(msg.sender, destPortAddr);
_sendPacket(channelId, IbcUtils.toBytes32(msg.sender), destPortAddr, 0, appData, timeoutTimestamp);
return _sendPacket(channelId, IbcUtils.toBytes32(msg.sender), destPortAddr, 0, appData, timeoutTimestamp);

Check warning on line 66 in contracts/base/GeneralMiddleware.sol

View workflow job for this annotation

GitHub Actions / lint

Named parameters missing. MIN unnamed argumenst is 4
}

function sendUniversalPacketWithFee(

Check warning on line 69 in contracts/base/GeneralMiddleware.sol

View workflow job for this annotation

GitHub Actions / lint

Function sendUniversalPacketWithFee() must match Foundry test naming convention
bytes32 channelId,
bytes32 destPortAddr,
bytes calldata appData,
uint64 timeoutTimestamp,
uint256[2] calldata gasLimits,
uint256[2] calldata gasPrices
) external payable override returns (uint64 sequence) {}

Check warning on line 76 in contracts/base/GeneralMiddleware.sol

View workflow job for this annotation

GitHub Actions / lint

Code contains empty blocks

function sendMWPacket(

Check warning on line 78 in contracts/base/GeneralMiddleware.sol

View workflow job for this annotation

GitHub Actions / lint

Function sendMWPacket() must match Foundry test naming convention
bytes32 channelId,
bytes32 srcPortAddr,
bytes32 destPortAddr,
uint256 srcMwIds,
bytes calldata appData,
uint64 timeoutTimestamp
) external override {
_sendPacket(channelId, srcPortAddr, destPortAddr, srcMwIds, appData, timeoutTimestamp);
) external override returns (uint64 sequence) {
return _sendPacket(channelId, srcPortAddr, destPortAddr, srcMwIds, appData, timeoutTimestamp);

Check warning on line 86 in contracts/base/GeneralMiddleware.sol

View workflow job for this annotation

GitHub Actions / lint

Named parameters missing. MIN unnamed argumenst is 4
}

function onRecvMWPacket(

Check warning on line 89 in contracts/base/GeneralMiddleware.sol

View workflow job for this annotation

GitHub Actions / lint

All public or external methods in a contract must override a definition from an interface

Check warning on line 89 in contracts/base/GeneralMiddleware.sol

View workflow job for this annotation

GitHub Actions / lint

Function onRecvMWPacket() must match Foundry test naming convention
Expand Down Expand Up @@ -191,7 +200,7 @@ contract GeneralMiddleware is IbcMwUser, IbcMiddleware, IbcMwEventsEmitter, IbcM
uint256 srcMwIds,
bytes calldata appData,
uint64 timeoutTimestamp
) internal virtual {
) internal virtual returns (uint64 sequence) {
// extra MW custom logic here to process packet, eg. emit MW events, mutate state, etc.
// implementer can emit custom data fields suitable for their use cases.
// Here we use MW_ID as the custom MW data field.
Expand All @@ -200,7 +209,7 @@ contract GeneralMiddleware is IbcMwUser, IbcMiddleware, IbcMwEventsEmitter, IbcM
);

// send packet to next MW
IbcMwPacketSender(mw).sendMWPacket(
return IbcMwPacketSender(mw).sendMWPacket(
channelId, srcPortAddr, destPortAddr, srcMwIds | MW_ID, appData, timeoutTimestamp
);
}
Expand Down
5 changes: 4 additions & 1 deletion contracts/core/Dispatcher.sol
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ import {ReentrancyGuard} from "@openzeppelin/contracts/security/ReentrancyGuard.
import {Channel, ChannelEnd, ChannelOrder, IbcPacket, ChannelState, AckPacket, Ibc} from "../libs/Ibc.sol";
import {IBCErrors} from "../libs/IbcErrors.sol";
import {IbcUtils} from "../libs/IbcUtils.sol";
import {IFeeVault} from "../interfaces/IFeeVault.sol";

/**
* @title Dispatcher
Expand Down Expand Up @@ -64,6 +65,7 @@ contract Dispatcher is OwnableUpgradeable, UUPSUpgradeable, ReentrancyGuard, IDi
ILightClient _UNUSED; // From previous dispatcher version
mapping(bytes32 => string) private _channelIdToConnection;
mapping(string => ILightClient) private _connectionToLightClient;
IFeeVault public feeVault;

constructor() {
_disableInitializers();
Expand All @@ -75,13 +77,14 @@ contract Dispatcher is OwnableUpgradeable, UUPSUpgradeable, ReentrancyGuard, IDi
* @dev This method should be called only once during contract deployment.
* @dev For contract upgarades, which need to reinitialize the contract, use the reinitializer modifier.
*/
function initialize(string memory initPortPrefix) public virtual initializer nonReentrant {
function initialize(string memory initPortPrefix, IFeeVault _feeVault) public virtual initializer nonReentrant {
if (bytes(initPortPrefix).length == 0) {
revert IBCErrors.invalidPortPrefix();
}
__Ownable_init();
portPrefix = initPortPrefix;
portPrefixLen = uint32(bytes(initPortPrefix).length);
feeVault = _feeVault;
}

/**
Expand Down
52 changes: 52 additions & 0 deletions contracts/core/FeeVault.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
// SPDX-License-Identifier: Apache-2.0
/*
* Copyright 2024, Polymer Labs
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

pragma solidity 0.8.15;

import {Ownable} from "@openzeppelin/contracts/access/Ownable.sol";
import {IFeeVault} from "../interfaces/IFeeVault.sol";
import {ReentrancyGuard} from "@openzeppelin/contracts/security/ReentrancyGuard.sol";

contract FeeVault is Ownable, ReentrancyGuard, IFeeVault {
function depositSendPacketFee(
bytes32 channelId,
uint64 sequence,
uint256[2] calldata gasLimits,
uint256[2] calldata gasPrices
) external payable nonReentrant {
uint256 fee = gasLimits[0] * gasPrices[0] + gasLimits[1] * gasPrices[1];
if ((fee) != msg.value) {
revert IncorrectFeeSent(fee, msg.value);
}
emit SendPacketFeeDeposited(channelId, sequence, gasLimits, gasPrices);
}

function depositOpenChannelFee(address src, string calldata destPortId, string[] calldata connectionHops)
external
payable
nonReentrant
{
if (msg.value == 0) {
revert NoFeeSent();
}
emit OpenChannelFeeDeposited(src, destPortId, connectionHops, msg.value);
}

function withdrawFeesToOwner() external {
payable(owner()).transfer(address(this).balance);
}
}
35 changes: 31 additions & 4 deletions contracts/core/UniversalChannelHandler.sol
Original file line number Diff line number Diff line change
Expand Up @@ -19,10 +19,11 @@ pragma solidity 0.8.15;

import {IbcDispatcher} from "../interfaces/IbcDispatcher.sol";
import {IbcUniversalChannelMW, IbcUniversalPacketReceiver} from "../interfaces/IbcMiddleware.sol";
import {IbcReceiverBaseUpgradeable} from "../interfaces/IbcReceiverUpgradeable.sol";
import {IbcReceiverBaseUpgradeable} from "../implementation_templates/IbcReceiverUpgradeable.sol";
import {ChannelOrder, IbcPacket, AckPacket, UniversalPacket} from "../libs/Ibc.sol";
import {IbcUtils} from "../libs/IbcUtils.sol";
import {UUPSUpgradeable} from "@openzeppelin/contracts/proxy/utils/UUPSUpgradeable.sol";
import {FeeSender} from "../implementation_templates/FeeSender.sol";

/**
* @title Universal Channel Handler
Expand All @@ -31,7 +32,7 @@ import {UUPSUpgradeable} from "@openzeppelin/contracts/proxy/utils/UUPSUpgradeab
* channel handshake to establish a channel.
* @dev This contract can integrate directly with dapps, or a middleware stack for packet routing.
*/
contract UniversalChannelHandler is IbcReceiverBaseUpgradeable, UUPSUpgradeable, IbcUniversalChannelMW {
contract UniversalChannelHandler is IbcReceiverBaseUpgradeable, FeeSender, UUPSUpgradeable, IbcUniversalChannelMW {
bytes32 private _UNUSED; // Storage placeholder to ensure upgrade from this version is backwards compatible

string public constant VERSION = "1.0";
Expand Down Expand Up @@ -82,12 +83,38 @@ contract UniversalChannelHandler is IbcReceiverBaseUpgradeable, UUPSUpgradeable,
bytes32 destPortAddr,
bytes calldata appData,
uint64 timeoutTimestamp
) external {
) external returns (uint64 sequence) {
bytes memory packetData = IbcUtils.toUniversalPacketBytes(
UniversalPacket(IbcUtils.toBytes32(msg.sender), MW_ID, destPortAddr, appData)
);
emit UCHPacketSent(msg.sender, destPortAddr);
dispatcher.sendPacket(channelId, packetData, timeoutTimestamp);
sequence = dispatcher.sendPacket(channelId, packetData, timeoutTimestamp);
}

/**
* @notice Sends a universal packet over an IBC channel
* @param channelId The channel ID through which the packet is sent on the dispatcher
* @param destPortAddr The destination port address
* @param appData The packet data to be sent
* @param timeoutTimestamp of when the packet can timeout
*/
function sendUniversalPacketWithFee(
bytes32 channelId,
bytes32 destPortAddr,
bytes calldata appData,
uint64 timeoutTimestamp,
uint256[2] calldata gasLimits,
uint256[2] calldata gasPrices
) external payable returns (uint64 sequence) {
// Cache dispatcher for gas savings
IbcDispatcher _dispatcher = dispatcher;

bytes memory packetData = IbcUtils.toUniversalPacketBytes(
UniversalPacket(IbcUtils.toBytes32(msg.sender), MW_ID, destPortAddr, appData)
);
emit UCHPacketSent(msg.sender, destPortAddr);
sequence = _dispatcher.sendPacket(channelId, packetData, timeoutTimestamp);
_depositSendPacketFee(dispatcher, channelId, sequence, gasLimits, gasPrices);
}

/**
Expand Down
22 changes: 22 additions & 0 deletions contracts/examples/Earth.sol
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ pragma solidity ^0.8.9;
import {UniversalPacket, AckPacket} from "../libs/Ibc.sol";
import {IbcUtils} from "../libs/IbcUtils.sol";
import {IbcUniversalPacketReceiverBase, IbcUniversalPacketSender} from "../interfaces/IbcMiddleware.sol";
import {IUniversalChannelHandler} from "../interfaces/IUniversalChannelHandler.sol";

/**
* @title Earth
Expand Down Expand Up @@ -48,12 +49,33 @@ contract Earth is IbcUniversalPacketReceiverBase {

constructor(address _middleware) IbcUniversalPacketReceiverBase(_middleware) {}

/**
* @notice Send a packet to a destination chain. without a fewe
* @notice this is useful for self-relaying apckets which don't rely on polymer to fund .
* @param destPortAddr The destination chain's port address.
* @param channelId The channel id to send the packet on.
* @param message The message to send.
* @param timeoutTimestamp The timeout timestamp for the packet.
*/
function greet(address destPortAddr, bytes32 channelId, bytes calldata message, uint64 timeoutTimestamp) external {
IbcUniversalPacketSender(mw).sendUniversalPacket(
channelId, IbcUtils.toBytes32(destPortAddr), message, timeoutTimestamp
);
}

function greetWithFee(
address destPortAddr,
bytes32 channelId,
bytes calldata message,
uint64 timeoutTimestamp,
uint256[2] memory gasLimits,
uint256[2] memory gasPrices
) external payable returns (uint64 sequence) {
return IUniversalChannelHandler(mw).sendUniversalPacketWithFee{value: msg.value}(
channelId, IbcUtils.toBytes32(destPortAddr), message, timeoutTimestamp, gasLimits, gasPrices
);
}

function onRecvUniversalPacket(bytes32 channelId, UniversalPacket calldata packet)
external
onlyIbcMw
Expand Down
41 changes: 38 additions & 3 deletions contracts/examples/Mars.sol
Original file line number Diff line number Diff line change
Expand Up @@ -20,14 +20,15 @@ pragma solidity ^0.8.9;
import {AckPacket, ChannelOrder} from "../libs/Ibc.sol";
import {IbcReceiverBase, IbcReceiver, IbcPacket} from "../interfaces/IbcReceiver.sol";
import {IbcDispatcher} from "../interfaces/IbcDispatcher.sol";
import {FeeSender} from "../implementation_templates/FeeSender.sol";

/**
* @title Mars
* @notice Mars is a simple IBC receiver contract that receives packets and sends acks.
* @dev This contract is used for only testing IBC functionality and as an example for dapp developers on how to
* integrate with the vibc protocol.
*/
contract Mars is IbcReceiverBase, IbcReceiver {
contract Mars is IbcReceiverBase, IbcReceiver, FeeSender {
// received packet as chain B
IbcPacket[] public recvedPackets;
// received ack packet as chain A
Expand All @@ -50,6 +51,18 @@ contract Mars is IbcReceiverBase, IbcReceiver {
dispatcher.channelOpenInit(version, ordering, feeEnabled, connectionHops, counterpartyPortId);
}

function triggerChannelInitWithFee(
string calldata version,
ChannelOrder ordering,
bool feeEnabled,
string[] calldata connectionHops,
string calldata counterpartyPortId
) external payable onlyOwner {
IbcDispatcher _dispatcher = dispatcher; // cache for gas savings to avoid 2 SLOADS
_dispatcher.channelOpenInit(version, ordering, feeEnabled, connectionHops, counterpartyPortId);
_depositOpenChannelFee(_dispatcher, counterpartyPortId, connectionHops);
}

function onRecvPacket(IbcPacket memory packet)
external
virtual
Expand Down Expand Up @@ -91,14 +104,36 @@ contract Mars is IbcReceiverBase, IbcReceiver {
dispatcher.channelCloseInit(channelId);
}

/**
* @dev Sends a packet with a greeting message over a specified channel.
* @param message The greeting message to be sent.
* @param channelId The ID of the channel to send the packet to.
* @param timeoutTimestamp The timestamp at which the packet will expire if not received.
* @dev This method also returns sequence from the dispatcher for easy testing
*/
function greet(string calldata message, bytes32 channelId, uint64 timeoutTimestamp)
external
payable
returns (uint64 sequence)
{
sequence = dispatcher.sendPacket(channelId, bytes(message), timeoutTimestamp);
}

/**
* @dev Sends a packet with a greeting message over a specified channel.
* @param message The greeting message to be sent.
* @param channelId The ID of the channel to send the packet to.
* @param timeoutTimestamp The timestamp at which the packet will expire if not received.
*/
function greet(string calldata message, bytes32 channelId, uint64 timeoutTimestamp) external {
dispatcher.sendPacket(channelId, bytes(message), timeoutTimestamp);
function greetWithFee(
string calldata message,
bytes32 channelId,
uint64 timeoutTimestamp,
uint256[2] calldata gasLimits,
uint256[2] calldata gasPrices
) external payable returns (uint64 sequence) {
sequence = dispatcher.sendPacket(channelId, bytes(message), timeoutTimestamp);
_depositSendPacketFee(dispatcher, channelId, sequence, gasLimits, gasPrices);
}

function onChanOpenInit(ChannelOrder, string[] calldata, string calldata, string calldata version)
Expand Down
41 changes: 41 additions & 0 deletions contracts/implementation_templates/FeeSender.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
// SPDX-License-Identifier: Apache-2.0
/*
* Copyright 2024, Polymer Labs
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

pragma solidity 0.8.15;

import {IbcDispatcher} from "../interfaces/IbcDispatcher.sol";

/// Contract with
abstract contract FeeSender {
function _depositSendPacketFee(
IbcDispatcher dispatcher,
bytes32 channelId,
uint64 sequence,
uint256[2] calldata gasLimits,
uint256[2] calldata gasPrices
) internal {
dispatcher.feeVault().depositSendPacketFee{value: msg.value}(channelId, sequence, gasLimits, gasPrices);
}

function _depositOpenChannelFee(
IbcDispatcher dispatcher,
string calldata destPortId,
string[] calldata connectionHops
) internal {
dispatcher.feeVault().depositOpenChannelFee{value: msg.value}(address(this), destPortId, connectionHops);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -18,12 +18,13 @@
pragma solidity ^0.8.9;

import {OwnableUpgradeable} from "@openzeppelin-upgradeable/contracts/access/OwnableUpgradeable.sol";
import {IbcDispatcher} from "./IbcDispatcher.sol";
import {IbcDispatcher} from "../interfaces/IbcDispatcher.sol";

contract IbcReceiverBaseUpgradeable is OwnableUpgradeable {
IbcDispatcher public dispatcher;

error notIbcDispatcher();
error invalidAddress();
error UnsupportedVersion();
error ChannelNotFound();

Expand Down
2 changes: 2 additions & 0 deletions contracts/interfaces/IDispatcher.sol
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import {IbcDispatcher, IbcEventsEmitter} from "./IbcDispatcher.sol";
import {L1Header, OpL2StateProof, Ics23Proof} from "./IProofVerifier.sol";
import {Channel, ChannelEnd, ChannelOrder, IbcPacket} from "../libs/Ibc.sol";
import {ILightClient} from "./ILightClient.sol";
import {IFeeVault} from "./IFeeVault.sol";

interface IDispatcher is IbcDispatcher, IbcEventsEmitter {
function setPortPrefix(string calldata _portPrefix) external;
Expand Down Expand Up @@ -105,6 +106,7 @@ interface IDispatcher is IbcDispatcher, IbcEventsEmitter {
function writeTimeoutPacket(IbcPacket calldata packet, Ics23Proof calldata proof) external;

function recvPacket(IbcPacket calldata packet, Ics23Proof calldata proof) external;
function feeVault() external returns (IFeeVault feeVault);

function getOptimisticConsensusState(uint256 height, string calldata connection)
external
Expand Down
Loading

0 comments on commit 42fcfb0

Please sign in to comment.