编译器
0.8.19+commit.7dd6d404
文件 1 的 27:AccessControlModule.sol
pragma solidity ^0.8.17;
import "../../Sickle.sol";
import "../../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();
_;
}
modifier checkOwnerOrInternal(address sickleAddress) {
Sickle sickle = Sickle(payable(sickleAddress));
if (factory.admins(sickleAddress) == address(0)) {
revert SickleNotDeployed();
}
if (
msg.sender != factory.admins(sickleAddress)
&& msg.sender != sickleAddress
) revert NotOwnerOrInternal();
_;
}
modifier checkOwnerOrApprovedOrInternal(address sickleAddress) {
Sickle sickle = Sickle(payable(sickleAddress));
if (factory.admins(sickleAddress) == address(0)) {
revert SickleNotDeployed();
}
if (
!sickle.isOwnerOrApproved(msg.sender) && msg.sender != sickleAddress
) revert NotOwnerOrApprovedOrInternal();
_;
}
function getSickle(address owner) internal view returns (Sickle) {
Sickle sickle = Sickle(payable(factory.sickles(owner)));
if (address(sickle) == address(0)) {
revert SickleNotDeployed();
}
return sickle;
}
}
文件 2 的 27: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 的 27: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 的 27: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 的 27:ConnectorRegistry.sol
pragma solidity ^0.8.17;
import "./base/Admin.sol";
import "./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 的 27: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 的 27: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 的 27:FarmStrategy.sol
pragma solidity ^0.8.17;
import "../Sickle.sol";
import "./modules/TransferModule.sol";
import "./modules/ZapModule.sol";
import "../interfaces/IFarmConnector.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 TransferModule, ZapModule {
error TokenOutRequired();
constructor(
SickleFactory factory,
FeesLib feesLib,
address wrappedNativeAddress,
ConnectorRegistry connectorRegistry
) ZapModule(factory, feesLib, wrappedNativeAddress, connectorRegistry) { }
struct DepositParams {
address stakingContractAddress;
address[] tokensIn;
uint256[] amountsIn;
ZapModule.ZapInData zapData;
bytes extraData;
}
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 TokenInRequired();
}
Sickle sickle = Sickle(
payable(factory.getOrDeploy(msg.sender, approved, referralCode))
);
address[] memory targets = new address[](4);
bytes[] memory data = new bytes[](4);
targets[0] = address(this);
data[0] = abi.encodeCall(
this._sickle_transfer_tokens_from_user,
(
params.tokensIn,
params.amountsIn,
address(this),
FarmStrategyFees.Deposit
)
);
targets[1] = address(this);
data[1] = abi.encodeCall(ZapModule._sickle_zap_in, (params.zapData));
targets[2] =
connectorRegistry.connectorOf(params.stakingContractAddress);
data[2] = abi.encodeCall(
IFarmConnector.deposit,
(
params.stakingContractAddress,
params.zapData.addLiquidityData.lpToken,
params.extraData
)
);
targets[3] = address(this);
data[3] =
abi.encodeCall(this._sickle_transfer_tokens_to_user, (sweepTokens));
sickle.multicall{ value: msg.value }(targets, data);
}
struct CompoundParams {
address claimContractAddress;
bytes claimExtraData;
address[] rewardTokens;
ZapModule.ZapInData zapData;
address depositContractAddress;
bytes depositExtraData;
}
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(this);
data[1] = abi.encodeCall(
this._sickle_charge_fees,
(address(this), FarmStrategyFees.Compound, params.rewardTokens)
);
targets[2] = address(this);
data[2] = abi.encodeCall(ZapModule._sickle_zap_in, (params.zapData));
targets[3] = farmConnector;
data[3] = abi.encodeCall(
IFarmConnector.deposit,
(
params.depositContractAddress,
params.zapData.addLiquidityData.lpToken,
params.depositExtraData
)
);
targets[4] = address(this);
data[4] =
abi.encodeCall(this._sickle_transfer_tokens_to_user, (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(this);
data[1] = abi.encodeCall(
this._sickle_charge_fees,
(address(this), FarmStrategyFees.CompoundFor, params.rewardTokens)
);
targets[2] = address(this);
data[2] = abi.encodeCall(ZapModule._sickle_zap_in, (params.zapData));
targets[3] = farmConnector;
data[3] = abi.encodeCall(
IFarmConnector.deposit,
(
params.depositContractAddress,
params.zapData.addLiquidityData.lpToken,
params.depositExtraData
)
);
targets[4] = address(this);
data[4] =
abi.encodeCall(this._sickle_transfer_tokens_to_user, (sweepTokens));
sickle.multicall(targets, data);
}
struct WithdrawParams {
address stakingContractAddress;
bytes extraData;
ZapModule.ZapOutData zapData;
address[] tokensOut;
}
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(this);
data[1] = abi.encodeCall(ZapModule._sickle_zap_out, (params.zapData));
targets[2] = address(this);
data[2] = abi.encodeCall(
this._sickle_charge_fees,
(address(this), FarmStrategyFees.Withdraw, params.tokensOut)
);
targets[3] = address(this);
data[3] =
abi.encodeCall(this._sickle_transfer_tokens_to_user, (sweepTokens));
sickle.multicall(targets, data);
}
struct HarvestParams {
address stakingContractAddress;
SwapData[] swaps;
bytes extraData;
address[] tokensOut;
}
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(this);
data[1] =
abi.encodeCall(SwapModule._sickle_swap_multiple, (params.swaps));
targets[2] = address(this);
data[2] = abi.encodeCall(
this._sickle_charge_fees,
(address(this), FarmStrategyFees.Harvest, params.tokensOut)
);
targets[3] = address(this);
data[3] =
abi.encodeCall(this._sickle_transfer_tokens_to_user, (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(this);
data[1] = abi.encodeCall(
this._sickle_charge_fees,
(address(this), 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(this);
data[3] =
abi.encodeCall(ZapModule._sickle_zap_out, (withdrawParams.zapData));
targets[4] = address(this);
data[4] = abi.encodeCall(
this._sickle_charge_fees,
(
address(this),
FarmStrategyFees.Rebalance,
withdrawParams.tokensOut
)
);
targets[5] = address(this);
data[5] =
abi.encodeCall(ZapModule._sickle_zap_in, (depositParams.zapData));
targets[6] =
connectorRegistry.connectorOf(depositParams.stakingContractAddress);
data[6] = abi.encodeCall(
IFarmConnector.deposit,
(
depositParams.stakingContractAddress,
depositParams.zapData.addLiquidityData.lpToken,
depositParams.extraData
)
);
targets[7] = address(this);
data[7] =
abi.encodeCall(this._sickle_transfer_tokens_to_user, (sweepTokens));
sickle.multicall(targets, data);
}
}
文件 9 的 27:FeesLib.sol
pragma solidity ^0.8.17;
import { SafeTransferLib } from "solmate/utils/SafeTransferLib.sol";
import { WETH } from "solmate/tokens/WETH.sol";
import "../SickleRegistry.sol";
contract FeesLib {
event FeeCharged(bytes32 feesHash, uint256 amount, address token);
event TransactionCostCharged(address recipient, uint256 amount);
uint256 public constant VERSION = 1;
SickleRegistry public immutable registry;
constructor(SickleRegistry registry_) {
registry = registry_;
}
function chargeFees(
bytes32 feeHash,
address tokenToCharge,
uint256 baseAmount
) public payable returns (uint256) {
uint256 fee = registry.feeRegistry(feeHash);
if (fee == 0) {
return baseAmount;
}
uint256 amountToCharge = baseAmount * fee / 10_000;
if (
tokenToCharge == address(0)
|| tokenToCharge == 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE
) {
SafeTransferLib.safeTransferETH(
registry.collector(), amountToCharge
);
} else {
SafeTransferLib.safeTransfer(
tokenToCharge, registry.collector(), amountToCharge
);
}
emit FeeCharged(feeHash, amountToCharge, tokenToCharge);
return baseAmount - amountToCharge;
}
function chargeTransactionCost(
address recipient,
address wrappedNative,
uint256 amountToCharge
) public payable {
WETH(payable(wrappedNative)).withdraw(amountToCharge);
SafeTransferLib.safeTransferETH(recipient, amountToCharge);
emit TransactionCostCharged(recipient, amountToCharge);
}
}
文件 10 的 27:FeesModule.sol
pragma solidity ^0.8.17;
import "../../libraries/FeesLib.sol";
import { IWETH9 } from "../../interfaces/external/IWETH.sol";
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "./DelegateModule.sol";
address constant ETH = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
contract FeesModule is DelegateModule {
FeesLib public immutable feesLib;
address public immutable wrappedNativeAddress;
constructor(FeesLib feesLib_, address wrappedNativeAddress_) {
feesLib = feesLib_;
wrappedNativeAddress = wrappedNativeAddress_;
}
function _charge_fees(
bytes32 feeHash,
address tokenToCharge,
uint256 baseAmount
) internal returns (uint256 remainingAmount) {
(remainingAmount) = abi.decode(
_delegateTo(
address(feesLib),
abi.encodeCall(
FeesLib.chargeFees, (feeHash, tokenToCharge, baseAmount)
)
),
(uint256)
);
}
function _sickle_charge_fee(
address strategy,
bytes4 feeDescriptor,
address feeToken
) public {
IWETH9 weth = IWETH9(wrappedNativeAddress);
uint256 feeBasis;
if (feeToken == ETH) {
weth.withdraw(weth.balanceOf(address(this)));
feeBasis = address(this).balance;
} else {
feeBasis = IERC20(feeToken).balanceOf(address(this));
}
_charge_fees(
keccak256(abi.encodePacked(strategy, feeDescriptor)),
feeToken,
feeBasis
);
}
function _sickle_charge_fees(
address strategy,
bytes4 feeDescriptor,
address[] memory feeTokens
) external {
for (uint256 i = 0; i < feeTokens.length;) {
_sickle_charge_fee(strategy, feeDescriptor, feeTokens[i]);
unchecked {
i++;
}
}
}
function _sickle_charge_transaction_cost(
address recipient,
address wrappedNative,
uint256 amountToCharge
) external {
_delegateTo(
address(feesLib),
abi.encodeCall(
FeesLib.chargeTransactionCost,
(recipient, wrappedNative, amountToCharge)
)
);
}
}
文件 11 的 27:IERC20.sol
pragma solidity ^0.8.0;
interface IERC20 {
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address to, uint256 amount) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transferFrom(
address from,
address to,
uint256 amount
) external returns (bool);
}
文件 12 的 27: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;
}
文件 13 的 27: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;
}
文件 14 的 27:IWETH.sol
pragma solidity >=0.8.9;
interface IWETH9 {
function deposit() external payable;
function withdraw(uint256 wad) external;
function approve(address guy, uint256 wad) external returns (bool);
function balanceOf(address account) external view returns (uint256);
function allowance(
address account,
address spender
) external view returns (uint256);
}
文件 15 的 27: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;
}
}
文件 16 的 27: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();
}
}
}
文件 17 的 27:Multicall.sol
pragma solidity ^0.8.17;
import "../base/SickleStorage.sol";
import "../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;
}
}
}
}
文件 18 的 27: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();
}
}
文件 19 的 27:Sickle.sol
pragma solidity ^0.8.17;
import "./base/SickleStorage.sol";
import "./base/Multicall.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;
}
}
文件 20 的 27:SickleFactory.sol
pragma solidity ^0.8.17;
import "@openzeppelin/contracts/proxy/Clones.sol";
import "./Sickle.sol";
import "./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);
}
}
文件 21 的 27:SickleRegistry.sol
pragma solidity ^0.8.17;
import "./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);
}
}
文件 22 的 27:SickleStorage.sol
pragma solidity ^0.8.17;
import "@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;
}
}
文件 23 的 27:SwapModule.sol
pragma solidity ^0.8.17;
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { SafeTransferLib } from "solmate/utils/SafeTransferLib.sol";
import "../../ConnectorRegistry.sol";
import "../../interfaces/ILiquidityConnector.sol";
import "./TransferModule.sol";
contract SwapModule is TransferModule {
error SwapAmountZero();
ConnectorRegistry immutable connectorRegistry;
constructor(
SickleFactory factory,
FeesLib feesLib,
address wrappedNativeAddress,
ConnectorRegistry connectorRegistry_
) TransferModule(factory, feesLib, wrappedNativeAddress) {
connectorRegistry = connectorRegistry_;
}
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);
}
function _sickle_swap(SwapData memory swapData)
external
onlyRegisteredSickle
{
_swap(swapData);
}
function _sickle_swap_multiple(SwapData[] memory swapData)
external
onlyRegisteredSickle
{
for (uint256 i; i < swapData.length;) {
_swap(swapData[i]);
unchecked {
i++;
}
}
}
}
文件 24 的 27: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;
}
}
文件 25 的 27:TransferModule.sol
pragma solidity ^0.8.17;
import "./AccessControlModule.sol";
import "./FeesModule.sol";
import "./MsgValueModule.sol";
contract TransferModule is AccessControlModule, FeesModule, MsgValueModule {
error ArrayLengthMismatch();
error TokenInRequired();
constructor(
SickleFactory factory_,
FeesLib feesLib_,
address wrappedNativeAddress_
)
FeesModule(feesLib_, wrappedNativeAddress_)
AccessControlModule(factory_)
{ }
function _sickle_transfer_token_to_user(address token)
public
payable
onlyRegisteredSickle
{
address recipient = Sickle(payable(address(this))).owner();
if (token == address(0)) {
return;
}
if (token == ETH) {
IWETH9(wrappedNativeAddress).withdraw(
IWETH9(wrappedNativeAddress).balanceOf(address(this))
);
SafeTransferLib.safeTransferETH(recipient, address(this).balance);
} else if (token == wrappedNativeAddress) {
IWETH9(wrappedNativeAddress).withdraw(
IWETH9(wrappedNativeAddress).balanceOf(address(this))
);
SafeTransferLib.safeTransferETH(recipient, address(this).balance);
} else {
SafeTransferLib.safeTransfer(
token, recipient, IERC20(token).balanceOf(address(this))
);
}
}
function _sickle_transfer_tokens_to_user(address[] memory tokens)
external
payable
onlyRegisteredSickle
{
for (uint256 i = 0; i != tokens.length; i++) {
_sickle_transfer_token_to_user(tokens[i]);
}
}
function _transfer_token_from_user(
address tokenIn,
uint256 amountIn,
address strategy,
bytes4 feeSelector
) internal {
if (tokenIn != ETH) {
SafeTransferLib.safeTransferFrom(
tokenIn,
Sickle(payable(address(this))).owner(),
address(this),
amountIn
);
}
amountIn = _charge_fees(
keccak256(abi.encodePacked(strategy, feeSelector)),
tokenIn,
amountIn
);
if (tokenIn == ETH) {
IWETH9 weth = IWETH9(wrappedNativeAddress);
weth.deposit{ value: amountIn }();
}
}
function _sickle_transfer_token_from_user(
address tokenIn,
uint256 amountIn,
address strategy,
bytes4 feeSelector
) public payable onlyRegisteredSickle {
_checkMsgValue(amountIn, tokenIn == ETH);
_transfer_token_from_user(tokenIn, amountIn, strategy, feeSelector);
}
function _sickle_transfer_tokens_from_user(
address[] memory tokensIn,
uint256[] memory amountsIn,
address strategy,
bytes4 feeSelector
) external payable onlyRegisteredSickle {
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;
}
_transfer_token_from_user(
tokensIn[i], amountsIn[i], strategy, feeSelector
);
}
if (!hasEth) {
_checkMsgValue(0, false);
}
}
}
文件 26 的 27: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();
}
}
文件 27 的 27:ZapModule.sol
pragma solidity ^0.8.17;
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { SafeTransferLib } from "solmate/utils/SafeTransferLib.sol";
import "../../interfaces/ILiquidityConnector.sol";
import "./SwapModule.sol";
abstract contract ZapModule is SwapModule {
error LiquidityAmountError();
struct ZapInData {
SwapData[] swaps;
AddLiquidityData addLiquidityData;
}
struct ZapOutData {
RemoveLiquidityData removeLiquidityData;
SwapData[] swaps;
}
constructor(
SickleFactory factory,
FeesLib feesLib,
address wrappedNativeAddress,
ConnectorRegistry connectorRegistry
) SwapModule(factory, feesLib, wrappedNativeAddress, connectorRegistry) { }
function _sickle_zap_in(ZapInData memory zapData)
external
payable
onlyRegisteredSickle
{
for (uint256 i; i < zapData.swaps.length;) {
_swap(zapData.swaps[i]);
unchecked {
i++;
}
}
if (zapData.addLiquidityData.lpToken == address(0)) {
return;
}
bool atLeastOneNonZero = false;
AddLiquidityData memory addLiquidityData = zapData.addLiquidityData;
for (uint256 i; i < addLiquidityData.tokens.length; 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 < addLiquidityData.tokens.length; i++) {
if (addLiquidityData.tokens[i] != address(0)) {
SafeTransferLib.safeApprove(
addLiquidityData.tokens[i], addLiquidityData.router, 0
);
}
}
}
function _sickle_zap_out(ZapOutData memory zapData)
external
onlyRegisteredSickle
{
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
)
);
}
for (uint256 i; i < zapData.swaps.length;) {
_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 FeesLib","name":"feesLib","type":"address"},{"internalType":"address","name":"wrappedNativeAddress","type":"address"},{"internalType":"contract ConnectorRegistry","name":"connectorRegistry","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApproveFailed","type":"error"},{"inputs":[],"name":"ArrayLengthMismatch","type":"error"},{"inputs":[],"name":"ArrayLengthMismatch","type":"error"},{"inputs":[],"name":"ETHTransferFailed","type":"error"},{"inputs":[],"name":"IncorrectMsgValue","type":"error"},{"inputs":[],"name":"LiquidityAmountError","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":"SwapAmountZero","type":"error"},{"inputs":[],"name":"TokenInRequired","type":"error"},{"inputs":[],"name":"TokenOutRequired","type":"error"},{"inputs":[],"name":"TransferFailed","type":"error"},{"inputs":[],"name":"TransferFromFailed","type":"error"},{"inputs":[{"internalType":"address","name":"strategy","type":"address"},{"internalType":"bytes4","name":"feeDescriptor","type":"bytes4"},{"internalType":"address","name":"feeToken","type":"address"}],"name":"_sickle_charge_fee","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"strategy","type":"address"},{"internalType":"bytes4","name":"feeDescriptor","type":"bytes4"},{"internalType":"address[]","name":"feeTokens","type":"address[]"}],"name":"_sickle_charge_fees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"address","name":"wrappedNative","type":"address"},{"internalType":"uint256","name":"amountToCharge","type":"uint256"}],"name":"_sickle_charge_transaction_cost","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"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":"swapData","type":"tuple"}],"name":"_sickle_swap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"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":"swapData","type":"tuple[]"}],"name":"_sickle_swap_multiple","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"address","name":"strategy","type":"address"},{"internalType":"bytes4","name":"feeSelector","type":"bytes4"}],"name":"_sickle_transfer_token_from_user","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"_sickle_transfer_token_to_user","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address[]","name":"tokensIn","type":"address[]"},{"internalType":"uint256[]","name":"amountsIn","type":"uint256[]"},{"internalType":"address","name":"strategy","type":"address"},{"internalType":"bytes4","name":"feeSelector","type":"bytes4"}],"name":"_sickle_transfer_tokens_from_user","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address[]","name":"tokens","type":"address[]"}],"name":"_sickle_transfer_tokens_to_user","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"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 ZapModule.ZapInData","name":"zapData","type":"tuple"}],"name":"_sickle_zap_in","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"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 ZapModule.ZapOutData","name":"zapData","type":"tuple"}],"name":"_sickle_zap_out","outputs":[],"stateMutability":"nonpayable","type":"function"},{"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 ZapModule.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 ZapModule.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":[{"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 ZapModule.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 ZapModule.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":[{"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 ZapModule.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 ZapModule.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":[{"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 ZapModule.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":"wrappedNativeAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"}]