编译器
0.8.17+commit.8df45f5f
文件 1 的 20: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 的 20:BoxImmutables.sol
pragma solidity 0.8.17;
contract BoxImmutables {
address internal immutable STARGATE_ROUTER;
address internal immutable UNISWAP_ROUTER;
address internal immutable WRAPPED_NATIVE;
address internal immutable SG_ETH;
address internal immutable EXECUTOR;
constructor(
address _executor,
address _stargateRouter,
address _uniswapRouter,
address _wrappedNative,
address _sgEth
) {
EXECUTOR = _executor;
WRAPPED_NATIVE = _wrappedNative;
SG_ETH = _sgEth;
STARGATE_ROUTER = _stargateRouter;
UNISWAP_ROUTER = _uniswapRouter;
}
}
文件 3 的 20:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 4 的 20:Core.sol
pragma solidity 0.8.17;
import {ICore} from "./interfaces/ICore.sol";
import {Dispatcher} from "./base/Dispatcher.sol";
import {IERC20} from "@openzeppelin-contracts/contracts/interfaces/IERC20.sol";
import {SafeERC20} from "@openzeppelin-contracts/contracts/token/ERC20/utils/SafeERC20.sol";
import {LzBridgeData, TokenData} from "./lib/CoreStructs.sol";
import {IStargateReceiver} from "./interfaces/stargate/IStargateReceiver.sol";
import {IStargateRouter} from "./interfaces/stargate/IStargateRouter.sol";
import {IWrappedToken} from "./interfaces/IWrappedToken.sol";
import {FeeOperator} from "./utils/FeeOperator.sol";
import {Context} from "@openzeppelin-contracts/contracts/utils/Context.sol";
contract Core is ICore, FeeOperator, Dispatcher, IStargateReceiver {
constructor(
address _executor,
address _stargateRouter,
address _uniswapRouter,
address _wrappedNative,
address _sgETH,
address _trustedForwarder
) Dispatcher(_executor, _stargateRouter, _uniswapRouter, _wrappedNative, _sgETH, _trustedForwarder) {}
function swapAndExecute(address target, address paymentOperator, TokenData calldata tokenData)
external
payable
handleFees(0, tokenData.amountIn, tokenData.tokenIn, tokenData.nativeOut)
{
_receiveErc20(tokenData.amountIn, tokenData.tokenIn);
_swapAndExecute(_msgSender(), target, paymentOperator, block.timestamp, tokenData);
}
function bridgeAndExecute(
LzBridgeData calldata lzBridgeData,
TokenData calldata tokenData,
IStargateRouter.lzTxObj calldata lzTxObj
) external payable handleFees(lzBridgeData.fee, tokenData.amountIn, tokenData.tokenIn, tokenData.nativeOut) {
if (tokenData.nativeOut > 0) {
revert BridgeNativeOutNonZero();
}
_receiveErc20(tokenData.amountIn, tokenData.tokenIn);
address tokenIn = tokenData.tokenIn;
if (tokenData.tokenIn == address(0) && tokenData.tokenOut != SG_ETH) {
tokenIn = WRAPPED_NATIVE;
IWrappedToken(tokenIn).deposit{value: tokenData.amountIn}();
}
if (tokenIn != tokenData.tokenOut && !(tokenIn == address(0) && tokenData.tokenOut == SG_ETH)) {
_swapExactOutput(
_msgSender(), tokenIn, tokenData.amountIn, tokenData.amountOut, block.timestamp, tokenData.path
);
}
if (tokenIn == tokenData.tokenOut && tokenData.amountOut > tokenData.amountIn) {
revert BridgeOutputExceedsInput();
}
_approveAndBridge(tokenData.tokenOut, tokenData.amountOut, lzBridgeData, lzTxObj, tokenData.payload);
}
function sgReceive(
uint16,
bytes memory,
uint256,
address _token,
uint256 amountLD,
bytes memory payload
) external override {
if (_msgSender() != address(STARGATE_ROUTER)) {
revert OnlyStargateRouter();
}
(
address sender,
address target,
address _paymentToken,
address paymentOperator,
uint256 _amountOutMin,
bytes memory path,
bytes memory callData
) = abi.decode(payload, (address, address, address, address, uint256, bytes, bytes));
TokenData memory tokenData = TokenData(amountLD, _amountOutMin, 0, _token, _paymentToken, path, callData);
_swapAndExecute(sender, target, paymentOperator, block.timestamp, tokenData);
emit ReceivedOnDestination(_token, amountLD);
}
function _msgSender() internal view override (Context, Dispatcher) returns (address) {
return Dispatcher._msgSender();
}
function _msgData() internal view override (Context, Dispatcher) returns (bytes calldata) {
return Dispatcher._msgData();
}
function withdraw(IERC20 token) external onlyOwner {
SafeERC20.safeTransfer(token, _msgSender(), token.balanceOf(address(this)));
}
function withdrawEth() external onlyOwner {
(bool success,) = payable(_msgSender()).call{value: address(this).balance}("");
require(success, "Could not drain ETH");
}
receive() external payable {}
}
文件 5 的 20:CoreStructs.sol
pragma solidity 0.8.17;
struct LzBridgeData {
uint120 _srcPoolId;
uint120 _dstPoolId;
uint16 _dstChainId;
address _bridgeAddress;
uint96 fee;
}
struct TokenData {
uint256 amountIn;
uint256 amountOut;
uint256 nativeOut;
address tokenIn;
address tokenOut;
bytes path;
bytes payload;
}
文件 6 的 20:Dispatcher.sol
pragma solidity 0.8.17;
import {BoxImmutables} from "./BoxImmutables.sol";
import {IWrappedToken} from "../interfaces/IWrappedToken.sol";
import {IERC20} from "@openzeppelin-contracts/contracts/interfaces/IERC20.sol";
import {Ownable} from "@openzeppelin-contracts/contracts/access/Ownable.sol";
import {ISwapRouter} from "@uniswap/v3-periphery/contracts/interfaces/ISwapRouter.sol";
import {SafeERC20} from "@openzeppelin-contracts/contracts/token/ERC20/utils/SafeERC20.sol";
import {LzBridgeData, TokenData} from "../lib/CoreStructs.sol";
import {IStargateRouter} from "../interfaces/stargate/IStargateRouter.sol";
import {IExecutor} from "../interfaces/IExecutor.sol";
import {ERC2771Context} from "@openzeppelin-contracts/contracts/metatx/ERC2771Context.sol";
import {Context} from "@openzeppelin-contracts/contracts/utils/Context.sol";
contract Dispatcher is ERC2771Context, BoxImmutables, Ownable {
constructor(
address _executor,
address _stargateRouter,
address _uniswapRouter,
address _wrappedNative,
address _sgEth,
address _trustedForwarder
)
ERC2771Context(_trustedForwarder)
BoxImmutables(_executor, _stargateRouter, _uniswapRouter, _wrappedNative, _sgEth)
{}
function _msgSender() internal view virtual override (Context, ERC2771Context) returns (address) {
return ERC2771Context._msgSender();
}
function _msgData() internal view virtual override (Context, ERC2771Context) returns (bytes calldata) {
return ERC2771Context._msgData();
}
function _receiveErc20(uint256 amountIn, address tokenIn) internal {
if (tokenIn != address(0)) {
SafeERC20.safeTransferFrom(IERC20(tokenIn), _msgSender(), address(this), amountIn);
}
}
event BridgedExecutionUnsuccessful();
event RefundUnsuccessful();
error ExecutionUnsuccessful();
error SwapOutputExceedsInput();
error SwapUnsuccessful();
error BridgeOutputExceedsInput();
error BridgeNativeOutNonZero();
function _approveAndBridge(
address bridgeToken,
uint256 amountOut,
LzBridgeData calldata lzBridgeData,
IStargateRouter.lzTxObj calldata lzTxObj,
bytes calldata payload
) internal {
if (bridgeToken != SG_ETH) {
SafeERC20.safeIncreaseAllowance(IERC20(bridgeToken), STARGATE_ROUTER, amountOut);
}
IStargateRouter(STARGATE_ROUTER).swap{
value: bridgeToken == SG_ETH ? (lzBridgeData.fee + amountOut) : lzBridgeData.fee
}(
lzBridgeData._dstChainId,
lzBridgeData._srcPoolId,
lzBridgeData._dstPoolId,
payable(_msgSender()),
amountOut,
(amountOut * 994) / 1000,
lzTxObj,
abi.encodePacked(lzBridgeData._bridgeAddress),
payload
);
}
function _swapExactOutput(
address sender,
address tokenIn,
uint256 amountInMaximum,
uint256 amountOut,
uint256 deadline,
bytes memory path
) internal returns (bool success) {
if (tokenIn == address(0)) {
IWrappedToken(WRAPPED_NATIVE).deposit{value: amountInMaximum}();
tokenIn = WRAPPED_NATIVE;
}
SafeERC20.safeIncreaseAllowance(IERC20(tokenIn), UNISWAP_ROUTER, amountInMaximum);
ISwapRouter.ExactOutputParams memory params = ISwapRouter.ExactOutputParams({
path: path,
recipient: address(this),
deadline: deadline,
amountOut: amountOut,
amountInMaximum: amountInMaximum
});
success = true;
uint256 refund;
if (_msgSender() == STARGATE_ROUTER) {
try ISwapRouter(UNISWAP_ROUTER).exactOutput(params) returns (uint256 amountIn) {
refund = amountInMaximum - amountIn;
} catch {
refund = amountInMaximum;
success = false;
}
} else {
uint256 amountIn = ISwapRouter(UNISWAP_ROUTER).exactOutput(params);
refund = amountInMaximum - amountIn;
}
if (refund > 0) {
SafeERC20.safeDecreaseAllowance(IERC20(tokenIn), UNISWAP_ROUTER, refund);
SafeERC20.safeTransfer(IERC20(tokenIn), sender, refund);
}
}
function _swapAndExecute(
address sender,
address target,
address paymentOperator,
uint256 deadline,
TokenData memory data
) internal {
bool success = true;
if (data.tokenIn == data.tokenOut && data.amountOut > data.amountIn) {
if (_msgSender() == STARGATE_ROUTER) {
_refund(sender, data.tokenIn, data.amountIn);
success = false;
} else {
revert SwapOutputExceedsInput();
}
}
if (data.tokenIn != data.tokenOut) {
if (data.tokenIn == WRAPPED_NATIVE && data.tokenOut == address(0)) {
IWrappedToken(WRAPPED_NATIVE).withdraw(data.amountOut);
} else if (data.tokenIn != SG_ETH || data.tokenOut != address(0)) {
success = _swapExactOutput(sender, data.tokenIn, data.amountIn, data.amountOut, deadline, data.path);
if (success && data.tokenOut == address(0)) {
IWrappedToken(WRAPPED_NATIVE).withdraw(data.amountOut);
} else if (!success) {
if (_msgSender() == STARGATE_ROUTER) {
_refund(sender, data.tokenIn, data.amountIn);
} else {
revert SwapUnsuccessful();
}
}
}
}
if (success) {
if (data.tokenOut == address(0)) {
try IExecutor(EXECUTOR).execute{value: data.amountOut + data.nativeOut}(target, paymentOperator, data)
returns (bool executionSuccess) {
success = executionSuccess;
} catch {
success = false;
}
} else {
SafeERC20.safeIncreaseAllowance(IERC20(data.tokenOut), EXECUTOR, data.amountOut);
try IExecutor(EXECUTOR).execute{value: data.nativeOut}(target, paymentOperator, data) returns (
bool executionSuccess
) {
success = executionSuccess;
} catch {
success = false;
}
}
if (!success) {
if (_msgSender() == STARGATE_ROUTER) {
_refund(sender, data.tokenOut, data.amountOut);
emit BridgedExecutionUnsuccessful();
} else {
revert ExecutionUnsuccessful();
}
}
}
}
function _refund(address to, address token, uint256 amount) internal {
if (token == address(0)) {
(bool success,) = payable(to).call{value: amount}("");
if (!success) {
emit RefundUnsuccessful();
}
} else {
SafeERC20.safeTransfer(IERC20(token), to, amount);
}
}
}
文件 7 的 20:ERC2771Context.sol
pragma solidity ^0.8.9;
import "../utils/Context.sol";
abstract contract ERC2771Context is Context {
address private immutable _trustedForwarder;
constructor(address trustedForwarder) {
_trustedForwarder = trustedForwarder;
}
function isTrustedForwarder(address forwarder) public view virtual returns (bool) {
return forwarder == _trustedForwarder;
}
function _msgSender() internal view virtual override returns (address sender) {
if (isTrustedForwarder(msg.sender)) {
assembly {
sender := shr(96, calldataload(sub(calldatasize(), 20)))
}
} else {
return super._msgSender();
}
}
function _msgData() internal view virtual override returns (bytes calldata) {
if (isTrustedForwarder(msg.sender)) {
return msg.data[:msg.data.length - 20];
} else {
return super._msgData();
}
}
}
文件 8 的 20:FeeOperator.sol
pragma solidity 0.8.17;
import "../interfaces/IFeeManager.sol";
import {Ownable} from "@openzeppelin-contracts/contracts/access/Ownable.sol";
abstract contract FeeOperator is Ownable {
address public feeManager;
event FeeManagerUpdated(address feeManager);
error InsufficientFees();
error FeeTransferFailed();
error RefundFailed();
modifier handleFees(uint256 bridgeFee, uint256 amountIn, address tokenIn, uint256 nativeOut) {
if (feeManager != address(0)) {
(uint256 fee, uint256 commission) = IFeeManager(feeManager).calculateFees(amountIn, tokenIn);
uint256 boxFees = fee + commission;
uint256 amountRequired =
tokenIn == address(0) ? amountIn + nativeOut + bridgeFee + boxFees : nativeOut + bridgeFee + boxFees;
if (msg.value < amountRequired) {
revert InsufficientFees();
}
_transferFees(boxFees);
_transferRefund(msg.value - amountRequired);
}
_;
}
function setFeeManager(address _feeManager) external onlyOwner {
feeManager = _feeManager;
emit FeeManagerUpdated(_feeManager);
}
function _transferFees(uint256 fees) internal {
(bool success,) = payable(feeManager).call{value: fees}("");
if (!success) {
revert FeeTransferFailed();
}
}
function _transferRefund(uint256 refund) internal {
if (refund > 0) {
(bool success,) = payable(_msgSender()).call{value: refund}("");
if (!success) {
revert RefundFailed();
}
}
}
}
文件 9 的 20:ICore.sol
pragma solidity 0.8.17;
import {LzBridgeData, TokenData} from "../lib/CoreStructs.sol";
import {IStargateRouter} from "../interfaces/stargate/IStargateRouter.sol";
interface ICore {
error OnlyStargateRouter();
function swapAndExecute(address target, address paymentOperator, TokenData calldata tokenData) external payable;
function bridgeAndExecute(
LzBridgeData calldata lzBridgeData,
TokenData calldata tokenData,
IStargateRouter.lzTxObj calldata lzTxObj
) external payable;
}
文件 10 的 20:IERC20.sol
pragma solidity ^0.8.0;
import "../token/ERC20/IERC20.sol";
文件 11 的 20: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);
}
文件 12 的 20:IExecutor.sol
pragma solidity 0.8.17;
import {TokenData} from "../lib/CoreStructs.sol";
interface IExecutor {
error OnlyCoreAuth();
event CoreUpdated(address newCore);
function execute(address target, address paymentOperator, TokenData memory data)
external
payable
returns (bool success);
function setCore(address core) external;
}
文件 13 的 20:IFeeManager.sol
pragma solidity 0.8.17;
interface IFeeManager {
error WithdrawFailed();
function setFees(uint256 _fee, uint256 _commissionBPS) external;
function calculateFees(uint256 amountIn, address tokenIn) external view returns (uint256 fee, uint256 commission);
function redeemFees() external;
}
文件 14 的 20:IStargateReceiver.sol
pragma solidity 0.8.17;
interface IStargateReceiver {
event ReceivedOnDestination(address token, uint256 amountLD);
function sgReceive(
uint16,
bytes memory,
uint256,
address _token,
uint256 amountLD,
bytes memory payload
) external;
}
文件 15 的 20:IStargateRouter.sol
pragma solidity 0.8.17;
pragma abicoder v2;
interface IStargateRouter {
struct lzTxObj {
uint256 dstGasForCall;
uint256 dstNativeAmount;
bytes dstNativeAddr;
}
function addLiquidity(uint256 _poolId, uint256 _amountLD, address _to) external;
function swap(
uint16 _dstChainId,
uint256 _srcPoolId,
uint256 _dstPoolId,
address payable _refundAddress,
uint256 _amountLD,
uint256 _minAmountLD,
lzTxObj memory _lzTxParams,
bytes calldata _to,
bytes calldata _payload
) external payable;
function redeemRemote(
uint16 _dstChainId,
uint256 _srcPoolId,
uint256 _dstPoolId,
address payable _refundAddress,
uint256 _amountLP,
uint256 _minAmountLD,
bytes calldata _to,
lzTxObj memory _lzTxParams
) external payable;
function instantRedeemLocal(uint16 _srcPoolId, uint256 _amountLP, address _to) external returns (uint256);
function redeemLocal(
uint16 _dstChainId,
uint256 _srcPoolId,
uint256 _dstPoolId,
address payable _refundAddress,
uint256 _amountLP,
bytes calldata _to,
lzTxObj memory _lzTxParams
) external payable;
function sendCredits(uint16 _dstChainId, uint256 _srcPoolId, uint256 _dstPoolId, address payable _refundAddress)
external
payable;
function quoteLayerZeroFee(
uint16 _dstChainId,
uint8 _functionType,
bytes calldata _toAddress,
bytes calldata _transferAndCallPayload,
lzTxObj memory _lzTxParams
) external view returns (uint256, uint256);
}
文件 16 的 20:ISwapRouter.sol
pragma solidity >=0.7.5;
pragma abicoder v2;
import '@uniswap/v3-core/contracts/interfaces/callback/IUniswapV3SwapCallback.sol';
interface ISwapRouter is IUniswapV3SwapCallback {
struct ExactInputSingleParams {
address tokenIn;
address tokenOut;
uint24 fee;
address recipient;
uint256 deadline;
uint256 amountIn;
uint256 amountOutMinimum;
uint160 sqrtPriceLimitX96;
}
function exactInputSingle(ExactInputSingleParams calldata params) external payable returns (uint256 amountOut);
struct ExactInputParams {
bytes path;
address recipient;
uint256 deadline;
uint256 amountIn;
uint256 amountOutMinimum;
}
function exactInput(ExactInputParams calldata params) external payable returns (uint256 amountOut);
struct ExactOutputSingleParams {
address tokenIn;
address tokenOut;
uint24 fee;
address recipient;
uint256 deadline;
uint256 amountOut;
uint256 amountInMaximum;
uint160 sqrtPriceLimitX96;
}
function exactOutputSingle(ExactOutputSingleParams calldata params) external payable returns (uint256 amountIn);
struct ExactOutputParams {
bytes path;
address recipient;
uint256 deadline;
uint256 amountOut;
uint256 amountInMaximum;
}
function exactOutput(ExactOutputParams calldata params) external payable returns (uint256 amountIn);
}
文件 17 的 20:IUniswapV3SwapCallback.sol
pragma solidity >=0.5.0;
interface IUniswapV3SwapCallback {
function uniswapV3SwapCallback(
int256 amount0Delta,
int256 amount1Delta,
bytes calldata data
) external;
}
文件 18 的 20:IWrappedToken.sol
pragma solidity 0.8.17;
interface IWrappedToken {
function withdraw(uint256 wad) external;
function deposit() external payable;
}
文件 19 的 20:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_transferOwnership(_msgSender());
}
modifier onlyOwner() {
_checkOwner();
_;
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
}
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 20 的 20: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));
}
}
{
"compilationTarget": {
"src/Core.sol": "Core"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":@openzeppelin-contracts/=lib/openzeppelin-contracts/",
":@uniswap/v3-core/=lib/v3-core/",
":@uniswap/v3-periphery/=lib/v3-periphery/",
":ds-test/=lib/forge-std/lib/ds-test/src/",
":erc4626-tests/=lib/openzeppelin-contracts/lib/erc4626-tests/",
":forge-std/=lib/forge-std/src/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/",
":solmate/=lib/solmate/src/",
":v3-core/=lib/v3-core/",
":v3-periphery/=lib/v3-periphery/contracts/"
]
}
[{"inputs":[{"internalType":"address","name":"_executor","type":"address"},{"internalType":"address","name":"_stargateRouter","type":"address"},{"internalType":"address","name":"_uniswapRouter","type":"address"},{"internalType":"address","name":"_wrappedNative","type":"address"},{"internalType":"address","name":"_sgETH","type":"address"},{"internalType":"address","name":"_trustedForwarder","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"BridgeNativeOutNonZero","type":"error"},{"inputs":[],"name":"BridgeOutputExceedsInput","type":"error"},{"inputs":[],"name":"ExecutionUnsuccessful","type":"error"},{"inputs":[],"name":"FeeTransferFailed","type":"error"},{"inputs":[],"name":"InsufficientFees","type":"error"},{"inputs":[],"name":"OnlyStargateRouter","type":"error"},{"inputs":[],"name":"RefundFailed","type":"error"},{"inputs":[],"name":"SwapOutputExceedsInput","type":"error"},{"inputs":[],"name":"SwapUnsuccessful","type":"error"},{"anonymous":false,"inputs":[],"name":"BridgedExecutionUnsuccessful","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"feeManager","type":"address"}],"name":"FeeManagerUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amountLD","type":"uint256"}],"name":"ReceivedOnDestination","type":"event"},{"anonymous":false,"inputs":[],"name":"RefundUnsuccessful","type":"event"},{"inputs":[{"components":[{"internalType":"uint120","name":"_srcPoolId","type":"uint120"},{"internalType":"uint120","name":"_dstPoolId","type":"uint120"},{"internalType":"uint16","name":"_dstChainId","type":"uint16"},{"internalType":"address","name":"_bridgeAddress","type":"address"},{"internalType":"uint96","name":"fee","type":"uint96"}],"internalType":"struct LzBridgeData","name":"lzBridgeData","type":"tuple"},{"components":[{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"amountOut","type":"uint256"},{"internalType":"uint256","name":"nativeOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"address","name":"tokenOut","type":"address"},{"internalType":"bytes","name":"path","type":"bytes"},{"internalType":"bytes","name":"payload","type":"bytes"}],"internalType":"struct TokenData","name":"tokenData","type":"tuple"},{"components":[{"internalType":"uint256","name":"dstGasForCall","type":"uint256"},{"internalType":"uint256","name":"dstNativeAmount","type":"uint256"},{"internalType":"bytes","name":"dstNativeAddr","type":"bytes"}],"internalType":"struct IStargateRouter.lzTxObj","name":"lzTxObj","type":"tuple"}],"name":"bridgeAndExecute","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"feeManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"forwarder","type":"address"}],"name":"isTrustedForwarder","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_feeManager","type":"address"}],"name":"setFeeManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"","type":"uint16"},{"internalType":"bytes","name":"","type":"bytes"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"amountLD","type":"uint256"},{"internalType":"bytes","name":"payload","type":"bytes"}],"name":"sgReceive","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"target","type":"address"},{"internalType":"address","name":"paymentOperator","type":"address"},{"components":[{"internalType":"uint256","name":"amountIn","type":"uint256"},{"internalType":"uint256","name":"amountOut","type":"uint256"},{"internalType":"uint256","name":"nativeOut","type":"uint256"},{"internalType":"address","name":"tokenIn","type":"address"},{"internalType":"address","name":"tokenOut","type":"address"},{"internalType":"bytes","name":"path","type":"bytes"},{"internalType":"bytes","name":"payload","type":"bytes"}],"internalType":"struct TokenData","name":"tokenData","type":"tuple"}],"name":"swapAndExecute","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IERC20","name":"token","type":"address"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawEth","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]