GovernorTimelockCompound.sol 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. // SPDX-License-Identifier: MIT
  2. // OpenZeppelin Contracts (last updated v4.9.0) (governance/extensions/GovernorTimelockCompound.sol)
  3. pragma solidity ^0.8.19;
  4. import "./IGovernorTimelock.sol";
  5. import "../Governor.sol";
  6. import "../../utils/math/SafeCast.sol";
  7. import "../../vendor/compound/ICompoundTimelock.sol";
  8. /**
  9. * @dev Extension of {Governor} that binds the execution process to a Compound Timelock. This adds a delay, enforced by
  10. * the external timelock to all successful proposal (in addition to the voting duration). The {Governor} needs to be
  11. * the admin of the timelock for any operation to be performed. A public, unrestricted,
  12. * {GovernorTimelockCompound-__acceptAdmin} is available to accept ownership of the timelock.
  13. *
  14. * Using this model means the proposal will be operated by the {TimelockController} and not by the {Governor}. Thus,
  15. * the assets and permissions must be attached to the {TimelockController}. Any asset sent to the {Governor} will be
  16. * inaccessible.
  17. *
  18. * _Available since v4.3._
  19. */
  20. abstract contract GovernorTimelockCompound is IGovernorTimelock, Governor {
  21. ICompoundTimelock private _timelock;
  22. /// @custom:oz-retyped-from mapping(uint256 => GovernorTimelockCompound.ProposalTimelock)
  23. mapping(uint256 => uint64) private _proposalTimelocks;
  24. /**
  25. * @dev Emitted when the timelock controller used for proposal execution is modified.
  26. */
  27. event TimelockChange(address oldTimelock, address newTimelock);
  28. /**
  29. * @dev Set the timelock.
  30. */
  31. constructor(ICompoundTimelock timelockAddress) {
  32. _updateTimelock(timelockAddress);
  33. }
  34. /**
  35. * @dev See {IERC165-supportsInterface}.
  36. */
  37. function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, Governor) returns (bool) {
  38. return interfaceId == type(IGovernorTimelock).interfaceId || super.supportsInterface(interfaceId);
  39. }
  40. /**
  41. * @dev Overridden version of the {Governor-state} function with added support for the `Queued` and `Expired` state.
  42. */
  43. function state(uint256 proposalId) public view virtual override(IGovernor, Governor) returns (ProposalState) {
  44. ProposalState currentState = super.state(proposalId);
  45. if (currentState != ProposalState.Succeeded) {
  46. return currentState;
  47. }
  48. uint256 eta = proposalEta(proposalId);
  49. if (eta == 0) {
  50. return currentState;
  51. } else if (block.timestamp >= eta + _timelock.GRACE_PERIOD()) {
  52. return ProposalState.Expired;
  53. } else {
  54. return ProposalState.Queued;
  55. }
  56. }
  57. /**
  58. * @dev Public accessor to check the address of the timelock
  59. */
  60. function timelock() public view virtual override returns (address) {
  61. return address(_timelock);
  62. }
  63. /**
  64. * @dev Public accessor to check the eta of a queued proposal
  65. */
  66. function proposalEta(uint256 proposalId) public view virtual override returns (uint256) {
  67. return _proposalTimelocks[proposalId];
  68. }
  69. /**
  70. * @dev Function to queue a proposal to the timelock.
  71. */
  72. function queue(
  73. address[] memory targets,
  74. uint256[] memory values,
  75. bytes[] memory calldatas,
  76. bytes32 descriptionHash
  77. ) public virtual override returns (uint256) {
  78. uint256 proposalId = hashProposal(targets, values, calldatas, descriptionHash);
  79. ProposalState currentState = state(proposalId);
  80. if (currentState != ProposalState.Succeeded) {
  81. revert GovernorIncorrectState(proposalId, currentState, _encodeState(ProposalState.Succeeded));
  82. }
  83. uint256 eta = block.timestamp + _timelock.delay();
  84. _proposalTimelocks[proposalId] = SafeCast.toUint64(eta);
  85. for (uint256 i = 0; i < targets.length; ++i) {
  86. if (_timelock.queuedTransactions(keccak256(abi.encode(targets[i], values[i], "", calldatas[i], eta)))) {
  87. revert GovernorProposalAlreadyQueued(proposalId);
  88. }
  89. _timelock.queueTransaction(targets[i], values[i], "", calldatas[i], eta);
  90. }
  91. emit ProposalQueued(proposalId, eta);
  92. return proposalId;
  93. }
  94. /**
  95. * @dev Overridden execute function that run the already queued proposal through the timelock.
  96. */
  97. function _execute(
  98. uint256 proposalId,
  99. address[] memory targets,
  100. uint256[] memory values,
  101. bytes[] memory calldatas,
  102. bytes32 /*descriptionHash*/
  103. ) internal virtual override {
  104. uint256 eta = proposalEta(proposalId);
  105. if (eta == 0) {
  106. revert GovernorProposalNotQueued(proposalId);
  107. }
  108. Address.sendValue(payable(_timelock), msg.value);
  109. for (uint256 i = 0; i < targets.length; ++i) {
  110. _timelock.executeTransaction(targets[i], values[i], "", calldatas[i], eta);
  111. }
  112. }
  113. /**
  114. * @dev Overridden version of the {Governor-_cancel} function to cancel the timelocked proposal if it as already
  115. * been queued.
  116. */
  117. function _cancel(
  118. address[] memory targets,
  119. uint256[] memory values,
  120. bytes[] memory calldatas,
  121. bytes32 descriptionHash
  122. ) internal virtual override returns (uint256) {
  123. uint256 proposalId = super._cancel(targets, values, calldatas, descriptionHash);
  124. uint256 eta = proposalEta(proposalId);
  125. if (eta > 0) {
  126. // update state first
  127. delete _proposalTimelocks[proposalId];
  128. // do external call later
  129. for (uint256 i = 0; i < targets.length; ++i) {
  130. _timelock.cancelTransaction(targets[i], values[i], "", calldatas[i], eta);
  131. }
  132. }
  133. return proposalId;
  134. }
  135. /**
  136. * @dev Address through which the governor executes action. In this case, the timelock.
  137. */
  138. function _executor() internal view virtual override returns (address) {
  139. return address(_timelock);
  140. }
  141. /**
  142. * @dev Accept admin right over the timelock.
  143. */
  144. // solhint-disable-next-line private-vars-leading-underscore
  145. function __acceptAdmin() public {
  146. _timelock.acceptAdmin();
  147. }
  148. /**
  149. * @dev Public endpoint to update the underlying timelock instance. Restricted to the timelock itself, so updates
  150. * must be proposed, scheduled, and executed through governance proposals.
  151. *
  152. * For security reasons, the timelock must be handed over to another admin before setting up a new one. The two
  153. * operations (hand over the timelock) and do the update can be batched in a single proposal.
  154. *
  155. * Note that if the timelock admin has been handed over in a previous operation, we refuse updates made through the
  156. * timelock if admin of the timelock has already been accepted and the operation is executed outside the scope of
  157. * governance.
  158. * CAUTION: It is not recommended to change the timelock while there are other queued governance proposals.
  159. */
  160. function updateTimelock(ICompoundTimelock newTimelock) external virtual onlyGovernance {
  161. _updateTimelock(newTimelock);
  162. }
  163. function _updateTimelock(ICompoundTimelock newTimelock) private {
  164. emit TimelockChange(address(_timelock), address(newTimelock));
  165. _timelock = newTimelock;
  166. }
  167. }