编译器
0.8.17+commit.8df45f5f
文件 1 的 10: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 functionCall(target, data, "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");
require(isContract(target), "Address: call to non-contract");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResult(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) {
require(isContract(target), "Address: static call to non-contract");
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResult(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) {
require(isContract(target), "Address: delegate call to non-contract");
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}
文件 2 的 10:GenericErrors.sol
pragma solidity 0.8.17;
error SwapFailed();
error LengthNotMatch();
error TokenAddressIsZero();
error TokenNotSupported();
error CannotBridgeToSameNetwork();
error ZeroPostSwapBalance();
error NoSwapDataProvided();
error NativeValueWithERC();
error ContractCallNotAllowed();
error NullAddrIsNotAValidSpender();
error NullAddrIsNotAnERC20Token();
error NoTransferToNullAddress();
error NativeAssetTransferFailed();
error InvalidBridgeConfigLength();
error InvalidAmount();
error InvalidContract();
error InvalidConfig();
error UnsupportedChainId(uint256 chainId);
error InvalidReceiver();
error InvalidDestinationChain();
error InvalidSendingToken();
error InvalidCaller();
error AlreadyInitialized();
error NotInitialized();
error OnlyContractOwner();
error CannotAuthoriseSelf();
error RecoveryAddressCannotBeZero();
error CannotDepositNativeToken();
error InvalidCallData();
error NativeAssetNotSupported();
error UnAuthorized();
error NoSwapFromZeroBalance();
error InvalidFallbackAddress();
error CumulativeSlippageTooHigh(uint256 minAmount, uint256 receivedAmount);
error InsufficientBalance(uint256 required, uint256 balance);
error ZeroAmount();
error InvalidFee();
error InformationMismatch();
error NotAContract();
error NotEnoughBalance(uint256 requested, uint256 available);
error ReentrancyError();
error NoAccessFacet();
error MoreAmountThanFee();
error RegisterFirst();
error ReceiveMessageFailed();
文件 3 的 10:IDiamondCut.sol
pragma solidity 0.8.17;
interface IDiamondCut {
enum FacetCutAction {
Add,
Replace,
Remove
}
struct FacetCut {
address facetAddress;
FacetCutAction action;
bytes4[] functionSelectors;
}
function diamondCut(FacetCut[] calldata _diamondCut, address _init, bytes calldata _calldata) external;
event DiamondCut(FacetCut[] _diamondCut, address _init, bytes _calldata);
}
文件 4 的 10:IERC20.sol
pragma solidity ^0.8.17;
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);
}
文件 5 的 10:LibData.sol
pragma solidity 0.8.17;
import "../interfaces/Structs.sol";
library LibData {
bytes32 internal constant BRIDGE_NAMESPACE = keccak256("diamond.standard.data.bridge");
bytes32 internal constant STARGATE_NAMESPACE = keccak256("diamond.standard.data.stargate");
bytes32 internal constant CONNEXT_NAMESPACE = keccak256("diamond.standard.data.connext");
bytes32 internal constant HOP_NAMESPACE = keccak256("com.plexus.facets.hop");
event Bridge(address user, uint64 chainId, address srcToken, uint256 fromAmount, bytes plexusData, string bridge);
event Swap(address user, InputToken[] input, OutputToken[] output, uint256[] returnAmount, bytes plexusData);
event Relayswap(address receiver, OutputToken[] output, uint256[] returnAmount, DupToken[] dupToken, bytes plexusData);
struct BridgeDesc {
mapping(bytes32 => BridgeInfo) transferInfo;
mapping(bytes32 => bool) transfers;
}
struct StargateData {
mapping(address => uint16) poolId;
mapping(address => mapping(uint256 => uint16)) dstPoolId;
mapping(uint256 => uint16) layerZeroId;
}
struct HopBridgeData {
mapping(address => address) bridge;
mapping(address => address) relayer;
mapping(address => bool) allowedRelayer;
mapping(address => bool) allowedBridge;
}
struct ConnextBridgeData {
mapping(uint64 => uint64) domainId;
}
function bridgeStorage() internal pure returns (BridgeDesc storage ds) {
bytes32 position = BRIDGE_NAMESPACE;
assembly {
ds.slot := position
}
}
function stargateStorage() internal pure returns (StargateData storage s) {
bytes32 position = STARGATE_NAMESPACE;
assembly {
s.slot := position
}
}
function hopStorage() internal pure returns (HopBridgeData storage s) {
bytes32 position = HOP_NAMESPACE;
assembly {
s.slot := position
}
}
function connextStorage() internal pure returns (ConnextBridgeData storage s) {
bytes32 position = CONNEXT_NAMESPACE;
assembly {
s.slot := position
}
}
}
文件 6 的 10:LibDiamond.sol
pragma solidity 0.8.17;
import {IDiamondCut} from "../interfaces/IDiamondCut.sol";
import {LibPlexusUtil} from "../libraries/LibPlexusUtil.sol";
import {OnlyContractOwner} from "../Errors/GenericErrors.sol";
library LibDiamond {
bytes32 internal constant DIAMOND_STORAGE_POSITION = keccak256("diamond.standard.diamond.storage");
error IncorrectFacetCutAction();
error NoSelectorsInFace();
error FunctionAlreadyExists();
error FacetAddressIsZero();
error FacetAddressIsNotZero();
error FacetContainsNoCode();
error FunctionDoesNotExist();
error FunctionIsImmutable();
error InitZeroButCalldataNotEmpty();
error CalldataEmptyButInitNotZero();
error InitReverted();
struct FacetAddressAndPosition {
address facetAddress;
uint96 functionSelectorPosition;
}
struct FacetFunctionSelectors {
bytes4[] functionSelectors;
uint256 facetAddressPosition;
}
struct DiamondStorage {
mapping(bytes4 => FacetAddressAndPosition) selectorToFacetAndPosition;
mapping(address => FacetFunctionSelectors) facetFunctionSelectors;
address[] facetAddresses;
mapping(bytes4 => bool) supportedInterfaces;
address contractOwner;
uint256 fee;
address feeReceiver;
}
function diamondStorage() internal pure returns (DiamondStorage storage ds) {
bytes32 position = DIAMOND_STORAGE_POSITION;
assembly {
ds.slot := position
}
}
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
event FeeReceiverTransferred(address indexed previousFeeReceiver, address indexed newFeeReceiver);
function setFeePercent(uint256 percent) internal {
DiamondStorage storage ds = diamondStorage();
ds.fee = percent;
}
function setContractOwner(address _newOwner) internal {
DiamondStorage storage ds = diamondStorage();
address previousOwner = ds.contractOwner;
ds.contractOwner = _newOwner;
emit OwnershipTransferred(previousOwner, _newOwner);
}
function setFeeReceiver(address _receiver) internal {
DiamondStorage storage ds = diamondStorage();
address previousFeeReceiver = ds.feeReceiver;
ds.feeReceiver = _receiver;
emit OwnershipTransferred(previousFeeReceiver, _receiver);
}
function contractOwner() internal view returns (address contractOwner_) {
contractOwner_ = diamondStorage().contractOwner;
}
function feeReceiver() internal view returns (address receiver) {
receiver = diamondStorage().feeReceiver;
}
function enforceIsContractOwner() internal view {
if (msg.sender != diamondStorage().contractOwner) revert OnlyContractOwner();
}
event DiamondCut(IDiamondCut.FacetCut[] _diamondCut, address _init, bytes _calldata);
function diamondCut(IDiamondCut.FacetCut[] memory _diamondCut, address _init, bytes memory _calldata) internal {
for (uint256 facetIndex; facetIndex < _diamondCut.length; ) {
IDiamondCut.FacetCutAction action = _diamondCut[facetIndex].action;
if (action == IDiamondCut.FacetCutAction.Add) {
addFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
} else if (action == IDiamondCut.FacetCutAction.Replace) {
replaceFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
} else if (action == IDiamondCut.FacetCutAction.Remove) {
removeFunctions(_diamondCut[facetIndex].facetAddress, _diamondCut[facetIndex].functionSelectors);
} else {
revert IncorrectFacetCutAction();
}
unchecked {
++facetIndex;
}
}
emit DiamondCut(_diamondCut, _init, _calldata);
initializeDiamondCut(_init, _calldata);
}
function addFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
if (_functionSelectors.length == 0) {
revert NoSelectorsInFace();
}
DiamondStorage storage ds = diamondStorage();
if (LibPlexusUtil.isZeroAddress(_facetAddress)) {
revert FacetAddressIsZero();
}
uint96 selectorPosition = uint96(ds.facetFunctionSelectors[_facetAddress].functionSelectors.length);
if (selectorPosition == 0) {
addFacet(ds, _facetAddress);
}
for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; ) {
bytes4 selector = _functionSelectors[selectorIndex];
address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
if (!LibPlexusUtil.isZeroAddress(oldFacetAddress)) {
revert FunctionAlreadyExists();
}
addFunction(ds, selector, selectorPosition, _facetAddress);
unchecked {
++selectorPosition;
++selectorIndex;
}
}
}
function replaceFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
if (_functionSelectors.length == 0) {
revert NoSelectorsInFace();
}
DiamondStorage storage ds = diamondStorage();
if (LibPlexusUtil.isZeroAddress(_facetAddress)) {
revert FacetAddressIsZero();
}
uint96 selectorPosition = uint96(ds.facetFunctionSelectors[_facetAddress].functionSelectors.length);
if (selectorPosition == 0) {
addFacet(ds, _facetAddress);
}
for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; ) {
bytes4 selector = _functionSelectors[selectorIndex];
address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
if (oldFacetAddress == _facetAddress) {
revert FunctionAlreadyExists();
}
removeFunction(ds, oldFacetAddress, selector);
addFunction(ds, selector, selectorPosition, _facetAddress);
unchecked {
++selectorPosition;
++selectorIndex;
}
}
}
function removeFunctions(address _facetAddress, bytes4[] memory _functionSelectors) internal {
if (_functionSelectors.length == 0) {
revert NoSelectorsInFace();
}
DiamondStorage storage ds = diamondStorage();
if (!LibPlexusUtil.isZeroAddress(_facetAddress)) {
revert FacetAddressIsNotZero();
}
for (uint256 selectorIndex; selectorIndex < _functionSelectors.length; ) {
bytes4 selector = _functionSelectors[selectorIndex];
address oldFacetAddress = ds.selectorToFacetAndPosition[selector].facetAddress;
removeFunction(ds, oldFacetAddress, selector);
unchecked {
++selectorIndex;
}
}
}
function addFacet(DiamondStorage storage ds, address _facetAddress) internal {
enforceHasContractCode(_facetAddress);
ds.facetFunctionSelectors[_facetAddress].facetAddressPosition = ds.facetAddresses.length;
ds.facetAddresses.push(_facetAddress);
}
function addFunction(DiamondStorage storage ds, bytes4 _selector, uint96 _selectorPosition, address _facetAddress) internal {
ds.selectorToFacetAndPosition[_selector].functionSelectorPosition = _selectorPosition;
ds.facetFunctionSelectors[_facetAddress].functionSelectors.push(_selector);
ds.selectorToFacetAndPosition[_selector].facetAddress = _facetAddress;
}
function removeFunction(DiamondStorage storage ds, address _facetAddress, bytes4 _selector) internal {
if (LibPlexusUtil.isZeroAddress(_facetAddress)) {
revert FunctionDoesNotExist();
}
if (_facetAddress == address(this)) {
revert FunctionIsImmutable();
}
uint256 selectorPosition = ds.selectorToFacetAndPosition[_selector].functionSelectorPosition;
uint256 lastSelectorPosition = ds.facetFunctionSelectors[_facetAddress].functionSelectors.length - 1;
if (selectorPosition != lastSelectorPosition) {
bytes4 lastSelector = ds.facetFunctionSelectors[_facetAddress].functionSelectors[lastSelectorPosition];
ds.facetFunctionSelectors[_facetAddress].functionSelectors[selectorPosition] = lastSelector;
ds.selectorToFacetAndPosition[lastSelector].functionSelectorPosition = uint96(selectorPosition);
}
ds.facetFunctionSelectors[_facetAddress].functionSelectors.pop();
delete ds.selectorToFacetAndPosition[_selector];
if (lastSelectorPosition == 0) {
uint256 lastFacetAddressPosition = ds.facetAddresses.length - 1;
uint256 facetAddressPosition = ds.facetFunctionSelectors[_facetAddress].facetAddressPosition;
if (facetAddressPosition != lastFacetAddressPosition) {
address lastFacetAddress = ds.facetAddresses[lastFacetAddressPosition];
ds.facetAddresses[facetAddressPosition] = lastFacetAddress;
ds.facetFunctionSelectors[lastFacetAddress].facetAddressPosition = facetAddressPosition;
}
ds.facetAddresses.pop();
delete ds.facetFunctionSelectors[_facetAddress].facetAddressPosition;
}
}
function initializeDiamondCut(address _init, bytes memory _calldata) internal {
if (LibPlexusUtil.isZeroAddress(_init)) {
if (_calldata.length != 0) {
revert InitZeroButCalldataNotEmpty();
}
} else {
if (_calldata.length == 0) {
revert CalldataEmptyButInitNotZero();
}
if (_init != address(this)) {
enforceHasContractCode(_init);
}
(bool success, bytes memory error) = _init.delegatecall(_calldata);
if (!success) {
if (error.length > 0) {
revert(string(error));
} else {
revert InitReverted();
}
}
}
}
function enforceHasContractCode(address _contract) internal view {
uint256 contractSize;
assembly {
contractSize := extcodesize(_contract)
}
if (contractSize == 0) {
revert FacetContainsNoCode();
}
}
}
文件 7 的 10:LibPlexusUtil.sol
pragma solidity 0.8.17;
import {SwapFailed} from "../Errors/GenericErrors.sol";
import "../libraries/SafeERC20.sol";
import "../interfaces/Structs.sol";
import "./LibDiamond.sol";
import "./LibData.sol";
library LibPlexusUtil {
using SafeERC20 for IERC20;
IERC20 private constant NATIVE_ADDRESS = IERC20(0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE);
bytes32 internal constant NAMESPACE = keccak256("com.plexus.facets.swap");
function getSwapStorage() internal pure returns (Dex storage s) {
bytes32 namespace = NAMESPACE;
assembly {
s.slot := namespace
}
}
function dexCheck(address dex) internal view returns (bool result) {
Dex storage s = LibPlexusUtil.getSwapStorage();
return s.allowedDex[dex];
}
function dexProxyCheck(address dex) internal view returns (address proxy) {
Dex storage s = LibPlexusUtil.getSwapStorage();
return s.proxy[dex];
}
function isZeroAddress(address addr) internal pure returns (bool) {
return addr == address(0);
}
function getBalance(address token) internal view returns (uint256) {
return token == address(NATIVE_ADDRESS) ? address(this).balance : IERC20(token).balanceOf(address(this));
}
function userBalance(address user, address token) internal view returns (uint256) {
return token == address(NATIVE_ADDRESS) ? user.balance : IERC20(token).balanceOf(user);
}
function _isNative(address _token) internal pure returns (bool) {
return (IERC20(_token) == NATIVE_ADDRESS);
}
function _isTokenDeposit(address _token, uint256 _amount) internal returns (bool isNotNative) {
isNotNative = !_isNative(_token);
if (isNotNative) {
IERC20(_token).safeTransferFrom(msg.sender, address(this), _amount);
} else {
require(msg.value >= _amount, "msg.value is missing.");
}
}
function _isSwapTokenDeposit(InputToken[] calldata input) internal {
uint256 nativeAmount = 0;
for (uint i; i < input.length; i++) {
bool isNotNative = !_isNative(input[i].srcToken);
if (isNotNative) {
IERC20(input[i].srcToken).safeTransferFrom(msg.sender, address(this), input[i].amount);
} else {
nativeAmount = nativeAmount + input[i].amount;
}
}
require(msg.value >= nativeAmount);
}
function _isTokenApprove(SwapData calldata swap) internal returns (uint256) {
Dex storage s = getSwapStorage();
require(s.allowedDex[swap.router]);
InputToken[] calldata input = swap.input;
uint256 nativeAmount = 0;
for (uint i; i < input.length; i++) {
bool isNotNative = !_isNative(input[i].srcToken);
if (isNotNative) {
if (s.proxy[swap.router] != address(0)) {
IERC20(input[i].srcToken).safeApprove(s.proxy[swap.router], 0);
IERC20(input[i].srcToken).safeApprove(s.proxy[swap.router], input[i].amount);
} else {
IERC20(input[i].srcToken).safeApprove(swap.router, 0);
IERC20(input[i].srcToken).safeApprove(swap.router, input[i].amount);
}
} else {
nativeAmount = input[i].amount;
}
}
require(msg.value >= nativeAmount);
return nativeAmount;
}
function _tokenDepositAndSwap(SwapData calldata swap) internal returns (uint256[] memory) {
_isSwapTokenDeposit(swap.input);
uint256[] memory dstAmount = new uint256[](swap.output.length);
dstAmount = _swapStart(swap);
return dstAmount;
}
function _swapStart(SwapData calldata swap) internal returns (uint256[] memory) {
_isTokenApprove(swap);
uint256 length = swap.output.length;
uint256[] memory initDstTokenBalance = new uint256[](length);
uint256[] memory dstTokenBalance = new uint256[](length);
for (uint i; i < length; i++) {
initDstTokenBalance[i] = getBalance(swap.output[i].dstToken);
}
(bool succ, ) = swap.router.call{value: msg.value}(swap.callData);
if (succ) {
for (uint i; i < length; i++) {
dstTokenBalance[i] = getBalance(swap.output[i].dstToken) - initDstTokenBalance[i];
}
return dstTokenBalance;
} else {
revert("swap failed");
}
}
function _bridgeSwapStart(SwapData calldata swap) internal returns (uint256) {
uint256 nativeAmount = _isTokenApprove(swap);
require(swap.output.length == 1);
uint256 initDstTokenBalance = getBalance(swap.output[0].dstToken);
(bool succ, ) = swap.router.call{value: nativeAmount}(swap.callData);
if (succ) {
uint256 dstTokenBalance = getBalance(swap.output[0].dstToken) - initDstTokenBalance;
return dstTokenBalance;
} else {
revert("swap failed");
}
}
function _safeNativeTransfer(address to_, uint256 amount_) internal {
(bool sent, ) = to_.call{value: amount_}("");
require(sent, "Safe safeTransfer fail");
}
function _fee(address dstToken, uint256 dstAmount) internal returns (uint256 returnAmount) {
LibDiamond.DiamondStorage storage ds = LibDiamond.diamondStorage();
uint256 getFee = (dstAmount * ds.fee) / 10000;
returnAmount = dstAmount - getFee;
if (getFee > 0) {
if (!_isNative(dstToken)) {
IERC20(dstToken).safeTransfer(ds.feeReceiver, getFee);
} else {
_safeNativeTransfer(ds.feeReceiver, getFee);
}
}
}
function _relaySwapStart(SwapData calldata swap) internal returns (uint256[] memory) {
uint256 nativeAmount = _isTokenApprove(swap);
uint256 length = swap.output.length;
uint256[] memory initDstTokenBalance = new uint256[](length);
uint256[] memory dstTokenBalance = new uint256[](length);
for (uint i; i < length; i++) {
initDstTokenBalance[i] = getBalance(swap.output[i].dstToken);
}
(bool succ, ) = swap.router.call{value: nativeAmount}(swap.callData);
if (succ) {
for (uint i; i < length; i++) {
dstTokenBalance[i] = getBalance(swap.output[i].dstToken) - initDstTokenBalance[i];
}
return dstTokenBalance;
} else {
revert("swap failed");
}
}
}
文件 8 的 10:PlexusDiamond.sol
pragma solidity 0.8.17;
import {LibDiamond} from "./libraries/LibDiamond.sol";
import {IDiamondCut} from "./interfaces/IDiamondCut.sol";
contract PlexusDiamond {
constructor(address _contractOwner, address _diamondCutFacet) payable {
LibDiamond.setContractOwner(_contractOwner);
IDiamondCut.FacetCut[] memory cut = new IDiamondCut.FacetCut[](1);
bytes4[] memory functionSelectors = new bytes4[](1);
functionSelectors[0] = IDiamondCut.diamondCut.selector;
cut[0] = IDiamondCut.FacetCut({facetAddress: _diamondCutFacet, action: IDiamondCut.FacetCutAction.Add, functionSelectors: functionSelectors});
LibDiamond.diamondCut(cut, address(0), "");
}
fallback() external payable {
LibDiamond.DiamondStorage storage ds;
bytes32 position = LibDiamond.DIAMOND_STORAGE_POSITION;
assembly {
ds.slot := position
}
address facet = ds.selectorToFacetAndPosition[msg.sig].facetAddress;
if (facet == address(0)) {
revert LibDiamond.FunctionDoesNotExist();
}
assembly {
calldatacopy(0, 0, calldatasize())
let result := delegatecall(gas(), facet, 0, calldatasize(), 0, 0)
returndatacopy(0, 0, returndatasize())
switch result
case 0 {
revert(0, returndatasize())
}
default {
return(0, returndatasize())
}
}
}
receive() external payable {}
}
文件 9 的 10:SafeERC20.sol
pragma solidity ^0.8.0;
import "../interfaces/IERC20.sol";
import "./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 newAllowance = token.allowance(address(this), spender) + value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
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");
uint256 newAllowance = oldAllowance - value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
}
function _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) {
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}
文件 10 的 10:Structs.sol
pragma solidity 0.8.17;
struct BridgeInfo {
string bridge;
address dstToken;
uint64 chainId;
uint256 amount;
address user;
}
struct InputToken {
address srcToken;
uint256 amount;
}
struct OutputToken {
address dstToken;
}
struct SwapData {
address router;
address user;
InputToken[] input;
OutputToken[] output;
DupToken[] dup;
bytes callData;
address feeToken;
bytes plexusData;
}
struct DupToken {
address token;
uint256 amount;
}
struct RelaySwapData {
SwapData swapData;
address feeTokenAddress;
}
struct Dex {
mapping(address => bool) allowedDex;
mapping(address => address) proxy;
}
{
"compilationTarget": {
"contracts/PlexusDiamond.sol": "PlexusDiamond"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_contractOwner","type":"address"},{"internalType":"address","name":"_diamondCutFacet","type":"address"}],"stateMutability":"payable","type":"constructor"},{"inputs":[],"name":"CalldataEmptyButInitNotZero","type":"error"},{"inputs":[],"name":"FacetAddressIsNotZero","type":"error"},{"inputs":[],"name":"FacetAddressIsZero","type":"error"},{"inputs":[],"name":"FacetContainsNoCode","type":"error"},{"inputs":[],"name":"FunctionAlreadyExists","type":"error"},{"inputs":[],"name":"FunctionDoesNotExist","type":"error"},{"inputs":[],"name":"FunctionIsImmutable","type":"error"},{"inputs":[],"name":"IncorrectFacetCutAction","type":"error"},{"inputs":[],"name":"InitReverted","type":"error"},{"inputs":[],"name":"InitZeroButCalldataNotEmpty","type":"error"},{"inputs":[],"name":"NoSelectorsInFace","type":"error"},{"stateMutability":"payable","type":"fallback"},{"stateMutability":"payable","type":"receive"}]