编译器
0.8.28+commit.7893614a
文件 1 的 6: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 的 6: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 的 6: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 的 6:Pausable.sol
pragma solidity ^0.8.20;
import {Context} from "../utils/Context.sol";
abstract contract Pausable is Context {
bool private _paused;
event Paused(address account);
event Unpaused(address account);
error EnforcedPause();
error ExpectedPause();
constructor() {
_paused = false;
}
modifier whenNotPaused() {
_requireNotPaused();
_;
}
modifier whenPaused() {
_requirePaused();
_;
}
function paused() public view virtual returns (bool) {
return _paused;
}
function _requireNotPaused() internal view virtual {
if (paused()) {
revert EnforcedPause();
}
}
function _requirePaused() internal view virtual {
if (!paused()) {
revert ExpectedPause();
}
}
function _pause() internal virtual whenNotPaused {
_paused = true;
emit Paused(_msgSender());
}
function _unpause() internal virtual whenPaused {
_paused = false;
emit Unpaused(_msgSender());
}
}
文件 5 的 6:ReentrancyGuard.sol
pragma solidity ^0.8.20;
abstract contract ReentrancyGuard {
uint256 private constant NOT_ENTERED = 1;
uint256 private constant ENTERED = 2;
uint256 private _status;
error ReentrancyGuardReentrantCall();
constructor() {
_status = NOT_ENTERED;
}
modifier nonReentrant() {
_nonReentrantBefore();
_;
_nonReentrantAfter();
}
function _nonReentrantBefore() private {
if (_status == ENTERED) {
revert ReentrancyGuardReentrantCall();
}
_status = ENTERED;
}
function _nonReentrantAfter() private {
_status = NOT_ENTERED;
}
function _reentrancyGuardEntered() internal view returns (bool) {
return _status == ENTERED;
}
}
文件 6 的 6:crai_migration_dapp.sol
pragma solidity ^0.8.28;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/utils/Pausable.sol";
import "@openzeppelin/contracts/utils/ReentrancyGuard.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
contract CraiMigrationDapp is Ownable, Pausable, ReentrancyGuard {
error BridgeClosed();
error InvalidAmount();
error NoClaimableTokens();
error ClaimPeriodEnded();
error TransferFailed();
error NotInSnapshot();
error ExceedsSnapshotBalance();
error V2NotSet();
error ClaimNotStarted();
error InvalidAddress();
error InvalidPeriod();
IERC20 public immutable v1Token;
IERC20 public v2Token;
uint256 public claimEnd;
bool public claimPhase;
uint256 public totalBridged;
uint256 public totalClaimed;
mapping(address => uint256) public snapshotBalances;
mapping(address => uint256) public deposited;
mapping(address => uint256) public hasClaimed;
event TokensDeposited(address indexed user, uint256 amount);
event TokensClaimed(address indexed user, uint256 amount);
event V2TokenSet(address indexed token);
event ClaimPhaseStarted(uint256 endTime);
event SnapshotUpdated(address indexed holder, uint256 amount);
event V1TokensWithdrawn(uint256 amount);
event V2TokensWithdrawn(uint256 amount);
event ClaimPeriodUpdated(uint256 newEndTime);
constructor(address _v1Token, address initialOwner) Ownable(initialOwner) {
v1Token = IERC20(_v1Token);
}
function deposit(uint256 amount) external whenNotPaused nonReentrant {
if (amount == 0) revert InvalidAmount();
uint256 snapshotBalance = snapshotBalances[msg.sender];
if (snapshotBalance == 0) revert NotInSnapshot();
uint256 totalDeposited = deposited[msg.sender] + amount;
if (totalDeposited > snapshotBalance) revert ExceedsSnapshotBalance();
deposited[msg.sender] = totalDeposited;
totalBridged += amount;
bool success = v1Token.transferFrom(msg.sender, address(this), amount);
if (!success) revert TransferFailed();
emit TokensDeposited(msg.sender, amount);
}
function claim() external whenNotPaused nonReentrant {
if (address(v2Token) == address(0)) revert V2NotSet();
if (!claimPhase) revert ClaimNotStarted();
if (block.timestamp > claimEnd) revert ClaimPeriodEnded();
if (hasClaimed[msg.sender] == deposited[msg.sender])
revert NoClaimableTokens();
uint256 amount = deposited[msg.sender] - hasClaimed[msg.sender];
if (amount == 0) revert NoClaimableTokens();
hasClaimed[msg.sender] += amount;
totalClaimed += amount;
bool success = v2Token.transfer(msg.sender, amount);
if (!success) revert TransferFailed();
emit TokensClaimed(msg.sender, amount);
}
function bulkUpdateSnapshot(
address[] calldata holders,
uint256[] calldata amounts
) external onlyOwner {
if (holders.length != amounts.length) revert InvalidAmount();
for (uint256 i = 0; i < holders.length; i++) {
if (holders[i] == address(0)) revert InvalidAddress();
snapshotBalances[holders[i]] = amounts[i];
emit SnapshotUpdated(holders[i], amounts[i]);
}
}
function updateSnapshotBalance(
address holder,
uint256 amount
) external onlyOwner {
if (holder == address(0)) revert InvalidAddress();
snapshotBalances[holder] = amount;
emit SnapshotUpdated(holder, amount);
}
function setV2Token(address _v2Token) external onlyOwner {
v2Token = IERC20(_v2Token);
emit V2TokenSet(_v2Token);
}
function startClaimPhase(uint256 claimPeriod) external onlyOwner {
if (address(v2Token) == address(0)) revert V2NotSet();
if (claimPeriod == 0) revert InvalidPeriod();
claimPhase = true;
claimEnd = block.timestamp + claimPeriod;
emit ClaimPhaseStarted(claimEnd);
}
function updateClaimPeriod(uint256 newEndTime) external onlyOwner {
if (newEndTime <= block.timestamp) revert InvalidPeriod();
claimEnd = newEndTime;
emit ClaimPeriodUpdated(newEndTime);
}
function withdrawV1Tokens() external onlyOwner {
uint256 balance = v1Token.balanceOf(address(this));
bool success = v1Token.transfer(msg.sender, balance);
if (!success) revert TransferFailed();
emit V1TokensWithdrawn(balance);
}
function withdrawV2Tokens() external onlyOwner {
if (address(v2Token) == address(0)) revert V2NotSet();
uint256 balance = v2Token.balanceOf(address(this));
bool success = v2Token.transfer(msg.sender, balance);
if (!success) revert TransferFailed();
emit V2TokensWithdrawn(balance);
}
function pause() external onlyOwner {
_pause();
}
function unpause() external onlyOwner {
_unpause();
}
function getSnapshotBalance(address user) external view returns (uint256) {
return snapshotBalances[user];
}
function getDepositedAmount(address user) external view returns (uint256) {
return deposited[user];
}
function hasUserClaimed(address user) external view returns (uint256) {
return hasClaimed[user];
}
}
{
"compilationTarget": {
"contracts/crai_migration_dapp.sol": "CraiMigrationDapp"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_v1Token","type":"address"},{"internalType":"address","name":"initialOwner","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"BridgeClosed","type":"error"},{"inputs":[],"name":"ClaimNotStarted","type":"error"},{"inputs":[],"name":"ClaimPeriodEnded","type":"error"},{"inputs":[],"name":"EnforcedPause","type":"error"},{"inputs":[],"name":"ExceedsSnapshotBalance","type":"error"},{"inputs":[],"name":"ExpectedPause","type":"error"},{"inputs":[],"name":"InvalidAddress","type":"error"},{"inputs":[],"name":"InvalidAmount","type":"error"},{"inputs":[],"name":"InvalidPeriod","type":"error"},{"inputs":[],"name":"NoClaimableTokens","type":"error"},{"inputs":[],"name":"NotInSnapshot","type":"error"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"OwnableInvalidOwner","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"OwnableUnauthorizedAccount","type":"error"},{"inputs":[],"name":"ReentrancyGuardReentrantCall","type":"error"},{"inputs":[],"name":"TransferFailed","type":"error"},{"inputs":[],"name":"V2NotSet","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"newEndTime","type":"uint256"}],"name":"ClaimPeriodUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"endTime","type":"uint256"}],"name":"ClaimPhaseStarted","type":"event"},{"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":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"holder","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"SnapshotUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"TokensClaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"TokensDeposited","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"V1TokensWithdrawn","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"token","type":"address"}],"name":"V2TokenSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"V2TokensWithdrawn","type":"event"},{"inputs":[{"internalType":"address[]","name":"holders","type":"address[]"},{"internalType":"uint256[]","name":"amounts","type":"uint256[]"}],"name":"bulkUpdateSnapshot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"claim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"claimEnd","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimPhase","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"deposit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"deposited","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"getDepositedAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"getSnapshotBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"hasClaimed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"hasUserClaimed","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":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_v2Token","type":"address"}],"name":"setV2Token","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"snapshotBalances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"claimPeriod","type":"uint256"}],"name":"startClaimPhase","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"totalBridged","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalClaimed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newEndTime","type":"uint256"}],"name":"updateClaimPeriod","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"holder","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"updateSnapshotBalance","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"v1Token","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"v2Token","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdrawV1Tokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawV2Tokens","outputs":[],"stateMutability":"nonpayable","type":"function"}]