编译器
0.8.28+commit.7893614a
文件 1 的 9:AccessControl.sol
pragma solidity ^0.8.20;
import {IAccessControl} from "./IAccessControl.sol";
import {Context} from "../utils/Context.sol";
import {ERC165} from "../utils/introspection/ERC165.sol";
abstract contract AccessControl is Context, IAccessControl, ERC165 {
struct RoleData {
mapping(address account => bool) hasRole;
bytes32 adminRole;
}
mapping(bytes32 role => RoleData) private _roles;
bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
modifier onlyRole(bytes32 role) {
_checkRole(role);
_;
}
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IAccessControl).interfaceId || super.supportsInterface(interfaceId);
}
function hasRole(bytes32 role, address account) public view virtual returns (bool) {
return _roles[role].hasRole[account];
}
function _checkRole(bytes32 role) internal view virtual {
_checkRole(role, _msgSender());
}
function _checkRole(bytes32 role, address account) internal view virtual {
if (!hasRole(role, account)) {
revert AccessControlUnauthorizedAccount(account, role);
}
}
function getRoleAdmin(bytes32 role) public view virtual returns (bytes32) {
return _roles[role].adminRole;
}
function grantRole(bytes32 role, address account) public virtual onlyRole(getRoleAdmin(role)) {
_grantRole(role, account);
}
function revokeRole(bytes32 role, address account) public virtual onlyRole(getRoleAdmin(role)) {
_revokeRole(role, account);
}
function renounceRole(bytes32 role, address callerConfirmation) public virtual {
if (callerConfirmation != _msgSender()) {
revert AccessControlBadConfirmation();
}
_revokeRole(role, callerConfirmation);
}
function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual {
bytes32 previousAdminRole = getRoleAdmin(role);
_roles[role].adminRole = adminRole;
emit RoleAdminChanged(role, previousAdminRole, adminRole);
}
function _grantRole(bytes32 role, address account) internal virtual returns (bool) {
if (!hasRole(role, account)) {
_roles[role].hasRole[account] = true;
emit RoleGranted(role, account, _msgSender());
return true;
} else {
return false;
}
}
function _revokeRole(bytes32 role, address account) internal virtual returns (bool) {
if (hasRole(role, account)) {
_roles[role].hasRole[account] = false;
emit RoleRevoked(role, account, _msgSender());
return true;
} else {
return false;
}
}
}
文件 2 的 9: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;
}
}
文件 3 的 9:ERC165.sol
pragma solidity ^0.8.20;
import {IERC165} from "./IERC165.sol";
abstract contract ERC165 is IERC165 {
function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) {
return interfaceId == type(IERC165).interfaceId;
}
}
文件 4 的 9:FreezerPayments.sol
pragma solidity ^0.8.23;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC1155/IERC1155.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/access/AccessControl.sol";
interface IPrizeVault {
function sendPrize(address winner) external;
function getAvailableTokenCount() external view returns (uint256);
}
contract FreezerPayments is
Ownable,
AccessControl
{
IPrizeVault public prizeVault;
struct ERC20Payment {
bool enabled;
uint256 cost;
bool burn;
}
mapping(address => ERC20Payment) public ERC20Payments;
address[] public AllowedERC20s;
struct WalletSplit {
address wallet;
uint256 percentage;
}
WalletSplit[] public splits;
uint256 public nativeCost = 5 * 10**18;
bool public paused = true;
bool public nativeEnabled = false;
address public burnAddress = 0x000000000000000000000000000000000000dEaD;
constructor() Ownable(msg.sender) {
_grantRole(DEFAULT_ADMIN_ROLE, msg.sender);
}
function erc20prize(address contractAddress) external {
if (paused) revert("Paused");
if (prizeVault.getAvailableTokenCount() == 0) revert("NoTokens");
if (!checkAllowedAddress(AllowedERC20s, contractAddress)) revert("PaymentNotAllowed");
ERC20Payment memory payment = ERC20Payments[contractAddress];
if (!payment.enabled) revert("PaymentDisabled");
IERC20 coin = IERC20(contractAddress);
if (coin.balanceOf(msg.sender) < payment.cost) revert("Cost");
if (coin.allowance(msg.sender, address(this)) < payment.cost) revert("Allow");
if (payment.burn) coin.transferFrom(msg.sender, burnAddress, payment.cost);
else coin.transferFrom(msg.sender, address(this), payment.cost);
prizeVault.sendPrize(msg.sender);
}
function nativePrize() external payable {
if (paused) revert("Paused");
if (!nativeEnabled) revert("NotEnabled");
if (msg.value < nativeCost) revert("Cost");
if (prizeVault.getAvailableTokenCount() == 0) revert("NoTokens");
prizeVault.sendPrize(msg.sender);
}
function removeAllowedAddress(address[] storage array,address removeAddress) internal {
uint256 length = array.length;
for (uint256 i; i < length; ) {
if (array[i] == removeAddress) {
if (i < length - 1) {
array[i] = array[length - 1];
}
array.pop();
return;
}
unchecked {
++i;
}
}
}
function checkAllowedAddress(address[] memory array, address contractAddress) internal pure returns (bool) {
uint256 length = array.length;
for (uint256 i; i < length; ) {
if (array[i] == contractAddress) return true;
unchecked {
++i;
}
}
return false;
}
function findIndex(uint256[] memory array, uint256 element) internal pure returns (int) {
uint length = array.length;
for (uint i = 0; i < length;) {
if (array[i] == element) {
return int(i);
}
unchecked {
++i;
}
}
return -1;
}
function setPrizeVault(address contractAddress) external onlyRole(DEFAULT_ADMIN_ROLE) {
prizeVault = IPrizeVault(contractAddress);
}
function getPrizeVault() external view returns (address) {
return address(prizeVault);
}
function getAllowedERC20Payments() external view returns (address[] memory) {
return AllowedERC20s;
}
function addERC20Payment(address contractAddress, uint256 cost, bool burn) external onlyRole(DEFAULT_ADMIN_ROLE){
if (checkAllowedAddress(AllowedERC20s, contractAddress)) revert ("Exists!");
AllowedERC20s.push(contractAddress);
ERC20Payments[contractAddress] = ERC20Payment(true, cost, burn);
}
function removeERC20Payment(address contractAddress) external onlyRole(DEFAULT_ADMIN_ROLE){
if (!checkAllowedAddress(AllowedERC20s, contractAddress)) revert ("NotExists!");
removeAllowedAddress(AllowedERC20s, contractAddress);
delete ERC20Payments[contractAddress];
}
function modifyERC20Cost(address contractAddress, uint256 cost) external onlyRole(DEFAULT_ADMIN_ROLE){
if (!checkAllowedAddress(AllowedERC20s, contractAddress)) revert ("NotExists!");
ERC20Payments[contractAddress].cost = cost;
}
function modifyERC20Burn(address contractAddress, bool burn) external onlyRole(DEFAULT_ADMIN_ROLE){
if (!checkAllowedAddress(AllowedERC20s, contractAddress)) revert ("NotExists!");
ERC20Payments[contractAddress].burn = burn;
}
function setNativeCostWEI(uint256 newCostWEI) external onlyRole(DEFAULT_ADMIN_ROLE)
{
nativeCost = newCostWEI;
}
function setPaused(bool state) external onlyRole(DEFAULT_ADMIN_ROLE)
{
paused = state;
}
function setNativeEnabled(bool state) external onlyRole(DEFAULT_ADMIN_ROLE) {
nativeEnabled = state;
}
function setSplits(address[] calldata wallets, uint256[] calldata percentages) external onlyOwner {
require(wallets.length == percentages.length, "Mismatched array lengths");
require(wallets.length > 0, "No wallets provided");
uint256 totalPercentage = 0;
for (uint256 i = 0; i < percentages.length; i++) {
require(percentages[i] > 0, "Percentage must be greater than 0");
totalPercentage += percentages[i];
}
require(totalPercentage == 100, "Total percentages must equal 100");
delete splits;
for (uint256 i = 0; i < wallets.length; i++) {
splits.push(WalletSplit({
wallet: wallets[i],
percentage: percentages[i]
}));
}
}
function withdrawERC20(address contractAddress) public onlyOwner {
IERC20 paymentToken = IERC20(contractAddress);
uint256 balance = paymentToken.balanceOf(address(this));
require(balance > 0, "No balance to withdraw");
require(splits.length > 0, "No splits set");
for (uint256 i = 0; i < splits.length; i++) {
uint256 amount = (balance * splits[i].percentage) / 100;
paymentToken.transfer(splits[i].wallet, amount);
}
}
function withdraw() public onlyOwner {
uint256 balance = address(this).balance;
require(balance > 0, "No balance to withdraw");
require(splits.length > 0, "No splits set");
for (uint256 i = 0; i < splits.length; i++) {
uint256 amount = (balance * splits[i].percentage) / 100;
payable(splits[i].wallet).transfer(amount);
}
}
}
文件 5 的 9:IAccessControl.sol
pragma solidity ^0.8.20;
interface IAccessControl {
error AccessControlUnauthorizedAccount(address account, bytes32 neededRole);
error AccessControlBadConfirmation();
event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
function hasRole(bytes32 role, address account) external view returns (bool);
function getRoleAdmin(bytes32 role) external view returns (bytes32);
function grantRole(bytes32 role, address account) external;
function revokeRole(bytes32 role, address account) external;
function renounceRole(bytes32 role, address callerConfirmation) external;
}
文件 6 的 9:IERC1155.sol
pragma solidity ^0.8.20;
import {IERC165} from "../../utils/introspection/IERC165.sol";
interface IERC1155 is IERC165 {
event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value);
event TransferBatch(
address indexed operator,
address indexed from,
address indexed to,
uint256[] ids,
uint256[] values
);
event ApprovalForAll(address indexed account, address indexed operator, bool approved);
event URI(string value, uint256 indexed id);
function balanceOf(address account, uint256 id) external view returns (uint256);
function balanceOfBatch(
address[] calldata accounts,
uint256[] calldata ids
) external view returns (uint256[] memory);
function setApprovalForAll(address operator, bool approved) external;
function isApprovedForAll(address account, address operator) external view returns (bool);
function safeTransferFrom(address from, address to, uint256 id, uint256 value, bytes calldata data) external;
function safeBatchTransferFrom(
address from,
address to,
uint256[] calldata ids,
uint256[] calldata values,
bytes calldata data
) external;
}
文件 7 的 9:IERC165.sol
pragma solidity ^0.8.20;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 8 的 9: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);
}
文件 9 的 9: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/Freezerverse/FreezerPayments.sol": "FreezerPayments"
},
"evmVersion": "cancun",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"AccessControlBadConfirmation","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"bytes32","name":"neededRole","type":"bytes32"}],"name":"AccessControlUnauthorizedAccount","type":"error"},{"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":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"previousAdminRole","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"newAdminRole","type":"bytes32"}],"name":"RoleAdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleGranted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleRevoked","type":"event"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"AllowedERC20s","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"DEFAULT_ADMIN_ROLE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"ERC20Payments","outputs":[{"internalType":"bool","name":"enabled","type":"bool"},{"internalType":"uint256","name":"cost","type":"uint256"},{"internalType":"bool","name":"burn","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"contractAddress","type":"address"},{"internalType":"uint256","name":"cost","type":"uint256"},{"internalType":"bool","name":"burn","type":"bool"}],"name":"addERC20Payment","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"burnAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"contractAddress","type":"address"}],"name":"erc20prize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getAllowedERC20Payments","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getPrizeVault","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"getRoleAdmin","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"grantRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"hasRole","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"contractAddress","type":"address"},{"internalType":"bool","name":"burn","type":"bool"}],"name":"modifyERC20Burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"contractAddress","type":"address"},{"internalType":"uint256","name":"cost","type":"uint256"}],"name":"modifyERC20Cost","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"nativeCost","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"nativeEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"nativePrize","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"prizeVault","outputs":[{"internalType":"contract IPrizeVault","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"contractAddress","type":"address"}],"name":"removeERC20Payment","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"callerConfirmation","type":"address"}],"name":"renounceRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"revokeRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newCostWEI","type":"uint256"}],"name":"setNativeCostWEI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"state","type":"bool"}],"name":"setNativeEnabled","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"state","type":"bool"}],"name":"setPaused","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"contractAddress","type":"address"}],"name":"setPrizeVault","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"wallets","type":"address[]"},{"internalType":"uint256[]","name":"percentages","type":"uint256[]"}],"name":"setSplits","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"splits","outputs":[{"internalType":"address","name":"wallet","type":"address"},{"internalType":"uint256","name":"percentage","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"contractAddress","type":"address"}],"name":"withdrawERC20","outputs":[],"stateMutability":"nonpayable","type":"function"}]