Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Iterations #50

Merged
merged 5 commits into from
Jan 31, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .npmrc
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
@angleprotocol:registry=https://npm.pkg.github.com
191 changes: 81 additions & 110 deletions contracts/DistributionCreator.sol

Large diffs are not rendered by default.

8 changes: 4 additions & 4 deletions contracts/Distributor.sol
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,7 @@ struct Claim {
}

/// @title Distributor
/// @notice Allows LPs on AMMs with concentrated liquidity to claim the rewards that were distributed to them
/// @notice Allows to claim rewards distributed to them through Merkl
/// @author Angle Labs. Inc
contract Distributor is UUPSHelper {
using SafeERC20 for IERC20;
Expand Down Expand Up @@ -290,20 +290,20 @@ contract Distributor is UUPSHelper {
}

/// @notice Sets the dispute period after which a tree update becomes effective
function setDisputePeriod(uint48 _disputePeriod) external onlyGovernorOrGuardian {
function setDisputePeriod(uint48 _disputePeriod) external onlyGovernor {
disputePeriod = uint48(_disputePeriod);
emit DisputePeriodUpdated(_disputePeriod);
}

/// @notice Sets the token used as a caution during disputes
function setDisputeToken(IERC20 _disputeToken) external onlyGovernorOrGuardian {
function setDisputeToken(IERC20 _disputeToken) external onlyGovernor {
if (disputer != address(0)) revert UnresolvedDispute();
disputeToken = _disputeToken;
emit DisputeTokenUpdated(address(_disputeToken));
}

/// @notice Sets the amount of `disputeToken` used as a caution during disputes
function setDisputeAmount(uint256 _disputeAmount) external onlyGovernorOrGuardian {
function setDisputeAmount(uint256 _disputeAmount) external onlyGovernor {
if (disputer != address(0)) revert UnresolvedDispute();
disputeAmount = _disputeAmount;
emit DisputeAmountUpdated(_disputeAmount);
Expand Down
307 changes: 261 additions & 46 deletions contracts/deprecated/OldDistributionCreator.sol

Large diffs are not rendered by default.

68 changes: 43 additions & 25 deletions contracts/deprecated/OldDistributor.sol
Original file line number Diff line number Diff line change
Expand Up @@ -35,9 +35,9 @@

pragma solidity ^0.8.17;

import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "@openzeppelin/contracts/utils/math/SafeCast.sol";
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { SafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import { SafeCast } from "@openzeppelin/contracts/utils/math/SafeCast.sol";

import "../utils/UUPSHelper.sol";

Expand All @@ -54,14 +54,18 @@ struct MerkleTree {
struct Claim {
uint208 amount;
uint48 timestamp;
bytes32 merkleRoot;
}

/// @title OldDistributor
/// @title Distributor
/// @notice Allows LPs on AMMs with concentrated liquidity to claim the rewards that were distributed to them
/// @author Angle Labs. Inc
contract OldDistributor is UUPSHelper {
using SafeERC20 for IERC20;

/// @notice Epoch duration
uint32 internal constant _EPOCH_DURATION = 3600;

// ================================= VARIABLES =================================

/// @notice Tree of claimable tokens through this contract
Expand All @@ -80,10 +84,10 @@ contract OldDistributor is UUPSHelper {
/// @dev Used to store if there is an ongoing dispute
address public disputer;

/// @notice Last time the `tree` was updated
uint48 public lastTreeUpdate;
/// @notice When the current tree will become valid
uint48 public endOfDisputePeriod;

/// @notice Time before which a change in a tree becomes effective
/// @notice Time after which a change in a tree becomes effective, in EPOCH_DURATION
uint48 public disputePeriod;

/// @notice Amount to deposit to freeze the roots update
Expand All @@ -105,16 +109,17 @@ contract OldDistributor is UUPSHelper {

// =================================== EVENTS ==================================

event Claimed(address user, address token, uint256 amount);
event Claimed(address indexed user, address indexed token, uint256 amount);
event DisputeAmountUpdated(uint256 _disputeAmount);
event Disputed(string reason);
event DisputePeriodUpdated(uint48 _disputePeriod);
event DisputeTokenUpdated(address indexed _disputeToken);
event DisputeAmountUpdated(uint256 _disputeAmount);
event DisputeResolved(bool valid);
event OperatorClaimingToggled(address user, bool isEnabled);
event OperatorToggled(address user, address operator, bool isWhitelisted);
event DisputeTokenUpdated(address indexed _disputeToken);
event OperatorClaimingToggled(address indexed user, bool isEnabled);
event OperatorToggled(address indexed user, address indexed operator, bool isWhitelisted);
event Recovered(address indexed token, address indexed to, uint256 amount);
event TreeUpdated(bytes32 merkleRoot, bytes32 ipfsHash);
event Revoked(); // With this event an indexer could maintain a table (timestamp, merkleRootUpdate)
event TreeUpdated(bytes32 merkleRoot, bytes32 ipfsHash, uint48 endOfDisputePeriod);
event TrustedToggled(address indexed eoa, bool trust);

// ================================= MODIFIERS =================================
Expand Down Expand Up @@ -181,7 +186,7 @@ contract OldDistributor is UUPSHelper {

// Closing reentrancy gate here
uint256 toSend = amount - claimed[user][token].amount;
claimed[user][token] = Claim(SafeCast.toUint208(amount), uint48(block.timestamp));
claimed[user][token] = Claim(SafeCast.toUint208(amount), uint48(block.timestamp), getMerkleRoot());

IERC20(token).safeTransfer(user, toSend);
emit Claimed(user, token, toSend);
Expand All @@ -193,7 +198,7 @@ contract OldDistributor is UUPSHelper {

/// @notice Returns the MerkleRoot that is currently live for the contract
function getMerkleRoot() public view returns (bytes32) {
if (block.timestamp - lastTreeUpdate >= disputePeriod) return tree.merkleRoot;
if (block.timestamp >= endOfDisputePeriod && disputer == address(0)) return tree.merkleRoot;
else return lastTree.merkleRoot;
}

Expand All @@ -212,21 +217,24 @@ contract OldDistributor is UUPSHelper {
disputer != address(0) ||
// A trusted address cannot update a tree right after a precedent tree update otherwise it can de facto
// validate a tree which has not passed the dispute period
((canUpdateMerkleRoot[msg.sender] != 1 || block.timestamp - lastTreeUpdate < disputePeriod) &&
((canUpdateMerkleRoot[msg.sender] != 1 || block.timestamp < endOfDisputePeriod) &&
!core.isGovernorOrGuardian(msg.sender))
) revert NotTrusted();
MerkleTree memory _lastTree = tree;
tree = _tree;
lastTree = _lastTree;
lastTreeUpdate = uint48(block.timestamp);
emit TreeUpdated(_tree.merkleRoot, _tree.ipfsHash);

uint48 _endOfPeriod = _endOfDisputePeriod(uint48(block.timestamp));
endOfDisputePeriod = _endOfPeriod;
emit TreeUpdated(_tree.merkleRoot, _tree.ipfsHash, _endOfPeriod);
}

/// @notice Freezes the Merkle tree update until the dispute is resolved
/// @dev Requires a deposit of `disputeToken` that'll be slashed if the dispute is not accepted
/// @dev It is only possible to create a dispute for `disputePeriod` after each tree update
/// @dev It is only possible to create a dispute within `disputePeriod` after each tree update
function disputeTree(string memory reason) external {
if (block.timestamp - lastTreeUpdate >= disputePeriod) revert InvalidDispute();
if (disputer != address(0)) revert UnresolvedDispute();
if (block.timestamp >= endOfDisputePeriod) revert InvalidDispute();
IERC20(disputeToken).safeTransferFrom(msg.sender, address(this), disputeAmount);
disputer = msg.sender;
emit Disputed(reason);
Expand All @@ -242,7 +250,7 @@ contract OldDistributor is UUPSHelper {
_revokeTree();
} else {
IERC20(disputeToken).safeTransfer(msg.sender, disputeAmount);
lastTreeUpdate = uint48(block.timestamp);
endOfDisputePeriod = _endOfDisputePeriod(uint48(block.timestamp));
}
disputer = address(0);
emit DisputeResolved(valid);
Expand Down Expand Up @@ -275,9 +283,8 @@ contract OldDistributor is UUPSHelper {
emit Recovered(tokenAddress, to, amountToRecover);
}

/// @notice Sets the dispute period before which a tree update becomes effective
/// @notice Sets the dispute period after which a tree update becomes effective
function setDisputePeriod(uint48 _disputePeriod) external onlyGovernorOrGuardian {
if (_disputePeriod > block.timestamp) revert InvalidParam();
disputePeriod = uint48(_disputePeriod);
emit DisputePeriodUpdated(_disputePeriod);
}
Expand All @@ -301,9 +308,20 @@ contract OldDistributor is UUPSHelper {
/// @notice Fallback to the last version of the tree
function _revokeTree() internal {
MerkleTree memory _tree = lastTree;
lastTreeUpdate = 0;
endOfDisputePeriod = 0;
tree = _tree;
emit TreeUpdated(_tree.merkleRoot, _tree.ipfsHash);
emit Revoked();
emit TreeUpdated(
_tree.merkleRoot,
_tree.ipfsHash,
(uint48(block.timestamp) / _EPOCH_DURATION) * (_EPOCH_DURATION) // Last hour
);
}

/// @notice Returns the end of the dispute period
/// @dev treeUpdate is rounded up to next hour and then `disputePeriod` hours are added
function _endOfDisputePeriod(uint48 treeUpdate) internal view returns (uint48) {
return ((treeUpdate - 1) / _EPOCH_DURATION + 1 + disputePeriod) * (_EPOCH_DURATION);
}

/// @notice Checks the validity of a proof
Expand Down
29 changes: 29 additions & 0 deletions contracts/struct/CampaignParameters.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
// SPDX-License-Identifier: BUSL-1.1

pragma solidity ^0.8.17;

struct CampaignParameters {
// POPULATED ONCE CREATED

// ID of the campaign. This can be left as a null bytes32 when creating campaigns
// on Merkl.
bytes32 campaignId;
// CHOSEN BY CAMPAIGN CREATOR

// Address of the campaign creator, if marked as address(0), it will be overriden with the
// address of the `msg.sender` creating the campaign
address creator;
// Address of the token used as a reward
address rewardToken;
// Amount of `rewardToken` to distribute across all the epochs
// Amount distributed per epoch is `amount/numEpoch`
uint256 amount;
// Type of campaign
uint32 campaignType;
// Timestamp at which the campaign should start
uint32 startTimestamp;
// Duration of the campaign in seconds. Has to be a multiple of EPOCH = 3600
uint32 duration;
// Extra data to pass to specify the campaign
bytes campaignData;
}
8 changes: 4 additions & 4 deletions hardhat.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -88,11 +88,9 @@ const config: HardhatUserConfig = {
hardfork: 'london',
forking: {
enabled: argv.fork || false,
/*
// Mainnet
url: nodeUrl('fork'),
blockNumber: 16671190,
*/
blockNumber: 19127150,
// Polygon
/*
url: nodeUrl('forkpolygon'),
Expand All @@ -116,8 +114,10 @@ const config: HardhatUserConfig = {
url: nodeUrl('polygonzkevm'),
blockNumber: 3214816,
*/
/*
url: nodeUrl('linea'),
// blockNumber: 14188687,
blockNumber: 14188687,
*/
},
mining: argv.disableAutoMining
? {
Expand Down
4 changes: 2 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@
"url": "https://github.com/AngleProtocol/merkl-contracts/issues"
},
"devDependencies": {
"@angleprotocol/sdk": "3.0.87",
"@angleprotocol/sdk": "0.21.1",
"@ethersproject/abi": "^5.7.0",
"@ethersproject/providers": "^5.7.1",
"@nomicfoundation/hardhat-chai-matchers": "^1.0.3",
Expand Down Expand Up @@ -73,7 +73,7 @@
"eslint-plugin-promise": "^6.0.0",
"eslint-plugin-standard": "5.0.0",
"ethers": "^5.7.1",
"hardhat": "^2.12.6",
"hardhat": "^2.19.4",
"hardhat-abi-exporter": "2.2.1",
"hardhat-contract-sizer": "2.0.3",
"hardhat-deploy": "^0.11.23",
Expand Down
8 changes: 8 additions & 0 deletions scripts/checkUpgradeability.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,14 @@ async function main() {

testUpgradeability('Distributor', 'contracts/Distributor.sol');
testStorage('OldDistributor', 'contracts/deprecated/OldDistributor.sol', 'Distributor', 'contracts/Distributor.sol');

testUpgradeability('DistributionCreator', 'contracts/DistributionCreator.sol');
testStorage(
'OldDistributionCreator',
'contracts/deprecated/OldDistributionCreator.sol',
'DistributionCreator',
'contracts/DistributionCreator.sol',
);
}

main().catch(error => {
Expand Down
Loading
Loading