Skip to content

Commit

Permalink
Merge pull request #12 from defi-wonderland/perf/opo-604-erc20-resolu…
Browse files Browse the repository at this point in the history
…tion-module

perf: optimize `ERC20ResolutionModule`
  • Loading branch information
0xmoebius authored Nov 12, 2023
2 parents 5c66eee + 0aec25b commit 23c9a60
Show file tree
Hide file tree
Showing 4 changed files with 687 additions and 668 deletions.
10 changes: 5 additions & 5 deletions docs/src/content/modules/resolution/erc20_resolution_module.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,11 +10,11 @@ The `ERC20ResolutionModule` is a dispute resolution module that decides on the o

### Key Methods

- `decodeRequestData(bytes32 _requestId)`: Decodes the request data associated with a given request ID.
- `startResolution(bytes32 _disputeId)`: Starts the resolution process for a given dispute.
- `castVote(bytes32 _requestId, bytes32 _disputeId, uint256 _numberOfVotes)`: Allows a user to cast votes for a dispute.
- `resolveDispute(bytes32 _disputeId)`: Resolves a dispute based on the votes cast.
- `getVoters(bytes32 _disputeId)`: Returns the addresses of the voters for a given dispute.
- `decodeRequestData`: Decodes the request data associated with a given request ID.
- `startResolution`: Starts the resolution process for a given dispute.
- `castVote`: Allows a user to cast votes for a dispute.
- `resolveDispute`: Resolves a dispute based on the votes cast.
- `getVoters`: Returns the addresses of the voters for a given dispute.

### Request Parameters

Expand Down
241 changes: 129 additions & 112 deletions solidity/contracts/modules/resolution/ERC20ResolutionModule.sol
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();
}
}
Loading

0 comments on commit 23c9a60

Please sign in to comment.