编译器
0.8.20+commit.a1b79de6
文件 1 的 7:4_RR_bridge.sol
pragma solidity ^0.8.20;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721Receiver.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
contract RRBridge is
IERC721Receiver,
Ownable
{
address public secret;
mapping(bytes => bool) public usedSignatures;
mapping(address => mapping(uint256 => address)) public tokenSender;
event WithdrawETH(uint256 amount, address recipient, address operator);
event WithdrawERC20(
address token,
uint256 amount,
address recipient,
address operator
);
event WithdrawNFT(
address token,
uint256 [] tokenIds,
address recipient,
address operator
);
event NftReceived(
address from,
uint256 [] tokenId,
address nftAddress
);
event SetSigner(address signer);
constructor() Ownable(msg.sender) {}
function startBridge(
uint256 [] memory ids,
address contractAddress
) external {
IERC721 contractInstance = IERC721(contractAddress);
for (uint256 i = 0; i < ids.length; i++) {
uint256 tokenId = ids[i];
require(
contractInstance.ownerOf(tokenId) == msg.sender,
"startBridge: sender is not the owner"
);
contractInstance.safeTransferFrom(msg.sender, address(this), tokenId);
tokenSender[contractAddress][tokenId] = msg.sender;
}
emit NftReceived(msg.sender, ids, contractAddress);
}
function finishBridge(
address tokenAddress,
uint256 [] memory tokenIds,
address recipient,
uint256 checkId,
bytes memory signature
) external {
require(block.timestamp < checkId, "finishBridge: Signature expired");
bytes32 hash = keccak256(
abi.encode(
tokenAddress,
tokenIds,
recipient,
checkId,
"finishBridge"
)
);
_handleSignature(hash, signature);
IERC721 contractInstance = IERC721(tokenAddress);
for (uint256 i = 0; i < tokenIds.length; i++) {
uint256 tokenId = tokenIds[i];
require(
contractInstance.ownerOf(tokenId) == address(this),
"finishBridge: Not enough ERC721 tokens in reserve to withdraw"
);
contractInstance.safeTransferFrom(address(this), recipient, tokenId);
}
emit WithdrawNFT(tokenAddress, tokenIds, recipient, msg.sender);
}
function _handleSignature(bytes32 hash, bytes memory signature) internal {
require(
_verifyHashSignature(hash, signature),
"handleSignature: Invalid signature"
);
require(
!usedSignatures[signature],
"handleSignature: Signature already used"
);
usedSignatures[signature] = true;
}
function emergencyWithdrawETH() external onlyOwner {
uint256 contractBalance = address(this).balance;
payable(msg.sender).transfer(contractBalance);
emit WithdrawETH(contractBalance, msg.sender, msg.sender);
}
function emergencyWithdrawERC20(address token) external onlyOwner {
uint256 contractBalance = IERC20(token).balanceOf(address(this));
require(
contractBalance > 0,
"emergencyWithdrawERC20: No tokens to withdraw"
);
IERC20(token).transfer(msg.sender, contractBalance);
emit WithdrawERC20(token, contractBalance, msg.sender, msg.sender);
}
function emergencyWithdrawNFT(
address token,
uint256 tokenId
) external onlyOwner {
IERC721(token).safeTransferFrom(address(this), msg.sender, tokenId);
}
function setSigner(address _signer) external onlyOwner {
secret = _signer;
emit SetSigner(_signer);
}
function _verifyHashSignature(
bytes32 freshHash,
bytes memory signature
) internal view returns (bool) {
bytes32 hash = keccak256(
abi.encodePacked("\x19Ethereum Signed Message:\n32", freshHash)
);
bytes32 r;
bytes32 s;
uint8 v;
if (signature.length != 65) {
return false;
}
assembly {
r := mload(add(signature, 32))
s := mload(add(signature, 64))
v := byte(0, mload(add(signature, 96)))
}
if (v < 27) {
v += 27;
}
address signer = address(0);
if (v == 27 || v == 28) {
signer = ecrecover(hash, v, r, s);
}
return secret == signer;
}
function onERC721Received(
address,
address from,
uint256 tokenId,
bytes memory
) external virtual override returns (bytes4) {
tokenSender[msg.sender][tokenId] = from;
uint256 [] memory ids = new uint256[](1);
ids[0] = tokenId;
emit NftReceived(from, ids, msg.sender);
return this.onERC721Received.selector;
}
}
文件 2 的 7: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;
}
}
文件 3 的 7:IERC165.sol
pragma solidity ^0.8.20;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 4 的 7: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);
}
文件 5 的 7:IERC721.sol
pragma solidity ^0.8.20;
import {IERC165} from "../../utils/introspection/IERC165.sol";
interface IERC721 is IERC165 {
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);
function balanceOf(address owner) external view returns (uint256 balance);
function ownerOf(uint256 tokenId) external view returns (address owner);
function safeTransferFrom(address from, address to, uint256 tokenId, bytes calldata data) external;
function safeTransferFrom(address from, address to, uint256 tokenId) external;
function transferFrom(address from, address to, uint256 tokenId) external;
function approve(address to, uint256 tokenId) external;
function setApprovalForAll(address operator, bool approved) external;
function getApproved(uint256 tokenId) external view returns (address operator);
function isApprovedForAll(address owner, address operator) external view returns (bool);
}
文件 6 的 7:IERC721Receiver.sol
pragma solidity ^0.8.20;
interface IERC721Receiver {
function onERC721Received(
address operator,
address from,
uint256 tokenId,
bytes calldata data
) external returns (bytes4);
}
文件 7 的 7: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);
}
}
{
"compilationTarget": {
"contracts/4_RR_bridge.sol": "RRBridge"
},
"evmVersion": "shanghai",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[],"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":false,"internalType":"address","name":"from","type":"address"},{"indexed":false,"internalType":"uint256[]","name":"tokenId","type":"uint256[]"},{"indexed":false,"internalType":"address","name":"nftAddress","type":"address"}],"name":"NftReceived","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":"signer","type":"address"}],"name":"SetSigner","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"address","name":"recipient","type":"address"},{"indexed":false,"internalType":"address","name":"operator","type":"address"}],"name":"WithdrawERC20","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"address","name":"recipient","type":"address"},{"indexed":false,"internalType":"address","name":"operator","type":"address"}],"name":"WithdrawETH","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"},{"indexed":false,"internalType":"address","name":"recipient","type":"address"},{"indexed":false,"internalType":"address","name":"operator","type":"address"}],"name":"WithdrawNFT","type":"event"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"emergencyWithdrawERC20","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"emergencyWithdrawETH","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"emergencyWithdrawNFT","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"tokenAddress","type":"address"},{"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"checkId","type":"uint256"},{"internalType":"bytes","name":"signature","type":"bytes"}],"name":"finishBridge","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"from","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC721Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"secret","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_signer","type":"address"}],"name":"setSigner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"internalType":"address","name":"contractAddress","type":"address"}],"name":"startBridge","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"tokenSender","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes","name":"","type":"bytes"}],"name":"usedSignatures","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"}]