Transaction Hash:
Block:
19581486 at Apr-04-2024 09:23:35 AM +UTC
Transaction Fee:
0.000851847325806092 ETH
$2.14
Gas Used:
46,151 Gas / 18.457830292 Gwei
Emitted Events:
218 |
Token.Approval( owner=[Sender] 0x3dfcfaac8ee248cf9976e866822f2daac2ac7d03, spender=0x00000000...43aC78BA3, amount=100000000000000000000 )
|
Account State Difference:
Address | Before | After | State Difference | ||
---|---|---|---|---|---|
0x1f9090aa...8e676c326
Miner
| 3.851166742142256747 Eth | 3.851169049692256747 Eth | 0.00000230755 | ||
0x3dfCfAAC...ac2Ac7D03 |
0.044381140040603191 Eth
Nonce: 798
|
0.043529292714797099 Eth
Nonce: 799
| 0.000851847325806092 | ||
0xaeB3607e...efb54309A |
Execution Trace
Token.approve( spender=0x000000000022D473030F116dDEE9F6B43aC78BA3, amount=100000000000000000000 ) => ( True )
approve[ERC20 (ln:216)]
Approval[ERC20 (ln:218)]
// SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.0; /// @notice Modified `Auth.sol`, where the contract is its own `authority` there is no `target` to `canCall`. /// @notice Provides a flexible and updatable auth pattern which is completely separate from application logic. /// @author Modified from Solmate (https://github.com/Rari-Capital/solmate/blob/main/src/auth/Auth.sol) /// @author Modified from Dappsys (https://github.com/dapphub/ds-auth/blob/master/src/auth.sol) abstract contract Auth { event OwnerUpdated(address indexed user, address indexed newOwner); address public owner; constructor(address _owner) { owner = _owner; emit OwnerUpdated(msg.sender, _owner); } modifier requiresAuth() virtual { require(isAuthorized(msg.sender, msg.sig), "UNAUTHORIZED"); _; } function isAuthorized(address user, bytes4 functionSig) internal view virtual returns (bool) { return canCall(user, functionSig) || user == owner; } function setOwner(address newOwner) external virtual requiresAuth { owner = newOwner; emit OwnerUpdated(msg.sender, newOwner); } function canCall( address user, bytes4 functionSig ) public view virtual returns (bool); } // SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.0; import {Auth} from "./Auth.sol"; /// @notice Role based Authority that supports up to 256 roles, target is always this. /// @author Modified from Solmate (https://github.com/Rari-Capital/solmate/blob/main/src/auth/authorities/RolesAuthority.sol) /// @author Modified from Dappsys (https://github.com/dapphub/ds-roles/blob/master/src/roles.sol) contract RolesAuthority is Auth { /*////////////////////////////////////////////////////////////// EVENTS //////////////////////////////////////////////////////////////*/ event UserRoleUpdated(address indexed user, uint8 indexed role, bool enabled); event PublicCapabilityUpdated(bytes4 indexed functionSig, bool enabled); event RoleCapabilityUpdated(uint8 indexed role, bytes4 indexed functionSig, bool enabled); /*////////////////////////////////////////////////////////////// CONSTRUCTOR //////////////////////////////////////////////////////////////*/ constructor(address _owner) Auth(_owner) {} /*////////////////////////////////////////////////////////////// ROLE/USER STORAGE //////////////////////////////////////////////////////////////*/ mapping(address => bytes32) public getUserRoles; mapping(bytes4 => bool) public isCapabilityPublic; mapping(bytes4 => bytes32) public getRolesWithCapability; function doesUserHaveRole(address user, uint8 role) public view virtual returns (bool) { return (uint256(getUserRoles[user]) >> role) & 1 != 0; } function doesRoleHaveCapability( uint8 role, bytes4 functionSig ) external view virtual returns (bool) { return (uint256(getRolesWithCapability[functionSig]) >> role) & 1 != 0; } /*////////////////////////////////////////////////////////////// AUTHORIZATION LOGIC //////////////////////////////////////////////////////////////*/ function canCall( address user, bytes4 functionSig ) public view virtual override returns (bool) { return isCapabilityPublic[functionSig] || bytes32(0) != getUserRoles[user] & getRolesWithCapability[functionSig]; } /*////////////////////////////////////////////////////////////// ROLE CAPABILITY CONFIGURATION LOGIC //////////////////////////////////////////////////////////////*/ function setPublicCapability( bytes4 functionSig, bool enabled ) external virtual requiresAuth { isCapabilityPublic[functionSig] = enabled; emit PublicCapabilityUpdated(functionSig, enabled); } function setRoleCapability( uint8 role, bytes4 functionSig, bool enabled ) external virtual requiresAuth { if (enabled) { getRolesWithCapability[functionSig] |= bytes32(1 << role); } else { getRolesWithCapability[functionSig] &= ~bytes32(1 << role); } emit RoleCapabilityUpdated(role, functionSig, enabled); } /*////////////////////////////////////////////////////////////// USER ROLE ASSIGNMENT LOGIC //////////////////////////////////////////////////////////////*/ function setUserRole( address user, uint8 role, bool enabled ) external virtual requiresAuth { if (enabled) { getUserRoles[user] |= bytes32(1 << role); } else { getUserRoles[user] &= ~bytes32(1 << role); } emit UserRoleUpdated(user, role, enabled); } } // SPDX-License-Identifier: AGPL-3.0 pragma solidity ^0.8.0; import {ERC20} from "./lib/solmate/ERC20.sol"; import {RolesAuthority} from "./RolesAuthority.sol"; /* The purpose of this token is to temporarily be nontransferable except for special cases. This is done by role-based access control. The token implements its own authorisation logic (by inheriting from RolesAuthority). It points to itself as its authority. The owner is then able to define who can call what functions of the token, then make the function public at a later stage (at an extra cost of 1 SLOAD). */ contract Token is ERC20, RolesAuthority { // We pass all arguments to the ancestors except the authority argument which is the token itself. constructor( string memory _name, string memory _symbol, uint8 _decimals, address _owner ) ERC20(_name, _symbol, _decimals) RolesAuthority(_owner) { _mint(_owner, 100_000_000 * 1e18); } // `transfer` now `requiresAuth`. function transfer(address to, uint256 amount) public override requiresAuth returns (bool) { return super.transfer(to, amount); } // `transferFrom` now `requiresAuth`. function transferFrom( address from, address to, uint256 amount ) public override requiresAuth returns (bool) { return super.transferFrom(from, to, amount); } // `mint` is added to the external interface, and also `requiresAuth` function mint(address to, uint256 amount) external requiresAuth { _mint(to, amount); } // `burn` is added to the external interface function burn(uint256 amount) external { _burn(msg.sender, amount); } // manually distribute the token based on the off-chain calculated balances. function migration(address[] calldata users, uint256[] calldata balances) external requiresAuth { uint256 length = users.length; address admin = owner; for (uint256 i; i < length; ++i) { address user = users[i]; uint256 balance = balances[i]; unchecked { balanceOf[user] += balance; balanceOf[admin] -= balance; } emit Transfer(admin, user, balance); } } } // SPDX-License-Identifier: AGPL-3.0 pragma solidity >=0.8.0; /// @notice Modern and gas efficient ERC20 + EIP-2612 implementation. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC20.sol) /// @author Modified from Uniswap (https://github.com/Uniswap/uniswap-v2-core/blob/master/contracts/UniswapV2ERC20.sol) /// @dev Do not manually set balances without updating totalSupply, as the sum of all user balances must not exceed it. abstract contract ERC20 { /*////////////////////////////////////////////////////////////// EVENTS //////////////////////////////////////////////////////////////*/ event Transfer(address indexed from, address indexed to, uint256 amount); event Approval(address indexed owner, address indexed spender, uint256 amount); /*////////////////////////////////////////////////////////////// METADATA STORAGE //////////////////////////////////////////////////////////////*/ string public name; string public symbol; uint8 public immutable decimals; /*////////////////////////////////////////////////////////////// ERC20 STORAGE //////////////////////////////////////////////////////////////*/ uint256 public totalSupply; mapping(address => uint256) public balanceOf; mapping(address => mapping(address => uint256)) public allowance; /*////////////////////////////////////////////////////////////// EIP-2612 STORAGE //////////////////////////////////////////////////////////////*/ uint256 internal immutable INITIAL_CHAIN_ID; bytes32 internal immutable INITIAL_DOMAIN_SEPARATOR; mapping(address => uint256) public nonces; /*////////////////////////////////////////////////////////////// CONSTRUCTOR //////////////////////////////////////////////////////////////*/ constructor( string memory _name, string memory _symbol, uint8 _decimals ) { name = _name; symbol = _symbol; decimals = _decimals; INITIAL_CHAIN_ID = block.chainid; INITIAL_DOMAIN_SEPARATOR = computeDomainSeparator(); } /*////////////////////////////////////////////////////////////// ERC20 LOGIC //////////////////////////////////////////////////////////////*/ function approve(address spender, uint256 amount) public virtual returns (bool) { allowance[msg.sender][spender] = amount; emit Approval(msg.sender, spender, amount); return true; } function transfer(address to, uint256 amount) public virtual returns (bool) { balanceOf[msg.sender] -= amount; // Cannot overflow because the sum of all user // balances can't exceed the max uint256 value. unchecked { balanceOf[to] += amount; } emit Transfer(msg.sender, to, amount); return true; } function transferFrom( address from, address to, uint256 amount ) public virtual returns (bool) { uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount; balanceOf[from] -= amount; // Cannot overflow because the sum of all user // balances can't exceed the max uint256 value. unchecked { balanceOf[to] += amount; } emit Transfer(from, to, amount); return true; } /*////////////////////////////////////////////////////////////// EIP-2612 LOGIC //////////////////////////////////////////////////////////////*/ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) public virtual { require(deadline >= block.timestamp, "PERMIT_DEADLINE_EXPIRED"); // Unchecked because the only math done is incrementing // the owner's nonce which cannot realistically overflow. unchecked { address recoveredAddress = ecrecover( keccak256( abi.encodePacked( "\\x19\\x01", DOMAIN_SEPARATOR(), keccak256( abi.encode( keccak256( "Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)" ), owner, spender, value, nonces[owner]++, deadline ) ) ) ), v, r, s ); require(recoveredAddress != address(0) && recoveredAddress == owner, "INVALID_SIGNER"); allowance[recoveredAddress][spender] = value; } emit Approval(owner, spender, value); } function DOMAIN_SEPARATOR() public view virtual returns (bytes32) { return block.chainid == INITIAL_CHAIN_ID ? INITIAL_DOMAIN_SEPARATOR : computeDomainSeparator(); } function computeDomainSeparator() internal view virtual returns (bytes32) { return keccak256( abi.encode( keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"), keccak256(bytes(name)), keccak256("1"), block.chainid, address(this) ) ); } /*////////////////////////////////////////////////////////////// INTERNAL MINT/BURN LOGIC //////////////////////////////////////////////////////////////*/ function _mint(address to, uint256 amount) internal virtual { totalSupply += amount; // Cannot overflow because the sum of all user // balances can't exceed the max uint256 value. unchecked { balanceOf[to] += amount; } emit Transfer(address(0), to, amount); } function _burn(address from, uint256 amount) internal virtual { balanceOf[from] -= amount; // Cannot underflow because a user's balance // will never be larger than the total supply. unchecked { totalSupply -= amount; } emit Transfer(from, address(0), amount); } }