You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
301 lines
9.8 KiB
301 lines
9.8 KiB
// SPDX-License-Identifier: MIT
|
|
|
|
pragma solidity ^0.8.0;
|
|
|
|
import "../../utils/Counters.sol";
|
|
import "../../utils/math/SafeCast.sol";
|
|
import "../extensions/IGovernorTimelock.sol";
|
|
import "../extensions/GovernorProposalThreshold.sol";
|
|
import "../Governor.sol";
|
|
import "./IGovernorCompatibilityBravo.sol";
|
|
|
|
/**
|
|
* @dev Compatibility layer that implements GovernorBravo compatibility on to of {Governor}.
|
|
*
|
|
* This compatibility layer includes a voting system and requires a {IGovernorTimelock} compatible module to be added
|
|
* through inheritance. It does not include token bindings, not does it include any variable upgrade patterns.
|
|
*
|
|
* _Available since v4.3._
|
|
*/
|
|
abstract contract GovernorCompatibilityBravo is
|
|
IGovernorTimelock,
|
|
IGovernorCompatibilityBravo,
|
|
Governor,
|
|
GovernorProposalThreshold
|
|
{
|
|
using Counters for Counters.Counter;
|
|
using Timers for Timers.BlockNumber;
|
|
|
|
enum VoteType {
|
|
Against,
|
|
For,
|
|
Abstain
|
|
}
|
|
|
|
struct ProposalDetails {
|
|
address proposer;
|
|
address[] targets;
|
|
uint256[] values;
|
|
string[] signatures;
|
|
bytes[] calldatas;
|
|
uint256 forVotes;
|
|
uint256 againstVotes;
|
|
uint256 abstainVotes;
|
|
mapping(address => Receipt) receipts;
|
|
bytes32 descriptionHash;
|
|
}
|
|
|
|
mapping(uint256 => ProposalDetails) private _proposalDetails;
|
|
|
|
// solhint-disable-next-line func-name-mixedcase
|
|
function COUNTING_MODE() public pure virtual override returns (string memory) {
|
|
return "support=bravo&quorum=bravo";
|
|
}
|
|
|
|
// ============================================== Proposal lifecycle ==============================================
|
|
/**
|
|
* @dev See {IGovernor-propose}.
|
|
*/
|
|
function propose(
|
|
address[] memory targets,
|
|
uint256[] memory values,
|
|
bytes[] memory calldatas,
|
|
string memory description
|
|
) public virtual override(IGovernor, Governor, GovernorProposalThreshold) returns (uint256) {
|
|
_storeProposal(_msgSender(), targets, values, new string[](calldatas.length), calldatas, description);
|
|
return super.propose(targets, values, calldatas, description);
|
|
}
|
|
|
|
/**
|
|
* @dev See {IGovernorCompatibilityBravo-propose}.
|
|
*/
|
|
function propose(
|
|
address[] memory targets,
|
|
uint256[] memory values,
|
|
string[] memory signatures,
|
|
bytes[] memory calldatas,
|
|
string memory description
|
|
) public virtual override returns (uint256) {
|
|
_storeProposal(_msgSender(), targets, values, signatures, calldatas, description);
|
|
return propose(targets, values, _encodeCalldata(signatures, calldatas), description);
|
|
}
|
|
|
|
/**
|
|
* @dev See {IGovernorCompatibilityBravo-queue}.
|
|
*/
|
|
function queue(uint256 proposalId) public virtual override {
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
queue(
|
|
details.targets,
|
|
details.values,
|
|
_encodeCalldata(details.signatures, details.calldatas),
|
|
details.descriptionHash
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @dev See {IGovernorCompatibilityBravo-execute}.
|
|
*/
|
|
function execute(uint256 proposalId) public payable virtual override {
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
execute(
|
|
details.targets,
|
|
details.values,
|
|
_encodeCalldata(details.signatures, details.calldatas),
|
|
details.descriptionHash
|
|
);
|
|
}
|
|
|
|
function cancel(uint256 proposalId) public virtual override {
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
|
|
require(
|
|
_msgSender() == details.proposer || getVotes(details.proposer, block.number - 1) < proposalThreshold(),
|
|
"GovernorBravo: proposer above threshold"
|
|
);
|
|
|
|
_cancel(
|
|
details.targets,
|
|
details.values,
|
|
_encodeCalldata(details.signatures, details.calldatas),
|
|
details.descriptionHash
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @dev Encodes calldatas with optional function signature.
|
|
*/
|
|
function _encodeCalldata(string[] memory signatures, bytes[] memory calldatas)
|
|
private
|
|
pure
|
|
returns (bytes[] memory)
|
|
{
|
|
bytes[] memory fullcalldatas = new bytes[](calldatas.length);
|
|
|
|
for (uint256 i = 0; i < signatures.length; ++i) {
|
|
fullcalldatas[i] = bytes(signatures[i]).length == 0
|
|
? calldatas[i]
|
|
: abi.encodeWithSignature(signatures[i], calldatas[i]);
|
|
}
|
|
|
|
return fullcalldatas;
|
|
}
|
|
|
|
/**
|
|
* @dev Store proposal metadata for later lookup
|
|
*/
|
|
function _storeProposal(
|
|
address proposer,
|
|
address[] memory targets,
|
|
uint256[] memory values,
|
|
string[] memory signatures,
|
|
bytes[] memory calldatas,
|
|
string memory description
|
|
) private {
|
|
bytes32 descriptionHash = keccak256(bytes(description));
|
|
uint256 proposalId = hashProposal(targets, values, _encodeCalldata(signatures, calldatas), descriptionHash);
|
|
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
if (details.descriptionHash == bytes32(0)) {
|
|
details.proposer = proposer;
|
|
details.targets = targets;
|
|
details.values = values;
|
|
details.signatures = signatures;
|
|
details.calldatas = calldatas;
|
|
details.descriptionHash = descriptionHash;
|
|
}
|
|
}
|
|
|
|
// ==================================================== Views =====================================================
|
|
/**
|
|
* @dev Part of the Governor Bravo's interface: _"The number of votes required in order for a voter to become a proposer"_.
|
|
*/
|
|
function proposalThreshold()
|
|
public
|
|
view
|
|
virtual
|
|
override(IGovernorCompatibilityBravo, GovernorProposalThreshold)
|
|
returns (uint256);
|
|
|
|
/**
|
|
* @dev See {IGovernorCompatibilityBravo-proposals}.
|
|
*/
|
|
function proposals(uint256 proposalId)
|
|
public
|
|
view
|
|
virtual
|
|
override
|
|
returns (
|
|
uint256 id,
|
|
address proposer,
|
|
uint256 eta,
|
|
uint256 startBlock,
|
|
uint256 endBlock,
|
|
uint256 forVotes,
|
|
uint256 againstVotes,
|
|
uint256 abstainVotes,
|
|
bool canceled,
|
|
bool executed
|
|
)
|
|
{
|
|
id = proposalId;
|
|
eta = proposalEta(proposalId);
|
|
startBlock = proposalSnapshot(proposalId);
|
|
endBlock = proposalDeadline(proposalId);
|
|
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
proposer = details.proposer;
|
|
forVotes = details.forVotes;
|
|
againstVotes = details.againstVotes;
|
|
abstainVotes = details.abstainVotes;
|
|
|
|
ProposalState status = state(proposalId);
|
|
canceled = status == ProposalState.Canceled;
|
|
executed = status == ProposalState.Executed;
|
|
}
|
|
|
|
/**
|
|
* @dev See {IGovernorCompatibilityBravo-getActions}.
|
|
*/
|
|
function getActions(uint256 proposalId)
|
|
public
|
|
view
|
|
virtual
|
|
override
|
|
returns (
|
|
address[] memory targets,
|
|
uint256[] memory values,
|
|
string[] memory signatures,
|
|
bytes[] memory calldatas
|
|
)
|
|
{
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
return (details.targets, details.values, details.signatures, details.calldatas);
|
|
}
|
|
|
|
/**
|
|
* @dev See {IGovernorCompatibilityBravo-getReceipt}.
|
|
*/
|
|
function getReceipt(uint256 proposalId, address voter) public view virtual override returns (Receipt memory) {
|
|
return _proposalDetails[proposalId].receipts[voter];
|
|
}
|
|
|
|
/**
|
|
* @dev See {IGovernorCompatibilityBravo-quorumVotes}.
|
|
*/
|
|
function quorumVotes() public view virtual override returns (uint256) {
|
|
return quorum(block.number - 1);
|
|
}
|
|
|
|
// ==================================================== Voting ====================================================
|
|
/**
|
|
* @dev See {IGovernor-hasVoted}.
|
|
*/
|
|
function hasVoted(uint256 proposalId, address account) public view virtual override returns (bool) {
|
|
return _proposalDetails[proposalId].receipts[account].hasVoted;
|
|
}
|
|
|
|
/**
|
|
* @dev See {Governor-_quorumReached}. In this module, only forVotes count toward the quorum.
|
|
*/
|
|
function _quorumReached(uint256 proposalId) internal view virtual override returns (bool) {
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
return quorum(proposalSnapshot(proposalId)) < details.forVotes;
|
|
}
|
|
|
|
/**
|
|
* @dev See {Governor-_voteSucceeded}. In this module, the forVotes must be scritly over the againstVotes.
|
|
*/
|
|
function _voteSucceeded(uint256 proposalId) internal view virtual override returns (bool) {
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
return details.forVotes > details.againstVotes;
|
|
}
|
|
|
|
/**
|
|
* @dev See {Governor-_countVote}. In this module, the support follows Governor Bravo.
|
|
*/
|
|
function _countVote(
|
|
uint256 proposalId,
|
|
address account,
|
|
uint8 support,
|
|
uint256 weight
|
|
) internal virtual override {
|
|
ProposalDetails storage details = _proposalDetails[proposalId];
|
|
Receipt storage receipt = details.receipts[account];
|
|
|
|
require(!receipt.hasVoted, "GovernorCompatibilityBravo: vote already casted");
|
|
receipt.hasVoted = true;
|
|
receipt.support = support;
|
|
receipt.votes = SafeCast.toUint96(weight);
|
|
|
|
if (support == uint8(VoteType.Against)) {
|
|
details.againstVotes += weight;
|
|
} else if (support == uint8(VoteType.For)) {
|
|
details.forVotes += weight;
|
|
} else if (support == uint8(VoteType.Abstain)) {
|
|
details.abstainVotes += weight;
|
|
} else {
|
|
revert("GovernorCompatibilityBravo: invalid vote type");
|
|
}
|
|
}
|
|
}
|
|
|