编译器
0.8.28+commit.7893614a
文件 1 的 11: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);
}
}
}
文件 2 的 11: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);
}
文件 3 的 11:IERC20Metadata.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
interface IERC20Metadata is IERC20 {
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
}
文件 4 的 11:IERC20Permit.sol
pragma solidity ^0.8.0;
interface IERC20Permit {
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
function nonces(address owner) external view returns (uint256);
function DOMAIN_SEPARATOR() external view returns (bytes32);
}
文件 5 的 11:IPSwapAggregator.sol
pragma solidity ^0.8.0;
struct SwapData {
SwapType swapType;
address extRouter;
bytes extCalldata;
bool needScale;
}
struct SwapDataExtra {
address tokenIn;
address tokenOut;
uint256 minOut;
SwapData swapData;
}
enum SwapType {
NONE,
KYBERSWAP,
ODOS,
ETH_WETH,
OKX,
ONE_INCH,
RESERVE_1,
RESERVE_2,
RESERVE_3,
RESERVE_4,
RESERVE_5
}
interface IPSwapAggregator {
event SwapSingle(SwapType indexed swapType, address indexed tokenIn, uint256 amountIn);
function swap(address tokenIn, uint256 amountIn, SwapData calldata swapData) external payable;
}
文件 6 的 11:IWETH.sol
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
interface IWETH is IERC20 {
event Deposit(address indexed dst, uint256 wad);
event Withdrawal(address indexed src, uint256 wad);
function deposit() external payable;
function withdraw(uint256 wad) external;
}
文件 7 的 11:OKXScaleHelper.sol
pragma solidity ^0.8.17;
import "../../core/libraries/TokenHelper.sol";
import "./IPSwapAggregator.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
abstract contract OKXScaleHelper {
function _okx_getTokenApprove() internal view returns (address) {
if (block.chainid == 1) {
return 0x40aA958dd87FC8305b97f2BA922CDdCa374bcD7f;
}
if (block.chainid == 10) {
return 0x68D6B739D2020067D1e2F713b999dA97E4d54812;
}
if (block.chainid == 56) {
return 0x2c34A2Fb1d0b4f55de51E1d0bDEfaDDce6b7cDD6;
}
if (block.chainid == 42161) {
return 0x70cBb871E8f30Fc8Ce23609E9E0Ea87B6b222F58;
}
if (block.chainid == 8453) {
return 0x57df6092665eb6058DE53939612413ff4B09114E;
}
if (block.chainid == 5000) {
return 0x57df6092665eb6058DE53939612413ff4B09114E;
}
revert("PendleSwap: OKX Chain not supported");
}
function _okxScaling(
bytes calldata rawCallData,
uint256 actualAmount
) internal pure returns (bytes memory scaledCallData) {
bytes4 selector = bytes4(rawCallData[:4]);
bytes calldata dataToDecode = rawCallData[4:];
if (selector == IOKXDexRouter.uniswapV3SwapTo.selector) {
(uint256 receiver, uint256 amount, uint256 minReturn, uint256[] memory pools) = abi.decode(
dataToDecode,
(uint256, uint256, uint256, uint256[])
);
minReturn = (minReturn * actualAmount) / amount;
amount = actualAmount;
return abi.encodeWithSelector(selector, receiver, amount, minReturn, pools);
} else if (selector == IOKXDexRouter.smartSwapTo.selector) {
(
uint256 orderId,
address receiver,
IOKXDexRouter.BaseRequest memory baseRequest,
uint256[] memory batchesAmount,
IOKXDexRouter.RouterPath[][] memory batches,
IOKXDexRouter.PMMSwapRequest[] memory extraData
) = abi.decode(
dataToDecode,
(
uint256,
address,
IOKXDexRouter.BaseRequest,
uint256[],
IOKXDexRouter.RouterPath[][],
IOKXDexRouter.PMMSwapRequest[]
)
);
batchesAmount = _scaleArray(batchesAmount, actualAmount, baseRequest.fromTokenAmount);
baseRequest.minReturnAmount = (baseRequest.minReturnAmount * actualAmount) / baseRequest.fromTokenAmount;
baseRequest.fromTokenAmount = actualAmount;
return abi.encodeWithSelector(selector, orderId, receiver, baseRequest, batchesAmount, batches, extraData);
} else if (selector == IOKXDexRouter.unxswapTo.selector) {
(uint256 srcToken, uint256 amount, uint256 minReturn, address receiver, bytes32[] memory pools) = abi
.decode(dataToDecode, (uint256, uint256, uint256, address, bytes32[]));
minReturn = (minReturn * actualAmount) / amount;
amount = actualAmount;
return abi.encodeWithSelector(selector, srcToken, amount, minReturn, receiver, pools);
} else if (selector == IOKXDexRouter.unxswapByOrderId.selector) {
(uint256 srcToken, uint256 amount, uint256 minReturn, bytes32[] memory pools) = abi.decode(
dataToDecode,
(uint256, uint256, uint256, bytes32[])
);
minReturn = (minReturn * actualAmount) / amount;
amount = actualAmount;
return abi.encodeWithSelector(selector, srcToken, amount, minReturn, pools);
} else if (selector == IOKXDexRouter.smartSwapByOrderId.selector) {
(
uint256 orderId,
IOKXDexRouter.BaseRequest memory baseRequest,
uint256[] memory batchesAmount,
IOKXDexRouter.RouterPath[][] memory batches,
IOKXDexRouter.PMMSwapRequest[] memory extraData
) = abi.decode(
dataToDecode,
(
uint256,
IOKXDexRouter.BaseRequest,
uint256[],
IOKXDexRouter.RouterPath[][],
IOKXDexRouter.PMMSwapRequest[]
)
);
batchesAmount = _scaleArray(batchesAmount, actualAmount, baseRequest.fromTokenAmount);
baseRequest.minReturnAmount = (baseRequest.minReturnAmount * actualAmount) / baseRequest.fromTokenAmount;
baseRequest.fromTokenAmount = actualAmount;
return abi.encodeWithSelector(selector, orderId, baseRequest, batchesAmount, batches, extraData);
} else {
revert("PendleSwap: OKX selector not supported");
}
}
function _scaleArray(
uint256[] memory arr,
uint256 newAmount,
uint256 oldAmount
) internal pure returns (uint256[] memory scaledArr) {
scaledArr = new uint256[](arr.length);
for (uint256 i = 0; i < arr.length; i++) {
scaledArr[i] = (arr[i] * newAmount) / oldAmount;
}
}
}
interface IOKXDexRouter {
struct BaseRequest {
uint256 fromToken;
address toToken;
uint256 fromTokenAmount;
uint256 minReturnAmount;
uint256 deadLine;
}
struct RouterPath {
address[] mixAdapters;
address[] assetTo;
uint256[] rawData;
bytes[] extraData;
uint256 fromToken;
}
struct PMMSwapRequest {
uint256 pathIndex;
address payer;
address fromToken;
address toToken;
uint256 fromTokenAmountMax;
uint256 toTokenAmountMax;
uint256 salt;
uint256 deadLine;
bool isPushOrder;
bytes extension;
}
function uniswapV3SwapTo(
uint256 receiver,
uint256 amount,
uint256 minReturn,
uint256[] calldata pools
) external payable returns (uint256 returnAmount);
function smartSwapTo(
uint256 orderId,
address receiver,
BaseRequest calldata baseRequest,
uint256[] calldata batchesAmount,
RouterPath[][] calldata batches,
PMMSwapRequest[] calldata extraData
) external payable;
function unxswapTo(
uint256 srcToken,
uint256 amount,
uint256 minReturn,
address receiver,
bytes32[] calldata pools
) external payable returns (uint256 returnAmount);
function unxswapByOrderId(
uint256 srcToken,
uint256 amount,
uint256 minReturn,
bytes32[] calldata pools
) external payable returns (uint256 returnAmount);
function smartSwapByOrderId(
uint256 orderId,
BaseRequest calldata baseRequest,
uint256[] calldata batchesAmount,
RouterPath[][] calldata batches,
PMMSwapRequest[] calldata extraData
) external payable returns (uint256 returnAmount);
}
文件 8 的 11:OneInchScaleHelper.sol
pragma solidity ^0.8.17;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
interface I1inchCommonType {
type Address256 is uint256;
}
abstract contract OneInchScaleHelper is I1inchCommonType {
function _rescaleMinAmount(
uint256 minAmount,
uint256 oldAmount,
uint256 newAmount
) internal pure returns (uint256) {
return (minAmount * newAmount) / oldAmount;
}
function _oneInchScaling(bytes calldata rawCallData, uint256 actualAmount) internal pure returns (bytes memory) {
bytes4 selector = bytes4(rawCallData[:4]);
bytes memory args = rawCallData[4:];
if (selector == I1inchAggregationRouterV6.unoswapTo.selector) {
(Address256 to, Address256 token, uint256 amount, uint256 minReturn, Address256 dex) = abi.decode(
args,
(Address256, Address256, uint256, uint256, Address256)
);
minReturn = (minReturn * actualAmount) / amount;
amount = actualAmount;
return abi.encodeWithSelector(selector, to, token, amount, minReturn, dex);
}
if (selector == I1inchAggregationRouterV6.unoswapTo2.selector) {
(Address256 to, Address256 token, uint256 amount, uint256 minReturn, Address256 dex, Address256 dex2) = abi
.decode(args, (Address256, Address256, uint256, uint256, Address256, Address256));
minReturn = (minReturn * actualAmount) / amount;
amount = actualAmount;
return abi.encodeWithSelector(selector, to, token, amount, minReturn, dex, dex2);
}
if (selector == I1inchAggregationRouterV6.unoswapTo3.selector) {
(
Address256 to,
Address256 token,
uint256 amount,
uint256 minReturn,
Address256 dex,
Address256 dex2,
Address256 dex3
) = abi.decode(args, (Address256, Address256, uint256, uint256, Address256, Address256, Address256));
minReturn = (minReturn * actualAmount) / amount;
amount = actualAmount;
return abi.encodeWithSelector(selector, to, token, amount, minReturn, dex, dex2, dex3);
}
if (selector == I1inchAggregationRouterV6.swap.selector) {
(address executor, I1inchAggregationRouterV6.SwapDescription memory desc, bytes memory data) = abi.decode(
args,
(address, I1inchAggregationRouterV6.SwapDescription, bytes)
);
desc.minReturnAmount = (desc.minReturnAmount * actualAmount) / desc.amount;
desc.amount = actualAmount;
return abi.encodeWithSelector(selector, executor, desc, data);
}
revert("1inch: unsupported selector");
}
}
interface I1inchAggregationRouterV6 is I1inchCommonType {
struct SwapDescription {
IERC20 srcToken;
IERC20 dstToken;
address payable srcReceiver;
address payable dstReceiver;
uint256 amount;
uint256 minReturnAmount;
uint256 flags;
}
function unoswapTo(
Address256 to,
Address256 token,
uint256 amount,
uint256 minReturn,
Address256 dex
) external returns (uint256 returnAmount);
function unoswapTo2(
Address256 to,
Address256 token,
uint256 amount,
uint256 minReturn,
Address256 dex,
Address256 dex2
) external returns (uint256 returnAmount);
function unoswapTo3(
Address256 to,
Address256 token,
uint256 amount,
uint256 minReturn,
Address256 dex,
Address256 dex2,
Address256 dex3
) external returns (uint256 returnAmount);
function swap(
address executor,
SwapDescription calldata desc,
bytes calldata data
) external payable returns (uint256 returnAmount, uint256 spentAmount);
}
文件 9 的 11:PendleSwap.sol
pragma solidity ^0.8.17;
import "../../core/libraries/TokenHelper.sol";
import "./IPSwapAggregator.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "./OKXScaleHelper.sol";
import "./OneInchScaleHelper.sol";
contract PendleSwap is IPSwapAggregator, TokenHelper, OKXScaleHelper, OneInchScaleHelper {
using Address for address;
using SafeERC20 for IERC20;
address private constant KYBER_SCALING_HELPER = 0x2f577A41BeC1BE1152AeEA12e73b7391d15f655D;
function swap(address tokenIn, uint256 amountIn, SwapData calldata data) external payable {
_approveForExtRouter(tokenIn, data);
data.extRouter.functionCallWithValue(
data.needScale ? _getScaledInputData(data.swapType, data.extCalldata, amountIn) : data.extCalldata,
tokenIn == NATIVE ? amountIn : 0
);
emit SwapSingle(data.swapType, tokenIn, amountIn);
}
function _approveForExtRouter(address token, SwapData calldata data) internal {
if (token == NATIVE) return;
if (data.swapType == SwapType.OKX) {
_safeApproveInfV2(IERC20(token), _okx_getTokenApprove());
} else {
_safeApproveInfV2(IERC20(token), data.extRouter);
}
}
function _safeApproveInfV2(IERC20 token, address spender) internal {
if (token.allowance(address(this), spender) < type(uint256).max) {
token.forceApprove(spender, type(uint256).max);
}
}
function _getScaledInputData(
SwapType swapType,
bytes calldata rawCallData,
uint256 amountIn
) internal view returns (bytes memory scaledCallData) {
if (swapType == SwapType.KYBERSWAP) {
bool isSuccess;
(isSuccess, scaledCallData) = IKyberScalingHelper(KYBER_SCALING_HELPER).getScaledInputData(
rawCallData,
amountIn
);
require(isSuccess, "PendleSwap: Kyber scaling failed");
} else if (swapType == SwapType.ODOS) {
scaledCallData = _odosScaling(rawCallData, amountIn);
} else if (swapType == SwapType.OKX) {
scaledCallData = _okxScaling(rawCallData, amountIn);
} else if (swapType == SwapType.ONE_INCH) {
scaledCallData = _oneInchScaling(rawCallData, amountIn);
} else {
assert(false);
}
}
function _odosScaling(
bytes calldata rawCallData,
uint256 amountIn
) internal pure returns (bytes memory scaledCallData) {
bytes4 selector = bytes4(rawCallData[:4]);
bytes calldata dataToDecode = rawCallData[4:];
assert(selector == IOdosRouterV2.swap.selector);
(
IOdosRouterV2.swapTokenInfo memory tokenInfo,
bytes memory pathDefinition,
address executor,
uint32 referralCode
) = abi.decode(dataToDecode, (IOdosRouterV2.swapTokenInfo, bytes, address, uint32));
tokenInfo.outputQuote = (tokenInfo.outputQuote * amountIn) / tokenInfo.inputAmount;
tokenInfo.outputMin = (tokenInfo.outputMin * amountIn) / tokenInfo.inputAmount;
tokenInfo.inputAmount = amountIn;
return abi.encodeWithSelector(selector, tokenInfo, pathDefinition, executor, referralCode);
}
receive() external payable {}
}
interface IKyberScalingHelper {
function getScaledInputData(
bytes calldata inputData,
uint256 newAmount
) external view returns (bool isSuccess, bytes memory data);
}
interface IOdosRouterV2 {
struct swapTokenInfo {
address inputToken;
uint256 inputAmount;
address inputReceiver;
address outputToken;
uint256 outputQuote;
uint256 outputMin;
address outputReceiver;
}
function swap(
swapTokenInfo memory tokenInfo,
bytes calldata pathDefinition,
address executor,
uint32 referralCode
) external payable returns (uint256 amountOut);
}
文件 10 的 11:SafeERC20.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/IERC20Permit.sol";
import "../../../utils/Address.sol";
library SafeERC20 {
using Address for address;
function safeTransfer(IERC20 token, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
function safeApprove(IERC20 token, address spender, uint256 value) internal {
require(
(value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
uint256 oldAllowance = token.allowance(address(this), spender);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
}
function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
}
}
function forceApprove(IERC20 token, address spender, uint256 value) internal {
bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value);
if (!_callOptionalReturnBool(token, approvalCall)) {
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
_callOptionalReturn(token, approvalCall);
}
}
function safePermit(
IERC20Permit token,
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) internal {
uint256 nonceBefore = token.nonces(owner);
token.permit(owner, spender, value, deadline, v, r, s);
uint256 nonceAfter = token.nonces(owner);
require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
}
function _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) {
(bool success, bytes memory returndata) = address(token).call(data);
return
success && (returndata.length == 0 || abi.decode(returndata, (bool))) && Address.isContract(address(token));
}
}
文件 11 的 11:TokenHelper.sol
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "../../interfaces/IWETH.sol";
abstract contract TokenHelper {
using SafeERC20 for IERC20;
address internal constant NATIVE = address(0);
uint256 internal constant LOWER_BOUND_APPROVAL = type(uint96).max / 2;
function _transferIn(address token, address from, uint256 amount) internal {
if (token == NATIVE) require(msg.value == amount, "eth mismatch");
else if (amount != 0) IERC20(token).safeTransferFrom(from, address(this), amount);
}
function _transferFrom(IERC20 token, address from, address to, uint256 amount) internal {
if (amount != 0) token.safeTransferFrom(from, to, amount);
}
function _transferOut(address token, address to, uint256 amount) internal {
if (amount == 0) return;
if (token == NATIVE) {
(bool success, ) = to.call{value: amount}("");
require(success, "eth send failed");
} else {
IERC20(token).safeTransfer(to, amount);
}
}
function _transferOut(address[] memory tokens, address to, uint256[] memory amounts) internal {
uint256 numTokens = tokens.length;
require(numTokens == amounts.length, "length mismatch");
for (uint256 i = 0; i < numTokens; ) {
_transferOut(tokens[i], to, amounts[i]);
unchecked {
i++;
}
}
}
function _selfBalance(address token) internal view returns (uint256) {
return (token == NATIVE) ? address(this).balance : IERC20(token).balanceOf(address(this));
}
function _selfBalance(IERC20 token) internal view returns (uint256) {
return token.balanceOf(address(this));
}
function _safeApprove(address token, address to, uint256 value) internal {
(bool success, bytes memory data) = token.call(abi.encodeWithSelector(IERC20.approve.selector, to, value));
require(success && (data.length == 0 || abi.decode(data, (bool))), "Safe Approve");
}
function _safeApproveInf(address token, address to) internal {
if (token == NATIVE) return;
if (IERC20(token).allowance(address(this), to) < LOWER_BOUND_APPROVAL) {
_safeApprove(token, to, 0);
_safeApprove(token, to, type(uint256).max);
}
}
function _wrap_unwrap_ETH(address tokenIn, address tokenOut, uint256 netTokenIn) internal {
if (tokenIn == NATIVE) IWETH(tokenOut).deposit{value: netTokenIn}();
else IWETH(tokenIn).withdraw(netTokenIn);
}
}
{
"compilationTarget": {
"contracts/pendle/contracts/router/swap-aggregator/PendleSwap.sol": "PendleSwap"
},
"evmVersion": "shanghai",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 1000000
},
"remappings": [],
"viaIR": true
}
[{"anonymous":false,"inputs":[{"indexed":true,"internalType":"enum SwapType","name":"swapType","type":"uint8"},{"indexed":true,"internalType":"address","name":"tokenIn","type":"address"},{"indexed":false,"internalType":"uint256","name":"amountIn","type":"uint256"}],"name":"SwapSingle","type":"event"},{"inputs":[{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"uint256","name":"amountIn","type":"uint256"},{"components":[{"internalType":"enum SwapType","name":"swapType","type":"uint8"},{"internalType":"address","name":"extRouter","type":"address"},{"internalType":"bytes","name":"extCalldata","type":"bytes"},{"internalType":"bool","name":"needScale","type":"bool"}],"internalType":"struct SwapData","name":"data","type":"tuple"}],"name":"swap","outputs":[],"stateMutability":"payable","type":"function"},{"stateMutability":"payable","type":"receive"}]