// SPDX-License-Identifier: MIT
pragma solidity ^0.8.4;
/// @dev Note: the ERC-165 identifier for this interface is 0x150b7a02.
interface IERC721TokenReceiver {
/// @notice Handle the receipt of an NFT
/// @dev The ERC721 smart contract calls this function on the recipient
/// after a `transfer`. This function MAY throw to revert and reject the
/// transfer. Return of other than the magic value MUST result in the
/// transaction being reverted.
/// Note: the contract address is always the message sender.
/// @param _operator The address which called `safeTransferFrom` function
/// @param _from The address which previously owned the token
/// @param _tokenId The NFT identifier which is being transferred
/// @param _data Additional data with no specified format
/// @return `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`
/// unless throwing
function onERC721Received(address _operator, address _from, uint256 _tokenId, bytes calldata _data) external returns(bytes4);
}
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.9;
interface IMarsMetadata {
function tokenURI(uint _tokenId,bytes32 _hash, uint _supplyAtMint, uint _opened) external view returns (string memory);
}
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.9;
import "./interfaces/IERC721TokenReceiver.sol";
import "./interfaces/IMarsMetadata.sol";
contract MarsMiners {
constructor(){
supportsInterface[0x80ac58cd] = true;
supportsInterface[0x5b5e139f] = true;
supportsInterface[0x01ffc9a7] = true;
owner = msg.sender;
_mint(1,msg.sender,bytes32(0));
mintCost = BASE_COSTS[0] + CREATOR_FEE;
}
uint256[8] BASE_COSTS = [
0.000010 ether,
0.000020 ether,
0.000025 ether,
0.000030 ether,
0.000040 ether,
0.000050 ether,
0.000070 ether,
0.000100 ether
];
uint constant BASE_DIFFICULTY = type(uint).max/uint(50000 * 300);
uint constant DIFFICULTY_RAMP = uint(50000 * 0.1);
uint constant CREATOR_FEE = 0.005 ether;
bytes32[] tokens;
uint public closed;
uint public ownerWithdrawn;
address public owner;
uint public mintCost;
mapping( uint => uint) public supplyAtMint;
event OpenMine( uint _tokenId, bytes32 _hash, address _miner, uint _newSupply, uint _newMintCost, uint _blockNumber);
event CloseMine(uint _tokenId, bytes32 _hash, uint _excavated, uint _supplyAtMint, uint _newSupply, uint _newMintCost, uint _blockNumber);
function getMineType(bytes32 hash, uint _supplyAtMint) public pure returns(uint){
uint mineTypeMax;
if(_supplyAtMint < 500){
mineTypeMax = 2;
}else if(_supplyAtMint < 1000){
mineTypeMax = 3;
}else if(_supplyAtMint < 1500){
mineTypeMax = 4;
}else if(_supplyAtMint < 2000){
mineTypeMax = 5;
}else if(_supplyAtMint < 2500){
mineTypeMax = 6;
}else{
mineTypeMax = 7;
}
return (uint(hash)%100)**3 *(mineTypeMax + 1) / 1000000;
}
function openMine(uint nonce) public payable {
uint tokenId = tokens.length + 1;
uint supply = totalSupply();
uint difficulty = BASE_DIFFICULTY - (DIFFICULTY_RAMP * supply);
bytes32 hash = keccak256(abi.encodePacked(
msg.sender,
tokens[tokens.length - 1],
nonce
));
require(uint(hash) < difficulty,"difficulty");
require(msg.value == mintCost,"cost");
supplyAtMint[tokenId] = supply;
hash = keccak256(abi.encodePacked(hash,block.timestamp));
_mint(tokenId,msg.sender,hash);
mintCost += BASE_COSTS[getMineType(hash,supply)];
emit OpenMine(tokenId, hash, msg.sender, totalSupply(), mintCost, block.number);
}
function closeMine(uint tokenId) public{
payable(msg.sender).transfer(_closeMine(tokenId));
}
function _closeMine(uint tokenId) private returns(uint){
require(msg.sender == ownerOf(tokenId),"ownerOf");
uint excavated = (tokens.length - tokenId);
uint BASE_COST = BASE_COSTS[getMineType(hashOf(tokenId),supplyAtMint[tokenId])];
uint earnings = excavated * BASE_COST;
closed++;
_burn(tokenId);
mintCost -= BASE_COST;
emit CloseMine(tokenId, tokens[tokenId - 1], excavated, supplyAtMint[tokenId], totalSupply(), mintCost, block.number);
return earnings;
}
function closeMultiple(uint[] calldata tokenIds) public{
require(tokenIds.length > 0,"tokenIds");
uint total;
for(uint i = 0; i < tokenIds.length; i++){
total += _closeMine(tokenIds[i]);
}
payable(msg.sender).transfer(total);
}
function hashOf(uint _tokenId) public view returns(bytes32){
require(isValidToken(_tokenId),"invalid");
return tokens[_tokenId - 1];
}
function getEthContained(uint _tokenId) public view returns(uint){
require(isValidToken(_tokenId),"invalid");
uint BASE_COST = BASE_COSTS[getMineType(hashOf(_tokenId),supplyAtMint[_tokenId])];
return (tokens.length - _tokenId) * BASE_COST;
}
function getLastHash() public view returns(bytes32){
return tokens[tokens.length - 1];
}
function getMiningState() public view returns(bytes32 _hash, uint _supply, uint _closed, uint blockNumber){
return (tokens[tokens.length - 1], totalSupply(), closed, block.number);
}
function mineData(uint _tokenId) public view returns(bytes32 _hash, uint _supplyAtMint, uint _opened){
require(isValidToken(_tokenId),"invalid");
return (tokens[_tokenId - 1],supplyAtMint[_tokenId],tokens.length);
}
event Transfer(address indexed _from, address indexed _to, uint256 indexed _tokenId);
event Approval(address indexed _owner, address indexed _approved, uint256 indexed _tokenId);
event ApprovalForAll(address indexed _owner, address indexed _operator, bool _approved);
mapping(address => uint256) public balanceOf;
mapping (uint256 => address) internal allowance;
mapping (address => mapping (address => bool)) public isApprovedForAll;
mapping(uint256 => address) owners;
string public name = "Mars Mining Company";
string public symbol = "MARS";
address private __metadata;
address constant VB = 0xd8dA6BF26964aF9D7eEd9e03E53415D37aA96045;
function _mint(uint _tokenId,address _to, bytes32 _hash) private{
owners[_tokenId] = msg.sender;
balanceOf[_to]++;
tokens.push(_hash);
emit Transfer(address(0),VB,_tokenId);
emit Transfer(VB,_to,_tokenId);
}
function _burn(uint _tokenId) private{
address _owner = owners[_tokenId];
balanceOf[ _owner ]--;
delete owners[_tokenId];
emit Transfer(_owner,address(0),_tokenId);
}
function isValidToken(uint256 _tokenId) internal view returns(bool){
return owners[_tokenId] != address(0);
}
function ownerOf(uint256 _tokenId) public view returns(address){
require(isValidToken(_tokenId),"invalid");
return owners[_tokenId];
}
function approve(address _approved, uint256 _tokenId) external{
address _owner = ownerOf(_tokenId);
require( _owner == msg.sender || isApprovedForAll[_owner][msg.sender],"permission");
emit Approval(_owner, _approved, _tokenId);
allowance[_tokenId] = _approved;
}
function getApproved(uint256 _tokenId) external view returns (address) {
require(isValidToken(_tokenId),"invalid");
return allowance[_tokenId];
}
function setApprovalForAll(address _operator, bool _approved) external {
emit ApprovalForAll(msg.sender,_operator, _approved);
isApprovedForAll[msg.sender][_operator] = _approved;
}
function transferFrom(address _from, address _to, uint256 _tokenId) public {
address _owner = ownerOf(_tokenId);
require ( _owner == msg.sender || allowance[_tokenId] == msg.sender || isApprovedForAll[_owner][msg.sender],"permission");
require(_owner == _from,"owner");
require(_to != address(0),"zero");
emit Transfer(_from, _to, _tokenId);
owners[_tokenId] =_to;
balanceOf[_from]--;
balanceOf[_to]++;
if(allowance[_tokenId] != address(0)){
delete allowance[_tokenId];
}
}
function safeTransferFrom(address _from, address _to, uint256 _tokenId, bytes memory data) public {
transferFrom(_from, _to, _tokenId);
uint32 size;
assembly {
size := extcodesize(_to)
}
if(size > 0){
IERC721TokenReceiver receiver = IERC721TokenReceiver(_to);
require(receiver.onERC721Received(msg.sender,_from,_tokenId,data) == bytes4(keccak256("onERC721Received(address,address,uint256,bytes)")),"receiver");
}
}
function safeTransferFrom(address _from, address _to, uint256 _tokenId) external {
safeTransferFrom(_from,_to,_tokenId,"");
}
function tokenURI(uint256 _tokenId) public view returns (string memory){
require(isValidToken(_tokenId),'tokenId');
return IMarsMetadata(__metadata).tokenURI(
_tokenId,
tokens[_tokenId-1],
supplyAtMint[_tokenId],
tokens.length);
}
function totalSupply() public view returns (uint256){
return tokens.length - closed;
}
mapping (bytes4 => bool) public supportsInterface;
function setOwner(address newOwner) public{
require(msg.sender == owner,"owner");
owner = newOwner;
}
function setMetadata(address _metadata) public{
require(msg.sender == owner,"owner");
__metadata = _metadata;
}
function ownerWithdraw() public{
require(msg.sender == owner,"owner");
uint toWithdraw = (tokens.length - ownerWithdrawn - 1) * CREATOR_FEE ;
require(toWithdraw > 0,"withdrawn");
ownerWithdrawn = tokens.length - 1;
payable(msg.sender).transfer(toWithdraw);
}
}
{
"compilationTarget": {
"contracts/MarsMiners.sol": "MarsMiners"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_owner","type":"address"},{"indexed":true,"internalType":"address","name":"_approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_owner","type":"address"},{"indexed":true,"internalType":"address","name":"_operator","type":"address"},{"indexed":false,"internalType":"bool","name":"_approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_tokenId","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"_hash","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"_excavated","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_supplyAtMint","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_newSupply","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_newMintCost","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_blockNumber","type":"uint256"}],"name":"CloseMine","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_tokenId","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"_hash","type":"bytes32"},{"indexed":false,"internalType":"address","name":"_miner","type":"address"},{"indexed":false,"internalType":"uint256","name":"_newSupply","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_newMintCost","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_blockNumber","type":"uint256"}],"name":"OpenMine","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_from","type":"address"},{"indexed":true,"internalType":"address","name":"_to","type":"address"},{"indexed":true,"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"_approved","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"closeMine","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"}],"name":"closeMultiple","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"closed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"getEthContained","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getLastHash","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"hash","type":"bytes32"},{"internalType":"uint256","name":"_supplyAtMint","type":"uint256"}],"name":"getMineType","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"getMiningState","outputs":[{"internalType":"bytes32","name":"_hash","type":"bytes32"},{"internalType":"uint256","name":"_supply","type":"uint256"},{"internalType":"uint256","name":"_closed","type":"uint256"},{"internalType":"uint256","name":"blockNumber","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"hashOf","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"mineData","outputs":[{"internalType":"bytes32","name":"_hash","type":"bytes32"},{"internalType":"uint256","name":"_supplyAtMint","type":"uint256"},{"internalType":"uint256","name":"_opened","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mintCost","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"nonce","type":"uint256"}],"name":"openMine","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ownerWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"ownerWithdrawn","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_operator","type":"address"},{"internalType":"bool","name":"_approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_metadata","type":"address"}],"name":"setMetadata","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"setOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"supplyAtMint","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"}]