From d6ded227e05935c8eff32622d9a4d01236346b9a Mon Sep 17 00:00:00 2001 From: Gas <86567384+gas1cent@users.noreply.github.com> Date: Thu, 26 Oct 2023 16:36:12 +0400 Subject: [PATCH] perf: store participants in a bytes field (#4) --- solidity/contracts/Oracle.sol | 30 +++++++++++++++---- solidity/test/integration/IntegrationBase.sol | 2 +- solidity/test/unit/Oracle.t.sol | 2 +- 3 files changed, 26 insertions(+), 8 deletions(-) diff --git a/solidity/contracts/Oracle.sol b/solidity/contracts/Oracle.sol index 459333a..e483326 100644 --- a/solidity/contracts/Oracle.sol +++ b/solidity/contracts/Oracle.sol @@ -6,7 +6,6 @@ import {EnumerableSet} from '@openzeppelin/contracts/utils/structs/EnumerableSet contract Oracle is IOracle { using EnumerableSet for EnumerableSet.Bytes32Set; - using EnumerableSet for EnumerableSet.AddressSet; /// @inheritdoc IOracle mapping(bytes32 _responseId => bytes32 _disputeId) public disputeOf; @@ -33,7 +32,7 @@ contract Oracle is IOracle { /** * @notice The list of the participants for each request */ - mapping(bytes32 _requestId => EnumerableSet.AddressSet _participants) internal _participants; + mapping(bytes32 _requestId => bytes _participants) internal _participants; /** * @notice The finalized response for each request @@ -194,7 +193,7 @@ contract Oracle is IOracle { } _responseId = keccak256(abi.encodePacked(_proposer, address(this), _requestId, _responseNonce++)); - _participants[_requestId].add(_proposer); + _participants[_requestId] = abi.encodePacked(_participants[_requestId], _proposer); _responses[_responseId] = _request.responseModule.propose(_requestId, _proposer, _responseData, msg.sender); _responseIds[_requestId].add(_responseId); @@ -241,7 +240,7 @@ contract Oracle is IOracle { } _disputeId = keccak256(abi.encodePacked(msg.sender, _requestId, _responseId)); - _participants[_requestId].add(msg.sender); + _participants[_requestId] = abi.encodePacked(_participants[_requestId], msg.sender); Dispute memory _dispute = _request.disputeModule.disputeResponse(_requestId, _responseId, msg.sender, _response.proposer); @@ -326,8 +325,27 @@ contract Oracle is IOracle { || address(_request.finalityModule) == _module; } + // @inheritdoc IOracle function isParticipant(bytes32 _requestId, address _user) external view returns (bool _isParticipant) { - _isParticipant = _participants[_requestId].contains(_user); + bytes memory _requestParticipants = _participants[_requestId]; + + assembly { + let length := mload(_requestParticipants) + let i := 0 + + // Iterate 20-bytes chunks of the participants data + for {} lt(i, length) { i := add(i, 20) } { + // Load the participant at index i + let _participant := mload(add(add(_requestParticipants, 0x20), i)) + + // Shift the participant to the right by 96 bits and compare with _user + if eq(shr(96, _participant), _user) { + // Set _isParticipant to true and return + mstore(0x00, 1) + return(0x00, 32) + } + } + } } /// @inheritdoc IOracle @@ -436,7 +454,7 @@ contract Oracle is IOracle { }); _requests[_requestId] = _storedRequest; - _participants[_requestId].add(msg.sender); + _participants[_requestId] = abi.encodePacked(_participants[_requestId], msg.sender); _request.requestModule.setupRequest(_requestId, _request.requestModuleData); _request.responseModule.setupRequest(_requestId, _request.responseModuleData); diff --git a/solidity/test/integration/IntegrationBase.sol b/solidity/test/integration/IntegrationBase.sol index c4ec1a3..46f5f11 100644 --- a/solidity/test/integration/IntegrationBase.sol +++ b/solidity/test/integration/IntegrationBase.sol @@ -29,7 +29,7 @@ import {TestConstants} from '../utils/TestConstants.sol'; // solhint-enable no-unused-import contract IntegrationBase is DSTestPlus, TestConstants, Helpers { - uint256 public constant FORK_BLOCK = 756_611; + uint256 public constant FORK_BLOCK = 111_361_902; uint256 internal _initialBalance = 100_000 ether; diff --git a/solidity/test/unit/Oracle.t.sol b/solidity/test/unit/Oracle.t.sol index ed68e19..e03d9fa 100644 --- a/solidity/test/unit/Oracle.t.sol +++ b/solidity/test/unit/Oracle.t.sol @@ -49,7 +49,7 @@ contract ForTest_Oracle is Oracle { } function forTest_addParticipant(bytes32 _requestId, address _participant) external { - _participants[_requestId].add(_participant); + _participants[_requestId] = abi.encodePacked(_participants[_requestId], _participant); } function forTest_setFinalizedResponseId(bytes32 _requestId, bytes32 _finalizedResponseId) external {