编译器
0.8.17+commit.8df45f5f
文件 1 的 12:EnumerableSet.sol
pragma solidity ^0.8.0;
library EnumerableSet {
struct Set {
bytes32[] _values;
mapping(bytes32 => uint256) _indexes;
}
function _add(Set storage set, bytes32 value) private returns (bool) {
if (!_contains(set, value)) {
set._values.push(value);
set._indexes[value] = set._values.length;
return true;
} else {
return false;
}
}
function _remove(Set storage set, bytes32 value) private returns (bool) {
uint256 valueIndex = set._indexes[value];
if (valueIndex != 0) {
uint256 toDeleteIndex = valueIndex - 1;
uint256 lastIndex = set._values.length - 1;
if (lastIndex != toDeleteIndex) {
bytes32 lastValue = set._values[lastIndex];
set._values[toDeleteIndex] = lastValue;
set._indexes[lastValue] = valueIndex;
}
set._values.pop();
delete set._indexes[value];
return true;
} else {
return false;
}
}
function _contains(Set storage set, bytes32 value) private view returns (bool) {
return set._indexes[value] != 0;
}
function _length(Set storage set) private view returns (uint256) {
return set._values.length;
}
function _at(Set storage set, uint256 index) private view returns (bytes32) {
return set._values[index];
}
function _values(Set storage set) private view returns (bytes32[] memory) {
return set._values;
}
struct Bytes32Set {
Set _inner;
}
function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
return _add(set._inner, value);
}
function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
return _remove(set._inner, value);
}
function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
return _contains(set._inner, value);
}
function length(Bytes32Set storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
return _at(set._inner, index);
}
function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
bytes32[] memory store = _values(set._inner);
bytes32[] memory result;
assembly {
result := store
}
return result;
}
struct AddressSet {
Set _inner;
}
function add(AddressSet storage set, address value) internal returns (bool) {
return _add(set._inner, bytes32(uint256(uint160(value))));
}
function remove(AddressSet storage set, address value) internal returns (bool) {
return _remove(set._inner, bytes32(uint256(uint160(value))));
}
function contains(AddressSet storage set, address value) internal view returns (bool) {
return _contains(set._inner, bytes32(uint256(uint160(value))));
}
function length(AddressSet storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(AddressSet storage set, uint256 index) internal view returns (address) {
return address(uint160(uint256(_at(set._inner, index))));
}
function values(AddressSet storage set) internal view returns (address[] memory) {
bytes32[] memory store = _values(set._inner);
address[] memory result;
assembly {
result := store
}
return result;
}
struct UintSet {
Set _inner;
}
function add(UintSet storage set, uint256 value) internal returns (bool) {
return _add(set._inner, bytes32(value));
}
function remove(UintSet storage set, uint256 value) internal returns (bool) {
return _remove(set._inner, bytes32(value));
}
function contains(UintSet storage set, uint256 value) internal view returns (bool) {
return _contains(set._inner, bytes32(value));
}
function length(UintSet storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(UintSet storage set, uint256 index) internal view returns (uint256) {
return uint256(_at(set._inner, index));
}
function values(UintSet storage set) internal view returns (uint256[] memory) {
bytes32[] memory store = _values(set._inner);
uint256[] memory result;
assembly {
result := store
}
return result;
}
}
文件 2 的 12:IERC20.sol
pragma solidity 0.8.17;
interface IERC20
{
function name () external view returns (string memory);
function symbol () external view returns (string memory);
function decimals () external view returns (uint8);
function totalSupply () external view returns (uint256);
function balanceOf (address account) external view returns (uint256);
function allowance (address owner, address spender) external view returns (uint256);
function approve (address spender, uint256 amount) external returns (bool);
function transfer (address to, uint256 amount) external returns (bool);
function transferFrom (address from, address to, uint256 amount) external returns (bool);
function mint (address account, uint256 amount) external;
function burn (address account, uint256 amount) external;
}
文件 3 的 12:IFrontender.sol
pragma solidity 0.8.17;
interface IFrontender
{
function isRegistered (address account) external view returns (bool);
function refer (address account, uint256 amount, address referrer) external;
}
文件 4 的 12:IGlove.sol
pragma solidity 0.8.17;
interface IGlove
{
function balanceOf (address account) external view returns (uint256);
function creditOf (address account) external view returns (uint256);
function creditlessOf (address account) external view returns (uint256);
function transfer (address to, uint256 amount) external returns (bool);
function transferFrom (address from, address to, uint256 amount) external returns (bool);
function transferCreditless (address to, uint256 amount) external returns (bool);
function mint (address account, uint256 amount) external;
function mintCreditless (address account, uint256 amount) external;
function creditize (address account, uint256 credits) external returns (bool);
function burn (address account, uint256 amount) external;
function decreditize (address account, uint256 credits) external returns (bool);
}
文件 5 的 12:IRegistry.sol
pragma solidity 0.8.17;
interface IRegistry
{
function get (string calldata name) external view returns (address);
function provisioner () external view returns (address);
function frontender () external view returns (address);
function collector () external view returns (address);
}
文件 6 的 12:ISwapRouter.sol
pragma solidity >=0.7.5;
pragma abicoder v2;
import '@uniswap/v3-core/contracts/interfaces/callback/IUniswapV3SwapCallback.sol';
interface ISwapRouter is IUniswapV3SwapCallback {
struct ExactInputSingleParams {
address tokenIn;
address tokenOut;
uint24 fee;
address recipient;
uint256 deadline;
uint256 amountIn;
uint256 amountOutMinimum;
uint160 sqrtPriceLimitX96;
}
function exactInputSingle(ExactInputSingleParams calldata params) external payable returns (uint256 amountOut);
struct ExactInputParams {
bytes path;
address recipient;
uint256 deadline;
uint256 amountIn;
uint256 amountOutMinimum;
}
function exactInput(ExactInputParams calldata params) external payable returns (uint256 amountOut);
struct ExactOutputSingleParams {
address tokenIn;
address tokenOut;
uint24 fee;
address recipient;
uint256 deadline;
uint256 amountOut;
uint256 amountInMaximum;
uint160 sqrtPriceLimitX96;
}
function exactOutputSingle(ExactOutputSingleParams calldata params) external payable returns (uint256 amountIn);
struct ExactOutputParams {
bytes path;
address recipient;
uint256 deadline;
uint256 amountOut;
uint256 amountInMaximum;
}
function exactOutput(ExactOutputParams calldata params) external payable returns (uint256 amountIn);
}
文件 7 的 12:IUniswapV3SwapCallback.sol
pragma solidity >=0.5.0;
interface IUniswapV3SwapCallback {
function uniswapV3SwapCallback(
int256 amount0Delta,
int256 amount1Delta,
bytes calldata data
) external;
}
文件 8 的 12:IWUSD.sol
pragma solidity 0.8.17;
struct Snapshot
{
uint32 epoch;
uint112 last;
uint112 cumulative;
}
interface IWUSD
{
function balanceOf (address account) external view returns (uint256);
function snapshot () external view returns (Snapshot memory);
function epochOf (address account) external view returns (uint256);
function allowance (address owner, address spender) external view returns (uint256);
function approve (address spender, uint256 amount) external returns (bool);
function transfer (address to, uint256 amount) external returns (bool);
function transferFrom (address from, address to, uint256 amount ) external returns (bool);
}
文件 9 的 12:Math.sol
pragma solidity ^0.8.0;
library Math {
enum Rounding {
Down,
Up,
Zero
}
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a > b ? a : b;
}
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a < b ? a : b;
}
function average(uint256 a, uint256 b) internal pure returns (uint256) {
return (a & b) + (a ^ b) / 2;
}
function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
return a == 0 ? 0 : (a - 1) / b + 1;
}
function mulDiv(
uint256 x,
uint256 y,
uint256 denominator
) internal pure returns (uint256 result) {
unchecked {
uint256 prod0;
uint256 prod1;
assembly {
let mm := mulmod(x, y, not(0))
prod0 := mul(x, y)
prod1 := sub(sub(mm, prod0), lt(mm, prod0))
}
if (prod1 == 0) {
return prod0 / denominator;
}
require(denominator > prod1);
uint256 remainder;
assembly {
remainder := mulmod(x, y, denominator)
prod1 := sub(prod1, gt(remainder, prod0))
prod0 := sub(prod0, remainder)
}
uint256 twos = denominator & (~denominator + 1);
assembly {
denominator := div(denominator, twos)
prod0 := div(prod0, twos)
twos := add(div(sub(0, twos), twos), 1)
}
prod0 |= prod1 * twos;
uint256 inverse = (3 * denominator) ^ 2;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
result = prod0 * inverse;
return result;
}
}
function mulDiv(
uint256 x,
uint256 y,
uint256 denominator,
Rounding rounding
) internal pure returns (uint256) {
uint256 result = mulDiv(x, y, denominator);
if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
result += 1;
}
return result;
}
function sqrt(uint256 a) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
uint256 result = 1 << (log2(a) >> 1);
unchecked {
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
return min(result, a / result);
}
}
function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = sqrt(a);
return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
}
}
function log2(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 128;
}
if (value >> 64 > 0) {
value >>= 64;
result += 64;
}
if (value >> 32 > 0) {
value >>= 32;
result += 32;
}
if (value >> 16 > 0) {
value >>= 16;
result += 16;
}
if (value >> 8 > 0) {
value >>= 8;
result += 8;
}
if (value >> 4 > 0) {
value >>= 4;
result += 4;
}
if (value >> 2 > 0) {
value >>= 2;
result += 2;
}
if (value >> 1 > 0) {
result += 1;
}
}
return result;
}
function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log2(value);
return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
}
}
function log10(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >= 10**64) {
value /= 10**64;
result += 64;
}
if (value >= 10**32) {
value /= 10**32;
result += 32;
}
if (value >= 10**16) {
value /= 10**16;
result += 16;
}
if (value >= 10**8) {
value /= 10**8;
result += 8;
}
if (value >= 10**4) {
value /= 10**4;
result += 4;
}
if (value >= 10**2) {
value /= 10**2;
result += 2;
}
if (value >= 10**1) {
result += 1;
}
}
return result;
}
function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log10(value);
return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0);
}
}
function log256(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 16;
}
if (value >> 64 > 0) {
value >>= 64;
result += 8;
}
if (value >> 32 > 0) {
value >>= 32;
result += 4;
}
if (value >> 16 > 0) {
value >>= 16;
result += 2;
}
if (value >> 8 > 0) {
result += 1;
}
}
return result;
}
function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log256(value);
return result + (rounding == Rounding.Up && 1 << (result * 8) < value ? 1 : 0);
}
}
}
文件 10 的 12:ReentrancyGuard.sol
pragma solidity 0.8.17;
abstract contract ReentrancyGuard
{
uint256 private _status = 1;
modifier nonReentrant ()
{
require(_status == 1, "reentrance");
_status = 2;
_;
_status = 1;
}
}
文件 11 的 12:SafeToken.sol
pragma solidity 0.8.17;
import { IERC20 } from "../interfaces/IERC20.sol";
library SafeToken
{
function _getRevertErr (bytes memory data, string memory message) private pure returns (string memory)
{
if (data.length < 68)
{
return message;
}
assembly
{
data := add(data, 0x04)
}
return abi.decode(data, (string));
}
function _call (address token, bytes memory encoded, string memory message) private
{
(bool success, bytes memory data) = token.call(encoded);
require(success && (data.length == 0 || abi.decode(data, (bool))), _getRevertErr(data, message));
}
function safeApprove (IERC20 token, address spender, uint256 amount) internal
{
_call(address(token), abi.encodeWithSelector(IERC20.approve.selector, spender, amount), "!sa");
}
function safeTransfer (IERC20 token, address to, uint256 amount) internal
{
_call(address(token), abi.encodeWithSelector(IERC20.transfer.selector, to, amount), "!st");
}
function safeTransferFrom (IERC20 token, address from, address to, uint256 amount) internal
{
_call(address(token), abi.encodeWithSelector(IERC20.transferFrom.selector, from, to, amount), "!stf");
}
}
文件 12 的 12:WUSD.sol
pragma solidity 0.8.17;
import { Math } from "@openzeppelin/contracts/utils/math/Math.sol";
import { EnumerableSet } from "@openzeppelin/contracts/utils/structs/EnumerableSet.sol";
import { ISwapRouter } from "@uniswap/v3-periphery/contracts/interfaces/ISwapRouter.sol";
import { ReentrancyGuard } from "./utils/ReentrancyGuard.sol";
import { SafeToken } from "./utils/SafeToken.sol";
import { IERC20 } from "./interfaces/IERC20.sol";
import { IGlove } from "./interfaces/IGlove.sol";
import { IRegistry } from "./interfaces/IRegistry.sol";
import { IFrontender } from "./interfaces/IFrontender.sol";
import { Snapshot, IWUSD } from "./interfaces/IWUSD.sol";
contract WUSD is IWUSD, ReentrancyGuard
{
using SafeToken for IERC20;
using EnumerableSet for EnumerableSet.AddressSet;
bytes32 private constant _PERMIT_TYPEHASH = keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
bytes32 private constant _DOMAIN_TYPEHASH = keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)");
bytes32 private constant _NAME_HASH = keccak256("Wrapped USD");
bytes32 private constant _VERSION_HASH = keccak256("1");
ISwapRouter private constant _ROUTER = ISwapRouter(0xE592427A0AEce92De3Edee1F18E0157C05861564);
IRegistry private constant _REGISTRY = IRegistry(0x4E23524aA15c689F2d100D49E27F28f8E5088C0D);
address private constant _GLOVE = 0x70c5f366dB60A2a0C59C4C24754803Ee47Ed7284;
address private constant _USDT = 0xdAC17F958D2ee523a2206206994597C13D831ec7;
address private constant _USDC = 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48;
uint256 private constant _MIN_GLOVABLE = 100e18;
uint256 private constant _MID_GLOVE = 0.01e18;
uint256 private constant _MAX_GLOVE = 2e18;
uint256 private constant _EPOCH = 100_000e18;
uint24 private constant _ROUTE = 500;
bytes32 private immutable _CACHED_DOMAIN_SEPARATOR;
uint256 private immutable _CACHED_CHAIN_ID;
address private immutable _CACHED_THIS;
Snapshot private _snapshot;
EnumerableSet.AddressSet private _fiatcoins;
uint256 private _totalSupply;
mapping(address => uint256) private _epoch;
mapping(address => uint256) private _decimal;
mapping(address => uint256) private _nonce;
mapping(address => uint256) private _balance;
mapping(address => mapping(address => uint256)) private _allowance;
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
event Wrap(address indexed account, address fiatcoin, uint256 amount, address referrer);
event Unwrap(address indexed account, address fiatcoin, uint256 amount);
constructor (address[] memory fiatcoins)
{
uint256 decimal;
address fiatcoin;
for (uint256 i; i < fiatcoins.length;)
{
fiatcoin = fiatcoins[i];
decimal = IERC20(fiatcoin).decimals();
_fiatcoins.add(fiatcoin);
_decimal[fiatcoin] = decimal;
IERC20(fiatcoin).safeApprove(address(_ROUTER), type(uint128).max);
unchecked { i++; }
}
_CACHED_THIS = address(this);
_CACHED_CHAIN_ID = block.chainid;
_CACHED_DOMAIN_SEPARATOR = _separator();
_snapshot = Snapshot({ epoch: 1, last: 0, cumulative: 0 });
}
function name () public pure returns (string memory)
{
return "Wrapped USD";
}
function symbol () public pure returns (string memory)
{
return "WUSD";
}
function decimals () public pure returns (uint8)
{
return 18;
}
function totalSupply () public view returns (uint256)
{
return _totalSupply;
}
function balanceOf (address account) public view returns (uint256)
{
return _balance[account];
}
function snapshot () public view returns (Snapshot memory)
{
return _snapshot;
}
function epochOf (address account) public view returns (uint256)
{
return _epoch[account];
}
function _separator () private view returns (bytes32)
{
return keccak256(abi.encode(_DOMAIN_TYPEHASH, _NAME_HASH, _VERSION_HASH, block.chainid, address(this)));
}
function DOMAIN_SEPARATOR () public view returns (bytes32)
{
return (address(this) == _CACHED_THIS && block.chainid == _CACHED_CHAIN_ID) ? _CACHED_DOMAIN_SEPARATOR : _separator();
}
function nonces (address owner) public view returns (uint256)
{
return _nonce[owner];
}
function allowance (address owner, address spender) public view returns (uint256)
{
return _allowance[owner][spender];
}
function _approve (address owner, address spender, uint256 amount) internal
{
_allowance[owner][spender] = amount;
emit Approval(owner, spender, amount);
}
function approve (address spender, uint256 amount) public returns (bool)
{
_approve(msg.sender, spender, amount);
return true;
}
function increaseAllowance (address spender, uint256 amount) public returns (bool)
{
_approve(msg.sender, spender, _allowance[msg.sender][spender] + amount);
return true;
}
function decreaseAllowance (address spender, uint256 amount) public returns (bool)
{
uint256 currentAllowance = _allowance[msg.sender][spender];
require(currentAllowance >= amount, "WUSD: decreasing < 0");
unchecked
{
_approve(msg.sender, spender, currentAllowance - amount);
}
return true;
}
function permit (address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public
{
require(block.timestamp <= deadline, "WUSD: expired deadline");
bytes32 hash = keccak256(abi.encode(_PERMIT_TYPEHASH, owner, spender, value, _nonce[owner]++, deadline));
address signer = ecrecover(keccak256(abi.encodePacked("\x19\x01", DOMAIN_SEPARATOR(), hash)), v, r, s);
require(signer != address(0) && signer == owner, "WUSD: !valid signature");
_approve(owner, spender, value);
}
function _transfer (address from, address to, uint256 amount) internal
{
require(to != address(0), "WUSD: transfer to 0 addr");
uint256 balance = _balance[from];
require(balance >= amount, "WUSD: amount > balance");
unchecked
{
_balance[from] = balance - amount;
_balance[to] += amount;
}
emit Transfer(from, to, amount);
}
function transfer (address to, uint256 amount) public returns (bool)
{
_transfer(msg.sender, to, amount);
return true;
}
function transferFrom (address from, address to, uint256 amount) public returns (bool)
{
uint256 currentAllowance = _allowance[from][msg.sender];
if (currentAllowance != type(uint256).max)
{
require(currentAllowance >= amount, "WUSD: !enough allowance");
unchecked
{
_approve(from, msg.sender, currentAllowance - amount);
}
}
_transfer(from, to, amount);
return true;
}
function _percent (uint256 amount, uint256 percent) internal pure returns (uint256)
{
return (amount * percent) / 100_00;
}
function _normalize (uint256 amount, uint256 decimal) internal pure returns (uint256)
{
return (amount * 1e18) / (10 ** decimal);
}
function _denormalize (uint256 amount, uint256 decimal) internal pure returns (uint256)
{
return (amount * (10 ** decimal)) / 1e18;
}
function _isFiatcoin (address token) internal view
{
require(_fiatcoins.contains(token), "WUSD: !fiatcoin");
}
function _snap (uint256 wrapping) internal
{
Snapshot memory snap = _snapshot;
if ((snap.cumulative - snap.last) >= _EPOCH)
{
_snapshot.epoch = snap.epoch + 1;
_snapshot.last = snap.cumulative;
}
if (wrapping >= _MIN_GLOVABLE || _epoch[msg.sender] > 0)
{
_epoch[msg.sender] = _snapshot.epoch;
}
_snapshot.cumulative = snap.cumulative + uint112(wrapping);
}
function _englove (uint256 wrapping) internal
{
uint256 gloves = IGlove(_GLOVE).balanceOf(msg.sender);
if (wrapping >= _MIN_GLOVABLE && gloves < _MAX_GLOVE)
{
IGlove(_GLOVE).mintCreditless(msg.sender, Math.min(_MAX_GLOVE - gloves, wrapping > 1_000e18 ? ((_MAX_GLOVE * wrapping) / _EPOCH) : ((_MID_GLOVE * wrapping) / 1_000e18)));
}
}
function _mint (address account, uint256 amount) internal
{
require(account != address(0), "WUSD: mint to 0 addr");
_totalSupply += amount;
unchecked
{
_balance[account] += amount;
}
emit Transfer(address(0), account, amount);
}
function _parse (uint256 amount, uint256 decimal) internal pure returns (uint256, uint256)
{
return (Math.max(10 ** decimal, _percent(amount, 1_00)), _normalize(amount, decimal));
}
function wrap (address fiatcoin, uint256 amount, address referrer) external nonReentrant
{
_isFiatcoin(fiatcoin);
require(amount > 0, "WUSD: wrap(0)");
(uint256 fee, uint256 wrapping) = _parse(amount, _decimal[fiatcoin]);
_snap(wrapping);
_mint(msg.sender, wrapping);
_englove(wrapping);
IERC20(fiatcoin).safeTransferFrom(msg.sender, address(this), amount + fee);
if (fiatcoin != _USDT && fiatcoin != _USDC)
{
_ROUTER.exactInputSingle(ISwapRouter.ExactInputSingleParams
({
tokenIn: fiatcoin,
tokenOut: _USDC,
fee: fiatcoin != 0x0000000000085d4780B73119b644AE5ecd22b376 ? _ROUTE : 100,
recipient: _REGISTRY.collector(),
deadline: block.timestamp,
amountIn: fee,
amountOutMinimum: _percent(_denormalize(_normalize(fee, _decimal[fiatcoin]), 6), 95_00),
sqrtPriceLimitX96: 0
}));
}
else
{
IERC20(fiatcoin).safeTransfer(_REGISTRY.collector(), fee);
}
if (referrer != address(0))
{
IFrontender(_REGISTRY.frontender()).refer(msg.sender, wrapping, referrer);
}
emit Wrap(msg.sender, fiatcoin, amount, referrer);
}
function _burn (address account, uint256 amount) internal
{
uint256 balance = _balance[account];
require(balance >= amount, "WUSD: burn > balance");
unchecked
{
_balance[account] = balance - amount;
_totalSupply -= amount;
}
emit Transfer(account, address(0), amount);
}
function _deglove (uint256 amount, uint256 balance) internal
{
uint256 creditless = IGlove(_GLOVE).creditlessOf(msg.sender);
uint256 credits = _percent(creditless, Math.min((amount * 100_00) / balance, (_snapshot.epoch - _epoch[msg.sender]) * 100));
if (_epoch[msg.sender] > 0)
{
if (amount == balance)
{
_epoch[msg.sender] = 0;
IGlove(_GLOVE).burn(msg.sender, creditless - credits);
}
else
{
_epoch[msg.sender] = _snapshot.epoch;
}
IGlove(_GLOVE).creditize(msg.sender, credits);
}
}
function unwrap (address fiatcoin, uint256 amount) external nonReentrant
{
_isFiatcoin(fiatcoin);
uint256 balance = _balance[msg.sender];
uint256 unwrapping = _denormalize(amount, _decimal[fiatcoin]);
require(amount > 0, "WUSD: unwrap(0)");
require((IERC20(fiatcoin).balanceOf(address(this)) - (10 ** _decimal[fiatcoin])) >= unwrapping, "WUSD: !enough fiatcoin");
_burn(msg.sender, amount);
_deglove(amount, balance);
IERC20(fiatcoin).safeTransfer(msg.sender, unwrapping);
emit Unwrap(msg.sender, fiatcoin, amount);
}
}
{
"compilationTarget": {
"contracts/WUSD.sol": "WUSD"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address[]","name":"fiatcoins","type":"address[]"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","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":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"fiatcoin","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Unwrap","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"fiatcoin","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"address","name":"referrer","type":"address"}],"name":"Wrap","type":"event"},{"inputs":[],"name":"DOMAIN_SEPARATOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"decreaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"epochOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"increaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"nonces","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"permit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"snapshot","outputs":[{"components":[{"internalType":"uint32","name":"epoch","type":"uint32"},{"internalType":"uint112","name":"last","type":"uint112"},{"internalType":"uint112","name":"cumulative","type":"uint112"}],"internalType":"struct Snapshot","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"fiatcoin","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"unwrap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"fiatcoin","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"address","name":"referrer","type":"address"}],"name":"wrap","outputs":[],"stateMutability":"nonpayable","type":"function"}]