ETH Price: $2,517.16 (-1.41%)

Transaction Decoder

Block:
21313949 at Dec-02-2024 09:31:59 AM +UTC
Transaction Fee:
0.000875925564308676 ETH $2.20
Gas Used:
66,402 Gas / 13.191252738 Gwei

Account State Difference:

  Address   Before After State Difference Code
(beaverbuild)
6.284920217028842371 Eth6.284982299189825857 Eth0.000062082160983486
0x98cDdeFA...11174368a
0.10205172307227899 Eth
Nonce: 26
0.101175797507970314 Eth
Nonce: 27
0.000875925564308676

Execution Trace

GnosisSafeProxy.6a761202( )
  • GnosisSafe.execTransaction( to=0x0e29BA1155cE103A07118c8912dA44B0507A982D, value=0, data=0x3E200D4B00000000000000000000000000000000000000000000001C4C348A5351E80000, operation=0, safeTxGas=0, baseGas=0, gasPrice=0, gasToken=0x0000000000000000000000000000000000000000, refundReceiver=0x0000000000000000000000000000000000000000, signatures=0xA51AC1875035F91CE550FE2800568B453C093978D43D29D63A78E7CD018BC89F5807D2ED8B6FDE45D1446C6C53878BE0A6B1BC03B5F92B2440D66E94815F51EA1C29C73E81BDBDA785532C4344F3E7407D85DA95E0CE36B769F6486CD34B9C14326DA871C80D56B03452767532433A2F61B0A985C73D8AB62962C672C75E60AAAC1B )
    • Null: 0x000...001.6c58ad19( )
    • Null: 0x000...001.6c58ad19( )
    • RocketNodeStaking.stakeRPL( _amount=522000000000000000000 )
      • RocketStorage.getAddress( _key=8FC06385DE84508EAF7EB3D75B93167987C9629589FE0A868A2B4E0E90862DD8 ) => ( r=0xF18Dc176C10Ff6D8b5A17974126D43301F8EEB95 )
        File 1 of 4: GnosisSafeProxy
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        
        /// @title IProxy - Helper interface to access masterCopy of the Proxy on-chain
        /// @author Richard Meissner - <[email protected]>
        interface IProxy {
            function masterCopy() external view returns (address);
        }
        
        /// @title GnosisSafeProxy - Generic proxy contract allows to execute all transactions applying the code of a master contract.
        /// @author Stefan George - <[email protected]>
        /// @author Richard Meissner - <[email protected]>
        contract GnosisSafeProxy {
            // singleton always needs to be first declared variable, to ensure that it is at the same location in the contracts to which calls are delegated.
            // To reduce deployment costs this variable is internal and needs to be retrieved via `getStorageAt`
            address internal singleton;
        
            /// @dev Constructor function sets address of singleton contract.
            /// @param _singleton Singleton address.
            constructor(address _singleton) {
                require(_singleton != address(0), "Invalid singleton address provided");
                singleton = _singleton;
            }
        
            /// @dev Fallback function forwards all transactions and returns all received return data.
            fallback() external payable {
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    let _singleton := and(sload(0), 0xffffffffffffffffffffffffffffffffffffffff)
                    // 0xa619486e == keccak("masterCopy()"). The value is right padded to 32-bytes with 0s
                    if eq(calldataload(0), 0xa619486e00000000000000000000000000000000000000000000000000000000) {
                        mstore(0, _singleton)
                        return(0, 0x20)
                    }
                    calldatacopy(0, 0, calldatasize())
                    let success := delegatecall(gas(), _singleton, 0, calldatasize(), 0, 0)
                    returndatacopy(0, 0, returndatasize())
                    if eq(success, 0) {
                        revert(0, returndatasize())
                    }
                    return(0, returndatasize())
                }
            }
        }
        
        /// @title Proxy Factory - Allows to create new proxy contact and execute a message call to the new proxy within one transaction.
        /// @author Stefan George - <[email protected]>
        contract GnosisSafeProxyFactory {
            event ProxyCreation(GnosisSafeProxy proxy, address singleton);
        
            /// @dev Allows to create new proxy contact and execute a message call to the new proxy within one transaction.
            /// @param singleton Address of singleton contract.
            /// @param data Payload for message call sent to new proxy contract.
            function createProxy(address singleton, bytes memory data) public returns (GnosisSafeProxy proxy) {
                proxy = new GnosisSafeProxy(singleton);
                if (data.length > 0)
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        if eq(call(gas(), proxy, 0, add(data, 0x20), mload(data), 0, 0), 0) {
                            revert(0, 0)
                        }
                    }
                emit ProxyCreation(proxy, singleton);
            }
        
            /// @dev Allows to retrieve the runtime code of a deployed Proxy. This can be used to check that the expected Proxy was deployed.
            function proxyRuntimeCode() public pure returns (bytes memory) {
                return type(GnosisSafeProxy).runtimeCode;
            }
        
            /// @dev Allows to retrieve the creation code used for the Proxy deployment. With this it is easily possible to calculate predicted address.
            function proxyCreationCode() public pure returns (bytes memory) {
                return type(GnosisSafeProxy).creationCode;
            }
        
            /// @dev Allows to create new proxy contact using CREATE2 but it doesn't run the initializer.
            ///      This method is only meant as an utility to be called from other methods
            /// @param _singleton Address of singleton contract.
            /// @param initializer Payload for message call sent to new proxy contract.
            /// @param saltNonce Nonce that will be used to generate the salt to calculate the address of the new proxy contract.
            function deployProxyWithNonce(
                address _singleton,
                bytes memory initializer,
                uint256 saltNonce
            ) internal returns (GnosisSafeProxy proxy) {
                // If the initializer changes the proxy address should change too. Hashing the initializer data is cheaper than just concatinating it
                bytes32 salt = keccak256(abi.encodePacked(keccak256(initializer), saltNonce));
                bytes memory deploymentData = abi.encodePacked(type(GnosisSafeProxy).creationCode, uint256(uint160(_singleton)));
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    proxy := create2(0x0, add(0x20, deploymentData), mload(deploymentData), salt)
                }
                require(address(proxy) != address(0), "Create2 call failed");
            }
        
            /// @dev Allows to create new proxy contact and execute a message call to the new proxy within one transaction.
            /// @param _singleton Address of singleton contract.
            /// @param initializer Payload for message call sent to new proxy contract.
            /// @param saltNonce Nonce that will be used to generate the salt to calculate the address of the new proxy contract.
            function createProxyWithNonce(
                address _singleton,
                bytes memory initializer,
                uint256 saltNonce
            ) public returns (GnosisSafeProxy proxy) {
                proxy = deployProxyWithNonce(_singleton, initializer, saltNonce);
                if (initializer.length > 0)
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        if eq(call(gas(), proxy, 0, add(initializer, 0x20), mload(initializer), 0, 0), 0) {
                            revert(0, 0)
                        }
                    }
                emit ProxyCreation(proxy, _singleton);
            }
        
            /// @dev Allows to create new proxy contact, execute a message call to the new proxy and call a specified callback within one transaction
            /// @param _singleton Address of singleton contract.
            /// @param initializer Payload for message call sent to new proxy contract.
            /// @param saltNonce Nonce that will be used to generate the salt to calculate the address of the new proxy contract.
            /// @param callback Callback that will be invoced after the new proxy contract has been successfully deployed and initialized.
            function createProxyWithCallback(
                address _singleton,
                bytes memory initializer,
                uint256 saltNonce,
                IProxyCreationCallback callback
            ) public returns (GnosisSafeProxy proxy) {
                uint256 saltNonceWithCallback = uint256(keccak256(abi.encodePacked(saltNonce, callback)));
                proxy = createProxyWithNonce(_singleton, initializer, saltNonceWithCallback);
                if (address(callback) != address(0)) callback.proxyCreated(proxy, _singleton, initializer, saltNonce);
            }
        
            /// @dev Allows to get the address for a new proxy contact created via `createProxyWithNonce`
            ///      This method is only meant for address calculation purpose when you use an initializer that would revert,
            ///      therefore the response is returned with a revert. When calling this method set `from` to the address of the proxy factory.
            /// @param _singleton Address of singleton contract.
            /// @param initializer Payload for message call sent to new proxy contract.
            /// @param saltNonce Nonce that will be used to generate the salt to calculate the address of the new proxy contract.
            function calculateCreateProxyWithNonceAddress(
                address _singleton,
                bytes calldata initializer,
                uint256 saltNonce
            ) external returns (GnosisSafeProxy proxy) {
                proxy = deployProxyWithNonce(_singleton, initializer, saltNonce);
                revert(string(abi.encodePacked(proxy)));
            }
        }
        
        interface IProxyCreationCallback {
            function proxyCreated(
                GnosisSafeProxy proxy,
                address _singleton,
                bytes calldata initializer,
                uint256 saltNonce
            ) external;
        }

        File 2 of 4: GnosisSafe
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        import "./base/ModuleManager.sol";
        import "./base/OwnerManager.sol";
        import "./base/FallbackManager.sol";
        import "./base/GuardManager.sol";
        import "./common/EtherPaymentFallback.sol";
        import "./common/Singleton.sol";
        import "./common/SignatureDecoder.sol";
        import "./common/SecuredTokenTransfer.sol";
        import "./common/StorageAccessible.sol";
        import "./interfaces/ISignatureValidator.sol";
        import "./external/GnosisSafeMath.sol";
        /// @title Gnosis Safe - A multisignature wallet with support for confirmations using signed messages based on ERC191.
        /// @author Stefan George - <[email protected]>
        /// @author Richard Meissner - <[email protected]>
        contract GnosisSafe is
            EtherPaymentFallback,
            Singleton,
            ModuleManager,
            OwnerManager,
            SignatureDecoder,
            SecuredTokenTransfer,
            ISignatureValidatorConstants,
            FallbackManager,
            StorageAccessible,
            GuardManager
        {
            using GnosisSafeMath for uint256;
            string public constant VERSION = "1.3.0";
            // keccak256(
            //     "EIP712Domain(uint256 chainId,address verifyingContract)"
            // );
            bytes32 private constant DOMAIN_SEPARATOR_TYPEHASH = 0x47e79534a245952e8b16893a336b85a3d9ea9fa8c573f3d803afb92a79469218;
            // keccak256(
            //     "SafeTx(address to,uint256 value,bytes data,uint8 operation,uint256 safeTxGas,uint256 baseGas,uint256 gasPrice,address gasToken,address refundReceiver,uint256 nonce)"
            // );
            bytes32 private constant SAFE_TX_TYPEHASH = 0xbb8310d486368db6bd6f849402fdd73ad53d316b5a4b2644ad6efe0f941286d8;
            event SafeSetup(address indexed initiator, address[] owners, uint256 threshold, address initializer, address fallbackHandler);
            event ApproveHash(bytes32 indexed approvedHash, address indexed owner);
            event SignMsg(bytes32 indexed msgHash);
            event ExecutionFailure(bytes32 txHash, uint256 payment);
            event ExecutionSuccess(bytes32 txHash, uint256 payment);
            uint256 public nonce;
            bytes32 private _deprecatedDomainSeparator;
            // Mapping to keep track of all message hashes that have been approve by ALL REQUIRED owners
            mapping(bytes32 => uint256) public signedMessages;
            // Mapping to keep track of all hashes (message or transaction) that have been approve by ANY owners
            mapping(address => mapping(bytes32 => uint256)) public approvedHashes;
            // This constructor ensures that this contract can only be used as a master copy for Proxy contracts
            constructor() {
                // By setting the threshold it is not possible to call setup anymore,
                // so we create a Safe with 0 owners and threshold 1.
                // This is an unusable Safe, perfect for the singleton
                threshold = 1;
            }
            /// @dev Setup function sets initial storage of contract.
            /// @param _owners List of Safe owners.
            /// @param _threshold Number of required confirmations for a Safe transaction.
            /// @param to Contract address for optional delegate call.
            /// @param data Data payload for optional delegate call.
            /// @param fallbackHandler Handler for fallback calls to this contract
            /// @param paymentToken Token that should be used for the payment (0 is ETH)
            /// @param payment Value that should be paid
            /// @param paymentReceiver Adddress that should receive the payment (or 0 if tx.origin)
            function setup(
                address[] calldata _owners,
                uint256 _threshold,
                address to,
                bytes calldata data,
                address fallbackHandler,
                address paymentToken,
                uint256 payment,
                address payable paymentReceiver
            ) external {
                // setupOwners checks if the Threshold is already set, therefore preventing that this method is called twice
                setupOwners(_owners, _threshold);
                if (fallbackHandler != address(0)) internalSetFallbackHandler(fallbackHandler);
                // As setupOwners can only be called if the contract has not been initialized we don't need a check for setupModules
                setupModules(to, data);
                if (payment > 0) {
                    // To avoid running into issues with EIP-170 we reuse the handlePayment function (to avoid adjusting code of that has been verified we do not adjust the method itself)
                    // baseGas = 0, gasPrice = 1 and gas = payment => amount = (payment + 0) * 1 = payment
                    handlePayment(payment, 0, 1, paymentToken, paymentReceiver);
                }
                emit SafeSetup(msg.sender, _owners, _threshold, to, fallbackHandler);
            }
            /// @dev Allows to execute a Safe transaction confirmed by required number of owners and then pays the account that submitted the transaction.
            ///      Note: The fees are always transferred, even if the user transaction fails.
            /// @param to Destination address of Safe transaction.
            /// @param value Ether value of Safe transaction.
            /// @param data Data payload of Safe transaction.
            /// @param operation Operation type of Safe transaction.
            /// @param safeTxGas Gas that should be used for the Safe transaction.
            /// @param baseGas Gas costs that are independent of the transaction execution(e.g. base transaction fee, signature check, payment of the refund)
            /// @param gasPrice Gas price that should be used for the payment calculation.
            /// @param gasToken Token address (or 0 if ETH) that is used for the payment.
            /// @param refundReceiver Address of receiver of gas payment (or 0 if tx.origin).
            /// @param signatures Packed signature data ({bytes32 r}{bytes32 s}{uint8 v})
            function execTransaction(
                address to,
                uint256 value,
                bytes calldata data,
                Enum.Operation operation,
                uint256 safeTxGas,
                uint256 baseGas,
                uint256 gasPrice,
                address gasToken,
                address payable refundReceiver,
                bytes memory signatures
            ) public payable virtual returns (bool success) {
                bytes32 txHash;
                // Use scope here to limit variable lifetime and prevent `stack too deep` errors
                {
                    bytes memory txHashData =
                        encodeTransactionData(
                            // Transaction info
                            to,
                            value,
                            data,
                            operation,
                            safeTxGas,
                            // Payment info
                            baseGas,
                            gasPrice,
                            gasToken,
                            refundReceiver,
                            // Signature info
                            nonce
                        );
                    // Increase nonce and execute transaction.
                    nonce++;
                    txHash = keccak256(txHashData);
                    checkSignatures(txHash, txHashData, signatures);
                }
                address guard = getGuard();
                {
                    if (guard != address(0)) {
                        Guard(guard).checkTransaction(
                            // Transaction info
                            to,
                            value,
                            data,
                            operation,
                            safeTxGas,
                            // Payment info
                            baseGas,
                            gasPrice,
                            gasToken,
                            refundReceiver,
                            // Signature info
                            signatures,
                            msg.sender
                        );
                    }
                }
                // We require some gas to emit the events (at least 2500) after the execution and some to perform code until the execution (500)
                // We also include the 1/64 in the check that is not send along with a call to counteract potential shortings because of EIP-150
                require(gasleft() >= ((safeTxGas * 64) / 63).max(safeTxGas + 2500) + 500, "GS010");
                // Use scope here to limit variable lifetime and prevent `stack too deep` errors
                {
                    uint256 gasUsed = gasleft();
                    // If the gasPrice is 0 we assume that nearly all available gas can be used (it is always more than safeTxGas)
                    // We only substract 2500 (compared to the 3000 before) to ensure that the amount passed is still higher than safeTxGas
                    success = execute(to, value, data, operation, gasPrice == 0 ? (gasleft() - 2500) : safeTxGas);
                    gasUsed = gasUsed.sub(gasleft());
                    // If no safeTxGas and no gasPrice was set (e.g. both are 0), then the internal tx is required to be successful
                    // This makes it possible to use `estimateGas` without issues, as it searches for the minimum gas where the tx doesn't revert
                    require(success || safeTxGas != 0 || gasPrice != 0, "GS013");
                    // We transfer the calculated tx costs to the tx.origin to avoid sending it to intermediate contracts that have made calls
                    uint256 payment = 0;
                    if (gasPrice > 0) {
                        payment = handlePayment(gasUsed, baseGas, gasPrice, gasToken, refundReceiver);
                    }
                    if (success) emit ExecutionSuccess(txHash, payment);
                    else emit ExecutionFailure(txHash, payment);
                }
                {
                    if (guard != address(0)) {
                        Guard(guard).checkAfterExecution(txHash, success);
                    }
                }
            }
            function handlePayment(
                uint256 gasUsed,
                uint256 baseGas,
                uint256 gasPrice,
                address gasToken,
                address payable refundReceiver
            ) private returns (uint256 payment) {
                // solhint-disable-next-line avoid-tx-origin
                address payable receiver = refundReceiver == address(0) ? payable(tx.origin) : refundReceiver;
                if (gasToken == address(0)) {
                    // For ETH we will only adjust the gas price to not be higher than the actual used gas price
                    payment = gasUsed.add(baseGas).mul(gasPrice < tx.gasprice ? gasPrice : tx.gasprice);
                    require(receiver.send(payment), "GS011");
                } else {
                    payment = gasUsed.add(baseGas).mul(gasPrice);
                    require(transferToken(gasToken, receiver, payment), "GS012");
                }
            }
            /**
             * @dev Checks whether the signature provided is valid for the provided data, hash. Will revert otherwise.
             * @param dataHash Hash of the data (could be either a message hash or transaction hash)
             * @param data That should be signed (this is passed to an external validator contract)
             * @param signatures Signature data that should be verified. Can be ECDSA signature, contract signature (EIP-1271) or approved hash.
             */
            function checkSignatures(
                bytes32 dataHash,
                bytes memory data,
                bytes memory signatures
            ) public view {
                // Load threshold to avoid multiple storage loads
                uint256 _threshold = threshold;
                // Check that a threshold is set
                require(_threshold > 0, "GS001");
                checkNSignatures(dataHash, data, signatures, _threshold);
            }
            /**
             * @dev Checks whether the signature provided is valid for the provided data, hash. Will revert otherwise.
             * @param dataHash Hash of the data (could be either a message hash or transaction hash)
             * @param data That should be signed (this is passed to an external validator contract)
             * @param signatures Signature data that should be verified. Can be ECDSA signature, contract signature (EIP-1271) or approved hash.
             * @param requiredSignatures Amount of required valid signatures.
             */
            function checkNSignatures(
                bytes32 dataHash,
                bytes memory data,
                bytes memory signatures,
                uint256 requiredSignatures
            ) public view {
                // Check that the provided signature data is not too short
                require(signatures.length >= requiredSignatures.mul(65), "GS020");
                // There cannot be an owner with address 0.
                address lastOwner = address(0);
                address currentOwner;
                uint8 v;
                bytes32 r;
                bytes32 s;
                uint256 i;
                for (i = 0; i < requiredSignatures; i++) {
                    (v, r, s) = signatureSplit(signatures, i);
                    if (v == 0) {
                        // If v is 0 then it is a contract signature
                        // When handling contract signatures the address of the contract is encoded into r
                        currentOwner = address(uint160(uint256(r)));
                        // Check that signature data pointer (s) is not pointing inside the static part of the signatures bytes
                        // This check is not completely accurate, since it is possible that more signatures than the threshold are send.
                        // Here we only check that the pointer is not pointing inside the part that is being processed
                        require(uint256(s) >= requiredSignatures.mul(65), "GS021");
                        // Check that signature data pointer (s) is in bounds (points to the length of data -> 32 bytes)
                        require(uint256(s).add(32) <= signatures.length, "GS022");
                        // Check if the contract signature is in bounds: start of data is s + 32 and end is start + signature length
                        uint256 contractSignatureLen;
                        // solhint-disable-next-line no-inline-assembly
                        assembly {
                            contractSignatureLen := mload(add(add(signatures, s), 0x20))
                        }
                        require(uint256(s).add(32).add(contractSignatureLen) <= signatures.length, "GS023");
                        // Check signature
                        bytes memory contractSignature;
                        // solhint-disable-next-line no-inline-assembly
                        assembly {
                            // The signature data for contract signatures is appended to the concatenated signatures and the offset is stored in s
                            contractSignature := add(add(signatures, s), 0x20)
                        }
                        require(ISignatureValidator(currentOwner).isValidSignature(data, contractSignature) == EIP1271_MAGIC_VALUE, "GS024");
                    } else if (v == 1) {
                        // If v is 1 then it is an approved hash
                        // When handling approved hashes the address of the approver is encoded into r
                        currentOwner = address(uint160(uint256(r)));
                        // Hashes are automatically approved by the sender of the message or when they have been pre-approved via a separate transaction
                        require(msg.sender == currentOwner || approvedHashes[currentOwner][dataHash] != 0, "GS025");
                    } else if (v > 30) {
                        // If v > 30 then default va (27,28) has been adjusted for eth_sign flow
                        // To support eth_sign and similar we adjust v and hash the messageHash with the Ethereum message prefix before applying ecrecover
                        currentOwner = ecrecover(keccak256(abi.encodePacked("\\x19Ethereum Signed Message:\
        32", dataHash)), v - 4, r, s);
                    } else {
                        // Default is the ecrecover flow with the provided data hash
                        // Use ecrecover with the messageHash for EOA signatures
                        currentOwner = ecrecover(dataHash, v, r, s);
                    }
                    require(currentOwner > lastOwner && owners[currentOwner] != address(0) && currentOwner != SENTINEL_OWNERS, "GS026");
                    lastOwner = currentOwner;
                }
            }
            /// @dev Allows to estimate a Safe transaction.
            ///      This method is only meant for estimation purpose, therefore the call will always revert and encode the result in the revert data.
            ///      Since the `estimateGas` function includes refunds, call this method to get an estimated of the costs that are deducted from the safe with `execTransaction`
            /// @param to Destination address of Safe transaction.
            /// @param value Ether value of Safe transaction.
            /// @param data Data payload of Safe transaction.
            /// @param operation Operation type of Safe transaction.
            /// @return Estimate without refunds and overhead fees (base transaction and payload data gas costs).
            /// @notice Deprecated in favor of common/StorageAccessible.sol and will be removed in next version.
            function requiredTxGas(
                address to,
                uint256 value,
                bytes calldata data,
                Enum.Operation operation
            ) external returns (uint256) {
                uint256 startGas = gasleft();
                // We don't provide an error message here, as we use it to return the estimate
                require(execute(to, value, data, operation, gasleft()));
                uint256 requiredGas = startGas - gasleft();
                // Convert response to string and return via error message
                revert(string(abi.encodePacked(requiredGas)));
            }
            /**
             * @dev Marks a hash as approved. This can be used to validate a hash that is used by a signature.
             * @param hashToApprove The hash that should be marked as approved for signatures that are verified by this contract.
             */
            function approveHash(bytes32 hashToApprove) external {
                require(owners[msg.sender] != address(0), "GS030");
                approvedHashes[msg.sender][hashToApprove] = 1;
                emit ApproveHash(hashToApprove, msg.sender);
            }
            /// @dev Returns the chain id used by this contract.
            function getChainId() public view returns (uint256) {
                uint256 id;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    id := chainid()
                }
                return id;
            }
            function domainSeparator() public view returns (bytes32) {
                return keccak256(abi.encode(DOMAIN_SEPARATOR_TYPEHASH, getChainId(), this));
            }
            /// @dev Returns the bytes that are hashed to be signed by owners.
            /// @param to Destination address.
            /// @param value Ether value.
            /// @param data Data payload.
            /// @param operation Operation type.
            /// @param safeTxGas Gas that should be used for the safe transaction.
            /// @param baseGas Gas costs for that are independent of the transaction execution(e.g. base transaction fee, signature check, payment of the refund)
            /// @param gasPrice Maximum gas price that should be used for this transaction.
            /// @param gasToken Token address (or 0 if ETH) that is used for the payment.
            /// @param refundReceiver Address of receiver of gas payment (or 0 if tx.origin).
            /// @param _nonce Transaction nonce.
            /// @return Transaction hash bytes.
            function encodeTransactionData(
                address to,
                uint256 value,
                bytes calldata data,
                Enum.Operation operation,
                uint256 safeTxGas,
                uint256 baseGas,
                uint256 gasPrice,
                address gasToken,
                address refundReceiver,
                uint256 _nonce
            ) public view returns (bytes memory) {
                bytes32 safeTxHash =
                    keccak256(
                        abi.encode(
                            SAFE_TX_TYPEHASH,
                            to,
                            value,
                            keccak256(data),
                            operation,
                            safeTxGas,
                            baseGas,
                            gasPrice,
                            gasToken,
                            refundReceiver,
                            _nonce
                        )
                    );
                return abi.encodePacked(bytes1(0x19), bytes1(0x01), domainSeparator(), safeTxHash);
            }
            /// @dev Returns hash to be signed by owners.
            /// @param to Destination address.
            /// @param value Ether value.
            /// @param data Data payload.
            /// @param operation Operation type.
            /// @param safeTxGas Fas that should be used for the safe transaction.
            /// @param baseGas Gas costs for data used to trigger the safe transaction.
            /// @param gasPrice Maximum gas price that should be used for this transaction.
            /// @param gasToken Token address (or 0 if ETH) that is used for the payment.
            /// @param refundReceiver Address of receiver of gas payment (or 0 if tx.origin).
            /// @param _nonce Transaction nonce.
            /// @return Transaction hash.
            function getTransactionHash(
                address to,
                uint256 value,
                bytes calldata data,
                Enum.Operation operation,
                uint256 safeTxGas,
                uint256 baseGas,
                uint256 gasPrice,
                address gasToken,
                address refundReceiver,
                uint256 _nonce
            ) public view returns (bytes32) {
                return keccak256(encodeTransactionData(to, value, data, operation, safeTxGas, baseGas, gasPrice, gasToken, refundReceiver, _nonce));
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        import "../common/Enum.sol";
        /// @title Executor - A contract that can execute transactions
        /// @author Richard Meissner - <[email protected]>
        contract Executor {
            function execute(
                address to,
                uint256 value,
                bytes memory data,
                Enum.Operation operation,
                uint256 txGas
            ) internal returns (bool success) {
                if (operation == Enum.Operation.DelegateCall) {
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        success := delegatecall(txGas, to, add(data, 0x20), mload(data), 0, 0)
                    }
                } else {
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        success := call(txGas, to, value, add(data, 0x20), mload(data), 0, 0)
                    }
                }
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        import "../common/SelfAuthorized.sol";
        /// @title Fallback Manager - A contract that manages fallback calls made to this contract
        /// @author Richard Meissner - <[email protected]>
        contract FallbackManager is SelfAuthorized {
            event ChangedFallbackHandler(address handler);
            // keccak256("fallback_manager.handler.address")
            bytes32 internal constant FALLBACK_HANDLER_STORAGE_SLOT = 0x6c9a6c4a39284e37ed1cf53d337577d14212a4870fb976a4366c693b939918d5;
            function internalSetFallbackHandler(address handler) internal {
                bytes32 slot = FALLBACK_HANDLER_STORAGE_SLOT;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    sstore(slot, handler)
                }
            }
            /// @dev Allows to add a contract to handle fallback calls.
            ///      Only fallback calls without value and with data will be forwarded.
            ///      This can only be done via a Safe transaction.
            /// @param handler contract to handle fallbacks calls.
            function setFallbackHandler(address handler) public authorized {
                internalSetFallbackHandler(handler);
                emit ChangedFallbackHandler(handler);
            }
            // solhint-disable-next-line payable-fallback,no-complex-fallback
            fallback() external {
                bytes32 slot = FALLBACK_HANDLER_STORAGE_SLOT;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    let handler := sload(slot)
                    if iszero(handler) {
                        return(0, 0)
                    }
                    calldatacopy(0, 0, calldatasize())
                    // The msg.sender address is shifted to the left by 12 bytes to remove the padding
                    // Then the address without padding is stored right after the calldata
                    mstore(calldatasize(), shl(96, caller()))
                    // Add 20 bytes for the address appended add the end
                    let success := call(gas(), handler, 0, 0, add(calldatasize(), 20), 0, 0)
                    returndatacopy(0, 0, returndatasize())
                    if iszero(success) {
                        revert(0, returndatasize())
                    }
                    return(0, returndatasize())
                }
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        import "../common/Enum.sol";
        import "../common/SelfAuthorized.sol";
        interface Guard {
            function checkTransaction(
                address to,
                uint256 value,
                bytes memory data,
                Enum.Operation operation,
                uint256 safeTxGas,
                uint256 baseGas,
                uint256 gasPrice,
                address gasToken,
                address payable refundReceiver,
                bytes memory signatures,
                address msgSender
            ) external;
            function checkAfterExecution(bytes32 txHash, bool success) external;
        }
        /// @title Fallback Manager - A contract that manages fallback calls made to this contract
        /// @author Richard Meissner - <[email protected]>
        contract GuardManager is SelfAuthorized {
            event ChangedGuard(address guard);
            // keccak256("guard_manager.guard.address")
            bytes32 internal constant GUARD_STORAGE_SLOT = 0x4a204f620c8c5ccdca3fd54d003badd85ba500436a431f0cbda4f558c93c34c8;
            /// @dev Set a guard that checks transactions before execution
            /// @param guard The address of the guard to be used or the 0 address to disable the guard
            function setGuard(address guard) external authorized {
                bytes32 slot = GUARD_STORAGE_SLOT;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    sstore(slot, guard)
                }
                emit ChangedGuard(guard);
            }
            function getGuard() internal view returns (address guard) {
                bytes32 slot = GUARD_STORAGE_SLOT;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    guard := sload(slot)
                }
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        import "../common/Enum.sol";
        import "../common/SelfAuthorized.sol";
        import "./Executor.sol";
        /// @title Module Manager - A contract that manages modules that can execute transactions via this contract
        /// @author Stefan George - <[email protected]>
        /// @author Richard Meissner - <[email protected]>
        contract ModuleManager is SelfAuthorized, Executor {
            event EnabledModule(address module);
            event DisabledModule(address module);
            event ExecutionFromModuleSuccess(address indexed module);
            event ExecutionFromModuleFailure(address indexed module);
            address internal constant SENTINEL_MODULES = address(0x1);
            mapping(address => address) internal modules;
            function setupModules(address to, bytes memory data) internal {
                require(modules[SENTINEL_MODULES] == address(0), "GS100");
                modules[SENTINEL_MODULES] = SENTINEL_MODULES;
                if (to != address(0))
                    // Setup has to complete successfully or transaction fails.
                    require(execute(to, 0, data, Enum.Operation.DelegateCall, gasleft()), "GS000");
            }
            /// @dev Allows to add a module to the whitelist.
            ///      This can only be done via a Safe transaction.
            /// @notice Enables the module `module` for the Safe.
            /// @param module Module to be whitelisted.
            function enableModule(address module) public authorized {
                // Module address cannot be null or sentinel.
                require(module != address(0) && module != SENTINEL_MODULES, "GS101");
                // Module cannot be added twice.
                require(modules[module] == address(0), "GS102");
                modules[module] = modules[SENTINEL_MODULES];
                modules[SENTINEL_MODULES] = module;
                emit EnabledModule(module);
            }
            /// @dev Allows to remove a module from the whitelist.
            ///      This can only be done via a Safe transaction.
            /// @notice Disables the module `module` for the Safe.
            /// @param prevModule Module that pointed to the module to be removed in the linked list
            /// @param module Module to be removed.
            function disableModule(address prevModule, address module) public authorized {
                // Validate module address and check that it corresponds to module index.
                require(module != address(0) && module != SENTINEL_MODULES, "GS101");
                require(modules[prevModule] == module, "GS103");
                modules[prevModule] = modules[module];
                modules[module] = address(0);
                emit DisabledModule(module);
            }
            /// @dev Allows a Module to execute a Safe transaction without any further confirmations.
            /// @param to Destination address of module transaction.
            /// @param value Ether value of module transaction.
            /// @param data Data payload of module transaction.
            /// @param operation Operation type of module transaction.
            function execTransactionFromModule(
                address to,
                uint256 value,
                bytes memory data,
                Enum.Operation operation
            ) public virtual returns (bool success) {
                // Only whitelisted modules are allowed.
                require(msg.sender != SENTINEL_MODULES && modules[msg.sender] != address(0), "GS104");
                // Execute transaction without further confirmations.
                success = execute(to, value, data, operation, gasleft());
                if (success) emit ExecutionFromModuleSuccess(msg.sender);
                else emit ExecutionFromModuleFailure(msg.sender);
            }
            /// @dev Allows a Module to execute a Safe transaction without any further confirmations and return data
            /// @param to Destination address of module transaction.
            /// @param value Ether value of module transaction.
            /// @param data Data payload of module transaction.
            /// @param operation Operation type of module transaction.
            function execTransactionFromModuleReturnData(
                address to,
                uint256 value,
                bytes memory data,
                Enum.Operation operation
            ) public returns (bool success, bytes memory returnData) {
                success = execTransactionFromModule(to, value, data, operation);
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    // Load free memory location
                    let ptr := mload(0x40)
                    // We allocate memory for the return data by setting the free memory location to
                    // current free memory location + data size + 32 bytes for data size value
                    mstore(0x40, add(ptr, add(returndatasize(), 0x20)))
                    // Store the size
                    mstore(ptr, returndatasize())
                    // Store the data
                    returndatacopy(add(ptr, 0x20), 0, returndatasize())
                    // Point the return data to the correct memory location
                    returnData := ptr
                }
            }
            /// @dev Returns if an module is enabled
            /// @return True if the module is enabled
            function isModuleEnabled(address module) public view returns (bool) {
                return SENTINEL_MODULES != module && modules[module] != address(0);
            }
            /// @dev Returns array of modules.
            /// @param start Start of the page.
            /// @param pageSize Maximum number of modules that should be returned.
            /// @return array Array of modules.
            /// @return next Start of the next page.
            function getModulesPaginated(address start, uint256 pageSize) external view returns (address[] memory array, address next) {
                // Init array with max page size
                array = new address[](pageSize);
                // Populate return array
                uint256 moduleCount = 0;
                address currentModule = modules[start];
                while (currentModule != address(0x0) && currentModule != SENTINEL_MODULES && moduleCount < pageSize) {
                    array[moduleCount] = currentModule;
                    currentModule = modules[currentModule];
                    moduleCount++;
                }
                next = currentModule;
                // Set correct size of returned array
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    mstore(array, moduleCount)
                }
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        import "../common/SelfAuthorized.sol";
        /// @title OwnerManager - Manages a set of owners and a threshold to perform actions.
        /// @author Stefan George - <[email protected]>
        /// @author Richard Meissner - <[email protected]>
        contract OwnerManager is SelfAuthorized {
            event AddedOwner(address owner);
            event RemovedOwner(address owner);
            event ChangedThreshold(uint256 threshold);
            address internal constant SENTINEL_OWNERS = address(0x1);
            mapping(address => address) internal owners;
            uint256 internal ownerCount;
            uint256 internal threshold;
            /// @dev Setup function sets initial storage of contract.
            /// @param _owners List of Safe owners.
            /// @param _threshold Number of required confirmations for a Safe transaction.
            function setupOwners(address[] memory _owners, uint256 _threshold) internal {
                // Threshold can only be 0 at initialization.
                // Check ensures that setup function can only be called once.
                require(threshold == 0, "GS200");
                // Validate that threshold is smaller than number of added owners.
                require(_threshold <= _owners.length, "GS201");
                // There has to be at least one Safe owner.
                require(_threshold >= 1, "GS202");
                // Initializing Safe owners.
                address currentOwner = SENTINEL_OWNERS;
                for (uint256 i = 0; i < _owners.length; i++) {
                    // Owner address cannot be null.
                    address owner = _owners[i];
                    require(owner != address(0) && owner != SENTINEL_OWNERS && owner != address(this) && currentOwner != owner, "GS203");
                    // No duplicate owners allowed.
                    require(owners[owner] == address(0), "GS204");
                    owners[currentOwner] = owner;
                    currentOwner = owner;
                }
                owners[currentOwner] = SENTINEL_OWNERS;
                ownerCount = _owners.length;
                threshold = _threshold;
            }
            /// @dev Allows to add a new owner to the Safe and update the threshold at the same time.
            ///      This can only be done via a Safe transaction.
            /// @notice Adds the owner `owner` to the Safe and updates the threshold to `_threshold`.
            /// @param owner New owner address.
            /// @param _threshold New threshold.
            function addOwnerWithThreshold(address owner, uint256 _threshold) public authorized {
                // Owner address cannot be null, the sentinel or the Safe itself.
                require(owner != address(0) && owner != SENTINEL_OWNERS && owner != address(this), "GS203");
                // No duplicate owners allowed.
                require(owners[owner] == address(0), "GS204");
                owners[owner] = owners[SENTINEL_OWNERS];
                owners[SENTINEL_OWNERS] = owner;
                ownerCount++;
                emit AddedOwner(owner);
                // Change threshold if threshold was changed.
                if (threshold != _threshold) changeThreshold(_threshold);
            }
            /// @dev Allows to remove an owner from the Safe and update the threshold at the same time.
            ///      This can only be done via a Safe transaction.
            /// @notice Removes the owner `owner` from the Safe and updates the threshold to `_threshold`.
            /// @param prevOwner Owner that pointed to the owner to be removed in the linked list
            /// @param owner Owner address to be removed.
            /// @param _threshold New threshold.
            function removeOwner(
                address prevOwner,
                address owner,
                uint256 _threshold
            ) public authorized {
                // Only allow to remove an owner, if threshold can still be reached.
                require(ownerCount - 1 >= _threshold, "GS201");
                // Validate owner address and check that it corresponds to owner index.
                require(owner != address(0) && owner != SENTINEL_OWNERS, "GS203");
                require(owners[prevOwner] == owner, "GS205");
                owners[prevOwner] = owners[owner];
                owners[owner] = address(0);
                ownerCount--;
                emit RemovedOwner(owner);
                // Change threshold if threshold was changed.
                if (threshold != _threshold) changeThreshold(_threshold);
            }
            /// @dev Allows to swap/replace an owner from the Safe with another address.
            ///      This can only be done via a Safe transaction.
            /// @notice Replaces the owner `oldOwner` in the Safe with `newOwner`.
            /// @param prevOwner Owner that pointed to the owner to be replaced in the linked list
            /// @param oldOwner Owner address to be replaced.
            /// @param newOwner New owner address.
            function swapOwner(
                address prevOwner,
                address oldOwner,
                address newOwner
            ) public authorized {
                // Owner address cannot be null, the sentinel or the Safe itself.
                require(newOwner != address(0) && newOwner != SENTINEL_OWNERS && newOwner != address(this), "GS203");
                // No duplicate owners allowed.
                require(owners[newOwner] == address(0), "GS204");
                // Validate oldOwner address and check that it corresponds to owner index.
                require(oldOwner != address(0) && oldOwner != SENTINEL_OWNERS, "GS203");
                require(owners[prevOwner] == oldOwner, "GS205");
                owners[newOwner] = owners[oldOwner];
                owners[prevOwner] = newOwner;
                owners[oldOwner] = address(0);
                emit RemovedOwner(oldOwner);
                emit AddedOwner(newOwner);
            }
            /// @dev Allows to update the number of required confirmations by Safe owners.
            ///      This can only be done via a Safe transaction.
            /// @notice Changes the threshold of the Safe to `_threshold`.
            /// @param _threshold New threshold.
            function changeThreshold(uint256 _threshold) public authorized {
                // Validate that threshold is smaller than number of owners.
                require(_threshold <= ownerCount, "GS201");
                // There has to be at least one Safe owner.
                require(_threshold >= 1, "GS202");
                threshold = _threshold;
                emit ChangedThreshold(threshold);
            }
            function getThreshold() public view returns (uint256) {
                return threshold;
            }
            function isOwner(address owner) public view returns (bool) {
                return owner != SENTINEL_OWNERS && owners[owner] != address(0);
            }
            /// @dev Returns array of owners.
            /// @return Array of Safe owners.
            function getOwners() public view returns (address[] memory) {
                address[] memory array = new address[](ownerCount);
                // populate return array
                uint256 index = 0;
                address currentOwner = owners[SENTINEL_OWNERS];
                while (currentOwner != SENTINEL_OWNERS) {
                    array[index] = currentOwner;
                    currentOwner = owners[currentOwner];
                    index++;
                }
                return array;
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        /// @title Enum - Collection of enums
        /// @author Richard Meissner - <[email protected]>
        contract Enum {
            enum Operation {Call, DelegateCall}
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        /// @title EtherPaymentFallback - A contract that has a fallback to accept ether payments
        /// @author Richard Meissner - <[email protected]>
        contract EtherPaymentFallback {
            event SafeReceived(address indexed sender, uint256 value);
            /// @dev Fallback function accepts Ether transactions.
            receive() external payable {
                emit SafeReceived(msg.sender, msg.value);
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        /// @title SecuredTokenTransfer - Secure token transfer
        /// @author Richard Meissner - <[email protected]>
        contract SecuredTokenTransfer {
            /// @dev Transfers a token and returns if it was a success
            /// @param token Token that should be transferred
            /// @param receiver Receiver to whom the token should be transferred
            /// @param amount The amount of tokens that should be transferred
            function transferToken(
                address token,
                address receiver,
                uint256 amount
            ) internal returns (bool transferred) {
                // 0xa9059cbb - keccack("transfer(address,uint256)")
                bytes memory data = abi.encodeWithSelector(0xa9059cbb, receiver, amount);
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    // We write the return value to scratch space.
                    // See https://docs.soliditylang.org/en/v0.7.6/internals/layout_in_memory.html#layout-in-memory
                    let success := call(sub(gas(), 10000), token, 0, add(data, 0x20), mload(data), 0, 0x20)
                    switch returndatasize()
                        case 0 {
                            transferred := success
                        }
                        case 0x20 {
                            transferred := iszero(or(iszero(success), iszero(mload(0))))
                        }
                        default {
                            transferred := 0
                        }
                }
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        /// @title SelfAuthorized - authorizes current contract to perform actions
        /// @author Richard Meissner - <[email protected]>
        contract SelfAuthorized {
            function requireSelfCall() private view {
                require(msg.sender == address(this), "GS031");
            }
            modifier authorized() {
                // This is a function call as it minimized the bytecode size
                requireSelfCall();
                _;
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        /// @title SignatureDecoder - Decodes signatures that a encoded as bytes
        /// @author Richard Meissner - <[email protected]>
        contract SignatureDecoder {
            /// @dev divides bytes signature into `uint8 v, bytes32 r, bytes32 s`.
            /// @notice Make sure to peform a bounds check for @param pos, to avoid out of bounds access on @param signatures
            /// @param pos which signature to read. A prior bounds check of this parameter should be performed, to avoid out of bounds access
            /// @param signatures concatenated rsv signatures
            function signatureSplit(bytes memory signatures, uint256 pos)
                internal
                pure
                returns (
                    uint8 v,
                    bytes32 r,
                    bytes32 s
                )
            {
                // The signature format is a compact form of:
                //   {bytes32 r}{bytes32 s}{uint8 v}
                // Compact means, uint8 is not padded to 32 bytes.
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    let signaturePos := mul(0x41, pos)
                    r := mload(add(signatures, add(signaturePos, 0x20)))
                    s := mload(add(signatures, add(signaturePos, 0x40)))
                    // Here we are loading the last 32 bytes, including 31 bytes
                    // of 's'. There is no 'mload8' to do this.
                    //
                    // 'byte' is not working due to the Solidity parser, so lets
                    // use the second best option, 'and'
                    v := and(mload(add(signatures, add(signaturePos, 0x41))), 0xff)
                }
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        /// @title Singleton - Base for singleton contracts (should always be first super contract)
        ///         This contract is tightly coupled to our proxy contract (see `proxies/GnosisSafeProxy.sol`)
        /// @author Richard Meissner - <[email protected]>
        contract Singleton {
            // singleton always needs to be first declared variable, to ensure that it is at the same location as in the Proxy contract.
            // It should also always be ensured that the address is stored alone (uses a full word)
            address private singleton;
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        /// @title StorageAccessible - generic base contract that allows callers to access all internal storage.
        /// @notice See https://github.com/gnosis/util-contracts/blob/bb5fe5fb5df6d8400998094fb1b32a178a47c3a1/contracts/StorageAccessible.sol
        contract StorageAccessible {
            /**
             * @dev Reads `length` bytes of storage in the currents contract
             * @param offset - the offset in the current contract's storage in words to start reading from
             * @param length - the number of words (32 bytes) of data to read
             * @return the bytes that were read.
             */
            function getStorageAt(uint256 offset, uint256 length) public view returns (bytes memory) {
                bytes memory result = new bytes(length * 32);
                for (uint256 index = 0; index < length; index++) {
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        let word := sload(add(offset, index))
                        mstore(add(add(result, 0x20), mul(index, 0x20)), word)
                    }
                }
                return result;
            }
            /**
             * @dev Performs a delegetecall on a targetContract in the context of self.
             * Internally reverts execution to avoid side effects (making it static).
             *
             * This method reverts with data equal to `abi.encode(bool(success), bytes(response))`.
             * Specifically, the `returndata` after a call to this method will be:
             * `success:bool || response.length:uint256 || response:bytes`.
             *
             * @param targetContract Address of the contract containing the code to execute.
             * @param calldataPayload Calldata that should be sent to the target contract (encoded method name and arguments).
             */
            function simulateAndRevert(address targetContract, bytes memory calldataPayload) external {
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    let success := delegatecall(gas(), targetContract, add(calldataPayload, 0x20), mload(calldataPayload), 0, 0)
                    mstore(0x00, success)
                    mstore(0x20, returndatasize())
                    returndatacopy(0x40, 0, returndatasize())
                    revert(0, add(returndatasize(), 0x40))
                }
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        /**
         * @title GnosisSafeMath
         * @dev Math operations with safety checks that revert on error
         * Renamed from SafeMath to GnosisSafeMath to avoid conflicts
         * TODO: remove once open zeppelin update to solc 0.5.0
         */
        library GnosisSafeMath {
            /**
             * @dev Multiplies two numbers, reverts on overflow.
             */
            function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                // benefit is lost if 'b' is also tested.
                // See: https://github.com/OpenZeppelin/openzeppelin-solidity/pull/522
                if (a == 0) {
                    return 0;
                }
                uint256 c = a * b;
                require(c / a == b);
                return c;
            }
            /**
             * @dev Subtracts two numbers, reverts on overflow (i.e. if subtrahend is greater than minuend).
             */
            function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                require(b <= a);
                uint256 c = a - b;
                return c;
            }
            /**
             * @dev Adds two numbers, reverts on overflow.
             */
            function add(uint256 a, uint256 b) internal pure returns (uint256) {
                uint256 c = a + b;
                require(c >= a);
                return c;
            }
            /**
             * @dev Returns the largest of two numbers.
             */
            function max(uint256 a, uint256 b) internal pure returns (uint256) {
                return a >= b ? a : b;
            }
        }
        // SPDX-License-Identifier: LGPL-3.0-only
        pragma solidity >=0.7.0 <0.9.0;
        contract ISignatureValidatorConstants {
            // bytes4(keccak256("isValidSignature(bytes,bytes)")
            bytes4 internal constant EIP1271_MAGIC_VALUE = 0x20c13b0b;
        }
        abstract contract ISignatureValidator is ISignatureValidatorConstants {
            /**
             * @dev Should return whether the signature provided is valid for the provided data
             * @param _data Arbitrary length data signed on the behalf of address(this)
             * @param _signature Signature byte array associated with _data
             *
             * MUST return the bytes4 magic value 0x20c13b0b when function passes.
             * MUST NOT modify state (using STATICCALL for solc < 0.5, view modifier for solc > 0.5)
             * MUST allow external calls
             */
            function isValidSignature(bytes memory _data, bytes memory _signature) public view virtual returns (bytes4);
        }
        

        File 3 of 4: RocketNodeStaking
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
        pragma solidity >0.5.0 <0.9.0;
        /**
         * @dev Interface of the ERC20 standard as defined in the EIP.
         */
        interface IERC20 {
            /**
             * @dev Emitted when `value` tokens are moved from one account (`from`) to
             * another (`to`).
             *
             * Note that `value` may be zero.
             */
            event Transfer(address indexed from, address indexed to, uint256 value);
            /**
             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
             * a call to {approve}. `value` is the new allowance.
             */
            event Approval(address indexed owner, address indexed spender, uint256 value);
            /**
             * @dev Returns the amount of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
            /**
             * @dev Returns the amount of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
            /**
             * @dev Moves `amount` tokens from the caller's account to `to`.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transfer(address to, uint256 amount) external returns (bool);
            /**
             * @dev Returns the remaining number of tokens that `spender` will be
             * allowed to spend on behalf of `owner` through {transferFrom}. This is
             * zero by default.
             *
             * This value changes when {approve} or {transferFrom} are called.
             */
            function allowance(address owner, address spender) external view returns (uint256);
            /**
             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * IMPORTANT: Beware that changing an allowance with this method brings the risk
             * that someone may use both the old and the new allowance by unfortunate
             * transaction ordering. One possible solution to mitigate this race
             * condition is to first reduce the spender's allowance to 0 and set the
             * desired value afterwards:
             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
             *
             * Emits an {Approval} event.
             */
            function approve(address spender, uint256 amount) external returns (bool);
            /**
             * @dev Moves `amount` tokens from `from` to `to` using the
             * allowance mechanism. `amount` is then deducted from the caller's
             * allowance.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(address from, address to, uint256 amount) external returns (bool);
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        interface RocketStorageInterface {
            // Deploy status
            function getDeployedStatus() external view returns (bool);
            // Guardian
            function getGuardian() external view returns(address);
            function setGuardian(address _newAddress) external;
            function confirmGuardian() external;
            // Getters
            function getAddress(bytes32 _key) external view returns (address);
            function getUint(bytes32 _key) external view returns (uint);
            function getString(bytes32 _key) external view returns (string memory);
            function getBytes(bytes32 _key) external view returns (bytes memory);
            function getBool(bytes32 _key) external view returns (bool);
            function getInt(bytes32 _key) external view returns (int);
            function getBytes32(bytes32 _key) external view returns (bytes32);
            // Setters
            function setAddress(bytes32 _key, address _value) external;
            function setUint(bytes32 _key, uint _value) external;
            function setString(bytes32 _key, string calldata _value) external;
            function setBytes(bytes32 _key, bytes calldata _value) external;
            function setBool(bytes32 _key, bool _value) external;
            function setInt(bytes32 _key, int _value) external;
            function setBytes32(bytes32 _key, bytes32 _value) external;
            // Deleters
            function deleteAddress(bytes32 _key) external;
            function deleteUint(bytes32 _key) external;
            function deleteString(bytes32 _key) external;
            function deleteBytes(bytes32 _key) external;
            function deleteBool(bytes32 _key) external;
            function deleteInt(bytes32 _key) external;
            function deleteBytes32(bytes32 _key) external;
            // Arithmetic
            function addUint(bytes32 _key, uint256 _amount) external;
            function subUint(bytes32 _key, uint256 _amount) external;
            // Protected storage
            function getNodeWithdrawalAddress(address _nodeAddress) external view returns (address);
            function getNodePendingWithdrawalAddress(address _nodeAddress) external view returns (address);
            function setWithdrawalAddress(address _nodeAddress, address _newWithdrawalAddress, bool _confirm) external;
            function confirmWithdrawalAddress(address _nodeAddress) external;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        import "../interface/RocketStorageInterface.sol";
        /// @title Base settings / modifiers for each contract in Rocket Pool
        /// @author David Rugendyke
        abstract contract RocketBase {
            // Calculate using this as the base
            uint256 constant calcBase = 1 ether;
            // Version of the contract
            uint8 public version;
            // The main storage contract where primary persistant storage is maintained
            RocketStorageInterface rocketStorage = RocketStorageInterface(address(0));
            /*** Modifiers **********************************************************/
            /**
            * @dev Throws if called by any sender that doesn't match a Rocket Pool network contract
            */
            modifier onlyLatestNetworkContract() {
                require(getBool(keccak256(abi.encodePacked("contract.exists", msg.sender))), "Invalid or outdated network contract");
                _;
            }
            /**
            * @dev Throws if called by any sender that doesn't match one of the supplied contract or is the latest version of that contract
            */
            modifier onlyLatestContract(string memory _contractName, address _contractAddress) {
                require(_contractAddress == getAddress(keccak256(abi.encodePacked("contract.address", _contractName))), "Invalid or outdated contract");
                _;
            }
            /**
            * @dev Throws if called by any sender that isn't a registered node
            */
            modifier onlyRegisteredNode(address _nodeAddress) {
                require(getBool(keccak256(abi.encodePacked("node.exists", _nodeAddress))), "Invalid node");
                _;
            }
            /**
            * @dev Throws if called by any sender that isn't a trusted node DAO member
            */
            modifier onlyTrustedNode(address _nodeAddress) {
                require(getBool(keccak256(abi.encodePacked("dao.trustednodes.", "member", _nodeAddress))), "Invalid trusted node");
                _;
            }
            /**
            * @dev Throws if called by any sender that isn't a registered minipool
            */
            modifier onlyRegisteredMinipool(address _minipoolAddress) {
                require(getBool(keccak256(abi.encodePacked("minipool.exists", _minipoolAddress))), "Invalid minipool");
                _;
            }
            
            /**
            * @dev Throws if called by any account other than a guardian account (temporary account allowed access to settings before DAO is fully enabled)
            */
            modifier onlyGuardian() {
                require(msg.sender == rocketStorage.getGuardian(), "Account is not a temporary guardian");
                _;
            }
            /*** Methods **********************************************************/
            /// @dev Set the main Rocket Storage address
            constructor(RocketStorageInterface _rocketStorageAddress) {
                // Update the contract address
                rocketStorage = RocketStorageInterface(_rocketStorageAddress);
            }
            /// @dev Get the address of a network contract by name
            function getContractAddress(string memory _contractName) internal view returns (address) {
                // Get the current contract address
                address contractAddress = getAddress(keccak256(abi.encodePacked("contract.address", _contractName)));
                // Check it
                require(contractAddress != address(0x0), "Contract not found");
                // Return
                return contractAddress;
            }
            /// @dev Get the address of a network contract by name (returns address(0x0) instead of reverting if contract does not exist)
            function getContractAddressUnsafe(string memory _contractName) internal view returns (address) {
                // Get the current contract address
                address contractAddress = getAddress(keccak256(abi.encodePacked("contract.address", _contractName)));
                // Return
                return contractAddress;
            }
            /// @dev Get the name of a network contract by address
            function getContractName(address _contractAddress) internal view returns (string memory) {
                // Get the contract name
                string memory contractName = getString(keccak256(abi.encodePacked("contract.name", _contractAddress)));
                // Check it
                require(bytes(contractName).length > 0, "Contract not found");
                // Return
                return contractName;
            }
            /// @dev Get revert error message from a .call method
            function getRevertMsg(bytes memory _returnData) internal pure returns (string memory) {
                // If the _res length is less than 68, then the transaction failed silently (without a revert message)
                if (_returnData.length < 68) return "Transaction reverted silently";
                assembly {
                    // Slice the sighash.
                    _returnData := add(_returnData, 0x04)
                }
                return abi.decode(_returnData, (string)); // All that remains is the revert string
            }
            /*** Rocket Storage Methods ****************************************/
            // Note: Unused helpers have been removed to keep contract sizes down
            /// @dev Storage get methods
            function getAddress(bytes32 _key) internal view returns (address) { return rocketStorage.getAddress(_key); }
            function getUint(bytes32 _key) internal view returns (uint) { return rocketStorage.getUint(_key); }
            function getString(bytes32 _key) internal view returns (string memory) { return rocketStorage.getString(_key); }
            function getBytes(bytes32 _key) internal view returns (bytes memory) { return rocketStorage.getBytes(_key); }
            function getBool(bytes32 _key) internal view returns (bool) { return rocketStorage.getBool(_key); }
            function getInt(bytes32 _key) internal view returns (int) { return rocketStorage.getInt(_key); }
            function getBytes32(bytes32 _key) internal view returns (bytes32) { return rocketStorage.getBytes32(_key); }
            /// @dev Storage set methods
            function setAddress(bytes32 _key, address _value) internal { rocketStorage.setAddress(_key, _value); }
            function setUint(bytes32 _key, uint _value) internal { rocketStorage.setUint(_key, _value); }
            function setString(bytes32 _key, string memory _value) internal { rocketStorage.setString(_key, _value); }
            function setBytes(bytes32 _key, bytes memory _value) internal { rocketStorage.setBytes(_key, _value); }
            function setBool(bytes32 _key, bool _value) internal { rocketStorage.setBool(_key, _value); }
            function setInt(bytes32 _key, int _value) internal { rocketStorage.setInt(_key, _value); }
            function setBytes32(bytes32 _key, bytes32 _value) internal { rocketStorage.setBytes32(_key, _value); }
            /// @dev Storage delete methods
            function deleteAddress(bytes32 _key) internal { rocketStorage.deleteAddress(_key); }
            function deleteUint(bytes32 _key) internal { rocketStorage.deleteUint(_key); }
            function deleteString(bytes32 _key) internal { rocketStorage.deleteString(_key); }
            function deleteBytes(bytes32 _key) internal { rocketStorage.deleteBytes(_key); }
            function deleteBool(bytes32 _key) internal { rocketStorage.deleteBool(_key); }
            function deleteInt(bytes32 _key) internal { rocketStorage.deleteInt(_key); }
            function deleteBytes32(bytes32 _key) internal { rocketStorage.deleteBytes32(_key); }
            /// @dev Storage arithmetic methods
            function addUint(bytes32 _key, uint256 _amount) internal { rocketStorage.addUint(_key, _amount); }
            function subUint(bytes32 _key, uint256 _amount) internal { rocketStorage.subUint(_key, _amount); }
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        // Represents the type of deposits required by a minipool
        enum MinipoolDeposit {
            None,       // Marks an invalid deposit type
            Full,       // The minipool requires 32 ETH from the node operator, 16 ETH of which will be refinanced from user deposits
            Half,       // The minipool required 16 ETH from the node operator to be matched with 16 ETH from user deposits
            Empty,      // The minipool requires 0 ETH from the node operator to be matched with 32 ETH from user deposits (trusted nodes only)
            Variable    // Indicates this minipool is of the new generation that supports a variable deposit amount
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        // Represents a minipool's status within the network
        enum MinipoolStatus {
            Initialised,    // The minipool has been initialised and is awaiting a deposit of user ETH
            Prelaunch,      // The minipool has enough ETH to begin staking and is awaiting launch by the node operator
            Staking,        // The minipool is currently staking
            Withdrawable,   // NO LONGER USED
            Dissolved       // The minipool has been dissolved and its user deposited ETH has been returned to the deposit pool
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        import "./MinipoolDeposit.sol";
        import "./MinipoolStatus.sol";
        // A struct containing all the information on-chain about a specific minipool
        struct MinipoolDetails {
            bool exists;
            address minipoolAddress;
            bytes pubkey;
            MinipoolStatus status;
            uint256 statusBlock;
            uint256 statusTime;
            bool finalised;
            MinipoolDeposit depositType;
            uint256 nodeFee;
            uint256 nodeDepositBalance;
            bool nodeDepositAssigned;
            uint256 userDepositBalance;
            bool userDepositAssigned;
            uint256 userDepositAssignedTime;
            bool useLatestDelegate;
            address delegate;
            address previousDelegate;
            address effectiveDelegate;
            uint256 penaltyCount;
            uint256 penaltyRate;
            address nodeAddress;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        import "../../types/MinipoolDeposit.sol";
        import "../../types/MinipoolStatus.sol";
        import "../RocketStorageInterface.sol";
        interface RocketMinipoolInterface {
            function version() external view returns (uint8);
            function initialise(address _nodeAddress) external;
            function getStatus() external view returns (MinipoolStatus);
            function getFinalised() external view returns (bool);
            function getStatusBlock() external view returns (uint256);
            function getStatusTime() external view returns (uint256);
            function getScrubVoted(address _member) external view returns (bool);
            function getDepositType() external view returns (MinipoolDeposit);
            function getNodeAddress() external view returns (address);
            function getNodeFee() external view returns (uint256);
            function getNodeDepositBalance() external view returns (uint256);
            function getNodeRefundBalance() external view returns (uint256);
            function getNodeDepositAssigned() external view returns (bool);
            function getPreLaunchValue() external view returns (uint256);
            function getNodeTopUpValue() external view returns (uint256);
            function getVacant() external view returns (bool);
            function getPreMigrationBalance() external view returns (uint256);
            function getUserDistributed() external view returns (bool);
            function getUserDepositBalance() external view returns (uint256);
            function getUserDepositAssigned() external view returns (bool);
            function getUserDepositAssignedTime() external view returns (uint256);
            function getTotalScrubVotes() external view returns (uint256);
            function calculateNodeShare(uint256 _balance) external view returns (uint256);
            function calculateUserShare(uint256 _balance) external view returns (uint256);
            function preDeposit(uint256 _bondingValue, bytes calldata _validatorPubkey, bytes calldata _validatorSignature, bytes32 _depositDataRoot) external payable;
            function deposit() external payable;
            function userDeposit() external payable;
            function distributeBalance(bool _rewardsOnly) external;
            function beginUserDistribute() external;
            function userDistributeAllowed() external view returns (bool);
            function refund() external;
            function slash() external;
            function finalise() external;
            function canStake() external view returns (bool);
            function canPromote() external view returns (bool);
            function stake(bytes calldata _validatorSignature, bytes32 _depositDataRoot) external;
            function prepareVacancy(uint256 _bondAmount, uint256 _currentBalance) external;
            function promote() external;
            function dissolve() external;
            function close() external;
            function voteScrub() external;
            function reduceBondAmount() external;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        pragma abicoder v2;
        // SPDX-License-Identifier: GPL-3.0-only
        import "../../types/MinipoolDeposit.sol";
        import "../../types/MinipoolDetails.sol";
        import "./RocketMinipoolInterface.sol";
        interface RocketMinipoolManagerInterface {
            function getMinipoolCount() external view returns (uint256);
            function getStakingMinipoolCount() external view returns (uint256);
            function getFinalisedMinipoolCount() external view returns (uint256);
            function getActiveMinipoolCount() external view returns (uint256);
            function getMinipoolRPLSlashed(address _minipoolAddress) external view returns (bool);
            function getMinipoolCountPerStatus(uint256 offset, uint256 limit) external view returns (uint256, uint256, uint256, uint256, uint256);
            function getPrelaunchMinipools(uint256 offset, uint256 limit) external view returns (address[] memory);
            function getMinipoolAt(uint256 _index) external view returns (address);
            function getNodeMinipoolCount(address _nodeAddress) external view returns (uint256);
            function getNodeActiveMinipoolCount(address _nodeAddress) external view returns (uint256);
            function getNodeFinalisedMinipoolCount(address _nodeAddress) external view returns (uint256);
            function getNodeStakingMinipoolCount(address _nodeAddress) external view returns (uint256);
            function getNodeStakingMinipoolCountBySize(address _nodeAddress, uint256 _depositSize) external view returns (uint256);
            function getNodeMinipoolAt(address _nodeAddress, uint256 _index) external view returns (address);
            function getNodeValidatingMinipoolCount(address _nodeAddress) external view returns (uint256);
            function getNodeValidatingMinipoolAt(address _nodeAddress, uint256 _index) external view returns (address);
            function getMinipoolByPubkey(bytes calldata _pubkey) external view returns (address);
            function getMinipoolExists(address _minipoolAddress) external view returns (bool);
            function getMinipoolDestroyed(address _minipoolAddress) external view returns (bool);
            function getMinipoolPubkey(address _minipoolAddress) external view returns (bytes memory);
            function updateNodeStakingMinipoolCount(uint256 _previousBond, uint256 _newBond, uint256 _previousFee, uint256 _newFee) external;
            function getMinipoolWithdrawalCredentials(address _minipoolAddress) external pure returns (bytes memory);
            function createMinipool(address _nodeAddress, uint256 _salt) external returns (RocketMinipoolInterface);
            function createVacantMinipool(address _nodeAddress, uint256 _salt, bytes calldata _validatorPubkey, uint256 _bondAmount, uint256 _currentBalance) external returns (RocketMinipoolInterface);
            function removeVacantMinipool() external;
            function getVacantMinipoolCount() external view returns (uint256);
            function getVacantMinipoolAt(uint256 _index) external view returns (address);
            function destroyMinipool() external;
            function incrementNodeStakingMinipoolCount(address _nodeAddress) external;
            function decrementNodeStakingMinipoolCount(address _nodeAddress) external;
            function tryDistribute(address _nodeAddress) external;
            function incrementNodeFinalisedMinipoolCount(address _nodeAddress) external;
            function setMinipoolPubkey(bytes calldata _pubkey) external;
            function getMinipoolDepositType(address _minipoolAddress) external view returns (MinipoolDeposit);
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        interface RocketNetworkPricesInterface {
            function getPricesBlock() external view returns (uint256);
            function getRPLPrice() external view returns (uint256);
            function submitPrices(uint256 _block, uint256 _slotTimestamp, uint256 _rplPrice) external;
            function executeUpdatePrices(uint256 _block, uint256 _slotTimestamp, uint256 _rplPrice) external;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        // SPDX-License-Identifier: GPL-3.0-only
        pragma solidity >0.5.0 <0.9.0;
        interface RocketNodeStakingInterface {
            function getTotalRPLStake() external view returns (uint256);
            function getNodeRPLStake(address _nodeAddress) external view returns (uint256);
            function getNodeETHMatched(address _nodeAddress) external view returns (uint256);
            function getNodeETHProvided(address _nodeAddress) external view returns (uint256);
            function getNodeETHCollateralisationRatio(address _nodeAddress) external view returns (uint256);
            function getNodeRPLStakedTime(address _nodeAddress) external view returns (uint256);
            function getNodeEffectiveRPLStake(address _nodeAddress) external view returns (uint256);
            function getNodeMinimumRPLStake(address _nodeAddress) external view returns (uint256);
            function getNodeMaximumRPLStake(address _nodeAddress) external view returns (uint256);
            function getNodeETHMatchedLimit(address _nodeAddress) external view returns (uint256);
            function getRPLLockingAllowed(address _nodeAddress) external view returns (bool);
            function stakeRPL(uint256 _amount) external;
            function stakeRPLFor(address _nodeAddress, uint256 _amount) external;
            function setRPLLockingAllowed(address _nodeAddress, bool _allowed) external;
            function setStakeRPLForAllowed(address _caller, bool _allowed) external;
            function setStakeRPLForAllowed(address _nodeAddress, address _caller, bool _allowed) external;
            function getNodeRPLLocked(address _nodeAddress) external view returns (uint256);
            function lockRPL(address _nodeAddress, uint256 _amount) external;
            function unlockRPL(address _nodeAddress, uint256 _amount) external;
            function transferRPL(address _from, address _to, uint256 _amount) external;
            function withdrawRPL(uint256 _amount) external;
            function withdrawRPL(address _nodeAddress, uint256 _amount) external;
            function slashRPL(address _nodeAddress, uint256 _ethSlashAmount) external;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        // SPDX-License-Identifier: GPL-3.0-only
        pragma solidity >0.5.0 <0.9.0;
        interface RocketDAOProtocolSettingsRewardsInterface {
            function setSettingRewardsClaimers(uint256 _trustedNodePercent, uint256 _protocolPercent, uint256 _nodePercent) external;
            function getRewardsClaimerPerc(string memory _contractName) external view returns (uint256);
            function getRewardsClaimersPerc() external view returns (uint256 _trustedNodePercent, uint256 _protocolPercent, uint256 _nodePercent);
            function getRewardsClaimersTrustedNodePerc() external view returns (uint256);
            function getRewardsClaimersProtocolPerc() external view returns (uint256);
            function getRewardsClaimersNodePerc() external view returns (uint256);
            function getRewardsClaimersTimeUpdated() external view returns (uint256);
            function getRewardsClaimIntervalPeriods() external view returns (uint256);
            function getRewardsClaimIntervalTime() external view returns (uint256);
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        import "../../../../types/MinipoolDeposit.sol";
        interface RocketDAOProtocolSettingsMinipoolInterface {
            function getLaunchBalance() external view returns (uint256);
            function getPreLaunchValue() external pure returns (uint256);
            function getDepositUserAmount(MinipoolDeposit _depositType) external view returns (uint256);
            function getFullDepositUserAmount() external view returns (uint256);
            function getHalfDepositUserAmount() external view returns (uint256);
            function getVariableDepositAmount() external view returns (uint256);
            function getSubmitWithdrawableEnabled() external view returns (bool);
            function getBondReductionEnabled() external view returns (bool);
            function getLaunchTimeout() external view returns (uint256);
            function getMaximumCount() external view returns (uint256);
            function isWithinUserDistributeWindow(uint256 _time) external view returns (bool);
            function hasUserDistributeWindowPassed(uint256 _time) external view returns (bool);
            function getUserDistributeWindowStart() external view returns (uint256);
            function getUserDistributeWindowLength() external view returns (uint256);
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        interface RocketDAOProtocolSettingsNodeInterface {
            function getRegistrationEnabled() external view returns (bool);
            function getSmoothingPoolRegistrationEnabled() external view returns (bool);
            function getDepositEnabled() external view returns (bool);
            function getVacantMinipoolsEnabled() external view returns (bool);
            function getMinimumPerMinipoolStake() external view returns (uint256);
            function getMaximumPerMinipoolStake() external view returns (uint256);
            function getMaximumStakeForVotingPower() external view returns (uint256);
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol)
        import "./IERC20.sol";
        pragma solidity >0.5.0 <0.9.0;
        interface IERC20Burnable is IERC20 {
            function burn(uint256 amount) external;
            function burnFrom(address account, uint256 amount) external;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        import "./util/IERC20Burnable.sol";
        interface RocketVaultInterface {
            function balanceOf(string memory _networkContractName) external view returns (uint256);
            function depositEther() external payable;
            function withdrawEther(uint256 _amount) external;
            function depositToken(string memory _networkContractName, IERC20 _tokenAddress, uint256 _amount) external;
            function withdrawToken(address _withdrawalAddress, IERC20 _tokenAddress, uint256 _amount) external;
            function balanceOfToken(string memory _networkContractName, IERC20 _tokenAddress) external view returns (uint256);
            function transferToken(string memory _networkContractName, IERC20 _tokenAddress, uint256 _amount) external;
            function burnToken(IERC20Burnable _tokenAddress, uint256 _amount) external;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        interface AddressSetStorageInterface {
            function getCount(bytes32 _key) external view returns (uint);
            function getItem(bytes32 _key, uint _index) external view returns (address);
            function getIndexOf(bytes32 _key, address _value) external view returns (int);
            function addItem(bytes32 _key, address _value) external;
            function removeItem(bytes32 _key, address _value) external;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        // SPDX-License-Identifier: GPL-3.0-only
        pragma solidity >0.5.0 <0.9.0;
        struct Checkpoint224 {
            uint32 _block;
            uint224 _value;
        }
        /// @notice Accounting for snapshotting of values based on block numbers
        interface RocketNetworkSnapshotsInterface {
            function push(bytes32 _key, uint224 _value) external;
            function length(bytes32 _key) external view returns (uint256);
            function latest(bytes32 _key) external view returns (bool, uint32, uint224);
            function latestBlock(bytes32 _key) external view returns (uint32);
            function latestValue(bytes32 _key) external view returns (uint224);
            function lookup(bytes32 _key, uint32 _block) external view returns (uint224);
            function lookupRecent(bytes32 _key, uint32 _block, uint256 _recency) external view returns (uint224);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/Math.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Standard math utilities missing in the Solidity language.
         */
        library Math {
            enum Rounding {
                Down, // Toward negative infinity
                Up, // Toward infinity
                Zero // Toward zero
            }
            /**
             * @dev Returns the largest of two numbers.
             */
            function max(uint256 a, uint256 b) internal pure returns (uint256) {
                return a > b ? a : b;
            }
            /**
             * @dev Returns the smallest of two numbers.
             */
            function min(uint256 a, uint256 b) internal pure returns (uint256) {
                return a < b ? a : b;
            }
            /**
             * @dev Returns the average of two numbers. The result is rounded towards
             * zero.
             */
            function average(uint256 a, uint256 b) internal pure returns (uint256) {
                // (a + b) / 2 can overflow.
                return (a & b) + (a ^ b) / 2;
            }
            /**
             * @dev Returns the ceiling of the division of two numbers.
             *
             * This differs from standard division with `/` in that it rounds up instead
             * of rounding down.
             */
            function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
                // (a + b - 1) / b can overflow on addition, so we distribute.
                return a == 0 ? 0 : (a - 1) / b + 1;
            }
            /**
             * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0
             * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv)
             * with further edits by Uniswap Labs also under MIT license.
             */
            function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) {
                unchecked {
                    // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use
                    // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256
                    // variables such that product = prod1 * 2^256 + prod0.
                    uint256 prod0; // Least significant 256 bits of the product
                    uint256 prod1; // Most significant 256 bits of the product
                    assembly {
                        let mm := mulmod(x, y, not(0))
                        prod0 := mul(x, y)
                        prod1 := sub(sub(mm, prod0), lt(mm, prod0))
                    }
                    // Handle non-overflow cases, 256 by 256 division.
                    if (prod1 == 0) {
                        // Solidity will revert if denominator == 0, unlike the div opcode on its own.
                        // The surrounding unchecked block does not change this fact.
                        // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic.
                        return prod0 / denominator;
                    }
                    // Make sure the result is less than 2^256. Also prevents denominator == 0.
                    require(denominator > prod1, "Math: mulDiv overflow");
                    ///////////////////////////////////////////////
                    // 512 by 256 division.
                    ///////////////////////////////////////////////
                    // Make division exact by subtracting the remainder from [prod1 prod0].
                    uint256 remainder;
                    assembly {
                        // Compute remainder using mulmod.
                        remainder := mulmod(x, y, denominator)
                        // Subtract 256 bit number from 512 bit number.
                        prod1 := sub(prod1, gt(remainder, prod0))
                        prod0 := sub(prod0, remainder)
                    }
                    // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1.
                    // See https://cs.stackexchange.com/q/138556/92363.
                    // Does not overflow because the denominator cannot be zero at this stage in the function.
                    uint256 twos = denominator & (~denominator + 1);
                    assembly {
                        // Divide denominator by twos.
                        denominator := div(denominator, twos)
                        // Divide [prod1 prod0] by twos.
                        prod0 := div(prod0, twos)
                        // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one.
                        twos := add(div(sub(0, twos), twos), 1)
                    }
                    // Shift in bits from prod1 into prod0.
                    prod0 |= prod1 * twos;
                    // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such
                    // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for
                    // four bits. That is, denominator * inv = 1 mod 2^4.
                    uint256 inverse = (3 * denominator) ^ 2;
                    // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works
                    // in modular arithmetic, doubling the correct bits in each step.
                    inverse *= 2 - denominator * inverse; // inverse mod 2^8
                    inverse *= 2 - denominator * inverse; // inverse mod 2^16
                    inverse *= 2 - denominator * inverse; // inverse mod 2^32
                    inverse *= 2 - denominator * inverse; // inverse mod 2^64
                    inverse *= 2 - denominator * inverse; // inverse mod 2^128
                    inverse *= 2 - denominator * inverse; // inverse mod 2^256
                    // Because the division is now exact we can divide by multiplying with the modular inverse of denominator.
                    // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is
                    // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1
                    // is no longer required.
                    result = prod0 * inverse;
                    return result;
                }
            }
            /**
             * @notice Calculates x * y / denominator with full precision, following the selected rounding direction.
             */
            function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) {
                uint256 result = mulDiv(x, y, denominator);
                if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
                    result += 1;
                }
                return result;
            }
            /**
             * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down.
             *
             * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11).
             */
            function sqrt(uint256 a) internal pure returns (uint256) {
                if (a == 0) {
                    return 0;
                }
                // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target.
                //
                // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have
                // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`.
                //
                // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)`
                // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))`
                // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)`
                //
                // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit.
                uint256 result = 1 << (log2(a) >> 1);
                // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128,
                // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at
                // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision
                // into the expected uint128 result.
                unchecked {
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    result = (result + a / result) >> 1;
                    return min(result, a / result);
                }
            }
            /**
             * @notice Calculates sqrt(a), following the selected rounding direction.
             */
            function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
                unchecked {
                    uint256 result = sqrt(a);
                    return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
                }
            }
            /**
             * @dev Return the log in base 2, rounded down, of a positive value.
             * Returns 0 if given 0.
             */
            function log2(uint256 value) internal pure returns (uint256) {
                uint256 result = 0;
                unchecked {
                    if (value >> 128 > 0) {
                        value >>= 128;
                        result += 128;
                    }
                    if (value >> 64 > 0) {
                        value >>= 64;
                        result += 64;
                    }
                    if (value >> 32 > 0) {
                        value >>= 32;
                        result += 32;
                    }
                    if (value >> 16 > 0) {
                        value >>= 16;
                        result += 16;
                    }
                    if (value >> 8 > 0) {
                        value >>= 8;
                        result += 8;
                    }
                    if (value >> 4 > 0) {
                        value >>= 4;
                        result += 4;
                    }
                    if (value >> 2 > 0) {
                        value >>= 2;
                        result += 2;
                    }
                    if (value >> 1 > 0) {
                        result += 1;
                    }
                }
                return result;
            }
            /**
             * @dev Return the log in base 2, following the selected rounding direction, of a positive value.
             * Returns 0 if given 0.
             */
            function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
                unchecked {
                    uint256 result = log2(value);
                    return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
                }
            }
            /**
             * @dev Return the log in base 10, rounded down, of a positive value.
             * Returns 0 if given 0.
             */
            function log10(uint256 value) internal pure returns (uint256) {
                uint256 result = 0;
                unchecked {
                    if (value >= 10 ** 64) {
                        value /= 10 ** 64;
                        result += 64;
                    }
                    if (value >= 10 ** 32) {
                        value /= 10 ** 32;
                        result += 32;
                    }
                    if (value >= 10 ** 16) {
                        value /= 10 ** 16;
                        result += 16;
                    }
                    if (value >= 10 ** 8) {
                        value /= 10 ** 8;
                        result += 8;
                    }
                    if (value >= 10 ** 4) {
                        value /= 10 ** 4;
                        result += 4;
                    }
                    if (value >= 10 ** 2) {
                        value /= 10 ** 2;
                        result += 2;
                    }
                    if (value >= 10 ** 1) {
                        result += 1;
                    }
                }
                return result;
            }
            /**
             * @dev Return the log in base 10, following the selected rounding direction, of a positive value.
             * Returns 0 if given 0.
             */
            function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
                unchecked {
                    uint256 result = log10(value);
                    return result + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0);
                }
            }
            /**
             * @dev Return the log in base 256, rounded down, of a positive value.
             * Returns 0 if given 0.
             *
             * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string.
             */
            function log256(uint256 value) internal pure returns (uint256) {
                uint256 result = 0;
                unchecked {
                    if (value >> 128 > 0) {
                        value >>= 128;
                        result += 16;
                    }
                    if (value >> 64 > 0) {
                        value >>= 64;
                        result += 8;
                    }
                    if (value >> 32 > 0) {
                        value >>= 32;
                        result += 4;
                    }
                    if (value >> 16 > 0) {
                        value >>= 16;
                        result += 2;
                    }
                    if (value >> 8 > 0) {
                        result += 1;
                    }
                }
                return result;
            }
            /**
             * @dev Return the log in base 256, following the selected rounding direction, of a positive value.
             * Returns 0 if given 0.
             */
            function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
                unchecked {
                    uint256 result = log256(value);
                    return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0);
                }
            }
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        // SPDX-License-Identifier: MIT
        // Copyright (c) 2016-2023 zOS Global Limited and contributors
        // Adapted from OpenZeppelin `Checkpoints` contract
        pragma solidity 0.8.18;
        import "@openzeppelin4/contracts/utils/math/Math.sol";
        import "../RocketBase.sol";
        import "../../interface/network/RocketNetworkSnapshotsInterface.sol";
        /// @notice Accounting for snapshotting of values based on block numbers
        contract RocketNetworkSnapshots is RocketBase, RocketNetworkSnapshotsInterface {
            constructor(RocketStorageInterface _rocketStorageAddress) RocketBase(_rocketStorageAddress) {
                // Set contract version
                version = 1;
            }
            function push(bytes32 _key, uint224 _value) onlyLatestContract("rocketNetworkSnapshots", address(this)) onlyLatestNetworkContract external {
                _insert(_key, _value);
            }
            function length(bytes32 _key) public view returns (uint256) {
                return rocketStorage.getUint(keccak256(abi.encodePacked("snapshot.length", _key)));
            }
            function latest(bytes32 _key) external view returns (bool, uint32, uint224) {
                uint256 len = length(_key);
                if (len == 0) {
                    return (false, 0, 0);
                }
                Checkpoint224 memory checkpoint = _load(_key, len - 1);
                return (true, checkpoint._block, checkpoint._value);
            }
            function latestBlock(bytes32 _key) external view returns (uint32) {
                uint256 len = length(_key);
                return len == 0 ? 0 : _blockAt(_key, len - 1);
            }
            function latestValue(bytes32 _key) external view returns (uint224) {
                uint256 len = length(_key);
                return len == 0 ? 0 : _valueAt(_key, len - 1);
            }
            function lookup(bytes32 _key, uint32 _block) external view returns (uint224) {
                uint256 len = length(_key);
                uint256 pos = _binaryLookup(_key, _block, 0, len);
                return pos == 0 ? 0 : _valueAt(_key, pos - 1);
            }
            function lookupRecent(bytes32 _key, uint32 _block, uint256 _recency) external view returns (uint224) {
                uint256 len = length(_key);
                uint256 low = 0;
                uint256 high = len;
                if (len > 5 && len > _recency) {
                    uint256 mid = len - _recency;
                    if (_block < _blockAt(_key, mid)) {
                        high = mid;
                    } else {
                        low = mid + 1;
                    }
                }
                uint256 pos = _binaryLookup(_key, _block, low, high);
                return pos == 0 ? 0 : _valueAt(_key, pos - 1);
            }
            function _insert(bytes32 _key, uint224 _value) private {
                uint32 blockNumber = uint32(block.number);
                uint256 pos = length(_key);
                if (pos > 0) {
                    Checkpoint224 memory last = _load(_key, pos - 1);
                    // Checkpoint keys must be non-decreasing.
                    require (last._block <= blockNumber, "Unordered snapshot insertion");
                    // Update or push new checkpoint
                    if (last._block == blockNumber) {
                        last._value = _value;
                        _set(_key, pos - 1, last);
                    } else {
                        _push(_key, Checkpoint224({_block: blockNumber, _value: _value}));
                    }
                } else {
                    _push(_key, Checkpoint224({_block: blockNumber, _value: _value}));
                }
            }
            function _binaryLookup(
                bytes32 _key,
                uint32 _block,
                uint256 _low,
                uint256 _high
            ) private view returns (uint256) {
                while (_low < _high) {
                    uint256 mid = Math.average(_low, _high);
                    if (_blockAt(_key, mid) > _block) {
                        _high = mid;
                    } else {
                        _low = mid + 1;
                    }
                }
                return _high;
            }
            function _load(bytes32 _key, uint256 _pos) private view returns (Checkpoint224 memory) {
                bytes32 key = bytes32(uint256(_key) + _pos);
                bytes32 raw = rocketStorage.getBytes32(key);
                Checkpoint224 memory result;
                result._block = uint32(uint256(raw) >> 224);
                result._value = uint224(uint256(raw));
                return result;
            }
            function _blockAt(bytes32 _key, uint256 _pos) private view returns (uint32) {
                bytes32 key = bytes32(uint256(_key) + _pos);
                bytes32 raw = rocketStorage.getBytes32(key);
                return uint32(uint256(raw) >> 224);
            }
            function _valueAt(bytes32 _key, uint256 _pos) private view returns (uint224) {
                bytes32 key = bytes32(uint256(_key) + _pos);
                bytes32 raw = rocketStorage.getBytes32(key);
                return uint224(uint256(raw));
            }
            function _push(bytes32 _key, Checkpoint224 memory _item) private {
                bytes32 lengthKey = keccak256(abi.encodePacked("snapshot.length", _key));
                uint256 snapshotLength = rocketStorage.getUint(lengthKey);
                bytes32 key = bytes32(uint256(_key) + snapshotLength);
                rocketStorage.setUint(lengthKey, snapshotLength + 1);
                rocketStorage.setBytes32(key, _encode(_item));
            }
            function _set(bytes32 _key, uint256 _pos, Checkpoint224 memory _item) private {
                bytes32 key = bytes32(uint256(_key) + _pos);
                rocketStorage.setBytes32(key, _encode(_item));
            }
            function _encode(Checkpoint224 memory _item) private pure returns (bytes32) {
                return bytes32(
                    uint256(_item._block) << 224 | uint256(_item._value)
                );
            }
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        pragma solidity >0.5.0 <0.9.0;
        // SPDX-License-Identifier: GPL-3.0-only
        // A struct containing all the information on-chain about a specific node
        struct NodeDetails {
            bool exists;
            uint256 registrationTime;
            string timezoneLocation;
            bool feeDistributorInitialised;
            address feeDistributorAddress;
            uint256 rewardNetwork;
            uint256 rplStake;
            uint256 effectiveRPLStake;
            uint256 minimumRPLStake;
            uint256 maximumRPLStake;
            uint256 ethMatched;
            uint256 ethMatchedLimit;
            uint256 minipoolCount;
            uint256 balanceETH;
            uint256 balanceRETH;
            uint256 balanceRPL;
            uint256 balanceOldRPL;
            uint256 depositCreditBalance;
            uint256 distributorBalanceUserETH;
            uint256 distributorBalanceNodeETH;
            address withdrawalAddress;
            address pendingWithdrawalAddress;
            bool smoothingPoolRegistrationState;
            uint256 smoothingPoolRegistrationChanged;
            address nodeAddress;
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        // SPDX-License-Identifier: GPL-3.0-only
        pragma solidity >0.5.0 <0.9.0;
        pragma abicoder v2;
        import "../../types/NodeDetails.sol";
        interface RocketNodeManagerInterface {
            // Structs
            struct TimezoneCount {
                string timezone;
                uint256 count;
            }
            function getNodeCount() external view returns (uint256);
            function getNodeCountPerTimezone(uint256 offset, uint256 limit) external view returns (TimezoneCount[] memory);
            function getNodeAt(uint256 _index) external view returns (address);
            function getNodeExists(address _nodeAddress) external view returns (bool);
            function getNodeWithdrawalAddress(address _nodeAddress) external view returns (address);
            function getNodePendingWithdrawalAddress(address _nodeAddress) external view returns (address);
            function getNodeRPLWithdrawalAddress(address _nodeAddress) external view returns (address);
            function getNodeRPLWithdrawalAddressIsSet(address _nodeAddress) external view returns (bool);
            function unsetRPLWithdrawalAddress(address _nodeAddress) external;
            function setRPLWithdrawalAddress(address _nodeAddress, address _newRPLWithdrawalAddress, bool _confirm) external;
            function confirmRPLWithdrawalAddress(address _nodeAddress) external;
            function getNodePendingRPLWithdrawalAddress(address _nodeAddress) external view returns (address);
            function getNodeTimezoneLocation(address _nodeAddress) external view returns (string memory);
            function registerNode(string calldata _timezoneLocation) external;
            function getNodeRegistrationTime(address _nodeAddress) external view returns (uint256);
            function setTimezoneLocation(string calldata _timezoneLocation) external;
            function setRewardNetwork(address _nodeAddress, uint256 network) external;
            function getRewardNetwork(address _nodeAddress) external view returns (uint256);
            function getFeeDistributorInitialised(address _nodeAddress) external view returns (bool);
            function initialiseFeeDistributor() external;
            function getAverageNodeFee(address _nodeAddress) external view returns (uint256);
            function setSmoothingPoolRegistrationState(bool _state) external;
            function getSmoothingPoolRegistrationState(address _nodeAddress) external returns (bool);
            function getSmoothingPoolRegistrationChanged(address _nodeAddress) external returns (uint256);
            function getSmoothingPoolRegisteredNodeCount(uint256 _offset, uint256 _limit) external view returns (uint256);
            function getNodeDetails(address _nodeAddress) external view returns (NodeDetails memory);
            function getNodeAddresses(uint256 _offset, uint256 _limit) external view returns (address[] memory);
        }
        /**
           *       .
           *      / \\
           *     |.'.|
           *     |'.'|
           *   ,'|   |'.
           *  |,-'-|-'-.|
           *   __|_| |         _        _      _____           _
           *  | ___ \\|        | |      | |    | ___ \\         | |
           *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
           *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
           *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
           *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
           * +---------------------------------------------------+
           * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
           * +---------------------------------------------------+
           *
           *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
           *  be community-owned, decentralised, permissionless, & trustless.
           *
           *  For more information about Rocket Pool, visit https://rocketpool.net
           *
           *  Authored by the Rocket Pool Core Team
           *  Contributors: https://github.com/rocket-pool/rocketpool/graphs/contributors
           *  A special thanks to the Rocket Pool community for all their contributions.
           *
           */
        // SPDX-License-Identifier: GPL-3.0-only
        pragma solidity 0.8.18;
        import "../../interface/util/IERC20.sol";
        import "../RocketBase.sol";
        import "../../interface/minipool/RocketMinipoolManagerInterface.sol";
        import "../../interface/network/RocketNetworkPricesInterface.sol";
        import "../../interface/node/RocketNodeStakingInterface.sol";
        import "../../interface/dao/protocol/settings/RocketDAOProtocolSettingsRewardsInterface.sol";
        import "../../interface/dao/protocol/settings/RocketDAOProtocolSettingsMinipoolInterface.sol";
        import "../../interface/dao/protocol/settings/RocketDAOProtocolSettingsNodeInterface.sol";
        import "../../interface/RocketVaultInterface.sol";
        import "../../interface/util/AddressSetStorageInterface.sol";
        import "../../interface/network/RocketNetworkSnapshotsInterface.sol";
        import "../network/RocketNetworkSnapshots.sol";
        import "../../interface/node/RocketNodeManagerInterface.sol";
        /// @notice Handles node deposits and minipool creation
        contract RocketNodeStaking is RocketBase, RocketNodeStakingInterface {
            // Constants
            bytes32 immutable internal totalKey;
            // Events
            event RPLStaked(address indexed from, uint256 amount, uint256 time);
            event RPLWithdrawn(address indexed to, uint256 amount, uint256 time);
            event RPLSlashed(address indexed node, uint256 amount, uint256 ethValue, uint256 time);
            event StakeRPLForAllowed(address indexed node, address indexed caller, bool allowed, uint256 time);
            event RPLLockingAllowed(address indexed node, bool allowed, uint256 time);
            event RPLLocked(address indexed from, uint256 amount, uint256 time);
            event RPLUnlocked(address indexed from, uint256 amount, uint256 time);
            event RPLTransferred(address indexed from, address indexed to, uint256 amount, uint256 time);
            modifier onlyRPLWithdrawalAddressOrNode(address _nodeAddress) {
                // Check that the call is coming from RPL withdrawal address (or node if unset)
                RocketNodeManagerInterface rocketNodeManager = RocketNodeManagerInterface(getContractAddress("rocketNodeManager"));
                if (rocketNodeManager.getNodeRPLWithdrawalAddressIsSet(_nodeAddress)) {
                    address rplWithdrawalAddress = rocketNodeManager.getNodeRPLWithdrawalAddress(_nodeAddress);
                    require(msg.sender == rplWithdrawalAddress, "Must be called from RPL withdrawal address");
                } else {
                    require(msg.sender == _nodeAddress, "Must be called from node address");
                }
                _;
            }
            constructor(RocketStorageInterface _rocketStorageAddress) RocketBase(_rocketStorageAddress) {
                version = 5;
                // Precompute keys
                totalKey = keccak256(abi.encodePacked("rpl.staked.total.amount"));
            }
            /// @notice Returns the total quantity of RPL staked on the network
            function getTotalRPLStake() override external view returns (uint256) {
                return getUint(totalKey);
            }
            /// @dev Increases the total network RPL stake
            /// @param _amount How much to increase by
            function increaseTotalRPLStake(uint256 _amount) private {
                addUint(totalKey, _amount);
            }
            /// @dev Decrease the total network RPL stake
            /// @param _amount How much to decrease by
            function decreaseTotalRPLStake(uint256 _amount) private {
                subUint(totalKey, _amount);
            }
            /// @notice Returns the amount a given node operator has staked
            /// @param _nodeAddress The address of the node operator to query
            function getNodeRPLStake(address _nodeAddress) override public view returns (uint256) {
                bytes32 key = keccak256(abi.encodePacked("rpl.staked.node.amount", _nodeAddress));
                RocketNetworkSnapshotsInterface rocketNetworkSnapshots = RocketNetworkSnapshotsInterface(getContractAddress("rocketNetworkSnapshots"));
                (bool exists,, uint224 value) = rocketNetworkSnapshots.latest(key);
                uint256 stake = uint256(value);
                if (!exists){
                    // Fallback to old value
                    stake = getUint(key);
                }
                return stake;
            }
            /// @dev Increases a node operator's RPL stake
            /// @param _amount How much to increase by
            function increaseNodeRPLStake(address _nodeAddress, uint256 _amount) private {
                RocketNetworkSnapshotsInterface rocketNetworkSnapshots = RocketNetworkSnapshotsInterface(getContractAddress("rocketNetworkSnapshots"));
                bytes32 key = keccak256(abi.encodePacked("rpl.staked.node.amount", _nodeAddress));
                (bool exists,, uint224 value) = rocketNetworkSnapshots.latest(key);
                if (!exists){
                    value = uint224(getUint(key));
                }
                rocketNetworkSnapshots.push(key, value + uint224(_amount));
            }
            /// @dev Decrease a node operator's RPL stake
            /// @param _amount How much to decrease by
            function decreaseNodeRPLStake(address _nodeAddress, uint256 _amount) private {
                RocketNetworkSnapshotsInterface rocketNetworkSnapshots = RocketNetworkSnapshotsInterface(getContractAddress("rocketNetworkSnapshots"));
                bytes32 key = keccak256(abi.encodePacked("rpl.staked.node.amount", _nodeAddress));
                (bool exists,, uint224 value) = rocketNetworkSnapshots.latest(key);
                if (!exists){
                    value = uint224(getUint(key));
                }
                rocketNetworkSnapshots.push(key, value - uint224(_amount));
            }
            /// @notice Returns a node's matched ETH amount (amount taken from protocol to stake)
            /// @param _nodeAddress The address of the node operator to query
            function getNodeETHMatched(address _nodeAddress) override public view returns (uint256) {
                RocketNetworkSnapshotsInterface rocketNetworkSnapshots = RocketNetworkSnapshotsInterface(getContractAddress("rocketNetworkSnapshots"));
                bytes32 key = keccak256(abi.encodePacked("eth.matched.node.amount", _nodeAddress));
                (bool exists, , uint224 value) = rocketNetworkSnapshots.latest(key);
                if (exists) {
                    // Value was previously set in a snapshot so return that
                    return value;
                } else {
                    // Fallback to old method
                    uint256 ethMatched = getUint(key);
                    if (ethMatched > 0) {
                        // Value was previously calculated and stored so return that
                        return ethMatched;
                    } else {
                        // Fallback for backwards compatibility before ETH matched was recorded (all legacy minipools matched 16 ETH from protocol)
                        RocketMinipoolManagerInterface rocketMinipoolManager = RocketMinipoolManagerInterface(getContractAddress("rocketMinipoolManager"));
                        return rocketMinipoolManager.getNodeActiveMinipoolCount(_nodeAddress) * 16 ether;
                    }
                }
            }
            /// @notice Returns a node's provided ETH amount (amount supplied to create minipools)
            /// @param _nodeAddress The address of the node operator to query
            function getNodeETHProvided(address _nodeAddress) override public view returns (uint256) {
                // Get contracts
                RocketMinipoolManagerInterface rocketMinipoolManager = RocketMinipoolManagerInterface(getContractAddress("rocketMinipoolManager"));
                uint256 activeMinipoolCount = rocketMinipoolManager.getNodeActiveMinipoolCount(_nodeAddress);
                // Retrieve stored ETH matched value
                uint256 ethMatched = getNodeETHMatched(_nodeAddress);
                if (ethMatched > 0) {
                    RocketDAOProtocolSettingsMinipoolInterface rocketDAOProtocolSettingsMinipool = RocketDAOProtocolSettingsMinipoolInterface(getContractAddress("rocketDAOProtocolSettingsMinipool"));
                    uint256 launchAmount = rocketDAOProtocolSettingsMinipool.getLaunchBalance();
                    // ETH provided is number of staking minipools * 32 - eth matched
                    uint256 totalEthStaked = activeMinipoolCount * launchAmount;
                    return totalEthStaked - ethMatched;
                } else {
                    // Fallback for legacy minipools is number of staking minipools * 16
                    return activeMinipoolCount * 16 ether;
                }
            }
            /// @notice Returns the ratio between capital taken from users and provided by a node operator.
            ///         The value is a 1e18 precision fixed point integer value of (node capital + user capital) / node capital.
            /// @param _nodeAddress The address of the node operator to query
            function getNodeETHCollateralisationRatio(address _nodeAddress) override public view returns (uint256) {
                uint256 ethMatched = getNodeETHMatched(_nodeAddress);
                if (ethMatched == 0) {
                    // Node operator only has legacy minipools and all legacy minipools had a 1:1 ratio
                    return calcBase * 2;
                } else {
                    RocketDAOProtocolSettingsMinipoolInterface rocketDAOProtocolSettingsMinipool = RocketDAOProtocolSettingsMinipoolInterface(getContractAddress("rocketDAOProtocolSettingsMinipool"));
                    uint256 launchAmount = rocketDAOProtocolSettingsMinipool.getLaunchBalance();
                    RocketMinipoolManagerInterface rocketMinipoolManager = RocketMinipoolManagerInterface(getContractAddress("rocketMinipoolManager"));
                    uint256 totalEthStaked = rocketMinipoolManager.getNodeActiveMinipoolCount(_nodeAddress) * launchAmount;
                    return (totalEthStaked * calcBase) / (totalEthStaked - ethMatched);
                }
            }
            /// @notice Returns the timestamp at which a node last staked RPL
            function getNodeRPLStakedTime(address _nodeAddress) override public view returns (uint256) {
                return getUint(keccak256(abi.encodePacked("rpl.staked.node.time", _nodeAddress)));
            }
            /// @dev Sets the timestamp at which a node last staked RPL
            /// @param _nodeAddress The address of the node operator to set the value for
            /// @param _time The timestamp to set
            function setNodeRPLStakedTime(address _nodeAddress, uint256 _time) private {
                setUint(keccak256(abi.encodePacked("rpl.staked.node.time", _nodeAddress)), _time);
            }
            /// @notice Calculate and return a node's effective RPL stake amount
            /// @param _nodeAddress The address of the node operator to calculate for
            function getNodeEffectiveRPLStake(address _nodeAddress) override public view returns (uint256) {
                // Load contracts
                RocketNetworkPricesInterface rocketNetworkPrices = RocketNetworkPricesInterface(getContractAddress("rocketNetworkPrices"));
                RocketDAOProtocolSettingsNodeInterface rocketDAOProtocolSettingsNode = RocketDAOProtocolSettingsNodeInterface(getContractAddress("rocketDAOProtocolSettingsNode"));
                // Get node's current RPL stake
                uint256 rplStake = getNodeRPLStake(_nodeAddress);
                // Retrieve variables for calculations
                uint256 matchedETH = getNodeETHMatched(_nodeAddress);
                uint256 providedETH = getNodeETHProvided(_nodeAddress);
                uint256 rplPrice = rocketNetworkPrices.getRPLPrice();
                // RPL stake cannot exceed maximum
                uint256 maximumStakePercent = rocketDAOProtocolSettingsNode.getMaximumPerMinipoolStake();
                uint256 maximumStake = providedETH * maximumStakePercent / rplPrice;
                if (rplStake > maximumStake) {
                    return maximumStake;
                }
                // If RPL stake is lower than minimum, node has no effective stake
                uint256 minimumStakePercent = rocketDAOProtocolSettingsNode.getMinimumPerMinipoolStake();
                uint256 minimumStake = matchedETH * minimumStakePercent / rplPrice;
                if (rplStake < minimumStake) {
                    return 0;
                }
                // Otherwise, return the actual stake
                return rplStake;
            }
            /// @notice Calculate and return a node's minimum RPL stake to collateralize their minipools
            /// @param _nodeAddress The address of the node operator to calculate for
            function getNodeMinimumRPLStake(address _nodeAddress) override external view returns (uint256) {
                // Load contracts
                RocketNetworkPricesInterface rocketNetworkPrices = RocketNetworkPricesInterface(getContractAddress("rocketNetworkPrices"));
                RocketDAOProtocolSettingsNodeInterface rocketDAOProtocolSettingsNode = RocketDAOProtocolSettingsNodeInterface(getContractAddress("rocketDAOProtocolSettingsNode"));
                // Retrieve variables
                uint256 minimumStakePercent = rocketDAOProtocolSettingsNode.getMinimumPerMinipoolStake();
                uint256 matchedETH = getNodeETHMatched(_nodeAddress);
                return matchedETH * minimumStakePercent / rocketNetworkPrices.getRPLPrice();
            }
            /// @notice Calculate and return a node's maximum RPL stake to fully collateralise their minipools
            /// @param _nodeAddress The address of the node operator to calculate for
            function getNodeMaximumRPLStake(address _nodeAddress) override public view returns (uint256) {
                // Load contracts
                RocketNetworkPricesInterface rocketNetworkPrices = RocketNetworkPricesInterface(getContractAddress("rocketNetworkPrices"));
                RocketDAOProtocolSettingsNodeInterface rocketDAOProtocolSettingsNode = RocketDAOProtocolSettingsNodeInterface(getContractAddress("rocketDAOProtocolSettingsNode"));
                // Retrieve variables
                uint256 maximumStakePercent = rocketDAOProtocolSettingsNode.getMaximumPerMinipoolStake();
                uint256 providedETH = getNodeETHProvided(_nodeAddress);
                return providedETH * maximumStakePercent / rocketNetworkPrices.getRPLPrice();
            }
            /// @notice Calculate and return a node's limit of how much user ETH they can use based on RPL stake
            /// @param _nodeAddress The address of the node operator to calculate for
            function getNodeETHMatchedLimit(address _nodeAddress) override external view returns (uint256) {
                // Load contracts
                RocketNetworkPricesInterface rocketNetworkPrices = RocketNetworkPricesInterface(getContractAddress("rocketNetworkPrices"));
                RocketDAOProtocolSettingsNodeInterface rocketDAOProtocolSettingsNode = RocketDAOProtocolSettingsNodeInterface(getContractAddress("rocketDAOProtocolSettingsNode"));
                // Calculate & return limit
                uint256 minimumStakePercent = rocketDAOProtocolSettingsNode.getMinimumPerMinipoolStake();
                return getNodeRPLStake(_nodeAddress) *rocketNetworkPrices.getRPLPrice() / minimumStakePercent;
            }
            /// @notice Returns whether this node allows RPL locking or not
            /// @param _nodeAddress The address of the node operator to query for
            function getRPLLockingAllowed(address _nodeAddress) external view returns (bool) {
                return getBool(keccak256(abi.encodePacked("rpl.locking.allowed", _nodeAddress)));
            }
            /// @notice Accept an RPL stake from the node operator's own address
            ///         Requires the node's RPL withdrawal address to be unset
            /// @param _amount The amount of RPL to stake
            function stakeRPL(uint256 _amount) override external {
                stakeRPLFor(msg.sender, _amount);
            }
            /// @notice Accept an RPL stake from any address for a specified node
            ///         Requires caller to have approved this contract to spend RPL
            ///         Requires caller to be on the node operator's allow list (see `setStakeForAllowed`)
            /// @param _nodeAddress The address of the node operator to stake on behalf of
            /// @param _amount The amount of RPL to stake
            function stakeRPLFor(address _nodeAddress, uint256 _amount) override public onlyLatestContract("rocketNodeStaking", address(this)) onlyRegisteredNode(_nodeAddress) {
                // Must be node's RPL withdrawal address if set or the node's address or an allow listed address or rocketMerkleDistributorMainnet
                if (msg.sender != getAddress(keccak256(abi.encodePacked("contract.address", "rocketMerkleDistributorMainnet")))) {
                    RocketNodeManagerInterface rocketNodeManager = RocketNodeManagerInterface(getContractAddress("rocketNodeManager"));
                    bool fromNode = false;
                    if (rocketNodeManager.getNodeRPLWithdrawalAddressIsSet(_nodeAddress)) {
                        address rplWithdrawalAddress = rocketNodeManager.getNodeRPLWithdrawalAddress(_nodeAddress);
                        fromNode = msg.sender == rplWithdrawalAddress;
                    } else {
                        address withdrawalAddress = rocketStorage.getNodeWithdrawalAddress(_nodeAddress);
                        fromNode = (msg.sender == _nodeAddress) || (msg.sender == withdrawalAddress);
                    }
                    if (!fromNode) {
                        require(getBool(keccak256(abi.encodePacked("node.stake.for.allowed", _nodeAddress, msg.sender))), "Not allowed to stake for");
                    }
                }
                _stakeRPL(_nodeAddress, _amount);
            }
            /// @notice Sets the allow state for this node to perform functions that require locking RPL
            /// @param _nodeAddress The address of the node operator to change the state for
            /// @param _allowed Whether locking is allowed or not
            function setRPLLockingAllowed(address _nodeAddress, bool _allowed) override external onlyLatestContract("rocketNodeStaking", address(this)) onlyRPLWithdrawalAddressOrNode(_nodeAddress) {
                // Set the value
                setBool(keccak256(abi.encodePacked("rpl.locking.allowed", _nodeAddress)), _allowed);
                // Log it
                emit RPLLockingAllowed(_nodeAddress, _allowed, block.timestamp);
            }
            /// @notice Explicitly allow or remove allowance of an address to be able to stake on behalf of a node
            /// @dev The node operator is determined by the address calling this method, it is here for backwards compatibility
            /// @param _caller The address you wish to allow
            /// @param _allowed Whether the address is allowed or denied
            function setStakeRPLForAllowed(address _caller, bool _allowed) override external {
                setStakeRPLForAllowed(msg.sender, _caller, _allowed);
            }
            /// @notice Explicitly allow or remove allowance of an address to be able to stake on behalf of a node
            /// @param _nodeAddress The address of the node operator allowing the caller
            /// @param _caller The address you wish to allow
            /// @param _allowed Whether the address is allowed or denied
            function setStakeRPLForAllowed(address _nodeAddress, address _caller, bool _allowed) override public onlyLatestContract("rocketNodeStaking", address(this)) onlyRPLWithdrawalAddressOrNode(_nodeAddress) {
                // Set the value
                setBool(keccak256(abi.encodePacked("node.stake.for.allowed", _nodeAddress, _caller)), _allowed);
                // Log it
                emit StakeRPLForAllowed(_nodeAddress, _caller, _allowed, block.timestamp);
            }
            /// @dev Internal logic for staking RPL
            /// @param _nodeAddress The address to increase the RPL stake of
            /// @param _amount The amount of RPL to stake
            function _stakeRPL(address _nodeAddress, uint256 _amount) internal {
                // Load contracts
                address rplTokenAddress = getContractAddress("rocketTokenRPL");
                address rocketVaultAddress = getContractAddress("rocketVault");
                IERC20 rplToken = IERC20(rplTokenAddress);
                RocketVaultInterface rocketVault = RocketVaultInterface(rocketVaultAddress);
                // Transfer RPL tokens
                require(rplToken.transferFrom(msg.sender, address(this), _amount), "Could not transfer RPL to staking contract");
                // Deposit RPL tokens to vault
                require(rplToken.approve(rocketVaultAddress, _amount), "Could not approve vault RPL deposit");
                rocketVault.depositToken("rocketNodeStaking", rplToken, _amount);
                // Update RPL stake amounts & node RPL staked block
                increaseTotalRPLStake(_amount);
                increaseNodeRPLStake(_nodeAddress, _amount);
                setNodeRPLStakedTime(_nodeAddress, block.timestamp);
                // Emit RPL staked event
                emit RPLStaked(_nodeAddress, _amount, block.timestamp);
            }
            /// @notice Returns the amount of RPL that is locked for a given node
            /// @param _nodeAddress The address of the node operator to query for
            function getNodeRPLLocked(address _nodeAddress) override public view returns (uint256) {
                return getUint(keccak256(abi.encodePacked("rpl.locked.node.amount", _nodeAddress)));
            }
            /// @notice Locks an amount of RPL from being withdrawn even if the node operator is over capitalised
            /// @param _nodeAddress The address of the node operator
            /// @param _amount The amount of RPL to lock
            function lockRPL(address _nodeAddress, uint256 _amount) override external onlyLatestContract("rocketNodeStaking", address(this)) onlyLatestNetworkContract() {
                // Check status
                require(getBool(keccak256(abi.encodePacked("rpl.locking.allowed", _nodeAddress))), "Node is not allowed to lock RPL");
                // The node must have unlocked stake equaling or greater than the amount
                uint256 rplStake = getNodeRPLStake(_nodeAddress);
                bytes32 lockedStakeKey = keccak256(abi.encodePacked("rpl.locked.node.amount", _nodeAddress));
                uint256 lockedStake = getUint(lockedStakeKey);
                require(rplStake - lockedStake >= _amount, "Not enough staked RPL");
                // Increase locked RPL
                setUint(lockedStakeKey, lockedStake + _amount);
                // Emit event
                emit RPLLocked(_nodeAddress, _amount, block.timestamp);
            }
            /// @notice Unlocks an amount of RPL making it possible to withdraw if the nod is over capitalised
            /// @param _nodeAddress The address of the node operator
            /// @param _amount The amount of RPL to unlock
            function unlockRPL(address _nodeAddress, uint256 _amount) override external onlyLatestContract("rocketNodeStaking", address(this)) onlyLatestNetworkContract() {
                // The node must have locked stake equaling or greater than the amount
                bytes32 lockedStakeKey = keccak256(abi.encodePacked("rpl.locked.node.amount", _nodeAddress));
                uint256 lockedStake = getUint(lockedStakeKey);
                require(_amount <= lockedStake, "Not enough locked RPL");
                // Decrease locked RPL
                setUint(lockedStakeKey, lockedStake - _amount);
                // Emit event
                emit RPLUnlocked(_nodeAddress, _amount, block.timestamp);
            }
            /// @notice Transfers RPL from one node to another
            /// @param _from The node to transfer from
            /// @param _to The node to transfer to
            /// @param _amount The amount of RPL to transfer
            function transferRPL(address _from, address _to, uint256 _amount) override external onlyLatestContract("rocketNodeStaking", address(this)) onlyLatestNetworkContract() onlyRegisteredNode(_from) {
                // Check sender has enough RPL
                require(getNodeRPLStake(_from) >= _amount, "Sender has insufficient RPL");
                // Transfer the stake
                decreaseNodeRPLStake(_from, _amount);
                increaseNodeRPLStake(_to, _amount);
                // Emit event
                emit RPLTransferred(_from, _to, _amount, block.timestamp);
            }
            /// @notice Withdraw staked RPL back to the node account or withdraw RPL address
            ///         Can only be called by a node if they have not set their RPL withdrawal address
            /// @param _amount The amount of RPL to withdraw
            function withdrawRPL(uint256 _amount) override external {
                withdrawRPL(msg.sender, _amount);
            }
            /// @notice Withdraw staked RPL back to the node account or withdraw RPL address
            ///         If RPL withdrawal address has been set, must be called from it. Otherwise, must be called from
            ///         node's primary withdrawal address or their node address.
            /// @param _nodeAddress The address of the node withdrawing
            /// @param _amount The amount of RPL to withdraw
            function withdrawRPL(address _nodeAddress, uint256 _amount) override public onlyLatestContract("rocketNodeStaking", address(this)) {
                // Check valid node
                require(getBool(keccak256(abi.encodePacked("node.exists", _nodeAddress))), "Invalid node");
                // Check address is permitted to withdraw
                RocketNodeManagerInterface rocketNodeManager = RocketNodeManagerInterface(getContractAddress("rocketNodeManager"));
                address rplWithdrawalAddress = rocketNodeManager.getNodeRPLWithdrawalAddress(_nodeAddress);
                if (rocketNodeManager.getNodeRPLWithdrawalAddressIsSet(_nodeAddress)) {
                    // If RPL withdrawal address is set, must be called from it
                    require(msg.sender == rplWithdrawalAddress, "Invalid caller");
                } else {
                    // Otherwise, must be called from node address or withdrawal address
                    address withdrawalAddress = rocketStorage.getNodeWithdrawalAddress(_nodeAddress);
                    require(msg.sender == _nodeAddress || msg.sender == withdrawalAddress, "Invalid caller");
                }
                // Load contracts
                RocketDAOProtocolSettingsRewardsInterface rocketDAOProtocolSettingsRewards = RocketDAOProtocolSettingsRewardsInterface(getContractAddress("rocketDAOProtocolSettingsRewards"));
                RocketVaultInterface rocketVault = RocketVaultInterface(getContractAddress("rocketVault"));
                // Check cooldown period (one claim period) has passed since RPL last staked
                require(block.timestamp - getNodeRPLStakedTime(_nodeAddress) >= rocketDAOProtocolSettingsRewards.getRewardsClaimIntervalTime(), "The withdrawal cooldown period has not passed");
                // Get & check node's current RPL stake
                uint256 rplStake = getNodeRPLStake(_nodeAddress);
                uint256 lockedStake = getNodeRPLLocked(_nodeAddress);
                require(rplStake - lockedStake >= _amount, "Withdrawal amount exceeds node's staked RPL balance");
                // Check withdrawal would not under collateralise node
                require(rplStake - _amount >= getNodeMaximumRPLStake(_nodeAddress) + lockedStake, "Node's staked RPL balance after withdrawal is less than required balance");
                // Update RPL stake amounts
                decreaseTotalRPLStake(_amount);
                decreaseNodeRPLStake(_nodeAddress, _amount);
                // Transfer RPL tokens to node's RPL withdrawal address (if unset, defaults to primary withdrawal address)
                rocketVault.withdrawToken(rplWithdrawalAddress, IERC20(getContractAddress("rocketTokenRPL")), _amount);
                // Emit RPL withdrawn event
                emit RPLWithdrawn(_nodeAddress, _amount, block.timestamp);
            }
            /// @notice Slash a node's RPL by an ETH amount
            ///         Only accepts calls from registered minipools
            /// @param _nodeAddress The address to slash RPL from
            /// @param _ethSlashAmount The amount of RPL to slash denominated in ETH value
            function slashRPL(address _nodeAddress, uint256 _ethSlashAmount) override external onlyLatestContract("rocketNodeStaking", address(this)) onlyRegisteredMinipool(msg.sender) {
                // Load contracts
                RocketNetworkPricesInterface rocketNetworkPrices = RocketNetworkPricesInterface(getContractAddress("rocketNetworkPrices"));
                RocketVaultInterface rocketVault = RocketVaultInterface(getContractAddress("rocketVault"));
                // Calculate RPL amount to slash
                uint256 rplSlashAmount = calcBase * _ethSlashAmount / rocketNetworkPrices.getRPLPrice();
                // Cap slashed amount to node's RPL stake
                uint256 rplStake = getNodeRPLStake(_nodeAddress);
                if (rplSlashAmount > rplStake) { rplSlashAmount = rplStake; }
                // Transfer slashed amount to auction contract
                if(rplSlashAmount > 0) rocketVault.transferToken("rocketAuctionManager", IERC20(getContractAddress("rocketTokenRPL")), rplSlashAmount);
                // Update RPL stake amounts
                decreaseTotalRPLStake(rplSlashAmount);
                decreaseNodeRPLStake(_nodeAddress, rplSlashAmount);
                // Mark minipool as slashed
                setBool(keccak256(abi.encodePacked("minipool.rpl.slashed", msg.sender)), true);
                // Emit RPL slashed event
                emit RPLSlashed(_nodeAddress, rplSlashAmount, _ethSlashAmount, block.timestamp);
            }
        }
        

        File 4 of 4: RocketStorage
        // SPDX-License-Identifier: MIT
        pragma solidity >=0.6.0 <0.8.0;
        /**
         * @dev Wrappers over Solidity's arithmetic operations with added overflow
         * checks.
         *
         * Arithmetic operations in Solidity wrap on overflow. This can easily result
         * in bugs, because programmers usually assume that an overflow raises an
         * error, which is the standard behavior in high level programming languages.
         * `SafeMath` restores this intuition by reverting the transaction when an
         * operation overflows.
         *
         * Using this library instead of the unchecked operations eliminates an entire
         * class of bugs, so it's recommended to use it always.
         */
        library SafeMath {
            /**
             * @dev Returns the addition of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                uint256 c = a + b;
                if (c < a) return (false, 0);
                return (true, c);
            }
            /**
             * @dev Returns the substraction of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                if (b > a) return (false, 0);
                return (true, a - b);
            }
            /**
             * @dev Returns the multiplication of two unsigned integers, with an overflow flag.
             *
             * _Available since v3.4._
             */
            function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
                // benefit is lost if 'b' is also tested.
                // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
                if (a == 0) return (true, 0);
                uint256 c = a * b;
                if (c / a != b) return (false, 0);
                return (true, c);
            }
            /**
             * @dev Returns the division of two unsigned integers, with a division by zero flag.
             *
             * _Available since v3.4._
             */
            function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                if (b == 0) return (false, 0);
                return (true, a / b);
            }
            /**
             * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag.
             *
             * _Available since v3.4._
             */
            function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
                if (b == 0) return (false, 0);
                return (true, a % b);
            }
            /**
             * @dev Returns the addition of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `+` operator.
             *
             * Requirements:
             *
             * - Addition cannot overflow.
             */
            function add(uint256 a, uint256 b) internal pure returns (uint256) {
                uint256 c = a + b;
                require(c >= a, "SafeMath: addition overflow");
                return c;
            }
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting on
             * overflow (when the result is negative).
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b) internal pure returns (uint256) {
                require(b <= a, "SafeMath: subtraction overflow");
                return a - b;
            }
            /**
             * @dev Returns the multiplication of two unsigned integers, reverting on
             * overflow.
             *
             * Counterpart to Solidity's `*` operator.
             *
             * Requirements:
             *
             * - Multiplication cannot overflow.
             */
            function mul(uint256 a, uint256 b) internal pure returns (uint256) {
                if (a == 0) return 0;
                uint256 c = a * b;
                require(c / a == b, "SafeMath: multiplication overflow");
                return c;
            }
            /**
             * @dev Returns the integer division of two unsigned integers, reverting on
             * division by zero. The result is rounded towards zero.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b) internal pure returns (uint256) {
                require(b > 0, "SafeMath: division by zero");
                return a / b;
            }
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * reverting when dividing by zero.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b) internal pure returns (uint256) {
                require(b > 0, "SafeMath: modulo by zero");
                return a % b;
            }
            /**
             * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
             * overflow (when the result is negative).
             *
             * CAUTION: This function is deprecated because it requires allocating memory for the error
             * message unnecessarily. For custom revert reasons use {trySub}.
             *
             * Counterpart to Solidity's `-` operator.
             *
             * Requirements:
             *
             * - Subtraction cannot overflow.
             */
            function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b <= a, errorMessage);
                return a - b;
            }
            /**
             * @dev Returns the integer division of two unsigned integers, reverting with custom message on
             * division by zero. The result is rounded towards zero.
             *
             * CAUTION: This function is deprecated because it requires allocating memory for the error
             * message unnecessarily. For custom revert reasons use {tryDiv}.
             *
             * Counterpart to Solidity's `/` operator. Note: this function uses a
             * `revert` opcode (which leaves remaining gas untouched) while Solidity
             * uses an invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b > 0, errorMessage);
                return a / b;
            }
            /**
             * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
             * reverting with custom message when dividing by zero.
             *
             * CAUTION: This function is deprecated because it requires allocating memory for the error
             * message unnecessarily. For custom revert reasons use {tryMod}.
             *
             * Counterpart to Solidity's `%` operator. This function uses a `revert`
             * opcode (which leaves remaining gas untouched) while Solidity uses an
             * invalid opcode to revert (consuming all remaining gas).
             *
             * Requirements:
             *
             * - The divisor cannot be zero.
             */
            function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
                require(b > 0, errorMessage);
                return a % b;
            }
        }
        /**
          *       .
          *      / \\
          *     |.'.|
          *     |'.'|
          *   ,'|   |`.
          *  |,-'-|-'-.|
          *   __|_| |         _        _      _____           _
          *  | ___ \\|        | |      | |    | ___ \\         | |
          *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
          *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
          *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
          *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
          * +---------------------------------------------------+
          * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
          * +---------------------------------------------------+
          *
          *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
          *  be community-owned, decentralised, and trustless.
          *
          *  For more information about Rocket Pool, visit https://rocketpool.net
          *
          *  Authors: David Rugendyke, Jake Pospischil, Kane Wallmann, Darren Langley, Joe Clapis, Nick Doherty
          *
          */
        pragma solidity 0.7.6;
        // SPDX-License-Identifier: GPL-3.0-only
        import "../interface/RocketStorageInterface.sol";
        import "@openzeppelin/contracts/math/SafeMath.sol";
        /// @title The primary persistent storage for Rocket Pool
        /// @author David Rugendyke
        contract RocketStorage is RocketStorageInterface {
            // Events
            event NodeWithdrawalAddressSet(address indexed node, address indexed withdrawalAddress, uint256 time);
            event GuardianChanged(address oldGuardian, address newGuardian);
            // Libraries
            using SafeMath for uint256;
            // Storage maps
            mapping(bytes32 => string)     private stringStorage;
            mapping(bytes32 => bytes)      private bytesStorage;
            mapping(bytes32 => uint256)    private uintStorage;
            mapping(bytes32 => int256)     private intStorage;
            mapping(bytes32 => address)    private addressStorage;
            mapping(bytes32 => bool)       private booleanStorage;
            mapping(bytes32 => bytes32)    private bytes32Storage;
            // Protected storage (not accessible by network contracts)
            mapping(address => address)    private withdrawalAddresses;
            mapping(address => address)    private pendingWithdrawalAddresses;
            // Guardian address
            address guardian;
            address newGuardian;
            // Flag storage has been initialised
            bool storageInit = false;
            /// @dev Only allow access from the latest version of a contract in the Rocket Pool network after deployment
            modifier onlyLatestRocketNetworkContract() {
                if (storageInit == true) {
                    // Make sure the access is permitted to only contracts in our Dapp
                    require(booleanStorage[keccak256(abi.encodePacked("contract.exists", msg.sender))], "Invalid or outdated network contract");
                } else {
                    // Only Dapp and the guardian account are allowed access during initialisation.
                    // tx.origin is only safe to use in this case for deployment since no external contracts are interacted with
                    require((
                        booleanStorage[keccak256(abi.encodePacked("contract.exists", msg.sender))] || tx.origin == guardian
                    ), "Invalid or outdated network contract attempting access during deployment");
                }
                _;
            }
            /// @dev Construct RocketStorage
            constructor() {
                // Set the guardian upon deployment
                guardian = msg.sender;
            }
            // Get guardian address
            function getGuardian() external override view returns (address) {
                return guardian;
            }
            // Transfers guardianship to a new address
            function setGuardian(address _newAddress) external override {
                // Check tx comes from current guardian
                require(msg.sender == guardian, "Is not guardian account");
                // Store new address awaiting confirmation
                newGuardian = _newAddress;
            }
            // Confirms change of guardian
            function confirmGuardian() external override {
                // Check tx came from new guardian address
                require(msg.sender == newGuardian, "Confirmation must come from new guardian address");
                // Store old guardian for event
                address oldGuardian = guardian;
                // Update guardian and clear storage
                guardian = newGuardian;
                delete newGuardian;
                // Emit event
                emit GuardianChanged(oldGuardian, guardian);
            }
            // Set this as being deployed now
            function getDeployedStatus() external override view returns (bool) {
                return storageInit;
            }
            // Set this as being deployed now
            function setDeployedStatus() external {
                // Only guardian can lock this down
                require(msg.sender == guardian, "Is not guardian account");
                // Set it now
                storageInit = true;
            }
            // Protected storage
            // Get a node's withdrawal address
            function getNodeWithdrawalAddress(address _nodeAddress) public override view returns (address) {
                // If no withdrawal address has been set, return the nodes address
                address withdrawalAddress = withdrawalAddresses[_nodeAddress];
                if (withdrawalAddress == address(0)) {
                    return _nodeAddress;
                }
                return withdrawalAddress;
            }
            // Get a node's pending withdrawal address
            function getNodePendingWithdrawalAddress(address _nodeAddress) external override view returns (address) {
                return pendingWithdrawalAddresses[_nodeAddress];
            }
            // Set a node's withdrawal address
            function setWithdrawalAddress(address _nodeAddress, address _newWithdrawalAddress, bool _confirm) external override {
                // Check new withdrawal address
                require(_newWithdrawalAddress != address(0x0), "Invalid withdrawal address");
                // Confirm the transaction is from the node's current withdrawal address
                address withdrawalAddress = getNodeWithdrawalAddress(_nodeAddress);
                require(withdrawalAddress == msg.sender, "Only a tx from a node's withdrawal address can update it");
                // Update immediately if confirmed
                if (_confirm) {
                    updateWithdrawalAddress(_nodeAddress, _newWithdrawalAddress);
                }
                // Set pending withdrawal address if not confirmed
                else {
                    pendingWithdrawalAddresses[_nodeAddress] = _newWithdrawalAddress;
                }
            }
            // Confirm a node's new withdrawal address
            function confirmWithdrawalAddress(address _nodeAddress) external override {
                // Get node by pending withdrawal address
                require(pendingWithdrawalAddresses[_nodeAddress] == msg.sender, "Confirmation must come from the pending withdrawal address");
                delete pendingWithdrawalAddresses[_nodeAddress];
                // Update withdrawal address
                updateWithdrawalAddress(_nodeAddress, msg.sender);
            }
            // Update a node's withdrawal address
            function updateWithdrawalAddress(address _nodeAddress, address _newWithdrawalAddress) private {
                // Set new withdrawal address
                withdrawalAddresses[_nodeAddress] = _newWithdrawalAddress;
                // Emit withdrawal address set event
                emit NodeWithdrawalAddressSet(_nodeAddress, _newWithdrawalAddress, block.timestamp);
            }
            /// @param _key The key for the record
            function getAddress(bytes32 _key) override external view returns (address r) {
                return addressStorage[_key];
            }
            /// @param _key The key for the record
            function getUint(bytes32 _key) override external view returns (uint256 r) {
                return uintStorage[_key];
            }
            /// @param _key The key for the record
            function getString(bytes32 _key) override external view returns (string memory) {
                return stringStorage[_key];
            }
            /// @param _key The key for the record
            function getBytes(bytes32 _key) override external view returns (bytes memory) {
                return bytesStorage[_key];
            }
            /// @param _key The key for the record
            function getBool(bytes32 _key) override external view returns (bool r) {
                return booleanStorage[_key];
            }
            /// @param _key The key for the record
            function getInt(bytes32 _key) override external view returns (int r) {
                return intStorage[_key];
            }
            /// @param _key The key for the record
            function getBytes32(bytes32 _key) override external view returns (bytes32 r) {
                return bytes32Storage[_key];
            }
            /// @param _key The key for the record
            function setAddress(bytes32 _key, address _value) onlyLatestRocketNetworkContract override external {
                addressStorage[_key] = _value;
            }
            /// @param _key The key for the record
            function setUint(bytes32 _key, uint _value) onlyLatestRocketNetworkContract override external {
                uintStorage[_key] = _value;
            }
            /// @param _key The key for the record
            function setString(bytes32 _key, string calldata _value) onlyLatestRocketNetworkContract override external {
                stringStorage[_key] = _value;
            }
            /// @param _key The key for the record
            function setBytes(bytes32 _key, bytes calldata _value) onlyLatestRocketNetworkContract override external {
                bytesStorage[_key] = _value;
            }
            /// @param _key The key for the record
            function setBool(bytes32 _key, bool _value) onlyLatestRocketNetworkContract override external {
                booleanStorage[_key] = _value;
            }
            /// @param _key The key for the record
            function setInt(bytes32 _key, int _value) onlyLatestRocketNetworkContract override external {
                intStorage[_key] = _value;
            }
            /// @param _key The key for the record
            function setBytes32(bytes32 _key, bytes32 _value) onlyLatestRocketNetworkContract override external {
                bytes32Storage[_key] = _value;
            }
            /// @param _key The key for the record
            function deleteAddress(bytes32 _key) onlyLatestRocketNetworkContract override external {
                delete addressStorage[_key];
            }
            /// @param _key The key for the record
            function deleteUint(bytes32 _key) onlyLatestRocketNetworkContract override external {
                delete uintStorage[_key];
            }
            /// @param _key The key for the record
            function deleteString(bytes32 _key) onlyLatestRocketNetworkContract override external {
                delete stringStorage[_key];
            }
            /// @param _key The key for the record
            function deleteBytes(bytes32 _key) onlyLatestRocketNetworkContract override external {
                delete bytesStorage[_key];
            }
            /// @param _key The key for the record
            function deleteBool(bytes32 _key) onlyLatestRocketNetworkContract override external {
                delete booleanStorage[_key];
            }
            /// @param _key The key for the record
            function deleteInt(bytes32 _key) onlyLatestRocketNetworkContract override external {
                delete intStorage[_key];
            }
            /// @param _key The key for the record
            function deleteBytes32(bytes32 _key) onlyLatestRocketNetworkContract override external {
                delete bytes32Storage[_key];
            }
            /// @param _key The key for the record
            /// @param _amount An amount to add to the record's value
            function addUint(bytes32 _key, uint256 _amount) onlyLatestRocketNetworkContract override external {
                uintStorage[_key] = uintStorage[_key].add(_amount);
            }
            /// @param _key The key for the record
            /// @param _amount An amount to subtract from the record's value
            function subUint(bytes32 _key, uint256 _amount) onlyLatestRocketNetworkContract override external {
                uintStorage[_key] = uintStorage[_key].sub(_amount);
            }
        }
        /**
          *       .
          *      / \\
          *     |.'.|
          *     |'.'|
          *   ,'|   |`.
          *  |,-'-|-'-.|
          *   __|_| |         _        _      _____           _
          *  | ___ \\|        | |      | |    | ___ \\         | |
          *  | |_/ /|__   ___| | _____| |_   | |_/ /__   ___ | |
          *  |    // _ \\ / __| |/ / _ \\ __|  |  __/ _ \\ / _ \\| |
          *  | |\\ \\ (_) | (__|   <  __/ |_   | | | (_) | (_) | |
          *  \\_| \\_\\___/ \\___|_|\\_\\___|\\__|  \\_|  \\___/ \\___/|_|
          * +---------------------------------------------------+
          * |    DECENTRALISED STAKING PROTOCOL FOR ETHEREUM    |
          * +---------------------------------------------------+
          *
          *  Rocket Pool is a first-of-its-kind Ethereum staking pool protocol, designed to
          *  be community-owned, decentralised, and trustless.
          *
          *  For more information about Rocket Pool, visit https://rocketpool.net
          *
          *  Authors: David Rugendyke, Jake Pospischil, Kane Wallmann, Darren Langley, Joe Clapis, Nick Doherty
          *
          */
        pragma solidity 0.7.6;
        // SPDX-License-Identifier: GPL-3.0-only
        interface RocketStorageInterface {
            // Deploy status
            function getDeployedStatus() external view returns (bool);
            // Guardian
            function getGuardian() external view returns(address);
            function setGuardian(address _newAddress) external;
            function confirmGuardian() external;
            // Getters
            function getAddress(bytes32 _key) external view returns (address);
            function getUint(bytes32 _key) external view returns (uint);
            function getString(bytes32 _key) external view returns (string memory);
            function getBytes(bytes32 _key) external view returns (bytes memory);
            function getBool(bytes32 _key) external view returns (bool);
            function getInt(bytes32 _key) external view returns (int);
            function getBytes32(bytes32 _key) external view returns (bytes32);
            // Setters
            function setAddress(bytes32 _key, address _value) external;
            function setUint(bytes32 _key, uint _value) external;
            function setString(bytes32 _key, string calldata _value) external;
            function setBytes(bytes32 _key, bytes calldata _value) external;
            function setBool(bytes32 _key, bool _value) external;
            function setInt(bytes32 _key, int _value) external;
            function setBytes32(bytes32 _key, bytes32 _value) external;
            // Deleters
            function deleteAddress(bytes32 _key) external;
            function deleteUint(bytes32 _key) external;
            function deleteString(bytes32 _key) external;
            function deleteBytes(bytes32 _key) external;
            function deleteBool(bytes32 _key) external;
            function deleteInt(bytes32 _key) external;
            function deleteBytes32(bytes32 _key) external;
            // Arithmetic
            function addUint(bytes32 _key, uint256 _amount) external;
            function subUint(bytes32 _key, uint256 _amount) external;
            // Protected storage
            function getNodeWithdrawalAddress(address _nodeAddress) external view returns (address);
            function getNodePendingWithdrawalAddress(address _nodeAddress) external view returns (address);
            function setWithdrawalAddress(address _nodeAddress, address _newWithdrawalAddress, bool _confirm) external;
            function confirmWithdrawalAddress(address _nodeAddress) external;
        }