Skip to content

Commit

Permalink
Upgradeable MezoAllocator (#338)
Browse files Browse the repository at this point in the history
Depends on: #326

Make the MezoAllocator contract upgradable and deploy it as a
transparent proxy with Open Zeppelin upgrades plugin.
  • Loading branch information
dimpar authored Apr 10, 2024
2 parents c64e9f8 + c13953b commit fb87c72
Show file tree
Hide file tree
Showing 4 changed files with 377 additions and 21 deletions.
33 changes: 21 additions & 12 deletions core/contracts/MezoAllocator.sol
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ pragma solidity ^0.8.21;

import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "@openzeppelin/contracts/access/Ownable2Step.sol";
import "@openzeppelin/contracts-upgradeable/access/Ownable2StepUpgradeable.sol";
import {ZeroAddress} from "./utils/Errors.sol";
import "./stBTC.sol";
import "./interfaces/IDispatcher.sol";
Expand Down Expand Up @@ -59,15 +59,15 @@ interface IMezoPortal {
}

/// @notice MezoAllocator routes tBTC to/from MezoPortal.
contract MezoAllocator is IDispatcher, Ownable2Step {
contract MezoAllocator is IDispatcher, Ownable2StepUpgradeable {
using SafeERC20 for IERC20;

/// @notice Address of the MezoPortal contract.
IMezoPortal public immutable mezoPortal;
IMezoPortal public mezoPortal;
/// @notice tBTC token contract.
IERC20 public immutable tbtc;
IERC20 public tbtc;
/// @notice stBTC token vault contract.
stBTC public immutable stbtc;
stBTC public stbtc;
/// @notice Keeps track of the addresses that are allowed to trigger deposit
/// allocations.
mapping(address => bool) public isMaintainer;
Expand Down Expand Up @@ -109,26 +109,35 @@ contract MezoAllocator is IDispatcher, Ownable2Step {
_;
}

/// @custom:oz-upgrades-unsafe-allow constructor
constructor() {
_disableInitializers();
}

/// @notice Initializes the MezoAllocator contract.
/// @param _mezoPortal Address of the MezoPortal contract.
/// @param _tbtc Address of the tBTC token contract.
constructor(
function initialize(
address _mezoPortal,
IERC20 _tbtc,
stBTC _stbtc
) Ownable(msg.sender) {
address _tbtc,
address _stbtc
) public initializer {
__Ownable2Step_init();
__Ownable_init(msg.sender);

if (_mezoPortal == address(0)) {
revert ZeroAddress();
}
if (address(_tbtc) == address(0)) {
if (_tbtc == address(0)) {
revert ZeroAddress();
}
if (address(_stbtc) == address(0)) {
revert ZeroAddress();
}

mezoPortal = IMezoPortal(_mezoPortal);
tbtc = _tbtc;
stbtc = _stbtc;
tbtc = IERC20(_tbtc);
stbtc = stBTC(_stbtc);
}

/// @notice Allocate tBTC to MezoPortal. Each allocation creates a new "rolling"
Expand Down
250 changes: 250 additions & 0 deletions core/contracts/test/upgrades/MezoAllocatorV2.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,250 @@
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.21;

import {IERC20} from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "@openzeppelin/contracts-upgradeable/access/Ownable2StepUpgradeable.sol";
import {ZeroAddress} from "../../utils/Errors.sol";
import "../../stBTC.sol";
import "../../interfaces/IDispatcher.sol";

/// @title IMezoPortal
/// @dev Interface for the Mezo's Portal contract.
interface IMezoPortal {
/// @notice DepositInfo keeps track of the deposit balance and unlock time.
/// Each deposit is tracked separately and associated with a specific
/// token. Some tokens can be deposited but can not be locked - in
/// that case the unlockAt is the block timestamp of when the deposit
/// was created. The same is true for tokens that can be locked but
/// the depositor decided not to lock them.
struct DepositInfo {
uint96 balance;
uint32 unlockAt;
}

/// @notice Deposit and optionally lock tokens for the given period.
/// @dev Lock period will be normalized to weeks. If non-zero, it must not
/// be shorter than the minimum lock period and must not be longer than
/// the maximum lock period.
/// @param token token address to deposit
/// @param amount amount of tokens to deposit
/// @param lockPeriod lock period in seconds, 0 to not lock the deposit
function deposit(address token, uint96 amount, uint32 lockPeriod) external;

/// @notice Withdraw deposited tokens.
/// Deposited lockable tokens can be withdrawn at any time if
/// there is no lock set on the deposit or the lock period has passed.
/// There is no way to withdraw locked deposit. Tokens that are not
/// lockable can be withdrawn at any time. Deposit can be withdrawn
/// partially.
/// @param token deposited token address
/// @param depositId id of the deposit
/// @param amount amount of the token to be withdrawn from the deposit
function withdraw(address token, uint256 depositId, uint96 amount) external;

/// @notice The number of deposits created. Includes the deposits that
/// were fully withdrawn. This is also the identifier of the most
/// recently created deposit.
function depositCount() external view returns (uint256);

/// @notice Get the balance and unlock time of a given deposit.
/// @param depositor depositor address
/// @param token token address to get the balance
/// @param depositId id of the deposit
function getDeposit(
address depositor,
address token,
uint256 depositId
) external view returns (DepositInfo memory);
}

/// @notice MezoAllocator routes tBTC to/from MezoPortal.
contract MezoAllocatorV2 is IDispatcher, Ownable2StepUpgradeable {
using SafeERC20 for IERC20;

/// @notice Address of the MezoPortal contract.
IMezoPortal public mezoPortal;
/// @notice tBTC token contract.
IERC20 public tbtc;
/// @notice stBTC token vault contract.
stBTC public stbtc;
/// @notice Keeps track of the addresses that are allowed to trigger deposit
/// allocations.
mapping(address => bool) public isMaintainer;
/// @notice List of maintainers.
address[] public maintainers;
/// @notice keeps track of the latest deposit ID assigned in Mezo Portal.
uint256 public depositId;
/// @notice Keeps track of the total amount of tBTC allocated to MezoPortal.
uint96 public depositBalance;

// TEST: New variable.
uint256 public newVariable;

/// @notice Emitted when tBTC is deposited to MezoPortal.
event DepositAllocated(
uint256 indexed oldDepositId,
uint256 indexed newDepositId,
uint256 addedAmount,
uint256 newDepositAmount
);
/// @notice Emitted when tBTC is withdrawn from MezoPortal.
event DepositWithdrawn(uint256 indexed depositId, uint256 amount);
/// @notice Emitted when the maintainer address is updated.
event MaintainerAdded(address indexed maintainer);
/// @notice Emitted when the maintainer address is updated.
event MaintainerRemoved(address indexed maintainer);
/// @notice Emitted when tBTC is released from MezoPortal.
event DepositReleased(uint256 indexed depositId, uint256 amount);
// TEST: New event.
event NewEvent();
/// @notice Reverts if the caller is not a maintainer.
error CallerNotMaintainer();
/// @notice Reverts if the caller is not the stBTC contract.
error CallerNotStbtc();
/// @notice Reverts if the maintainer is already registered.
error MaintainerNotRegistered();
/// @notice Reverts if the caller is already a maintainer.
error MaintainerAlreadyRegistered();

modifier onlyMaintainer() {
if (!isMaintainer[msg.sender]) {
revert CallerNotMaintainer();
}
_;
}

/// @custom:oz-upgrades-unsafe-allow constructor
constructor() {
_disableInitializers();
}

function initialize(
address _mezoPortal,
address _tbtc,
address _stbtc
) public initializer {
// TEST: Removed content of initialize function. Initialize shouldn't be
// called again during the upgrade because of the `initializer`
// modifier.
}

// TEST: Initializer for V2.
function initializeV2(uint256 _newVariable) public reinitializer(2) {
newVariable = _newVariable;
}

/// @notice Allocate tBTC to MezoPortal. Each allocation creates a new "rolling"
/// deposit meaning that the previous Acre's deposit is fully withdrawn
/// before a new deposit with added amount is created. This mimics a
/// "top up" functionality with the difference that a new deposit id
/// is created and the previous deposit id is no longer in use.
/// @dev This function can be invoked periodically by a maintainer.
function allocate() external onlyMaintainer {
if (depositBalance > 0) {
// Free all Acre's tBTC from MezoPortal before creating a new deposit.
// slither-disable-next-line reentrancy-no-eth
mezoPortal.withdraw(address(tbtc), depositId, depositBalance);
}

// Fetch unallocated tBTC from stBTC contract.
uint256 addedAmount = tbtc.balanceOf(address(stbtc));
// slither-disable-next-line arbitrary-send-erc20
tbtc.safeTransferFrom(address(stbtc), address(this), addedAmount);

// Create a new deposit in the MezoPortal.
depositBalance = uint96(tbtc.balanceOf(address(this)));
tbtc.forceApprove(address(mezoPortal), depositBalance);
// 0 denotes no lock period for this deposit.
mezoPortal.deposit(address(tbtc), depositBalance, 0);
uint256 oldDepositId = depositId;
// MezoPortal doesn't return depositId, so we have to read depositCounter
// which assigns depositId to the current deposit.
depositId = mezoPortal.depositCount();

// slither-disable-next-line reentrancy-events
emit DepositAllocated(
oldDepositId,
depositId,
addedAmount,
depositBalance
);
}

/// @notice Withdraws tBTC from MezoPortal and transfers it to stBTC.
/// This function can withdraw partial or a full amount of tBTC from
/// MezoPortal for a given deposit id.
/// @param amount Amount of tBTC to withdraw.
function withdraw(uint256 amount) external {
if (msg.sender != address(stbtc)) revert CallerNotStbtc();

emit DepositWithdrawn(depositId, amount);
mezoPortal.withdraw(address(tbtc), depositId, uint96(amount));
// slither-disable-next-line reentrancy-benign
depositBalance -= uint96(amount);
tbtc.safeTransfer(address(stbtc), amount);
}

/// @notice Releases deposit in full from MezoPortal.
/// @dev This is a special function that can be used to migrate funds during
/// allocator upgrade or in case of emergencies.
function releaseDeposit() external onlyOwner {
uint96 amount = mezoPortal
.getDeposit(address(this), address(tbtc), depositId)
.balance;

emit DepositReleased(depositId, amount);
depositBalance = 0;
mezoPortal.withdraw(address(tbtc), depositId, amount);
tbtc.safeTransfer(address(stbtc), tbtc.balanceOf(address(this)));
}

/// @notice Updates the maintainer address.
/// @param maintainerToAdd Address of the new maintainer.
// TEST: Modified function.
function addMaintainer(address maintainerToAdd) external onlyOwner {
if (maintainerToAdd == address(0)) {
revert ZeroAddress();
}
if (isMaintainer[maintainerToAdd]) {
revert MaintainerAlreadyRegistered();
}
maintainers.push(maintainerToAdd);
isMaintainer[maintainerToAdd] = true;

emit MaintainerAdded(maintainerToAdd);

// TEST: Emit new event.
emit NewEvent();
}

/// @notice Removes the maintainer address.
/// @param maintainerToRemove Address of the maintainer to remove.
function removeMaintainer(address maintainerToRemove) external onlyOwner {
if (!isMaintainer[maintainerToRemove]) {
revert MaintainerNotRegistered();
}
delete (isMaintainer[maintainerToRemove]);

for (uint256 i = 0; i < maintainers.length; i++) {
if (maintainers[i] == maintainerToRemove) {
maintainers[i] = maintainers[maintainers.length - 1];
// slither-disable-next-line costly-loop
maintainers.pop();
break;
}
}

emit MaintainerRemoved(maintainerToRemove);
}

/// @notice Returns the total amount of tBTC allocated to MezoPortal.
function totalAssets() external view returns (uint256) {
return depositBalance;
}

/// @notice Returns the list of maintainers.
function getMaintainers() external view returns (address[] memory) {
return maintainers;
}
}
24 changes: 15 additions & 9 deletions core/deploy/02_deploy_mezo_allocator.ts
Original file line number Diff line number Diff line change
@@ -1,24 +1,30 @@
import type { HardhatRuntimeEnvironment } from "hardhat/types"
import type { DeployFunction } from "hardhat-deploy/types"
import { waitConfirmationsNumber } from "../helpers/deployment"
import { waitForTransaction } from "../helpers/deployment"

const func: DeployFunction = async (hre: HardhatRuntimeEnvironment) => {
const { getNamedAccounts, deployments, helpers } = hre
const { deployer } = await getNamedAccounts()
const { governance } = await getNamedAccounts()
const { deployer } = await helpers.signers.getNamedSigners()

const tbtc = await deployments.get("TBTC")
const stbtc = await deployments.get("stBTC")
const mezoPortal = await deployments.get("MezoPortal")

const mezoAllocator = await deployments.deploy("MezoAllocator", {
from: deployer,
args: [mezoPortal.address, tbtc.address, stbtc.address],
log: true,
waitConfirmations: waitConfirmationsNumber(hre),
const [, deployment] = await helpers.upgrades.deployProxy("MezoAllocator", {
factoryOpts: {
signer: deployer,
},
initializerArgs: [mezoPortal.address, tbtc.address, stbtc.address],
proxyOpts: {
kind: "transparent",
initialOwner: governance,
},
})

if (hre.network.tags.etherscan) {
await helpers.etherscan.verify(mezoAllocator)
if (deployment.transactionHash && hre.network.tags.etherscan) {
await waitForTransaction(hre, deployment.transactionHash)
await helpers.etherscan.verify(deployment)
}

// TODO: Add Tenderly verification
Expand Down
Loading

0 comments on commit fb87c72

Please sign in to comment.