Contract
0x3F861649a7517af171ff845a5cb7aE6ACeEbd6aA
1
Contract Overview
Balance:
0 Ether
More Info
My Name Tag:
Not Available
[ Download CSV Export ]
Latest 25 internal transaction
[ Download CSV Export ]
Contract Name:
PodsUpkeep
Compiler Version
v0.7.6+commit.7338295f
Contract Source Code (Solidity Standard Json-Input format)
// 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 SafeMathUpgradeable { /** * @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; } }
// SPDX-License-Identifier: MIT pragma solidity >=0.6.0 <0.8.0; import "../utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor () internal { address msgSender = _msgSender(); _owner = msgSender; emit OwnershipTransferred(address(0), msgSender); } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { require(owner() == _msgSender(), "Ownable: caller is not the owner"); _; } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { emit OwnershipTransferred(_owner, address(0)); _owner = address(0); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); emit OwnershipTransferred(_owner, newOwner); _owner = newOwner; } }
// SPDX-License-Identifier: MIT pragma solidity >=0.6.0 <0.8.0; /* * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with GSN meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address payable) { return msg.sender; } function _msgData() internal view virtual returns (bytes memory) { this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691 return msg.data; } }
// SPDX-License-Identifier: MIT pragma solidity >=0.6.12 <=0.7.6; import "@openzeppelin/contracts/access/Ownable.sol"; import "./utils/MappedSinglyLinkedList.sol"; ///@notice A registry to hold Contract addresses. Underlying data structure is a singly linked list. contract AddressRegistry is Ownable { using MappedSinglyLinkedList for MappedSinglyLinkedList.Mapping; MappedSinglyLinkedList.Mapping internal addressList; /// @notice Emmitted when a contract has been added to the registry event AddressAdded(address indexed _address); /// @notice Emmitted when a contract has been removed to the registry event AddressRemoved(address indexed _address); /// @notice Emitted when all the registry addresses are cleared event AllAddressesCleared(); /// @notice Storage field for what type of contract this Registry is storing string public addressType; /// @notice Contract constructor sets addressType, intializes list and transfers ownership /// @param _addressType The type of contracts stored in this registry /// @param _owner The address to set as owner of the contract constructor(string memory _addressType, address _owner) Ownable() { addressType = _addressType; addressList.initialize(); transferOwnership(_owner); } /// @notice Returns an array of all contract addresses in the linked list /// @return Array of contract addresses function getAddresses() view external returns(address[] memory) { return addressList.addressArray(); } /// @notice Adds addresses to the linked list. Will revert if the address is already in the list. Can only be called by the Registry owner. /// @param _addresses Array of contract addresses to be added function addAddresses(address[] calldata _addresses) public onlyOwner { for(uint256 _address = 0; _address < _addresses.length; _address++ ){ addressList.addAddress(_addresses[_address]); emit AddressAdded(_addresses[_address]); } } /// @notice Removes an address from the linked list. Can only be called by the Registry owner. /// @param _previousContract The address positionally located before the address that will be deleted. This may be the SENTINEL address if the list contains one contract address /// @param _address The address to remove from the linked list. function removeAddress(address _previousContract, address _address) public onlyOwner { addressList.removeAddress(_previousContract, _address); emit AddressRemoved(_address); } /// @notice Removes every address from the list function clearAll() public onlyOwner { addressList.clearAll(); emit AllAddressesCleared(); } /// @notice Determines whether the list contains the given address /// @param _addr The address to check /// @return True if the address is contained, false otherwise. function contains(address _addr) public returns (bool) { return addressList.contains(_addr); } /// @notice Gives the address at the start of the list /// @return The address at the start of the list function start() public view returns (address) { return addressList.addressMap[MappedSinglyLinkedList.SENTINEL]; } /// @notice Exposes the internal next() iterator /// @param current The current address /// @return Returns the next address in the list function next(address current) public view returns (address) { return addressList.addressMap[current]; } /// @notice Exposes the end of the list /// @return The sentinel address function end() public pure returns (address) { return MappedSinglyLinkedList.SENTINEL; } }
// SPDX-License-Identifier: GPL-3.0 pragma solidity ^0.7.6; /// @notice An efficient implementation of a singly linked list of addresses /// @dev A mapping(address => address) tracks the 'next' pointer. A special address called the SENTINEL is used to denote the beginning and end of the list. library MappedSinglyLinkedList { /// @notice The special value address used to denote the end of the list address public constant SENTINEL = address(0x1); /// @notice The data structure to use for the list. struct Mapping { uint256 count; mapping(address => address) addressMap; } /// @notice Initializes the list. /// @dev It is important that this is called so that the SENTINEL is correctly setup. function initialize(Mapping storage self) internal { require(self.count == 0, "Already init"); self.addressMap[SENTINEL] = SENTINEL; } function start(Mapping storage self) internal view returns (address) { return self.addressMap[SENTINEL]; } function next(Mapping storage self, address current) internal view returns (address) { return self.addressMap[current]; } function end(Mapping storage) internal pure returns (address) { return SENTINEL; } function addAddresses(Mapping storage self, address[] memory addresses) internal { for (uint256 i = 0; i < addresses.length; i++) { addAddress(self, addresses[i]); } } /// @notice Adds an address to the front of the list. /// @param self The Mapping struct that this function is attached to /// @param newAddress The address to shift to the front of the list function addAddress(Mapping storage self, address newAddress) internal { require(newAddress != SENTINEL && newAddress != address(0), "Invalid address"); require(self.addressMap[newAddress] == address(0), "Already added"); self.addressMap[newAddress] = self.addressMap[SENTINEL]; self.addressMap[SENTINEL] = newAddress; self.count = self.count + 1; } /// @notice Removes an address from the list /// @param self The Mapping struct that this function is attached to /// @param prevAddress The address that precedes the address to be removed. This may be the SENTINEL if at the start. /// @param addr The address to remove from the list. function removeAddress(Mapping storage self, address prevAddress, address addr) internal { require(addr != SENTINEL && addr != address(0), "Invalid address"); require(self.addressMap[prevAddress] == addr, "Invalid prevAddress"); self.addressMap[prevAddress] = self.addressMap[addr]; delete self.addressMap[addr]; self.count = self.count - 1; } /// @notice Determines whether the list contains the given address /// @param self The Mapping struct that this function is attached to /// @param addr The address to check /// @return True if the address is contained, false otherwise. function contains(Mapping storage self, address addr) internal view returns (bool) { return addr != SENTINEL && addr != address(0) && self.addressMap[addr] != address(0); } /// @notice Returns an address array of all the addresses in this list /// @dev Contains a for loop, so complexity is O(n) wrt the list size /// @param self The Mapping struct that this function is attached to /// @return An array of all the addresses function addressArray(Mapping storage self) internal view returns (address[] memory) { address[] memory array = new address[](self.count); uint256 count; address currentAddress = self.addressMap[SENTINEL]; while (currentAddress != address(0) && currentAddress != SENTINEL) { array[count] = currentAddress; currentAddress = self.addressMap[currentAddress]; count++; } return array; } /// @notice Removes every address from the list /// @param self The Mapping struct that this function is attached to function clearAll(Mapping storage self) internal { address currentAddress = self.addressMap[SENTINEL]; while (currentAddress != address(0) && currentAddress != SENTINEL) { address nextAddress = self.addressMap[currentAddress]; delete self.addressMap[currentAddress]; currentAddress = nextAddress; } self.addressMap[SENTINEL] = SENTINEL; self.count = 0; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.7.6; import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts-upgradeable/math/SafeMathUpgradeable.sol"; import "@pooltogether/pooltogether-generic-registry/contracts/AddressRegistry.sol"; import "./interfaces/IPod.sol"; import "./interfaces/KeeperCompatibleInterface.sol"; /// @notice Contract implements Chainlink's Upkeep system interface, automating the upkeep of a registry of Pod contracts contract PodsUpkeep is KeeperCompatibleInterface, Ownable { using SafeMathUpgradeable for uint256; /// @notice Address of the registry of pods contract which require upkeep AddressRegistry public podsRegistry; /// @dev Fixed length of the last upkeep block number (multiple this by 8 to get the maximum number of pods for storage) uint8 constant PODS_PACKED_ARRAY_SIZE = 10; uint256[PODS_PACKED_ARRAY_SIZE] internal lastUpkeepBlockNumber; /// @notice Global upkeep interval expressed in blocks at which pods.batch() will be called uint256 public upkeepBlockInterval; /// @notice Emitted when the upkeep block interval is updated event UpkeepBlockIntervalUpdated(uint upkeepBlockInterval); /// @notice Emitted when the upkeep max batch is updated event UpkeepBatchLimitUpdated(uint upkeepBatchLimit); /// @notice Emitted when the address registry is updated event PodsRegistryUpdated(AddressRegistry addressRegistry); /// @notice Maximum number of pods that performUpkeep can be called on uint256 public upkeepBatchLimit; /// @notice Contract Constructor. No initializer. constructor(AddressRegistry _podsRegistry, address _owner, uint256 _upkeepBlockInterval, uint256 _upkeepBatchLimit) Ownable() { podsRegistry = _podsRegistry; emit PodsRegistryUpdated(_podsRegistry); transferOwnership(_owner); upkeepBlockInterval = _upkeepBlockInterval; emit UpkeepBlockIntervalUpdated(_upkeepBlockInterval); upkeepBatchLimit = _upkeepBatchLimit; emit UpkeepBatchLimitUpdated(_upkeepBatchLimit); } /// @notice Updates a 256 bit word with a 32 bit representation of a block number at a particular index /// @param _existingUpkeepBlockNumbers The 256 word /// @param _podIndex The index within that word (0 to 7) /// @param _value The block number value to be inserted function _updateLastBlockNumberForPodIndex(uint256 _existingUpkeepBlockNumbers, uint8 _podIndex, uint32 _value) internal pure returns (uint256) { uint256 mask = (type(uint32).max | uint256(0)) << (_podIndex * 32); // get a mask of all 1's at the pod index uint256 updateBits = (uint256(0) | _value) << (_podIndex * 32); // position value at index with 0's in every other position /* (updateBits | ~mask) negation of the mask is 0's at the location of the block number, 1's everywhere else OR'ing it with updateBits will give 1's everywhere else, block number intact (_existingUpkeepBlockNumbers | mask) OR'ing the exstingUpkeepBlockNumbers with mask will give maintain other blocknumber, put all 1's at podIndex finally AND'ing the two halves will filter through 1's if they are supposed to be there */ return (updateBits | ~mask) & (_existingUpkeepBlockNumbers | mask); } /// @notice Takes a 256 bit word and 0 to 7 index within and returns the uint32 value at that index /// @param _existingUpkeepBlockNumbers The 256 word /// @param _podIndex The index within that word function _readLastBlockNumberForPodIndex(uint256 _existingUpkeepBlockNumbers, uint8 _podIndex) internal pure returns (uint32) { uint256 mask = (type(uint32).max | uint256(0)) << (_podIndex * 32); return uint32((_existingUpkeepBlockNumbers & mask) >> (_podIndex * 32)); } /// @notice Get the last Upkeep block number for a pod /// @param podIndex The position of the pod in the Registry function readLastBlockNumberForPodIndex(uint256 podIndex) public view returns (uint32) { uint256 wordIndex = podIndex / 8; return _readLastBlockNumberForPodIndex(lastUpkeepBlockNumber[wordIndex], uint8(podIndex % 8)); } /// @notice Checks if Pods require upkeep. Call in a static manner every block by the Chainlink Upkeep network. /// @param checkData Not used in this implementation. /// @return upkeepNeeded as true if performUpkeep() needs to be called, false otherwise. performData returned empty. function checkUpkeep(bytes calldata checkData) override external view returns (bool upkeepNeeded, bytes memory performData) { address[] memory pods = podsRegistry.getAddresses(); uint256 _upkeepBlockInterval = upkeepBlockInterval; uint256 podsLength = pods.length; for(uint256 podWord = 0; podWord <= podsLength / 8; podWord++){ uint256 _lastUpkeep = lastUpkeepBlockNumber[podWord]; // this performs the SLOAD for(uint256 i = 0; i + (podWord * 8) < podsLength; i++){ uint32 podLastUpkeepBlockNumber = _readLastBlockNumberForPodIndex(_lastUpkeep, uint8(i)); if(block.number > podLastUpkeepBlockNumber + _upkeepBlockInterval){ return (true, ""); } } } return (false, ""); } /// @notice Performs upkeep on the pods contract and updates lastUpkeepBlockNumbers /// @param performData Not used in this implementation. function performUpkeep(bytes calldata performData) override external { address[] memory pods = podsRegistry.getAddresses(); uint256 podsLength = pods.length; uint256 _batchLimit = upkeepBatchLimit; uint256 batchesPerformed = 0; for(uint8 podWord = 0; podWord <= podsLength / 8; podWord++){ // give word index uint256 _updateBlockNumber = lastUpkeepBlockNumber[podWord]; // this performs the SLOAD for(uint8 i = 0; i + (podWord * 8) < podsLength; i++){ // pod index within word if(batchesPerformed >= _batchLimit) { break; } // get the 32 bit block number from the 256 bit word uint32 podLastUpkeepBlockNumber = _readLastBlockNumberForPodIndex(_updateBlockNumber, i); if(block.number > podLastUpkeepBlockNumber + upkeepBlockInterval) { IPod(pods[i + (podWord * 8)]).drop(); batchesPerformed++; // updated pod's most recent upkeep block number and store update to that 256 bit word _updateBlockNumber = _updateLastBlockNumberForPodIndex(_updateBlockNumber, i, uint32(block.number)); } } lastUpkeepBlockNumber[podWord] = _updateBlockNumber; // update the entire 256 bit word at once } } /// @notice Updates the upkeepBlockInterval. Can only be called by the contract owner /// @param _upkeepBlockInterval The new upkeepBlockInterval (in blocks) function updateBlockUpkeepInterval(uint256 _upkeepBlockInterval) external onlyOwner { upkeepBlockInterval = _upkeepBlockInterval; emit UpkeepBlockIntervalUpdated(_upkeepBlockInterval); } /// @notice Updates the upkeep max batch. Can only be called by the contract owner /// @param _upkeepBatchLimit The new _upkeepBatchLimit function updateUpkeepBatchLimit(uint256 _upkeepBatchLimit) external onlyOwner { upkeepBatchLimit = _upkeepBatchLimit; emit UpkeepBatchLimitUpdated(_upkeepBatchLimit); } /// @notice Updates the address registry. Can only be called by the contract owner /// @param _addressRegistry The new podsRegistry function updatePodsRegistry(AddressRegistry _addressRegistry) external onlyOwner { podsRegistry = _addressRegistry; emit PodsRegistryUpdated(_addressRegistry); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.7.6; interface IPod { function drop() external returns (uint256); }
// SPDX-License-Identifier: MIT pragma solidity 0.7.6; interface KeeperCompatibleInterface { /** * @notice method that is simulated by the keepers to see if any work actually * needs to be performed. This method does does not actually need to be * executable, and since it is only ever simulated it can consume lots of gas. * @dev To ensure that it is never called, you may want to add the * cannotExecute modifier from KeeperBase to your implementation of this * method. * @param checkData specified in the upkeep registration so it is always the * same for a registered upkeep. This can easily be broken down into specific * arguments using `abi.decode`, so multiple upkeeps can be registered on the * same contract and easily differentiated by the contract. * @return upkeepNeeded boolean to indicate whether the keeper should call * performUpkeep or not. * @return performData bytes that the keeper should call performUpkeep with, if * upkeep is needed. If you would like to encode data to decode later, try * `abi.encode`. */ function checkUpkeep( bytes calldata checkData ) external returns ( bool upkeepNeeded, bytes memory performData ); /** * @notice method that is actually executed by the keepers, via the registry. * The data returned by the checkUpkeep simulation will be passed into * this method to actually be executed. * @dev The input to this method should not be trusted, and the caller of the * method should not even be restricted to any single registry. Anyone should * be able call it, and the input should be validated, there is no guarantee * that the data passed in is the performData returned from checkUpkeep. This * could happen due to malicious keepers, racing keepers, or simply a state * change while the performUpkeep transaction is waiting for confirmation. * Always validate the data passed in. * @param performData is the data which was passed back from the checkData * simulation. If it is encoded, it can easily be decoded into other types by * calling `abi.decode`. This data should not be trusted, and should be * validated against the contract's current state. */ function performUpkeep( bytes calldata performData ) external; }
{ "evmVersion": "istanbul", "libraries": {}, "metadata": { "bytecodeHash": "ipfs", "useLiteralContent": true }, "optimizer": { "enabled": true, "runs": 200 }, "remappings": [], "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "abi" ] } } }
[{"inputs":[{"internalType":"contract AddressRegistry","name":"_podsRegistry","type":"address"},{"internalType":"address","name":"_owner","type":"address"},{"internalType":"uint256","name":"_upkeepBlockInterval","type":"uint256"},{"internalType":"uint256","name":"_upkeepBatchLimit","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"contract AddressRegistry","name":"addressRegistry","type":"address"}],"name":"PodsRegistryUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"upkeepBatchLimit","type":"uint256"}],"name":"UpkeepBatchLimitUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"upkeepBlockInterval","type":"uint256"}],"name":"UpkeepBlockIntervalUpdated","type":"event"},{"inputs":[{"internalType":"bytes","name":"checkData","type":"bytes"}],"name":"checkUpkeep","outputs":[{"internalType":"bool","name":"upkeepNeeded","type":"bool"},{"internalType":"bytes","name":"performData","type":"bytes"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes","name":"performData","type":"bytes"}],"name":"performUpkeep","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"podsRegistry","outputs":[{"internalType":"contract AddressRegistry","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"podIndex","type":"uint256"}],"name":"readLastBlockNumberForPodIndex","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_upkeepBlockInterval","type":"uint256"}],"name":"updateBlockUpkeepInterval","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract AddressRegistry","name":"_addressRegistry","type":"address"}],"name":"updatePodsRegistry","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_upkeepBatchLimit","type":"uint256"}],"name":"updateUpkeepBatchLimit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"upkeepBatchLimit","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"upkeepBlockInterval","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"}]
Contract Creation Code
608060405234801561001057600080fd5b50604051610eb3380380610eb38339818101604052608081101561003357600080fd5b5080516020820151604083015160609093015191929091600061005461015f565b600080546001600160a01b0319166001600160a01b038316908117825560405192935091600080516020610e93833981519152908290a350600180546001600160a01b0386166001600160a01b0319909116811790915560408051918252517f18bd423236e58da961144fbace94c4bc95a7adc67dd2d7d85187a6fb194af92d9181900360200190a16100e683610163565b600c8290556040805183815290517f90a69c64360b7faa15b24f7c6bf7886de2ae300b4405895b8c4a4beabe67f7879181900360200190a1600d8190556040805182815290517f65a9cb1bf62947e219360bb6e0c9bdf929470e2d9c3dbd10600089b13ef7c0c89181900360200190a150505050610274565b3390565b61016b61015f565b6001600160a01b031661017c610265565b6001600160a01b0316146101d7576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b03811661021c5760405162461bcd60e51b8152600401808060200182810382526026815260200180610e6d6026913960400191505060405180910390fd5b600080546040516001600160a01b0380851693921691600080516020610e9383398151915291a3600080546001600160a01b0319166001600160a01b0392909216919091179055565b6000546001600160a01b031690565b610bea806102836000396000f3fe608060405234801561001057600080fd5b50600436106100b45760003560e01c8063715018a611610071578063715018a6146102a957806378fa10ac146102b157806385f5695c146102d75780638da5cb5b146102df578063cc112329146102e7578063f2fde38b14610304576100b4565b8063241a9a65146100b95780634043511c146100d35780634585e33b146100f257806358c81ec414610160578063691b166e146101845780636e04ff0d146101ba575b600080fd5b6100c161032a565b60408051918252519081900360200190f35b6100f0600480360360208110156100e957600080fd5b5035610330565b005b6100f06004803603602081101561010857600080fd5b810190602081018135600160201b81111561012257600080fd5b82018360208201111561013457600080fd5b803590602001918460018302840111600160201b8311171561015557600080fd5b5090925090506103cd565b610168610612565b604080516001600160a01b039092168252519081900360200190f35b6101a16004803603602081101561019a57600080fd5b5035610621565b6040805163ffffffff9092168252519081900360200190f35b610228600480360360208110156101d057600080fd5b810190602081018135600160201b8111156101ea57600080fd5b8201836020820111156101fc57600080fd5b803590602001918460018302840111600160201b8311171561021d57600080fd5b509092509050610649565b60405180831515815260200180602001828103825283818151815260200191508051906020019080838360005b8381101561026d578181015183820152602001610255565b50505050905090810190601f16801561029a5780820380516001836020036101000a031916815260200191505b50935050505060405180910390f35b6100f0610818565b6100f0600480360360208110156102c757600080fd5b50356001600160a01b03166108c4565b6100c161097a565b610168610980565b6100f0600480360360208110156102fd57600080fd5b503561098f565b6100f06004803603602081101561031a57600080fd5b50356001600160a01b0316610a2c565b600d5481565b610338610b2e565b6001600160a01b0316610349610980565b6001600160a01b031614610392576040805162461bcd60e51b81526020600482018190526024820152600080516020610b95833981519152604482015290519081900360640190fd5b600c8190556040805182815290517f90a69c64360b7faa15b24f7c6bf7886de2ae300b4405895b8c4a4beabe67f7879181900360200190a150565b600154604080516351cfd60960e11b815290516000926001600160a01b03169163a39fac129160048083019286929190829003018186803b15801561041157600080fd5b505afa158015610425573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052602081101561044e57600080fd5b8101908080516040519392919084600160201b82111561046d57600080fd5b90830190602082018581111561048257600080fd5b82518660208202830111600160201b8211171561049e57600080fd5b82525081516020918201928201910280838360005b838110156104cb5781810151838201526020016104b3565b5050505090500160405250505090506000815190506000600d5490506000805b600884048160ff161161060957600060028260ff16600a811061050a57fe5b0154905060005b8583600802820160ff1610156105ec5784841061052d576105ec565b60006105398383610b32565b9050600c548163ffffffff16014311156105e3578784600802830160ff168151811061056157fe5b60200260200101516001600160a01b031663f751cd8f6040518163ffffffff1660e01b8152600401602060405180830381600087803b1580156105a357600080fd5b505af11580156105b7573d6000803e3d6000fd5b505050506040513d60208110156105cd57600080fd5b50506001909401936105e0838343610b4b565b92505b50600101610511565b508060028360ff16600a81106105fe57fe5b0155506001016104eb565b50505050505050565b6001546001600160a01b031681565b600060088204610642600282600a811061063757fe5b015460088506610b32565b9392505050565b600060606000600160009054906101000a90046001600160a01b03166001600160a01b031663a39fac126040518163ffffffff1660e01b815260040160006040518083038186803b15801561069d57600080fd5b505afa1580156106b1573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405260208110156106da57600080fd5b8101908080516040519392919084600160201b8211156106f957600080fd5b90830190602082018581111561070e57600080fd5b82518660208202830111600160201b8211171561072a57600080fd5b82525081516020918201928201910280838360005b8381101561075757818101518382015260200161073f565b5050505090500160405250505090506000600c54905060008251905060005b6008820481116107f6576000600282600a811061078f57fe5b0154905060005b8383600802820110156107ec5760006107af8383610b32565b9050858163ffffffff16014311156107e3576001604051806020016040528060008152509850985050505050505050610811565b50600101610796565b5050600101610776565b50600060405180602001604052806000815250945094505050505b9250929050565b610820610b2e565b6001600160a01b0316610831610980565b6001600160a01b03161461087a576040805162461bcd60e51b81526020600482018190526024820152600080516020610b95833981519152604482015290519081900360640190fd5b600080546040516001600160a01b03909116907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600080546001600160a01b0319169055565b6108cc610b2e565b6001600160a01b03166108dd610980565b6001600160a01b031614610926576040805162461bcd60e51b81526020600482018190526024820152600080516020610b95833981519152604482015290519081900360640190fd5b600180546001600160a01b0383166001600160a01b0319909116811790915560408051918252517f18bd423236e58da961144fbace94c4bc95a7adc67dd2d7d85187a6fb194af92d9181900360200190a150565b600c5481565b6000546001600160a01b031690565b610997610b2e565b6001600160a01b03166109a8610980565b6001600160a01b0316146109f1576040805162461bcd60e51b81526020600482018190526024820152600080516020610b95833981519152604482015290519081900360640190fd5b600d8190556040805182815290517f65a9cb1bf62947e219360bb6e0c9bdf929470e2d9c3dbd10600089b13ef7c0c89181900360200190a150565b610a34610b2e565b6001600160a01b0316610a45610980565b6001600160a01b031614610a8e576040805162461bcd60e51b81526020600482018190526024820152600080516020610b95833981519152604482015290519081900360640190fd5b6001600160a01b038116610ad35760405162461bcd60e51b8152600401808060200182810382526026815260200180610b6f6026913960400191505060405180910390fd5b600080546040516001600160a01b03808516939216917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e091a3600080546001600160a01b0319166001600160a01b0392909216919091179055565b3390565b63ffffffff60ff602083021690811b8316901c92915050565b63ffffffff60ff6020939093029290921682811b801992909316901b179117169056fe4f776e61626c653a206e6577206f776e657220697320746865207a65726f20616464726573734f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572a26469706673582212200c20dcc521457b3c02d81ffa2e470c2a1f8cdc19f4f946a37af5dbe9182b3ecc64736f6c634300070600334f776e61626c653a206e6577206f776e657220697320746865207a65726f20616464726573738be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0000000000000000000000000e89f13fd3e5f13f49b6c4c48ae7104a01f2e70cf00000000000000000000000072c9aa4c753fc36cbf3d1ff6fec0bc44ad41d7f2000000000000000000000000000000000000000000000000000000000000000a0000000000000000000000000000000000000000000000000000000000000003
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000e89f13fd3e5f13f49b6c4c48ae7104a01f2e70cf00000000000000000000000072c9aa4c753fc36cbf3d1ff6fec0bc44ad41d7f2000000000000000000000000000000000000000000000000000000000000000a0000000000000000000000000000000000000000000000000000000000000003
-----Decoded View---------------
Arg [0] : _podsRegistry (address): 0xe89f13fd3e5f13f49b6c4c48ae7104a01f2e70cf
Arg [1] : _owner (address): 0x72c9aa4c753fc36cbf3d1ff6fec0bc44ad41d7f2
Arg [2] : _upkeepBlockInterval (uint256): 10
Arg [3] : _upkeepBatchLimit (uint256): 3
-----Encoded View---------------
4 Constructor Arguments found :
Arg [0] : 000000000000000000000000e89f13fd3e5f13f49b6c4c48ae7104a01f2e70cf
Arg [1] : 00000000000000000000000072c9aa4c753fc36cbf3d1ff6fec0bc44ad41d7f2
Arg [2] : 000000000000000000000000000000000000000000000000000000000000000a
Arg [3] : 0000000000000000000000000000000000000000000000000000000000000003
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.