// SPDX-License-Identifier: Apache-2.0 // https://docs.soliditylang.org/en/v0.8.10/style-guide.html pragma solidity ^0.8.10; import "./IAccounts.sol"; import "./ILockedGold.sol"; import "./IElection.sol"; import "./IRegistry.sol"; import "./IValidators.sol"; /// /// ▐▌ ▛ /// █ /// ▗▛▀▀▌ ▗▛▀▀▀▙ ▐▌ ▟▀▀▘▗▛▀▀▀▙ ▙ ▟▀▀▀▖ /// ▝▙▄ ▛ ▝▌ ▐▌ ▌ ▛ ▝▌ ▛ ▐▄▖▖ /// ▀▙ █ ▐▌ ▐▌ ▛ █ ▐▌ █ ▝▀▌ /// ▐▄▄▄▛ █▚▄▄▄▀ ▐▌ ▛ ▝▙▄▄▄▀▙ ▙ ▜▄▄▄▘ /// ▙ /// ▌ /// /// /// @title SpiralsStaking /// @author @douglasqian /// @notice This smart contract implements a simple staking protocol that /// plugs into Celo's L1 staking. On a high-level, the contract allows callers /// to stake with a designated validator group and unstake from it as well. /// The lockup period is the same as what's on the L1. /// /// Staking through this contract sends value directly to Celo's /// LockedGold contract so there should never be a large amount of /// value accrued in this contract's balance. From a security perspective /// it also means that we are leveraging the testing & auditing done on /// Celo's smart contracts to ensure funds are secured. /// /// Callers who staked through this contract can only redeem the original /// staking amount which means the rewards from staking are managed by the /// contract. /// /// @dev This contract simplifies the staking experience for Spirals users /// because it pushes the responsibility of activation & withdrawal to /// the contract admin. For activation, this means that it's up to us to /// activate the pending votes after the next epoch passes. Every call to /// "stake" resets the timer on behalf of this contract in Celo's Election /// smart contract. /// /// On the other side, users looking to unstake are able to after the /// same lockup period on the L1. This means that Spirals bears the /// responsibility of withdrawing on behalf of the protocol in a timely /// manner to ensure there's enough liquidity. Otherwise, users eligible to /// withdraw from Spirals will not be able to. We also address this by /// allowing deposits into a buffer pool. In the long-run though, building /// some automation around this would be ideal. /// contract SpiralsStaking { /* * Struct Definitions */ struct StakerInfo { uint256 stakedValue; uint256 withdrawalValue; uint256 withdrawalTimestamp; } /* * State Variables */ address public validatorGroup; address private ownerDeprecated; // don't remove this, will change slot assignments uint256 public bufferPool; uint256 public totalPendingWithdrawal; IRegistry constant c_celoRegistry = IRegistry(0x000000000000000000000000000000000000ce10); // TODO: change to nested mapping if we want to support multiple validator groups mapping(address => StakerInfo) stakers; /* * Events */ event Deposit( address indexed sender, uint256 indexed amount, bool isBuffer ); event UserCeloStaked( address indexed _address, address indexed _validatorGroup, uint256 indexed amount ); event ProtocolCeloActivated( address indexed _validatorGroup, uint256 indexed amount ); event UserCeloUnstaked( address indexed _address, address indexed _validatorGroup, uint256 indexed amount ); event UserCeloWithdrawn(address indexed _address, uint256 indexed amount); event ProtocolCeloWithdrawn( uint256 indexed totalAmount, uint256 indexed timestamp ); function initialize(address _validatorGroup) public virtual { validatorGroup = _validatorGroup; require(getAccounts().createAccount(), "CREATE_ACCOUNT_FAILED"); } receive() external payable { emit Deposit(msg.sender, msg.value, false); } /// @notice Allows deposits into the protocol's buffer pool to facilitate /// unstaking. function depositBP() public payable { bufferPool += msg.value; emit Deposit(msg.sender, msg.value, true); } /* * STAKING */ /// @notice Main function for staking with Spirals protocol. /// @dev Since contract call is atomic, staked Celo should never /// end up in this contract (goes straight to LockedGold). function stake() external payable virtual { require(msg.value > 0, "STAKING_ZERO"); lock(msg.value); vote(msg.value); stakers[msg.sender].stakedValue += msg.value; emit UserCeloStaked(msg.sender, validatorGroup, msg.value); } /// @dev Helper function for locking CELO function lock(uint256 _value) internal { require(_value > 0, "LOCKING_ZERO"); getLockedGold().lock{value: _value}(); } /// @dev Helper function for casting votes with a given validator group function vote(uint256 _value) internal { (address lesser, address greater) = getLesserGreater(); require( !(lesser == address(0) && greater == address(0)), "INVALID_LESSER_GREATER" ); // Can't both be null address require( getElection().vote(validatorGroup, _value, lesser, greater), "VOTE_FAILED" ); } /// @dev Helper function for getting the 2 validator groups that /// our target validator group is sandwiched between. function getLesserGreater() internal view returns (address, address) { (address[] memory validatorGroups, ) = getElection() .getTotalVotesForEligibleValidatorGroups(); // sorted by votes desc address lesser = address(0); address greater = address(0); for (uint256 i = 0; i < validatorGroups.length; i++) { if (validatorGroup == validatorGroups[i]) { if (i > 0) { greater = validatorGroups[i - 1]; } if (i < validatorGroups.length - 1) { lesser = validatorGroups[i + 1]; } break; } } return (lesser, greater); } /// @dev Helper function for finding the index of the validator /// group in a list of validator groups voted on by an account. function getGroupIndex(address _group) internal view returns (uint256, bool) { address[] memory votedValidatorGroups = getElection() .getGroupsVotedForByAccount(address(this)); for (uint256 i = 0; i < votedValidatorGroups.length; i++) { if (votedValidatorGroups[i] == _group) { return (i, true); } } return (0, false); } /* * UNSTAKING */ /// @notice Main function for unstaking from Spirals protocol /// @dev A particular user calling "unstake" adds a pending withdrawal /// for Spirals in the Celo smart contracts. After calling this function, /// a user officially unstakes but still needs to "withdraw" after /// the unlocking period is over. function unstake(uint256 _value) public virtual { require( stakers[msg.sender].stakedValue >= _value, "EXCEEDS_USER_STAKE" ); uint256 activeVotes = getActiveVotes(); (uint256 pendingVotes, ) = getPendingVotes(); require(activeVotes + pendingVotes >= _value, "EXCEEDS_PROTOCOL_STAKE"); // Can only support 1 outstanding unstake request at a time (without // rebuilding all of how Celo unstaking works) require( stakers[msg.sender].withdrawalValue == 0, "OUTSTANDING_PENDING_WITHDRAWAL" ); if (activeVotes >= _value) { revokeActive(_value); } else { revokePending(_value); } unlock(_value); StakerInfo memory newStaker = stakers[msg.sender]; newStaker.stakedValue -= _value; newStaker.withdrawalValue = _value; newStaker.withdrawalTimestamp = block.timestamp + getLockedGold().unlockingPeriod(); totalPendingWithdrawal += _value; stakers[msg.sender] = newStaker; emit UserCeloUnstaked(msg.sender, validatorGroup, _value); } /// @notice Helper function for revoking active votes CELO function revokeActive(uint256 _value) internal { (address lesser, address greater) = getLesserGreater(); (uint256 index, bool found) = getGroupIndex(validatorGroup); require(found, "UNSUPPORTED_VALIDATOR_GROUP"); require( getElection().revokeActive( validatorGroup, _value, lesser, greater, index ) ); } /// @notice Helper function for revoking pending votes CELO function revokePending(uint256 _value) internal { (address lesser, address greater) = getLesserGreater(); (uint256 index, bool found) = getGroupIndex(validatorGroup); require(found, "UNSUPPORTED_VALIDATOR_GROUP"); require( getElection().revokePending( validatorGroup, _value, lesser, greater, index ) ); } /// @notice Helper function for unlocking CELO function unlock(uint256 _value) internal { getLockedGold().unlock(_value); } /// @notice Allow user to withdraw their stake back to wallet. /// @dev Withdraws from this contracts balance directly. function withdraw() public virtual { StakerInfo memory s = stakers[msg.sender]; require(s.withdrawalValue > 0, "NO_PENDING_WITHDRAWALS"); require(userCanWithdraw(msg.sender), "WITHDRAWAL_NOT_READY"); payable(msg.sender).transfer(s.withdrawalValue); // should fail if protocol doesn't have enough emit UserCeloWithdrawn(msg.sender, s.withdrawalValue); totalPendingWithdrawal -= s.withdrawalValue; s.withdrawalValue = 0; s.withdrawalTimestamp = 0; stakers[msg.sender] = s; } /// @notice Helper function for checking whether protocol can support /// a user who wants to withdraw. function userCanWithdraw(address _address) public view returns (bool) { StakerInfo memory s = stakers[_address]; return address(this).balance >= s.withdrawalValue && s.withdrawalTimestamp <= block.timestamp; } /* * ADMIN */ /// @notice Activates pending votes (if ready) with a given validator group. /// @dev Onus is on the protocol owners to activate to make sure CELO /// staked with protocol is staked on CELO L1. function activateForProtocol() external { IElection c_election = getElection(); require( c_election.hasActivatablePendingVotes( address(this), validatorGroup ), "NOT_READY_TO_ACTIVATE" ); uint256 pendingVotes = getElection().getPendingVotesForGroupByAccount( validatorGroup, address(this) ); require(c_election.activate(validatorGroup), "ACTIVATE_FAILED"); // all pending -> active for this group emit ProtocolCeloActivated(validatorGroup, pendingVotes); } /// @notice Attemps to withdraw CELO for all pending withdrawals that are /// available for the proxy contract address. /// @dev Onus is on the protocol owners to withdraw CELO from LockedGold function withdrawForProtocol() external { (uint256[] memory values, uint256[] memory timestamps) = getLockedGold() .getPendingWithdrawals(address(this)); require(values.length > 0, "NO_PENDING_WITHDRAWALS"); // loop backwards so withdrawing at a single index doesn't shift indices uint256 withdrawnTotal; for (uint256 i = timestamps.length; i > 0; i--) { if (block.timestamp >= timestamps[i - 1]) { getLockedGold().withdraw(i - 1); withdrawnTotal += values[i - 1]; } } require(withdrawnTotal > 0, "NO_WITHDRAWALS_READY"); if (withdrawnTotal > 0) { emit ProtocolCeloWithdrawn(withdrawnTotal, block.timestamp); } } /* * OTHER */ /// @notice Returns all details relevant to an account staking with Spirals. function getAccount(address _address) public view returns ( uint256, uint256, uint256 ) { StakerInfo memory s = stakers[_address]; return (s.stakedValue, s.withdrawalValue, s.withdrawalTimestamp); } /// @notice For updating with validator group we stake with. Performs /// some simple checks to make sure address given is an eligible /// validator group (limited to 1 for now). function setValidatorGroup(address _newValidatorGroup) external virtual { require( getValidators().isValidatorGroup(_newValidatorGroup), "NOT_VALIDATOR_GROUP" ); require( getElection().getGroupEligibility(_newValidatorGroup), "NOT_ELIGIBLE_VALIDATOR_GROUP" ); validatorGroup = _newValidatorGroup; } /// @notice Get pending votes for this smart contract. function getPendingVotes() public view returns (uint256, bool) { return ( getElection().getPendingVotesForGroupByAccount( validatorGroup, address(this) ), getElection().hasActivatablePendingVotes( address(this), validatorGroup ) ); } /// @notice Get active votes (staked + rewards) for this smart contract. function getActiveVotes() public view returns (uint256) { return getElection().getActiveVotesForGroupByAccount( validatorGroup, address(this) ); } /* * CELO SMART CONTRACT HELPERS */ /// @dev Returns a Accounts.sol interface for interacting with the smart contract. function getAccounts() internal view returns (IAccounts) { address accountsAddr = c_celoRegistry.getAddressForStringOrDie( "Accounts" ); return IAccounts(accountsAddr); } /// @dev Returns an Election.sol interface for interacting with the smart contract. function getElection() internal view returns (IElection) { address electionAddr = c_celoRegistry.getAddressForStringOrDie( "Election" ); return IElection(electionAddr); } /// @dev Returns a LockedGold.sol interface for interacting with the smart contract. function getLockedGold() internal view returns (ILockedGold) { address lockedGoldAddr = c_celoRegistry.getAddressForStringOrDie( "LockedGold" ); return ILockedGold(lockedGoldAddr); } /// @dev Returns a Validators.sol interface for interacting with the smart contract. function getValidators() internal view returns (IValidators) { address validatorsAddr = c_celoRegistry.getAddressForStringOrDie( "Validators" ); return IValidators(validatorsAddr); } }