编译器
0.8.20+commit.a1b79de6
文件 1 的 4:Context.sol
pragma solidity ^0.8.20;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
function _contextSuffixLength() internal view virtual returns (uint256) {
return 0;
}
}
文件 2 的 4:IERC20.sol
pragma solidity ^0.8.20;
interface IERC20 {
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address to, uint256 value) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 value) external returns (bool);
function transferFrom(address from, address to, uint256 value) external returns (bool);
}
文件 3 的 4:Ownable.sol
pragma solidity ^0.8.20;
import {Context} from "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
error OwnableUnauthorizedAccount(address account);
error OwnableInvalidOwner(address owner);
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor(address initialOwner) {
if (initialOwner == address(0)) {
revert OwnableInvalidOwner(address(0));
}
_transferOwnership(initialOwner);
}
modifier onlyOwner() {
_checkOwner();
_;
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
if (owner() != _msgSender()) {
revert OwnableUnauthorizedAccount(_msgSender());
}
}
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
if (newOwner == address(0)) {
revert OwnableInvalidOwner(address(0));
}
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 4 的 4:Staking.sol
pragma solidity ^0.8.19;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
contract Staking is Ownable {
struct Stake {
uint256 amount;
uint256 unlocksAt;
}
event Staked(
address staker,
uint256 amount
);
event Unstaked(
address staker,
uint256 amount
);
IERC20 public token;
uint256 public minStakingAmount;
uint256 public stakingDuration;
mapping(address => Stake) public stakes;
constructor(address _tokenAddress) Ownable(msg.sender) {
token = IERC20(_tokenAddress);
minStakingAmount = 0;
stakingDuration = 0;
}
function setStakingParameters(
uint256 _amount,
uint256 _duration
) external onlyOwner {
require(_amount > 0, "Amount must be greater than 0");
require(_duration > 0, "Duration must be greater than 0");
minStakingAmount = _amount;
stakingDuration = _duration;
}
function stake(uint256 _amount) external {
require(minStakingAmount > 0, "Staking amount not set");
require(stakingDuration > 0, "Staking duration not set");
require(_amount >= minStakingAmount, "Amount must be greater than or equal to the minimum staking amount");
Stake memory existingStake = stakes[msg.sender];
if (existingStake.amount > 0) {
require(block.timestamp >= existingStake.unlocksAt, "Existing stake has not expired");
stakes[msg.sender] = Stake({
amount: _amount,
unlocksAt: block.timestamp + stakingDuration
});
if (_amount > existingStake.amount) {
token.transferFrom(msg.sender, address(this), _amount - existingStake.amount);
} else {
token.transfer(msg.sender, existingStake.amount - _amount);
}
emit Unstaked(msg.sender, existingStake.amount);
} else {
stakes[msg.sender] = Stake({
amount: _amount,
unlocksAt: block.timestamp + stakingDuration
});
token.transferFrom(msg.sender, address(this), _amount);
}
emit Staked(msg.sender, _amount);
}
function unstake(uint256 _amount) external {
require(stakes[msg.sender].unlocksAt <= block.timestamp, "Stake has not expired");
require(stakes[msg.sender].amount > 0, "Stake does not exist");
require(stakes[msg.sender].amount >= _amount, "Amount must be less than or equal to the stake amount");
if (stakes[msg.sender].amount == _amount) {
delete stakes[msg.sender];
} else {
stakes[msg.sender].amount -= _amount;
}
token.transfer(msg.sender, _amount);
emit Unstaked(msg.sender, _amount);
}
}
{
"compilationTarget": {
"Staking.sol": "Staking"
},
"evmVersion": "shanghai",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_tokenAddress","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"OwnableInvalidOwner","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"OwnableUnauthorizedAccount","type":"error"},{"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":"address","name":"staker","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Staked","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"staker","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Unstaked","type":"event"},{"inputs":[],"name":"minStakingAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"uint256","name":"_duration","type":"uint256"}],"name":"setStakingParameters","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"stake","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"stakes","outputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"unlocksAt","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stakingDuration","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"token","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"unstake","outputs":[],"stateMutability":"nonpayable","type":"function"}]