编译器
0.8.19+commit.7dd6d404
文件 1 的 3:IMint.sol
pragma solidity 0.8.19;
interface IMint {
function mint(address to, uint256 amount) external;
}
文件 2 的 3:NonERC20.sol
pragma solidity >=0.8.0;
abstract contract NonERC20 {
event Transfer(address indexed from, address indexed to, uint256 amount);
string public name;
string public symbol;
uint8 public immutable decimals;
uint256 public totalSupply;
mapping(address => uint256) public balanceOf;
constructor(
string memory _name,
string memory _symbol,
uint8 _decimals
) {
name = _name;
symbol = _symbol;
decimals = _decimals;
}
function _mint(address to, uint256 amount) internal virtual {
totalSupply += amount;
unchecked {
balanceOf[to] += amount;
}
emit Transfer(address(0), to, amount);
}
function _burn(address from, uint256 amount) internal virtual {
balanceOf[from] -= amount;
unchecked {
totalSupply -= amount;
}
emit Transfer(from, address(0), amount);
}
}
文件 3 的 3:esZTH.sol
pragma solidity 0.8.19;
import "./interfaces/IMint.sol";
import "./NonERC20.sol";
contract ESZTH is NonERC20("Zenith Escrowed Token", "esZTH", 18) {
uint256 private constant _SLEEP_TIMES = 3 days;
uint256 private constant _MAX_LOCK_TIMES = 14 days;
address private constant _DEAD = 0x000000000000000000000000000000000000dEaD;
address public immutable MASTERCHEF;
address public immutable TREASURY;
address public immutable ZTHTOKEN;
LockInfo[] private _lockPositions;
struct LockInfo {
address holder;
uint256 amount;
uint256 unlockTime;
}
constructor(address masterChef, address treasury, address zthToken) {
if (masterChef == address(0)) revert AddressIsZero();
if (treasury == address(0)) revert AddressIsZero();
if (zthToken == address(0)) revert AddressIsZero();
MASTERCHEF = masterChef;
TREASURY = treasury;
ZTHTOKEN = zthToken;
}
function mint(address to, uint256 amount) external onlyMasterChef {
_mint(to, amount);
}
function swap(uint256 amount, uint256 lockTimes) external {
_burn(msg.sender, amount);
uint256 amountOut = getAmountOut(amount, lockTimes);
uint256 loss;
unchecked {
loss = amount - amountOut;
}
if (loss > 0) {
IMint(ZTHTOKEN).mint(_DEAD, loss / 2);
IMint(ZTHTOKEN).mint(TREASURY, loss / 2);
}
uint256 lid = _lockPositions.length;
unchecked {
_lockPositions.push(LockInfo(msg.sender, amountOut, block.timestamp + lockTimes));
}
emit Swap(msg.sender, lid, amount, lockTimes, amountOut);
}
function redeem(uint256 lockId) public {
LockInfo storage lockInfo = _lockPositions[lockId];
if (lockInfo.unlockTime > block.timestamp) revert UnlockTimeNotArrived();
uint256 amount = lockInfo.amount;
if (amount == 0) revert LockedAmountIsZero();
lockInfo.amount = 0;
IMint(ZTHTOKEN).mint(lockInfo.holder, amount);
emit Redeem(lockInfo.holder, lockId, amount);
}
function batchRedeem(uint256[] calldata lockIds) external {
for (uint256 i = 0; i < lockIds.length; i++) {
redeem(lockIds[i]);
}
}
function getAmountOut(uint256 amount, uint256 lockTimes) public pure returns (uint256) {
if (lockTimes < _SLEEP_TIMES || lockTimes > _MAX_LOCK_TIMES) revert InvalidLockTimes();
unchecked {
return (amount * 2 + amount * 8 * (lockTimes - _SLEEP_TIMES) / (_MAX_LOCK_TIMES - _SLEEP_TIMES)) / 10;
}
}
function getLockPosition(uint256 lockId) external view returns (LockInfo memory) {
return _lockPositions[lockId];
}
modifier onlyMasterChef() {
if (msg.sender != MASTERCHEF) revert OnlyCallByMasterChef();
_;
}
event Swap(address indexed holder, uint256 lockId, uint256 amount, uint256 lockTimes, uint256 amountOut);
event Redeem(address indexed holder, uint256 lockId, uint256 amount);
error AddressIsZero();
error OnlyCallByMasterChef();
error InvalidLockTimes();
error LockedAmountIsZero();
error UnlockTimeNotArrived();
}
{
"compilationTarget": {
"src/esZTH.sol": "ESZTH"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 20000
},
"remappings": [
":@openzeppelin/=lib/openzeppelin-contracts/",
":@solmate/=lib/solmate/src/",
":ds-test/=lib/forge-std/lib/ds-test/src/",
":forge-std/=lib/forge-std/src/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/",
":solmate/=lib/solmate/src/"
]
}
[{"inputs":[{"internalType":"address","name":"masterChef","type":"address"},{"internalType":"address","name":"treasury","type":"address"},{"internalType":"address","name":"zthToken","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"AddressIsZero","type":"error"},{"inputs":[],"name":"InvalidLockTimes","type":"error"},{"inputs":[],"name":"LockedAmountIsZero","type":"error"},{"inputs":[],"name":"OnlyCallByMasterChef","type":"error"},{"inputs":[],"name":"UnlockTimeNotArrived","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"holder","type":"address"},{"indexed":false,"internalType":"uint256","name":"lockId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Redeem","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"holder","type":"address"},{"indexed":false,"internalType":"uint256","name":"lockId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"lockTimes","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amountOut","type":"uint256"}],"name":"Swap","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"MASTERCHEF","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"TREASURY","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ZTHTOKEN","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"lockIds","type":"uint256[]"}],"name":"batchRedeem","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"lockTimes","type":"uint256"}],"name":"getAmountOut","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"lockId","type":"uint256"}],"name":"getLockPosition","outputs":[{"components":[{"internalType":"address","name":"holder","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"unlockTime","type":"uint256"}],"internalType":"struct ESZTH.LockInfo","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"lockId","type":"uint256"}],"name":"redeem","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"lockTimes","type":"uint256"}],"name":"swap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"}]