-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #12 from defi-wonderland/perf/opo-604-erc20-resolu…
…tion-module perf: optimize `ERC20ResolutionModule`
- Loading branch information
Showing
4 changed files
with
687 additions
and
668 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
241 changes: 129 additions & 112 deletions
241
solidity/contracts/modules/resolution/ERC20ResolutionModule.sol
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,112 +1,129 @@ | ||
// // SPDX-License-Identifier: MIT | ||
// pragma solidity ^0.8.19; | ||
|
||
// // solhint-disable-next-line no-unused-import | ||
// import {IERC20} from '@openzeppelin/contracts/token/ERC20/IERC20.sol'; | ||
// import {SafeERC20} from '@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol'; | ||
// import {EnumerableSet} from '@openzeppelin/contracts/utils/structs/EnumerableSet.sol'; | ||
|
||
// // solhint-disable-next-line no-unused-import | ||
// import {Module, IModule} from '@defi-wonderland/prophet-core-contracts/solidity/contracts/Module.sol'; | ||
// import {IOracle} from '@defi-wonderland/prophet-core-contracts/solidity/interfaces/IOracle.sol'; | ||
|
||
// import {IERC20ResolutionModule} from '../../../interfaces/modules/resolution/IERC20ResolutionModule.sol'; | ||
|
||
// contract ERC20ResolutionModule is Module, IERC20ResolutionModule { | ||
// using SafeERC20 for IERC20; | ||
// using EnumerableSet for EnumerableSet.AddressSet; | ||
|
||
// /// @inheritdoc IERC20ResolutionModule | ||
// mapping(bytes32 _disputeId => Escalation _escalation) public escalations; | ||
|
||
// /// @inheritdoc IERC20ResolutionModule | ||
// mapping(bytes32 _disputeId => mapping(address _voter => uint256 _numOfVotes)) public votes; | ||
|
||
// mapping(bytes32 _disputeId => EnumerableSet.AddressSet _votersSet) private _voters; | ||
|
||
// constructor(IOracle _oracle) Module(_oracle) {} | ||
|
||
// /// @inheritdoc IModule | ||
// function moduleName() external pure returns (string memory _moduleName) { | ||
// return 'ERC20ResolutionModule'; | ||
// } | ||
|
||
// /// @inheritdoc IERC20ResolutionModule | ||
// function decodeRequestData(bytes32 _requestId) public view returns (RequestParameters memory _params) { | ||
// _params = abi.decode(requestData[_requestId], (RequestParameters)); | ||
// } | ||
|
||
// /// @inheritdoc IERC20ResolutionModule | ||
// function startResolution(bytes32 _disputeId) external onlyOracle { | ||
// escalations[_disputeId].startTime = block.timestamp; | ||
// emit VotingPhaseStarted(block.timestamp, _disputeId); | ||
// } | ||
|
||
// /// @inheritdoc IERC20ResolutionModule | ||
// function castVote(bytes32 _requestId, bytes32 _disputeId, uint256 _numberOfVotes) public { | ||
// IOracle.Dispute memory _dispute = ORACLE.getDispute(_disputeId); | ||
// if (_dispute.createdAt == 0) revert ERC20ResolutionModule_NonExistentDispute(); | ||
// if (_dispute.status != IOracle.DisputeStatus.None) revert ERC20ResolutionModule_AlreadyResolved(); | ||
|
||
// Escalation memory _escalation = escalations[_disputeId]; | ||
// if (_escalation.startTime == 0) revert ERC20ResolutionModule_DisputeNotEscalated(); | ||
|
||
// RequestParameters memory _params = decodeRequestData(_requestId); | ||
// uint256 _deadline = _escalation.startTime + _params.timeUntilDeadline; | ||
// if (block.timestamp >= _deadline) revert ERC20ResolutionModule_VotingPhaseOver(); | ||
|
||
// votes[_disputeId][msg.sender] += _numberOfVotes; | ||
|
||
// _voters[_disputeId].add(msg.sender); | ||
// escalations[_disputeId].totalVotes += _numberOfVotes; | ||
|
||
// _params.votingToken.safeTransferFrom(msg.sender, address(this), _numberOfVotes); | ||
// emit VoteCast(msg.sender, _disputeId, _numberOfVotes); | ||
// } | ||
|
||
// /// @inheritdoc IERC20ResolutionModule | ||
// function resolveDispute(bytes32 _disputeId) external onlyOracle { | ||
// // 0. Check disputeId actually exists and that it isn't resolved already | ||
// IOracle.Dispute memory _dispute = ORACLE.getDispute(_disputeId); | ||
// if (_dispute.createdAt == 0) revert ERC20ResolutionModule_NonExistentDispute(); | ||
// if (_dispute.status != IOracle.DisputeStatus.None) revert ERC20ResolutionModule_AlreadyResolved(); | ||
|
||
// Escalation memory _escalation = escalations[_disputeId]; | ||
// // 1. Check that the dispute is actually escalated | ||
// if (_escalation.startTime == 0) revert ERC20ResolutionModule_DisputeNotEscalated(); | ||
|
||
// // 2. Check that voting deadline is over | ||
// RequestParameters memory _params = decodeRequestData(_dispute.requestId); | ||
// uint256 _deadline = _escalation.startTime + _params.timeUntilDeadline; | ||
// if (block.timestamp < _deadline) revert ERC20ResolutionModule_OnGoingVotingPhase(); | ||
|
||
// uint256 _quorumReached = _escalation.totalVotes >= _params.minVotesForQuorum ? 1 : 0; | ||
|
||
// address[] memory __voters = _voters[_disputeId].values(); | ||
|
||
// // 5. Update status | ||
// if (_quorumReached == 1) { | ||
// ORACLE.updateDisputeStatus(_disputeId, IOracle.DisputeStatus.Won); | ||
// emit DisputeResolved(_dispute.requestId, _disputeId, IOracle.DisputeStatus.Won); | ||
// } else { | ||
// ORACLE.updateDisputeStatus(_disputeId, IOracle.DisputeStatus.Lost); | ||
// emit DisputeResolved(_dispute.requestId, _disputeId, IOracle.DisputeStatus.Lost); | ||
// } | ||
|
||
// uint256 _votersLength = __voters.length; | ||
|
||
// // 6. Return tokens | ||
// for (uint256 _i; _i < _votersLength;) { | ||
// address _voter = __voters[_i]; | ||
// _params.votingToken.safeTransfer(_voter, votes[_disputeId][_voter]); | ||
// unchecked { | ||
// ++_i; | ||
// } | ||
// } | ||
// } | ||
|
||
// /// @inheritdoc IERC20ResolutionModule | ||
// function getVoters(bytes32 _disputeId) external view returns (address[] memory __voters) { | ||
// __voters = _voters[_disputeId].values(); | ||
// } | ||
// } | ||
// SPDX-License-Identifier: MIT | ||
pragma solidity ^0.8.19; | ||
|
||
// solhint-disable-next-line no-unused-import | ||
import {IERC20} from '@openzeppelin/contracts/token/ERC20/IERC20.sol'; | ||
import {SafeERC20} from '@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol'; | ||
import {EnumerableSet} from '@openzeppelin/contracts/utils/structs/EnumerableSet.sol'; | ||
|
||
// solhint-disable-next-line no-unused-import | ||
import {Module, IModule} from '@defi-wonderland/prophet-core-contracts/solidity/contracts/Module.sol'; | ||
import {IOracle} from '@defi-wonderland/prophet-core-contracts/solidity/interfaces/IOracle.sol'; | ||
|
||
import {IERC20ResolutionModule} from '../../../interfaces/modules/resolution/IERC20ResolutionModule.sol'; | ||
|
||
contract ERC20ResolutionModule is Module, IERC20ResolutionModule { | ||
using SafeERC20 for IERC20; | ||
using EnumerableSet for EnumerableSet.AddressSet; | ||
|
||
/// @inheritdoc IERC20ResolutionModule | ||
mapping(bytes32 _disputeId => Escalation _escalation) public escalations; | ||
|
||
/// @inheritdoc IERC20ResolutionModule | ||
mapping(bytes32 _disputeId => mapping(address _voter => uint256 _numOfVotes)) public votes; | ||
|
||
/** | ||
* @notice The list of voters for each dispute | ||
*/ | ||
mapping(bytes32 _disputeId => EnumerableSet.AddressSet _votersSet) internal _voters; | ||
|
||
constructor(IOracle _oracle) Module(_oracle) {} | ||
|
||
/// @inheritdoc IModule | ||
function moduleName() external pure returns (string memory _moduleName) { | ||
return 'ERC20ResolutionModule'; | ||
} | ||
|
||
/// @inheritdoc IERC20ResolutionModule | ||
function decodeRequestData(bytes calldata _data) public pure returns (RequestParameters memory _params) { | ||
_params = abi.decode(_data, (RequestParameters)); | ||
} | ||
|
||
/// @inheritdoc IERC20ResolutionModule | ||
function startResolution( | ||
bytes32 _disputeId, | ||
IOracle.Request calldata _request, | ||
IOracle.Response calldata _response, | ||
IOracle.Dispute calldata _dispute | ||
) external onlyOracle { | ||
escalations[_disputeId].startTime = block.timestamp; | ||
emit VotingPhaseStarted(block.timestamp, _disputeId); | ||
} | ||
|
||
/// @inheritdoc IERC20ResolutionModule | ||
function castVote( | ||
IOracle.Request calldata _request, | ||
IOracle.Dispute calldata _dispute, | ||
uint256 _numberOfVotes | ||
) public { | ||
bytes32 _disputeId = _getId(_dispute); | ||
Escalation memory _escalation = escalations[_disputeId]; | ||
if (_escalation.startTime == 0) revert ERC20ResolutionModule_DisputeNotEscalated(); | ||
if (ORACLE.disputeStatus(_disputeId) != IOracle.DisputeStatus.Escalated) { | ||
revert ERC20ResolutionModule_AlreadyResolved(); | ||
} | ||
|
||
RequestParameters memory _params = decodeRequestData(_request.resolutionModuleData); | ||
uint256 _deadline = _escalation.startTime + _params.timeUntilDeadline; | ||
if (block.timestamp >= _deadline) revert ERC20ResolutionModule_VotingPhaseOver(); | ||
|
||
votes[_disputeId][msg.sender] += _numberOfVotes; | ||
|
||
_voters[_disputeId].add(msg.sender); | ||
escalations[_disputeId].totalVotes += _numberOfVotes; | ||
|
||
_params.votingToken.safeTransferFrom(msg.sender, address(this), _numberOfVotes); | ||
emit VoteCast(msg.sender, _disputeId, _numberOfVotes); | ||
} | ||
|
||
/// @inheritdoc IERC20ResolutionModule | ||
function resolveDispute( | ||
bytes32 _disputeId, | ||
IOracle.Request calldata _request, | ||
IOracle.Response calldata _response, | ||
IOracle.Dispute calldata _dispute | ||
) external onlyOracle { | ||
// 0. Check disputeId actually exists and that it isn't resolved already | ||
if (ORACLE.disputeStatus(_disputeId) != IOracle.DisputeStatus.Escalated) { | ||
revert ERC20ResolutionModule_AlreadyResolved(); | ||
} | ||
|
||
// 1. Check that the dispute is actually escalated | ||
Escalation memory _escalation = escalations[_disputeId]; | ||
if (_escalation.startTime == 0) revert ERC20ResolutionModule_DisputeNotEscalated(); | ||
|
||
// 2. Check that voting deadline is over | ||
RequestParameters memory _params = decodeRequestData(_request.resolutionModuleData); | ||
uint256 _deadline = _escalation.startTime + _params.timeUntilDeadline; | ||
if (block.timestamp < _deadline) revert ERC20ResolutionModule_OnGoingVotingPhase(); | ||
|
||
uint256 _quorumReached = _escalation.totalVotes >= _params.minVotesForQuorum ? 1 : 0; | ||
|
||
address[] memory __voters = _voters[_disputeId].values(); | ||
|
||
// 5. Update status | ||
if (_quorumReached == 1) { | ||
ORACLE.updateDisputeStatus(_request, _response, _dispute, IOracle.DisputeStatus.Won); | ||
emit DisputeResolved(_dispute.requestId, _disputeId, IOracle.DisputeStatus.Won); | ||
} else { | ||
ORACLE.updateDisputeStatus(_request, _response, _dispute, IOracle.DisputeStatus.Lost); | ||
emit DisputeResolved(_dispute.requestId, _disputeId, IOracle.DisputeStatus.Lost); | ||
} | ||
|
||
uint256 _votersLength = __voters.length; | ||
|
||
// 6. Return tokens | ||
for (uint256 _i; _i < _votersLength;) { | ||
address _voter = __voters[_i]; | ||
_params.votingToken.safeTransfer(_voter, votes[_disputeId][_voter]); | ||
unchecked { | ||
++_i; | ||
} | ||
} | ||
} | ||
|
||
/// @inheritdoc IERC20ResolutionModule | ||
function getVoters(bytes32 _disputeId) external view returns (address[] memory __voters) { | ||
__voters = _voters[_disputeId].values(); | ||
} | ||
} |
Oops, something went wrong.