编译器
0.8.16+commit.07a7930e
文件 1 的 3:ERC20.sol
pragma solidity >=0.8.0;
abstract contract ERC20 {
event Transfer(address indexed from, address indexed to, uint256 amount);
event Approval(address indexed owner, address indexed spender, uint256 amount);
string public name;
string public symbol;
uint8 public immutable decimals;
uint256 public totalSupply;
mapping(address => uint256) public balanceOf;
mapping(address => mapping(address => uint256)) public allowance;
uint256 internal immutable INITIAL_CHAIN_ID;
bytes32 internal immutable INITIAL_DOMAIN_SEPARATOR;
mapping(address => uint256) public nonces;
constructor(
string memory _name,
string memory _symbol,
uint8 _decimals
) {
name = _name;
symbol = _symbol;
decimals = _decimals;
INITIAL_CHAIN_ID = block.chainid;
INITIAL_DOMAIN_SEPARATOR = computeDomainSeparator();
}
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;
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];
if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount;
balanceOf[from] -= amount;
unchecked {
balanceOf[to] += amount;
}
emit Transfer(from, to, amount);
return true;
}
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 {
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)
)
);
}
function _mint(address to, uint256 amount) internal virtual {
totalSupply += amount;
unchecked {
balanceOf[to] += amount;
}
emit Transfer(address(0), to, amount);
}
function _burn(address from, uint256 amount) internal virtual {
balanceOf[from] -= amount;
unchecked {
totalSupply -= amount;
}
emit Transfer(from, address(0), amount);
}
}
文件 2 的 3:LlamaPayBot.sol
pragma solidity ^0.8.0;
import {ERC20} from "solmate/tokens/ERC20.sol";
import {SafeTransferLib} from "solmate/utils/SafeTransferLib.sol";
interface LlamaPay {
function withdraw(
address from,
address to,
uint216 amountPerSec
) external;
function withdrawable(
address from,
address to,
uint216 amountPerSec
)
external
view
returns (
uint256 withdrawableAmount,
uint256 lastUpdate,
uint256 owed
);
}
interface LlamaPayFactory {
function getLlamaPayContractByToken(address _token)
external
view
returns (address predictedAddress, bool isDeployed);
}
contract LlamaPayBot {
using SafeTransferLib for ERC20;
address public immutable factory;
address public bot;
address public llama;
address public newLlama = address(0);
uint256 public fee = 50000;
event WithdrawScheduled(
address owner,
address token,
address from,
address to,
uint216 amountPerSec,
uint40 starts,
uint40 frequency,
bytes32 id
);
event WithdrawCancelled(
address owner,
address token,
address from,
address to,
uint216 amountPerSec,
uint40 starts,
uint40 frequency,
bytes32 id
);
event WithdrawExecuted(
address owner,
address token,
address from,
address to,
uint216 amountPerSec,
uint40 starts,
uint40 frequency,
bytes32 id
);
mapping(address => uint256) public balances;
mapping(bytes32 => address) public owners;
mapping(address => address) public redirects;
constructor(
address _factory,
address _bot,
address _llama
) {
factory = _factory;
bot = _bot;
llama = _llama;
}
function deposit() external payable {
require(msg.sender != bot, "bot cannot deposit");
balances[msg.sender] += msg.value;
}
function refund() external {
uint256 toSend = balances[msg.sender];
balances[msg.sender] = 0;
(bool sent, ) = msg.sender.call{value: toSend}("");
require(sent, "failed to send ether");
}
function scheduleWithdraw(
address _token,
address _from,
address _to,
uint216 _amountPerSec,
uint40 _starts,
uint40 _frequency
) external returns (bytes32 id) {
id = calcWithdrawId(
_token,
_from,
_to,
_amountPerSec,
_starts,
_frequency
);
require(owners[id] == address(0), "already exists");
owners[id] = msg.sender;
emit WithdrawScheduled(
msg.sender,
_token,
_from,
_to,
_amountPerSec,
_starts,
_frequency,
id
);
}
function cancelWithdraw(
address _token,
address _from,
address _to,
uint216 _amountPerSec,
uint40 _starts,
uint40 _frequency
) external returns (bytes32 id) {
id = calcWithdrawId(
_token,
_from,
_to,
_amountPerSec,
_starts,
_frequency
);
require(msg.sender == owners[id], "not owner");
owners[id] = address(0);
emit WithdrawCancelled(
msg.sender,
_token,
_from,
_to,
_amountPerSec,
_starts,
_frequency,
id
);
}
function setRedirect(address _to) external {
redirects[msg.sender] = _to;
}
function cancelRedirect() external {
redirects[msg.sender] = address(0);
}
function executeWithdraw(
address _owner,
address _token,
address _from,
address _to,
uint216 _amountPerSec,
uint40 _starts,
uint40 _frequency,
bytes32 _id,
bool _execute,
bool _emitEvent
) external {
require(msg.sender == bot, "not bot");
if (_execute) {
(address llamapay, bool isDeployed) = LlamaPayFactory(factory)
.getLlamaPayContractByToken(_token);
require(isDeployed, "invalid llamapay contract");
if (redirects[_to] != address(0)) {
(uint256 withdrawableAmount, , ) = LlamaPay(llamapay)
.withdrawable(_from, _to, _amountPerSec);
LlamaPay(llamapay).withdraw(_from, _to, _amountPerSec);
ERC20(_token).safeTransferFrom(
_to,
redirects[_to],
withdrawableAmount
);
} else {
LlamaPay(llamapay).withdraw(_from, _to, _amountPerSec);
}
}
if (_emitEvent) {
emit WithdrawExecuted(
_owner,
_token,
_from,
_to,
_amountPerSec,
_starts,
_frequency,
_id
);
}
}
function execute(bytes[] calldata _calls, address _from) external {
require(msg.sender == bot, "not bot");
uint256 i;
uint256 len = _calls.length;
uint256 startGas = gasleft();
for (i = 0; i < len; ++i) {
address(this).delegatecall(_calls[i]);
}
uint256 gasUsed = ((startGas - gasleft()) + 21000) + fee;
uint256 totalSpent = gasUsed * tx.gasprice;
balances[_from] -= totalSpent;
(bool sent, ) = bot.call{value: totalSpent}("");
require(sent, "failed to send ether to bot");
}
function batchExecute(bytes[] calldata _calls) external {
require(msg.sender == bot, "not bot");
uint256 i;
uint256 len = _calls.length;
for (i = 0; i < len; ++i) {
address(this).delegatecall(_calls[i]);
}
}
function changeBot(address _newBot) external {
require(msg.sender == llama, "not llama");
bot = _newBot;
}
function changeLlama(address _newLlama) external {
require(msg.sender == llama, "not llama");
newLlama = _newLlama;
}
function confirmNewLlama() external {
require(msg.sender == newLlama, "not new llama");
llama = newLlama;
}
function changeFee(uint256 _newFee) external {
require(msg.sender == llama, "not llama");
fee = _newFee;
}
function calcWithdrawId(
address _token,
address _from,
address _to,
uint216 _amountPerSec,
uint40 _starts,
uint40 _frequency
) public pure returns (bytes32) {
return
keccak256(
abi.encodePacked(
_token,
_from,
_to,
_amountPerSec,
_starts,
_frequency
)
);
}
}
文件 3 的 3:SafeTransferLib.sol
pragma solidity >=0.8.0;
import {ERC20} from "../tokens/ERC20.sol";
library SafeTransferLib {
function safeTransferETH(address to, uint256 amount) internal {
bool success;
assembly {
success := call(gas(), to, amount, 0, 0, 0, 0)
}
require(success, "ETH_TRANSFER_FAILED");
}
function safeTransferFrom(
ERC20 token,
address from,
address to,
uint256 amount
) internal {
bool success;
assembly {
let freeMemoryPointer := mload(0x40)
mstore(freeMemoryPointer, 0x23b872dd00000000000000000000000000000000000000000000000000000000)
mstore(add(freeMemoryPointer, 4), from)
mstore(add(freeMemoryPointer, 36), to)
mstore(add(freeMemoryPointer, 68), amount)
success := and(
or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())),
call(gas(), token, 0, freeMemoryPointer, 100, 0, 32)
)
}
require(success, "TRANSFER_FROM_FAILED");
}
function safeTransfer(
ERC20 token,
address to,
uint256 amount
) internal {
bool success;
assembly {
let freeMemoryPointer := mload(0x40)
mstore(freeMemoryPointer, 0xa9059cbb00000000000000000000000000000000000000000000000000000000)
mstore(add(freeMemoryPointer, 4), to)
mstore(add(freeMemoryPointer, 36), amount)
success := and(
or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())),
call(gas(), token, 0, freeMemoryPointer, 68, 0, 32)
)
}
require(success, "TRANSFER_FAILED");
}
function safeApprove(
ERC20 token,
address to,
uint256 amount
) internal {
bool success;
assembly {
let freeMemoryPointer := mload(0x40)
mstore(freeMemoryPointer, 0x095ea7b300000000000000000000000000000000000000000000000000000000)
mstore(add(freeMemoryPointer, 4), to)
mstore(add(freeMemoryPointer, 36), amount)
success := and(
or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())),
call(gas(), token, 0, freeMemoryPointer, 68, 0, 32)
)
}
require(success, "APPROVE_FAILED");
}
}
{
"compilationTarget": {
"src/LlamaPayBot.sol": "LlamaPayBot"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":ds-test/=lib/solmate/lib/ds-test/src/",
":forge-std/=lib/forge-std/src/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/",
":solmate/=lib/solmate/src/"
]
}
[{"inputs":[{"internalType":"address","name":"_factory","type":"address"},{"internalType":"address","name":"_bot","type":"address"},{"internalType":"address","name":"_llama","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"owner","type":"address"},{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"address","name":"from","type":"address"},{"indexed":false,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint216","name":"amountPerSec","type":"uint216"},{"indexed":false,"internalType":"uint40","name":"starts","type":"uint40"},{"indexed":false,"internalType":"uint40","name":"frequency","type":"uint40"},{"indexed":false,"internalType":"bytes32","name":"id","type":"bytes32"}],"name":"WithdrawCancelled","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"owner","type":"address"},{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"address","name":"from","type":"address"},{"indexed":false,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint216","name":"amountPerSec","type":"uint216"},{"indexed":false,"internalType":"uint40","name":"starts","type":"uint40"},{"indexed":false,"internalType":"uint40","name":"frequency","type":"uint40"},{"indexed":false,"internalType":"bytes32","name":"id","type":"bytes32"}],"name":"WithdrawExecuted","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"owner","type":"address"},{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"address","name":"from","type":"address"},{"indexed":false,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint216","name":"amountPerSec","type":"uint216"},{"indexed":false,"internalType":"uint40","name":"starts","type":"uint40"},{"indexed":false,"internalType":"uint40","name":"frequency","type":"uint40"},{"indexed":false,"internalType":"bytes32","name":"id","type":"bytes32"}],"name":"WithdrawScheduled","type":"event"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"balances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes[]","name":"_calls","type":"bytes[]"}],"name":"batchExecute","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"bot","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint216","name":"_amountPerSec","type":"uint216"},{"internalType":"uint40","name":"_starts","type":"uint40"},{"internalType":"uint40","name":"_frequency","type":"uint40"}],"name":"calcWithdrawId","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"cancelRedirect","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint216","name":"_amountPerSec","type":"uint216"},{"internalType":"uint40","name":"_starts","type":"uint40"},{"internalType":"uint40","name":"_frequency","type":"uint40"}],"name":"cancelWithdraw","outputs":[{"internalType":"bytes32","name":"id","type":"bytes32"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newBot","type":"address"}],"name":"changeBot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newFee","type":"uint256"}],"name":"changeFee","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newLlama","type":"address"}],"name":"changeLlama","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"confirmNewLlama","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"deposit","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"bytes[]","name":"_calls","type":"bytes[]"},{"internalType":"address","name":"_from","type":"address"}],"name":"execute","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_owner","type":"address"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint216","name":"_amountPerSec","type":"uint216"},{"internalType":"uint40","name":"_starts","type":"uint40"},{"internalType":"uint40","name":"_frequency","type":"uint40"},{"internalType":"bytes32","name":"_id","type":"bytes32"},{"internalType":"bool","name":"_execute","type":"bool"},{"internalType":"bool","name":"_emitEvent","type":"bool"}],"name":"executeWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"factory","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"fee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"llama","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"newLlama","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"name":"owners","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"redirects","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"refund","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint216","name":"_amountPerSec","type":"uint216"},{"internalType":"uint40","name":"_starts","type":"uint40"},{"internalType":"uint40","name":"_frequency","type":"uint40"}],"name":"scheduleWithdraw","outputs":[{"internalType":"bytes32","name":"id","type":"bytes32"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"}],"name":"setRedirect","outputs":[],"stateMutability":"nonpayable","type":"function"}]