编译器
0.8.19+commit.7dd6d404
文件 1 的 26:AccessControlModule.sol
pragma solidity ^0.8.17;
import { Sickle } from "contracts/Sickle.sol";
import { SickleFactory } from "contracts/SickleFactory.sol";
contract AccessControlModule {
SickleFactory public immutable factory;
error NotOwner(address sender);
error NotOwnerOrInternal();
error NotOwnerOrApproved();
error NotOwnerOrApprovedOrInternal();
error SickleNotDeployed();
error NotRegisteredSickle();
constructor(SickleFactory factory_) {
factory = factory_;
}
modifier onlyRegisteredSickle() {
if (factory.admins(address(this)) == address(0)) {
revert NotRegisteredSickle();
}
_;
}
modifier checkOwner(address sickleAddress) {
if (msg.sender != factory.admins(sickleAddress)) {
revert NotOwner(msg.sender);
}
_;
}
modifier checkOwnerOrApproved(address sickleAddress) {
Sickle sickle = Sickle(payable(sickleAddress));
if (factory.admins(sickleAddress) == address(0)) {
revert SickleNotDeployed();
}
if (!sickle.isOwnerOrApproved(msg.sender)) revert NotOwnerOrApproved();
_;
}
}
文件 2 的 26:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value
) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function _revert(bytes memory returndata, string memory errorMessage) private pure {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
文件 3 的 26:Admin.sol
pragma solidity ^0.8.17;
abstract contract Admin {
error NotAdminError();
event AdminSet(address oldAdmin, address newAdmin);
address public admin;
modifier onlyAdmin() {
if (msg.sender != admin) revert NotAdminError();
_;
}
constructor(address admin_) {
emit AdminSet(admin, admin_);
admin = admin_;
}
function setAdmin(address newAdmin) external onlyAdmin {
emit AdminSet(admin, newAdmin);
admin = newAdmin;
}
}
文件 4 的 26:Clones.sol
pragma solidity ^0.8.0;
library Clones {
function clone(address implementation) internal returns (address instance) {
assembly {
mstore(0x00, or(shr(0xe8, shl(0x60, implementation)), 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000))
mstore(0x20, or(shl(0x78, implementation), 0x5af43d82803e903d91602b57fd5bf3))
instance := create(0, 0x09, 0x37)
}
require(instance != address(0), "ERC1167: create failed");
}
function cloneDeterministic(address implementation, bytes32 salt) internal returns (address instance) {
assembly {
mstore(0x00, or(shr(0xe8, shl(0x60, implementation)), 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000))
mstore(0x20, or(shl(0x78, implementation), 0x5af43d82803e903d91602b57fd5bf3))
instance := create2(0, 0x09, 0x37, salt)
}
require(instance != address(0), "ERC1167: create2 failed");
}
function predictDeterministicAddress(
address implementation,
bytes32 salt,
address deployer
) internal pure returns (address predicted) {
assembly {
let ptr := mload(0x40)
mstore(add(ptr, 0x38), deployer)
mstore(add(ptr, 0x24), 0x5af43d82803e903d91602b57fd5bf3ff)
mstore(add(ptr, 0x14), implementation)
mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73)
mstore(add(ptr, 0x58), salt)
mstore(add(ptr, 0x78), keccak256(add(ptr, 0x0c), 0x37))
predicted := keccak256(add(ptr, 0x43), 0x55)
}
}
function predictDeterministicAddress(address implementation, bytes32 salt)
internal
view
returns (address predicted)
{
return predictDeterministicAddress(implementation, salt, address(this));
}
}
文件 5 的 26:ConnectorRegistry.sol
pragma solidity ^0.8.17;
import { Admin } from "contracts/base/Admin.sol";
import { TimelockAdmin } from "contracts/base/TimelockAdmin.sol";
error ConnectorNotRegistered(address target);
interface ICustomConnectorRegistry {
function connectorOf(address target) external view returns (address);
}
contract ConnectorRegistry is Admin, TimelockAdmin {
event ConnectorChanged(address target, address connector);
event CustomRegistryAdded(address registry);
event CustomRegistryRemoved(address registry);
error ConnectorAlreadySet(address target);
error ConnectorNotSet(address target);
ICustomConnectorRegistry[] public customRegistries;
mapping(ICustomConnectorRegistry => bool) public isCustomRegistry;
mapping(address target => address connector) private connectors_;
constructor(
address admin_,
address timelockAdmin_
) Admin(admin_) TimelockAdmin(timelockAdmin_) { }
function setConnectors(
address[] calldata targets,
address[] calldata connectors
) external onlyAdmin {
for (uint256 i; i != targets.length;) {
if (connectors_[targets[i]] != address(0)) {
revert ConnectorAlreadySet(targets[i]);
}
connectors_[targets[i]] = connectors[i];
emit ConnectorChanged(targets[i], connectors[i]);
unchecked {
++i;
}
}
}
function updateConnectors(
address[] calldata targets,
address[] calldata connectors
) external onlyTimelockAdmin {
for (uint256 i; i != targets.length;) {
if (connectors_[targets[i]] == address(0)) {
revert ConnectorNotSet(targets[i]);
}
connectors_[targets[i]] = connectors[i];
emit ConnectorChanged(targets[i], connectors[i]);
unchecked {
++i;
}
}
}
function addCustomRegistry(ICustomConnectorRegistry registry)
external
onlyAdmin
{
customRegistries.push(registry);
isCustomRegistry[registry] = true;
emit CustomRegistryAdded(address(registry));
}
function updateCustomRegistry(
uint256 index,
ICustomConnectorRegistry newRegistry
) external onlyTimelockAdmin {
address oldRegistry = address(customRegistries[index]);
isCustomRegistry[customRegistries[index]] = false;
emit CustomRegistryRemoved(oldRegistry);
customRegistries[index] = newRegistry;
isCustomRegistry[newRegistry] = true;
if (address(newRegistry) != address(0)) {
emit CustomRegistryAdded(address(newRegistry));
}
}
function connectorOf(address target) external view returns (address) {
address connector = connectors_[target];
if (connector != address(0)) {
return connector;
}
uint256 length = customRegistries.length;
for (uint256 i; i != length;) {
if (address(customRegistries[i]) != address(0)) {
try customRegistries[i].connectorOf(target) returns (
address _connector
) {
if (_connector != address(0)) {
return _connector;
}
} catch {
}
}
unchecked {
++i;
}
}
revert ConnectorNotRegistered(target);
}
function hasConnector(address target) external view returns (bool) {
if (connectors_[target] != address(0)) {
return true;
}
uint256 length = customRegistries.length;
for (uint256 i; i != length;) {
if (address(customRegistries[i]) != address(0)) {
try customRegistries[i].connectorOf(target) returns (
address _connector
) {
if (_connector != address(0)) {
return true;
}
} catch {
}
unchecked {
++i;
}
}
}
return false;
}
}
文件 6 的 26:DelegateModule.sol
pragma solidity ^0.8.17;
contract DelegateModule {
function _delegateTo(
address to,
bytes memory data
) internal returns (bytes memory) {
(bool success, bytes memory result) = to.delegatecall(data);
if (!success) {
if (result.length == 0) revert();
assembly {
revert(add(32, result), mload(result))
}
}
return result;
}
}
文件 7 的 26: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);
}
}
文件 8 的 26:FarmStrategy.sol
pragma solidity ^0.8.17;
import {
StrategyModule,
SickleFactory,
Sickle,
ConnectorRegistry
} from "contracts/modules/StrategyModule.sol";
import { IFarmConnector } from "contracts/interfaces/IFarmConnector.sol";
import { ZapLib, ZapInData, ZapOutData } from "contracts/libraries/ZapLib.sol";
import { FeesLib } from "contracts/libraries/FeesLib.sol";
import { TransferLib } from "contracts/libraries/TransferLib.sol";
import { SwapData } from "contracts/interfaces/ILiquidityConnector.sol";
import { SickleRegistry } from "contracts/SickleRegistry.sol";
import { SwapLib } from "contracts/libraries/SwapLib.sol";
library FarmStrategyFees {
bytes4 constant Deposit = bytes4(keccak256("FarmDepositFee"));
bytes4 constant Harvest = bytes4(keccak256("FarmHarvestFee"));
bytes4 constant Compound = bytes4(keccak256("FarmCompoundFee"));
bytes4 constant CompoundFor = bytes4(keccak256("FarmCompoundForFee"));
bytes4 constant Withdraw = bytes4(keccak256("FarmWithdrawFee"));
bytes4 constant Rebalance = bytes4(keccak256("FarmRebalanceFee"));
}
contract FarmStrategy is StrategyModule {
struct Libraries {
TransferLib transferLib;
SwapLib swapLib;
FeesLib feesLib;
ZapLib zapLib;
}
struct DepositParams {
address stakingContractAddress;
address[] tokensIn;
uint256[] amountsIn;
ZapInData zapData;
bytes extraData;
}
struct CompoundParams {
address claimContractAddress;
bytes claimExtraData;
address[] rewardTokens;
ZapInData zapData;
address depositContractAddress;
bytes depositExtraData;
}
struct WithdrawParams {
address stakingContractAddress;
bytes extraData;
ZapOutData zapData;
address[] tokensOut;
}
struct HarvestParams {
address stakingContractAddress;
SwapData[] swaps;
bytes extraData;
address[] tokensOut;
}
error TokenOutRequired();
ZapLib public immutable zapLib;
SwapLib public immutable swapLib;
TransferLib public immutable transferLib;
FeesLib public immutable feesLib;
address public immutable strategyAddress;
constructor(
SickleFactory factory,
ConnectorRegistry connectorRegistry,
Libraries memory libraries
) StrategyModule(factory, connectorRegistry) {
zapLib = libraries.zapLib;
swapLib = libraries.swapLib;
transferLib = libraries.transferLib;
feesLib = libraries.feesLib;
strategyAddress = address(this);
}
function deposit(
DepositParams calldata params,
address[] memory sweepTokens,
address approved,
bytes32 referralCode
) public payable {
if (params.tokensIn.length != params.amountsIn.length) {
revert SickleRegistry.ArrayLengthMismatch();
}
if (params.tokensIn.length == 0) {
revert TransferLib.TokenInRequired();
}
Sickle sickle = getOrDeploySickle(msg.sender, approved, referralCode);
address[] memory targets = new address[](4);
bytes[] memory data = new bytes[](4);
targets[0] = address(transferLib);
data[0] = abi.encodeCall(
TransferLib.transferTokensFromUser,
(
params.tokensIn,
params.amountsIn,
strategyAddress,
FarmStrategyFees.Deposit
)
);
targets[1] = address(zapLib);
data[1] = abi.encodeCall(ZapLib.zapIn, (params.zapData));
targets[2] =
connectorRegistry.connectorOf(params.stakingContractAddress);
data[2] = abi.encodeCall(
IFarmConnector.deposit,
(
params.stakingContractAddress,
params.zapData.addLiquidityData.lpToken,
params.extraData
)
);
if (sweepTokens.length > 0) {
targets[3] = address(transferLib);
data[3] =
abi.encodeCall(TransferLib.transferTokensToUser, (sweepTokens));
}
sickle.multicall{ value: msg.value }(targets, data);
}
function compound(
CompoundParams calldata params,
address[] memory sweepTokens
) external {
Sickle sickle = getSickle(msg.sender);
address[] memory targets = new address[](5);
bytes[] memory data = new bytes[](5);
address farmConnector =
connectorRegistry.connectorOf(params.claimContractAddress);
targets[0] = farmConnector;
data[0] = abi.encodeCall(
IFarmConnector.claim,
(params.claimContractAddress, params.claimExtraData)
);
targets[1] = address(feesLib);
data[1] = abi.encodeCall(
FeesLib.chargeFees,
(strategyAddress, FarmStrategyFees.Compound, params.rewardTokens)
);
targets[2] = address(zapLib);
data[2] = abi.encodeCall(ZapLib.zapIn, (params.zapData));
targets[3] = farmConnector;
data[3] = abi.encodeCall(
IFarmConnector.deposit,
(
params.depositContractAddress,
params.zapData.addLiquidityData.lpToken,
params.depositExtraData
)
);
if (sweepTokens.length > 0) {
targets[4] = address(transferLib);
data[4] =
abi.encodeCall(TransferLib.transferTokensToUser, (sweepTokens));
}
sickle.multicall(targets, data);
}
function compoundFor(
address sickleAddress,
CompoundParams calldata params,
address[] memory sweepTokens
) external checkOwnerOrApproved(sickleAddress) {
Sickle sickle = Sickle(payable(sickleAddress));
address[] memory targets = new address[](5);
bytes[] memory data = new bytes[](5);
address farmConnector =
connectorRegistry.connectorOf(params.claimContractAddress);
targets[0] = farmConnector;
data[0] = abi.encodeCall(
IFarmConnector.claim,
(params.claimContractAddress, params.claimExtraData)
);
targets[1] = address(feesLib);
data[1] = abi.encodeCall(
FeesLib.chargeFees,
(strategyAddress, FarmStrategyFees.CompoundFor, params.rewardTokens)
);
targets[2] = address(zapLib);
data[2] = abi.encodeCall(ZapLib.zapIn, (params.zapData));
targets[3] = farmConnector;
data[3] = abi.encodeCall(
IFarmConnector.deposit,
(
params.depositContractAddress,
params.zapData.addLiquidityData.lpToken,
params.depositExtraData
)
);
if (sweepTokens.length > 0) {
targets[4] = address(transferLib);
data[4] =
abi.encodeCall(TransferLib.transferTokensToUser, (sweepTokens));
}
sickle.multicall(targets, data);
}
function withdraw(
WithdrawParams calldata params,
address[] memory sweepTokens
) public {
if (params.tokensOut.length == 0) {
revert TokenOutRequired();
}
Sickle sickle = getSickle(msg.sender);
address[] memory targets = new address[](4);
bytes[] memory data = new bytes[](4);
address farmConnector =
connectorRegistry.connectorOf(params.stakingContractAddress);
targets[0] = farmConnector;
data[0] = abi.encodeCall(
IFarmConnector.withdraw,
(
params.stakingContractAddress,
params.zapData.removeLiquidityData.lpAmountIn,
params.extraData
)
);
targets[1] = address(zapLib);
data[1] = abi.encodeCall(ZapLib.zapOut, (params.zapData));
targets[2] = address(feesLib);
data[2] = abi.encodeCall(
FeesLib.chargeFees,
(strategyAddress, FarmStrategyFees.Withdraw, params.tokensOut)
);
if (sweepTokens.length > 0) {
targets[3] = address(transferLib);
data[3] =
abi.encodeCall(TransferLib.transferTokensToUser, (sweepTokens));
}
sickle.multicall(targets, data);
}
function harvest(
HarvestParams calldata params,
address[] memory sweepTokens
) public {
Sickle sickle = getSickle(msg.sender);
address[] memory targets = new address[](4);
bytes[] memory data = new bytes[](4);
address farmConnector =
connectorRegistry.connectorOf(params.stakingContractAddress);
targets[0] = farmConnector;
data[0] = abi.encodeCall(
IFarmConnector.claim,
(params.stakingContractAddress, params.extraData)
);
targets[1] = address(swapLib);
data[1] = abi.encodeCall(SwapLib.swapMultiple, (params.swaps));
targets[2] = address(feesLib);
data[2] = abi.encodeCall(
FeesLib.chargeFees,
(strategyAddress, FarmStrategyFees.Harvest, params.tokensOut)
);
if (sweepTokens.length > 0) {
targets[3] = address(transferLib);
data[3] =
abi.encodeCall(TransferLib.transferTokensToUser, (sweepTokens));
}
sickle.multicall(targets, data);
}
function exit(
HarvestParams calldata harvestParams,
WithdrawParams calldata withdrawParams,
address[] memory sweepTokens
) external {
harvest(harvestParams, new address[](0));
withdraw(withdrawParams, sweepTokens);
}
function rebalance(
HarvestParams calldata harvestParams,
WithdrawParams calldata withdrawParams,
DepositParams calldata depositParams,
address[] memory sweepTokens
) external {
if (withdrawParams.tokensOut.length == 0) {
revert TokenOutRequired();
}
Sickle sickle = getSickle(msg.sender);
address[] memory targets = new address[](8);
bytes[] memory data = new bytes[](8);
targets[0] =
connectorRegistry.connectorOf(harvestParams.stakingContractAddress);
data[0] = abi.encodeCall(
IFarmConnector.claim,
(harvestParams.stakingContractAddress, harvestParams.extraData)
);
targets[1] = address(feesLib);
data[1] = abi.encodeCall(
FeesLib.chargeFees,
(strategyAddress, FarmStrategyFees.Harvest, harvestParams.tokensOut)
);
targets[2] =
connectorRegistry.connectorOf(withdrawParams.stakingContractAddress);
data[2] = abi.encodeCall(
IFarmConnector.withdraw,
(
withdrawParams.stakingContractAddress,
withdrawParams.zapData.removeLiquidityData.lpAmountIn,
withdrawParams.extraData
)
);
targets[3] = address(zapLib);
data[3] = abi.encodeCall(ZapLib.zapOut, (withdrawParams.zapData));
targets[4] = address(feesLib);
data[4] = abi.encodeCall(
FeesLib.chargeFees,
(
strategyAddress,
FarmStrategyFees.Rebalance,
withdrawParams.tokensOut
)
);
targets[5] = address(zapLib);
data[5] = abi.encodeCall(ZapLib.zapIn, (depositParams.zapData));
targets[6] =
connectorRegistry.connectorOf(depositParams.stakingContractAddress);
data[6] = abi.encodeCall(
IFarmConnector.deposit,
(
depositParams.stakingContractAddress,
depositParams.zapData.addLiquidityData.lpToken,
depositParams.extraData
)
);
if (sweepTokens.length > 0) {
targets[7] = address(transferLib);
data[7] =
abi.encodeCall(TransferLib.transferTokensToUser, (sweepTokens));
}
sickle.multicall(targets, data);
}
}
文件 9 的 26:FeesLib.sol
pragma solidity ^0.8.17;
import { SafeTransferLib } from "solmate/utils/SafeTransferLib.sol";
import { WETH } from "solmate/tokens/WETH.sol";
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { Sickle } from "contracts/Sickle.sol";
import { SickleRegistry } from "contracts/SickleRegistry.sol";
contract FeesLib {
event FeeCharged(
address strategy, bytes4 feeDescriptor, uint256 amount, address token
);
event TransactionCostCharged(address recipient, uint256 amount);
uint256 public constant VERSION = 1;
SickleRegistry public immutable registry;
WETH public immutable weth;
address public constant ETH = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
constructor(SickleRegistry registry_, WETH weth_) {
registry = registry_;
weth = weth_;
}
function chargeFee(
address strategy,
bytes4 feeDescriptor,
address feeToken,
uint256 feeBasis
) public payable returns (uint256 remainder) {
uint256 fee = registry.feeRegistry(
keccak256(abi.encodePacked(strategy, feeDescriptor))
);
if (feeBasis == 0) {
if (feeToken == ETH) {
uint256 wethBalance = weth.balanceOf(address(this));
if (wethBalance > 0) {
weth.withdraw(wethBalance);
}
feeBasis = address(this).balance;
} else {
feeBasis = IERC20(feeToken).balanceOf(address(this));
}
}
if (fee == 0) {
return feeBasis;
}
uint256 amountToCharge = feeBasis * fee / 10_000;
if (feeToken == 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE) {
SafeTransferLib.safeTransferETH(
registry.collector(), amountToCharge
);
} else {
SafeTransferLib.safeTransfer(
feeToken, registry.collector(), amountToCharge
);
}
emit FeeCharged(strategy, feeDescriptor, amountToCharge, feeToken);
return feeBasis - amountToCharge;
}
function chargeFees(
address strategy,
bytes4 feeDescriptor,
address[] memory feeTokens
) external {
for (uint256 i = 0; i < feeTokens.length;) {
chargeFee(strategy, feeDescriptor, feeTokens[i], 0);
unchecked {
i++;
}
}
}
function getBalance(
Sickle sickle,
address token
) public view returns (uint256) {
if (token == ETH) {
return weth.balanceOf(address(sickle));
}
return IERC20(token).balanceOf(address(sickle));
}
}
文件 10 的 26:IERC20.sol
pragma solidity ^0.8.0;
import "../token/ERC20/IERC20.sol";
文件 11 的 26:IFarmConnector.sol
pragma solidity ^0.8.0;
interface IFarmConnector {
function deposit(
address target,
address token,
bytes memory extraData
) external payable;
function withdraw(
address target,
uint256 amount,
bytes memory extraData
) external;
function claim(address target, bytes memory extraData) external;
}
文件 12 的 26:ILiquidityConnector.sol
pragma solidity ^0.8.0;
struct AddLiquidityData {
address router;
address lpToken;
address[] tokens;
uint256[] desiredAmounts;
uint256[] minAmounts;
bytes extraData;
}
struct RemoveLiquidityData {
address router;
address lpToken;
address[] tokens;
uint256 lpAmountIn;
uint256[] minAmountsOut;
bytes extraData;
}
struct SwapData {
address router;
uint256 amountIn;
uint256 minAmountOut;
address tokenIn;
bytes extraData;
}
interface ILiquidityConnector {
function addLiquidity(AddLiquidityData memory addLiquidityData)
external
payable;
function removeLiquidity(RemoveLiquidityData memory removeLiquidityData)
external;
function swapExactTokensForTokens(SwapData memory swapData)
external
payable;
}
文件 13 的 26:Initializable.sol
pragma solidity ^0.8.2;
import "../../utils/Address.sol";
abstract contract Initializable {
uint8 private _initialized;
bool private _initializing;
event Initialized(uint8 version);
modifier initializer() {
bool isTopLevelCall = !_initializing;
require(
(isTopLevelCall && _initialized < 1) || (!Address.isContract(address(this)) && _initialized == 1),
"Initializable: contract is already initialized"
);
_initialized = 1;
if (isTopLevelCall) {
_initializing = true;
}
_;
if (isTopLevelCall) {
_initializing = false;
emit Initialized(1);
}
}
modifier reinitializer(uint8 version) {
require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
_initialized = version;
_initializing = true;
_;
_initializing = false;
emit Initialized(version);
}
modifier onlyInitializing() {
require(_initializing, "Initializable: contract is not initializing");
_;
}
function _disableInitializers() internal virtual {
require(!_initializing, "Initializable: contract is initializing");
if (_initialized < type(uint8).max) {
_initialized = type(uint8).max;
emit Initialized(type(uint8).max);
}
}
function _getInitializedVersion() internal view returns (uint8) {
return _initialized;
}
function _isInitializing() internal view returns (bool) {
return _initializing;
}
}
文件 14 的 26:MsgValueModule.sol
pragma solidity ^0.8.17;
contract MsgValueModule {
error IncorrectMsgValue();
function _checkMsgValue(uint256 inputAmount, bool isNative) internal {
if (
(isNative && inputAmount != msg.value)
|| (!isNative && msg.value > 0)
) {
revert IncorrectMsgValue();
}
}
}
文件 15 的 26:Multicall.sol
pragma solidity ^0.8.17;
import { SickleStorage } from "contracts/base/SickleStorage.sol";
import { SickleRegistry } from "contracts/SickleRegistry.sol";
abstract contract Multicall is SickleStorage {
error MulticallParamsMismatchError();
error TargetNotWhitelisted(address target);
error CallerNotWhitelisted(address caller);
SickleRegistry public immutable registry;
constructor(SickleRegistry registry_) initializer {
registry = registry_;
}
function multicall(
address[] calldata targets,
bytes[] calldata data
) external payable {
if (targets.length != data.length) {
revert MulticallParamsMismatchError();
}
if (!registry.isWhitelistedCaller(msg.sender)) {
revert CallerNotWhitelisted(msg.sender);
}
for (uint256 i = 0; i != data.length;) {
if (targets[i] == address(0)) {
unchecked {
++i;
}
continue;
}
if (targets[i] != address(this)) {
if (!registry.isWhitelistedTarget(targets[i])) {
revert TargetNotWhitelisted(targets[i]);
}
}
(bool success, bytes memory result) =
targets[i].delegatecall(data[i]);
if (!success) {
if (result.length == 0) revert();
assembly {
revert(add(32, result), mload(result))
}
}
unchecked {
++i;
}
}
}
}
文件 16 的 26:SafeTransferLib.sol
pragma solidity >=0.8.0;
import {ERC20} from "../tokens/ERC20.sol";
library SafeTransferLib {
error ETHTransferFailed();
error TransferFromFailed();
error TransferFailed();
error ApproveFailed();
function safeTransferETH(address to, uint256 amount) internal {
bool success;
assembly {
success := call(gas(), to, amount, 0, 0, 0, 0)
}
if (!success) revert ETHTransferFailed();
}
function safeTransferFrom(
address 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)
)
}
if (!success) revert TransferFromFailed();
}
function safeTransfer(
address 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)
)
}
if (!success) revert TransferFailed();
}
function safeApprove(
address 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)
)
}
if (!success) revert ApproveFailed();
}
}
文件 17 的 26:Sickle.sol
pragma solidity ^0.8.17;
import { SickleStorage } from "contracts/base/SickleStorage.sol";
import { Multicall } from "contracts/base/Multicall.sol";
import { SickleRegistry } from "contracts/SickleRegistry.sol";
contract Sickle is SickleStorage, Multicall {
receive() external payable { }
constructor(SickleRegistry sickleRegistry_)
initializer
Multicall(sickleRegistry_)
{
_Sickle_initialize(address(0), address(0));
}
function initialize(
address sickleOwner_,
address approved_
) external initializer {
_Sickle_initialize(sickleOwner_, approved_);
}
function _Sickle_initialize(
address sickleOwner_,
address approved_
) internal {
SickleStorage._SickleStorage_initialize(sickleOwner_, approved_);
}
function onERC721Received(
address,
address,
uint256,
bytes calldata
) external pure returns (bytes4) {
return this.onERC721Received.selector;
}
function onERC1155Received(
address,
address,
uint256,
uint256,
bytes calldata
) external pure returns (bytes4) {
return this.onERC1155Received.selector;
}
function onERC1155BatchReceived(
address,
address,
uint256[] calldata,
uint256[] calldata,
bytes calldata
) external pure returns (bytes4) {
return this.onERC1155BatchReceived.selector;
}
}
文件 18 的 26:SickleFactory.sol
pragma solidity ^0.8.17;
import { Clones } from "@openzeppelin/contracts/proxy/Clones.sol";
import { SickleRegistry } from "contracts/SickleRegistry.sol";
import { Sickle } from "contracts/Sickle.sol";
import { Admin } from "contracts/base/Admin.sol";
contract SickleFactory is Admin {
event Deploy(address indexed admin, address sickle);
error CallerNotWhitelisted(address caller);
error NotActive();
error SickleAlreadyDeployed();
mapping(address => address) private _sickles;
mapping(address => address) private _admins;
mapping(address => bytes32) public _referralCodes;
SickleRegistry public immutable registry;
address public immutable implementation;
SickleFactory public immutable previousFactory;
bool public isActive = true;
constructor(
address admin_,
address sickleRegistry_,
address sickleImplementation_,
address previousFactory_
) Admin(admin_) {
registry = SickleRegistry(sickleRegistry_);
implementation = sickleImplementation_;
previousFactory = SickleFactory(previousFactory_);
}
function setActive(bool active) external onlyAdmin {
isActive = active;
}
function _deploy(
address admin,
address approved,
bytes32 referralCode
) internal returns (address sickle) {
sickle = Clones.cloneDeterministic(
implementation, keccak256(abi.encode(admin))
);
Sickle(payable(sickle)).initialize(admin, approved);
_sickles[admin] = sickle;
_admins[sickle] = admin;
if (referralCode != bytes32(0)) {
_referralCodes[sickle] = referralCode;
}
emit Deploy(admin, sickle);
}
function _getSickle(address admin) internal returns (address sickle) {
sickle = _sickles[admin];
if (sickle != address(0)) {
return sickle;
}
if (address(previousFactory) != address(0)) {
sickle = previousFactory.sickles(admin);
if (sickle != address(0)) {
_sickles[admin] = sickle;
_admins[sickle] = admin;
_referralCodes[sickle] = previousFactory.referralCodes(sickle);
return sickle;
}
}
}
function predict(address admin) external view returns (address) {
bytes32 salt = keccak256(abi.encode(admin));
return Clones.predictDeterministicAddress(implementation, salt);
}
function sickles(address admin) external view returns (address sickle) {
sickle = _sickles[admin];
if (sickle == address(0) && address(previousFactory) != address(0)) {
sickle = previousFactory.sickles(admin);
}
}
function admins(address sickle) external view returns (address admin) {
admin = _admins[sickle];
if (admin == address(0) && address(previousFactory) != address(0)) {
admin = previousFactory.admins(sickle);
}
}
function referralCodes(address sickle)
external
view
returns (bytes32 referralCode)
{
referralCode = _referralCodes[sickle];
if (
referralCode == bytes32(0) && address(previousFactory) != address(0)
) {
referralCode = previousFactory.referralCodes(sickle);
}
}
function getOrDeploy(
address admin,
address approved,
bytes32 referralCode
) external returns (address sickle) {
if (!isActive) {
revert NotActive();
}
if (!registry.isWhitelistedCaller(msg.sender)) {
revert CallerNotWhitelisted(msg.sender);
}
if ((sickle = _getSickle(admin)) != address(0)) {
return sickle;
}
return _deploy(admin, approved, referralCode);
}
function deploy(
address approved,
bytes32 referralCode
) external returns (address sickle) {
if (!isActive) {
revert NotActive();
}
if (_getSickle(msg.sender) != address(0)) {
revert SickleAlreadyDeployed();
}
return _deploy(msg.sender, approved, referralCode);
}
}
文件 19 的 26:SickleRegistry.sol
pragma solidity ^0.8.17;
import { Admin } from "contracts/base/Admin.sol";
library SickleRegistryEvents {
event CollectorChanged(address newCollector);
event FeesUpdated(bytes32[] feeHashes, uint256[] feesInBP);
event ReferralCodeCreated(bytes32 indexed code, address indexed referrer);
event CallerStatusChanged(address caller, bool isWhitelisted);
event TargetStatusChanged(address target, bool isWhitelisted);
}
contract SickleRegistry is Admin {
error ArrayLengthMismatch();
error FeeAboveMaxLimit();
error InvalidReferralCode();
address public collector;
mapping(address => bool) public isWhitelistedTarget;
mapping(address => bool) public isWhitelistedCaller;
mapping(bytes32 => address) public referralCodes;
mapping(bytes32 => uint256) public feeRegistry;
constructor(address admin_, address collector_) Admin(admin_) {
collector = collector_;
}
function setWhitelistedTargets(
address[] calldata targets,
bool isApproved
) external onlyAdmin {
for (uint256 i; i < targets.length;) {
isWhitelistedTarget[targets[i]] = isApproved;
emit SickleRegistryEvents.TargetStatusChanged(
targets[i], isApproved
);
unchecked {
++i;
}
}
}
function updateCollector(address newCollector) external onlyAdmin {
collector = newCollector;
emit SickleRegistryEvents.CollectorChanged(newCollector);
}
function setWhitelistedCallers(
address[] calldata callers,
bool isApproved
) external onlyAdmin {
for (uint256 i; i < callers.length;) {
isWhitelistedCaller[callers[i]] = isApproved;
emit SickleRegistryEvents.CallerStatusChanged(
callers[i], isApproved
);
unchecked {
++i;
}
}
}
function setReferralCode(bytes32 referralCode) external {
if (referralCodes[referralCode] != address(0)) {
revert InvalidReferralCode();
}
referralCodes[referralCode] = msg.sender;
emit SickleRegistryEvents.ReferralCodeCreated(referralCode, msg.sender);
}
function setFees(
bytes32[] calldata feeHashes,
uint256[] calldata feesArray
) external onlyAdmin {
if (feeHashes.length != feesArray.length) {
revert ArrayLengthMismatch();
}
for (uint256 i = 0; i < feeHashes.length;) {
if (feesArray[i] <= 500) {
feeRegistry[feeHashes[i]] = feesArray[i];
} else {
revert FeeAboveMaxLimit();
}
unchecked {
++i;
}
}
emit SickleRegistryEvents.FeesUpdated(feeHashes, feesArray);
}
}
文件 20 的 26:SickleStorage.sol
pragma solidity ^0.8.17;
import { Initializable } from
"@openzeppelin/contracts/proxy/utils/Initializable.sol";
library SickleStorageEvents {
event ApprovedAddressChanged(address newApproved);
}
abstract contract SickleStorage is Initializable {
error NotOwnerError();
error NotStrategyError();
address public owner;
address public approved;
modifier onlyOwner() {
if (msg.sender != owner) revert NotOwnerError();
_;
}
function _SickleStorage_initialize(
address owner_,
address approved_
) internal onlyInitializing {
owner = owner_;
approved = approved_;
}
function setApproved(address newApproved) external onlyOwner {
approved = newApproved;
emit SickleStorageEvents.ApprovedAddressChanged(newApproved);
}
function isOwnerOrApproved(address caller) public view returns (bool) {
return caller == owner || caller == approved;
}
}
文件 21 的 26:StrategyModule.sol
pragma solidity ^0.8.17;
import { SickleFactory, Sickle } from "contracts/SickleFactory.sol";
import { ConnectorRegistry } from "contracts/ConnectorRegistry.sol";
import { AccessControlModule } from "contracts/modules/AccessControlModule.sol";
contract StrategyModule is AccessControlModule {
ConnectorRegistry public immutable connectorRegistry;
constructor(
SickleFactory factory,
ConnectorRegistry connectorRegistry_
) AccessControlModule(factory) {
connectorRegistry = connectorRegistry_;
}
function getSickle(address owner) public view returns (Sickle) {
Sickle sickle = Sickle(payable(factory.sickles(owner)));
if (address(sickle) == address(0)) {
revert SickleNotDeployed();
}
return sickle;
}
function getOrDeploySickle(
address owner,
address approved,
bytes32 referralCode
) public returns (Sickle) {
return
Sickle(payable(factory.getOrDeploy(owner, approved, referralCode)));
}
}
文件 22 的 26:SwapLib.sol
pragma solidity ^0.8.17;
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { SafeTransferLib } from "solmate/utils/SafeTransferLib.sol";
import { DelegateModule } from "contracts/modules/DelegateModule.sol";
import { ConnectorRegistry } from "contracts/ConnectorRegistry.sol";
import {
ILiquidityConnector,
SwapData
} from "contracts/interfaces/ILiquidityConnector.sol";
contract SwapLib is DelegateModule {
error SwapAmountZero();
ConnectorRegistry immutable connectorRegistry;
constructor(ConnectorRegistry connectorRegistry_) {
connectorRegistry = connectorRegistry_;
}
function swap(SwapData memory swapData) external payable {
_swap(swapData);
}
function swapMultiple(SwapData[] memory swapData) external {
uint256 swapDataLength = swapData.length;
for (uint256 i; i < swapDataLength;) {
_swap(swapData[i]);
unchecked {
i++;
}
}
}
function _swap(SwapData memory swapData) internal {
address tokenIn = swapData.tokenIn;
if (swapData.amountIn == 0) {
swapData.amountIn = IERC20(tokenIn).balanceOf(address(this));
}
if (swapData.amountIn == 0) {
revert SwapAmountZero();
}
SafeTransferLib.safeApprove(tokenIn, swapData.router, 0);
SafeTransferLib.safeApprove(tokenIn, swapData.router, swapData.amountIn);
address connectorAddress =
connectorRegistry.connectorOf(swapData.router);
ILiquidityConnector routerConnector =
ILiquidityConnector(connectorAddress);
_delegateTo(
address(routerConnector),
abi.encodeCall(routerConnector.swapExactTokensForTokens, swapData)
);
SafeTransferLib.safeApprove(tokenIn, swapData.router, 0);
}
}
文件 23 的 26:TimelockAdmin.sol
pragma solidity ^0.8.17;
abstract contract TimelockAdmin {
error NotTimelockAdminError();
event TimelockAdminSet(address oldTimelockAdmin, address newTimelockAdmin);
address public timelockAdmin;
modifier onlyTimelockAdmin() {
if (msg.sender != timelockAdmin) revert NotTimelockAdminError();
_;
}
constructor(address timelockAdmin_) {
emit TimelockAdminSet(timelockAdmin, timelockAdmin_);
timelockAdmin = timelockAdmin_;
}
function setTimelockAdmin(address newTimelockAdmin)
external
onlyTimelockAdmin
{
emit TimelockAdminSet(timelockAdmin, newTimelockAdmin);
timelockAdmin = newTimelockAdmin;
}
}
文件 24 的 26:TransferLib.sol
pragma solidity ^0.8.17;
import { MsgValueModule } from "contracts/modules/MsgValueModule.sol";
import { WETH } from "lib/solmate/src/tokens/WETH.sol";
import { Sickle } from "contracts/Sickle.sol";
import { SafeTransferLib } from "lib/solmate/src/utils/SafeTransferLib.sol";
import { IERC20 } from
"lib/openzeppelin-contracts/contracts/interfaces/IERC20.sol";
import { FeesLib } from "contracts/libraries/FeesLib.sol";
import { DelegateModule } from "contracts/modules/DelegateModule.sol";
contract TransferLib is MsgValueModule, DelegateModule {
error ArrayLengthMismatch();
error TokenInRequired();
address public constant ETH = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
WETH public immutable weth;
FeesLib public immutable feesLib;
constructor(FeesLib feseLib_, WETH weth_) {
feesLib = feseLib_;
weth = weth_;
}
function transferTokenToUser(address token) public payable {
address recipient = Sickle(payable(address(this))).owner();
if (token == address(0)) {
return;
}
if (token == ETH) {
uint256 wethBalance = weth.balanceOf(address(this));
if (wethBalance > 0) {
weth.withdraw(wethBalance);
}
if (address(this).balance > 0) {
SafeTransferLib.safeTransferETH(
recipient, address(this).balance
);
}
} else {
uint256 balance = IERC20(token).balanceOf(address(this));
if (balance > 0) {
SafeTransferLib.safeTransfer(token, recipient, balance);
}
}
}
function transferTokensToUser(address[] memory tokens) external payable {
for (uint256 i = 0; i != tokens.length;) {
transferTokenToUser(tokens[i]);
unchecked {
i++;
}
}
}
function transferTokenFromUser(
address tokenIn,
uint256 amountIn,
address strategy,
bytes4 feeSelector
) public payable {
_checkMsgValue(amountIn, tokenIn == ETH);
_transferTokenFromUser(tokenIn, amountIn, strategy, feeSelector);
}
function transferTokensFromUser(
address[] memory tokensIn,
uint256[] memory amountsIn,
address strategy,
bytes4 feeSelector
) external payable {
if (tokensIn.length != amountsIn.length) {
revert ArrayLengthMismatch();
}
if (tokensIn.length == 0) {
revert TokenInRequired();
}
bool hasEth = false;
for (uint256 i = 0; i < tokensIn.length; i++) {
if (tokensIn[i] == ETH) {
_checkMsgValue(amountsIn[i], true);
hasEth = true;
}
_transferTokenFromUser(
tokensIn[i], amountsIn[i], strategy, feeSelector
);
}
if (!hasEth) {
_checkMsgValue(0, false);
}
}
function _transferTokenFromUser(
address tokenIn,
uint256 amountIn,
address strategy,
bytes4 feeSelector
) internal {
if (tokenIn != ETH) {
SafeTransferLib.safeTransferFrom(
tokenIn,
Sickle(payable(address(this))).owner(),
address(this),
amountIn
);
}
bytes memory result = _delegateTo(
address(feesLib),
abi.encodeCall(
FeesLib.chargeFee, (strategy, feeSelector, tokenIn, 0)
)
);
uint256 remainder = abi.decode(result, (uint256));
if (tokenIn == ETH) {
weth.deposit{ value: remainder }();
}
}
}
文件 25 的 26:WETH.sol
pragma solidity >=0.8.0;
import {ERC20} from "./ERC20.sol";
import {SafeTransferLib} from "../utils/SafeTransferLib.sol";
contract WETH is ERC20("Wrapped Ether", "WETH", 18) {
using SafeTransferLib for address;
event Deposit(address indexed from, uint256 amount);
event Withdrawal(address indexed to, uint256 amount);
function deposit() public payable virtual {
_mint(msg.sender, msg.value);
emit Deposit(msg.sender, msg.value);
}
function withdraw(uint256 amount) public virtual {
_burn(msg.sender, amount);
emit Withdrawal(msg.sender, amount);
msg.sender.safeTransferETH(amount);
}
receive() external payable virtual {
deposit();
}
}
文件 26 的 26:ZapLib.sol
pragma solidity ^0.8.17;
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { SafeTransferLib } from "solmate/utils/SafeTransferLib.sol";
import {
ILiquidityConnector,
SwapData,
AddLiquidityData,
RemoveLiquidityData
} from "contracts/interfaces/ILiquidityConnector.sol";
import { SwapLib } from "contracts/libraries/SwapLib.sol";
import { ConnectorRegistry } from "contracts/ConnectorRegistry.sol";
import { DelegateModule } from "contracts/modules/DelegateModule.sol";
struct ZapInData {
SwapData[] swaps;
AddLiquidityData addLiquidityData;
}
struct ZapOutData {
RemoveLiquidityData removeLiquidityData;
SwapData[] swaps;
}
contract ZapLib is DelegateModule {
error LiquidityAmountError();
SwapLib public immutable swapLib;
ConnectorRegistry public immutable connectorRegistry;
constructor(ConnectorRegistry connectorRegistry_, SwapLib swapLib_) {
connectorRegistry = connectorRegistry_;
swapLib = swapLib_;
}
function zapIn(ZapInData memory zapData) external payable {
uint256 swapDataLength = zapData.swaps.length;
for (uint256 i; i < swapDataLength;) {
_delegateTo(
address(swapLib),
abi.encodeCall(SwapLib.swap, (zapData.swaps[i]))
);
unchecked {
i++;
}
}
if (zapData.addLiquidityData.lpToken == address(0)) {
return;
}
bool atLeastOneNonZero = false;
AddLiquidityData memory addLiquidityData = zapData.addLiquidityData;
uint256 addLiquidityDataTokensLength = addLiquidityData.tokens.length;
for (uint256 i; i < addLiquidityDataTokensLength; i++) {
if (addLiquidityData.tokens[i] == address(0)) {
continue;
}
if (addLiquidityData.desiredAmounts[i] == 0) {
addLiquidityData.desiredAmounts[i] =
IERC20(addLiquidityData.tokens[i]).balanceOf(address(this));
}
if (addLiquidityData.desiredAmounts[i] > 0) {
atLeastOneNonZero = true;
SafeTransferLib.safeApprove(
addLiquidityData.tokens[i], addLiquidityData.router, 0
);
SafeTransferLib.safeApprove(
addLiquidityData.tokens[i],
addLiquidityData.router,
addLiquidityData.desiredAmounts[i]
);
}
}
if (!atLeastOneNonZero) {
revert LiquidityAmountError();
}
address routerConnector =
connectorRegistry.connectorOf(addLiquidityData.router);
_delegateTo(
routerConnector,
abi.encodeCall(ILiquidityConnector.addLiquidity, (addLiquidityData))
);
for (uint256 i; i < addLiquidityDataTokensLength;) {
if (addLiquidityData.tokens[i] != address(0)) {
SafeTransferLib.safeApprove(
addLiquidityData.tokens[i], addLiquidityData.router, 0
);
}
unchecked {
i++;
}
}
}
function zapOut(ZapOutData memory zapData) external {
if (zapData.removeLiquidityData.lpToken != address(0)) {
if (zapData.removeLiquidityData.lpAmountIn > 0) {
SafeTransferLib.safeApprove(
zapData.removeLiquidityData.lpToken,
zapData.removeLiquidityData.router,
zapData.removeLiquidityData.lpAmountIn
);
}
address routerConnector = connectorRegistry.connectorOf(
zapData.removeLiquidityData.router
);
_delegateTo(
address(routerConnector),
abi.encodeCall(
ILiquidityConnector.removeLiquidity,
zapData.removeLiquidityData
)
);
}
uint256 swapDataLength = zapData.swaps.length;
for (uint256 i; i < swapDataLength;) {
_delegateTo(
address(swapLib),
abi.encodeCall(SwapLib.swap, (zapData.swaps[i]))
);
unchecked {
i++;
}
}
}
}
{
"compilationTarget": {
"contracts/strategies/FarmStrategy.sol": "FarmStrategy"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":@morpho-blue/=lib/morpho-blue/src/",
":@openzeppelin/=lib/openzeppelin-contracts/",
":@uniswap/v3-core/=lib/v3-core/",
":@uniswap/v3-periphery/=lib/v3-periphery/",
":ds-test/=lib/solmate/lib/ds-test/src/",
":forge-std/=lib/forge-std/src/",
":morpho-blue/=lib/morpho-blue/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/",
":solmate/=lib/solmate/src/"
]
}
[{"inputs":[{"internalType":"contract SickleFactory","name":"factory","type":"address"},{"internalType":"contract ConnectorRegistry","name":"connectorRegistry","type":"address"},{"components":[{"internalType":"contract TransferLib","name":"transferLib","type":"address"},{"internalType":"contract SwapLib","name":"swapLib","type":"address"},{"internalType":"contract FeesLib","name":"feesLib","type":"address"},{"internalType":"contract ZapLib","name":"zapLib","type":"address"}],"internalType":"struct FarmStrategy.Libraries","name":"libraries","type":"tuple"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ArrayLengthMismatch","type":"error"},{"inputs":[{"internalType":"address","name":"sender","type":"address"}],"name":"NotOwner","type":"error"},{"inputs":[],"name":"NotOwnerOrApproved","type":"error"},{"inputs":[],"name":"NotOwnerOrApprovedOrInternal","type":"error"},{"inputs":[],"name":"NotOwnerOrInternal","type":"error"},{"inputs":[],"name":"NotRegisteredSickle","type":"error"},{"inputs":[],"name":"SickleNotDeployed","type":"error"},{"inputs":[],"name":"TokenInRequired","type":"error"},{"inputs":[],"name":"TokenOutRequired","type":"error"},{"inputs":[{"components":[{"internalType":"address","name":"claimContractAddress","type":"address"},{"internalType":"bytes","name":"claimExtraData","type":"bytes"},{"internalType":"address[]","name":"rewardTokens","type":"address[]"},{"components":[{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address","name":"lpToken","type":"address"},{"internalType":"address[]","name":"tokens","type":"address[]"},{"internalType":"uint256[]","name":"desiredAmounts","type":"uint256[]"},{"internalType":"uint256[]","name":"minAmounts","type":"uint256[]"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct AddLiquidityData","name":"addLiquidityData","type":"tuple"}],"internalType":"struct ZapInData","name":"zapData","type":"tuple"},{"internalType":"address","name":"depositContractAddress","type":"address"},{"internalType":"bytes","name":"depositExtraData","type":"bytes"}],"internalType":"struct FarmStrategy.CompoundParams","name":"params","type":"tuple"},{"internalType":"address[]","name":"sweepTokens","type":"address[]"}],"name":"compound","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"sickleAddress","type":"address"},{"components":[{"internalType":"address","name":"claimContractAddress","type":"address"},{"internalType":"bytes","name":"claimExtraData","type":"bytes"},{"internalType":"address[]","name":"rewardTokens","type":"address[]"},{"components":[{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address","name":"lpToken","type":"address"},{"internalType":"address[]","name":"tokens","type":"address[]"},{"internalType":"uint256[]","name":"desiredAmounts","type":"uint256[]"},{"internalType":"uint256[]","name":"minAmounts","type":"uint256[]"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct AddLiquidityData","name":"addLiquidityData","type":"tuple"}],"internalType":"struct ZapInData","name":"zapData","type":"tuple"},{"internalType":"address","name":"depositContractAddress","type":"address"},{"internalType":"bytes","name":"depositExtraData","type":"bytes"}],"internalType":"struct FarmStrategy.CompoundParams","name":"params","type":"tuple"},{"internalType":"address[]","name":"sweepTokens","type":"address[]"}],"name":"compoundFor","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"connectorRegistry","outputs":[{"internalType":"contract ConnectorRegistry","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"stakingContractAddress","type":"address"},{"internalType":"address[]","name":"tokensIn","type":"address[]"},{"internalType":"uint256[]","name":"amountsIn","type":"uint256[]"},{"components":[{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address","name":"lpToken","type":"address"},{"internalType":"address[]","name":"tokens","type":"address[]"},{"internalType":"uint256[]","name":"desiredAmounts","type":"uint256[]"},{"internalType":"uint256[]","name":"minAmounts","type":"uint256[]"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct AddLiquidityData","name":"addLiquidityData","type":"tuple"}],"internalType":"struct ZapInData","name":"zapData","type":"tuple"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct FarmStrategy.DepositParams","name":"params","type":"tuple"},{"internalType":"address[]","name":"sweepTokens","type":"address[]"},{"internalType":"address","name":"approved","type":"address"},{"internalType":"bytes32","name":"referralCode","type":"bytes32"}],"name":"deposit","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"stakingContractAddress","type":"address"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"},{"internalType":"bytes","name":"extraData","type":"bytes"},{"internalType":"address[]","name":"tokensOut","type":"address[]"}],"internalType":"struct FarmStrategy.HarvestParams","name":"harvestParams","type":"tuple"},{"components":[{"internalType":"address","name":"stakingContractAddress","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"},{"components":[{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address","name":"lpToken","type":"address"},{"internalType":"address[]","name":"tokens","type":"address[]"},{"internalType":"uint256","name":"lpAmountIn","type":"uint256"},{"internalType":"uint256[]","name":"minAmountsOut","type":"uint256[]"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct RemoveLiquidityData","name":"removeLiquidityData","type":"tuple"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"}],"internalType":"struct ZapOutData","name":"zapData","type":"tuple"},{"internalType":"address[]","name":"tokensOut","type":"address[]"}],"internalType":"struct FarmStrategy.WithdrawParams","name":"withdrawParams","type":"tuple"},{"internalType":"address[]","name":"sweepTokens","type":"address[]"}],"name":"exit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"factory","outputs":[{"internalType":"contract SickleFactory","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"feesLib","outputs":[{"internalType":"contract FeesLib","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"approved","type":"address"},{"internalType":"bytes32","name":"referralCode","type":"bytes32"}],"name":"getOrDeploySickle","outputs":[{"internalType":"contract Sickle","name":"","type":"address"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"getSickle","outputs":[{"internalType":"contract Sickle","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"stakingContractAddress","type":"address"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"},{"internalType":"bytes","name":"extraData","type":"bytes"},{"internalType":"address[]","name":"tokensOut","type":"address[]"}],"internalType":"struct FarmStrategy.HarvestParams","name":"params","type":"tuple"},{"internalType":"address[]","name":"sweepTokens","type":"address[]"}],"name":"harvest","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"stakingContractAddress","type":"address"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"},{"internalType":"bytes","name":"extraData","type":"bytes"},{"internalType":"address[]","name":"tokensOut","type":"address[]"}],"internalType":"struct FarmStrategy.HarvestParams","name":"harvestParams","type":"tuple"},{"components":[{"internalType":"address","name":"stakingContractAddress","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"},{"components":[{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address","name":"lpToken","type":"address"},{"internalType":"address[]","name":"tokens","type":"address[]"},{"internalType":"uint256","name":"lpAmountIn","type":"uint256"},{"internalType":"uint256[]","name":"minAmountsOut","type":"uint256[]"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct RemoveLiquidityData","name":"removeLiquidityData","type":"tuple"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"}],"internalType":"struct ZapOutData","name":"zapData","type":"tuple"},{"internalType":"address[]","name":"tokensOut","type":"address[]"}],"internalType":"struct FarmStrategy.WithdrawParams","name":"withdrawParams","type":"tuple"},{"components":[{"internalType":"address","name":"stakingContractAddress","type":"address"},{"internalType":"address[]","name":"tokensIn","type":"address[]"},{"internalType":"uint256[]","name":"amountsIn","type":"uint256[]"},{"components":[{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address","name":"lpToken","type":"address"},{"internalType":"address[]","name":"tokens","type":"address[]"},{"internalType":"uint256[]","name":"desiredAmounts","type":"uint256[]"},{"internalType":"uint256[]","name":"minAmounts","type":"uint256[]"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct AddLiquidityData","name":"addLiquidityData","type":"tuple"}],"internalType":"struct ZapInData","name":"zapData","type":"tuple"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct FarmStrategy.DepositParams","name":"depositParams","type":"tuple"},{"internalType":"address[]","name":"sweepTokens","type":"address[]"}],"name":"rebalance","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"strategyAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"swapLib","outputs":[{"internalType":"contract SwapLib","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"transferLib","outputs":[{"internalType":"contract TransferLib","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"stakingContractAddress","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"},{"components":[{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address","name":"lpToken","type":"address"},{"internalType":"address[]","name":"tokens","type":"address[]"},{"internalType":"uint256","name":"lpAmountIn","type":"uint256"},{"internalType":"uint256[]","name":"minAmountsOut","type":"uint256[]"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct RemoveLiquidityData","name":"removeLiquidityData","type":"tuple"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"minAmountOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"bytes","name":"extraData","type":"bytes"}],"internalType":"struct SwapData[]","name":"swaps","type":"tuple[]"}],"internalType":"struct ZapOutData","name":"zapData","type":"tuple"},{"internalType":"address[]","name":"tokensOut","type":"address[]"}],"internalType":"struct FarmStrategy.WithdrawParams","name":"params","type":"tuple"},{"internalType":"address[]","name":"sweepTokens","type":"address[]"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"zapLib","outputs":[{"internalType":"contract ZapLib","name":"","type":"address"}],"stateMutability":"view","type":"function"}]