-
Notifications
You must be signed in to change notification settings - Fork 106
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #342 from 1inch/feature/fee-taker-upgrade
[SC-1198] Upgrade FeeTaker
- Loading branch information
Showing
6 changed files
with
349 additions
and
242 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
// SPDX-License-Identifier: MIT | ||
|
||
pragma solidity 0.8.23; | ||
|
||
import { Math } from "@openzeppelin/contracts/utils/math/Math.sol"; | ||
|
||
import { IAmountGetter } from "../interfaces/IAmountGetter.sol"; | ||
import { IOrderMixin } from "../interfaces/IOrderMixin.sol"; | ||
|
||
/// @title Base price getter contract that either calls external getter or applies linear formula | ||
contract AmountGetterBase is IAmountGetter { | ||
function getMakingAmount( | ||
IOrderMixin.Order calldata order, | ||
bytes calldata extension, | ||
bytes32 orderHash, | ||
address taker, | ||
uint256 takingAmount, | ||
uint256 remainingMakingAmount, | ||
bytes calldata extraData | ||
) external view returns (uint256) { | ||
return _getMakingAmount(order, extension, orderHash, taker, takingAmount, remainingMakingAmount, extraData); | ||
} | ||
|
||
function getTakingAmount( | ||
IOrderMixin.Order calldata order, | ||
bytes calldata extension, | ||
bytes32 orderHash, | ||
address taker, | ||
uint256 makingAmount, | ||
uint256 remainingMakingAmount, | ||
bytes calldata extraData | ||
) external view returns (uint256) { | ||
return _getTakingAmount(order, extension, orderHash, taker, makingAmount, remainingMakingAmount, extraData); | ||
} | ||
|
||
function _getMakingAmount( | ||
IOrderMixin.Order calldata order, | ||
bytes calldata extension, | ||
bytes32 orderHash, | ||
address taker, | ||
uint256 takingAmount, | ||
uint256 remainingMakingAmount, | ||
bytes calldata extraData | ||
) internal view virtual returns (uint256) { | ||
if (extraData.length >= 20) { | ||
return IAmountGetter(address(bytes20(extraData))).getMakingAmount( | ||
order, extension, orderHash, taker, takingAmount, remainingMakingAmount, extraData[20:] | ||
); | ||
} else { | ||
return Math.mulDiv(order.makingAmount, takingAmount, order.takingAmount); | ||
} | ||
} | ||
|
||
function _getTakingAmount( | ||
IOrderMixin.Order calldata order, | ||
bytes calldata extension, | ||
bytes32 orderHash, | ||
address taker, | ||
uint256 makingAmount, | ||
uint256 remainingMakingAmount, | ||
bytes calldata extraData | ||
) internal view virtual returns (uint256) { | ||
if (extraData.length >= 20) { | ||
return IAmountGetter(address(bytes20(extraData))).getTakingAmount( | ||
order, extension, orderHash, taker, makingAmount, remainingMakingAmount, extraData[20:] | ||
); | ||
} else { | ||
return Math.mulDiv(order.takingAmount, makingAmount, order.makingAmount, Math.Rounding.Ceil); | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,113 @@ | ||
// SPDX-License-Identifier: MIT | ||
|
||
pragma solidity 0.8.23; | ||
|
||
import { Math } from "@openzeppelin/contracts/utils/math/Math.sol"; | ||
|
||
import { IOrderMixin } from "../interfaces/IOrderMixin.sol"; | ||
import { AmountGetterBase } from "./AmountGetterBase.sol"; | ||
|
||
/// @title Price getter contract that adds fee calculation | ||
contract AmountGetterWithFee is AmountGetterBase { | ||
/// @dev Allows fees in range [1e-5, 0.65535] | ||
uint256 internal constant _FEE_BASE = 1e5; | ||
uint256 internal constant _DISCOUNT_BASE = 100; | ||
|
||
/** | ||
* @dev Calculates makingAmount with fee. | ||
*/ | ||
function _getMakingAmount( | ||
IOrderMixin.Order calldata order, | ||
bytes calldata extension, | ||
bytes32 orderHash, | ||
address taker, | ||
uint256 takingAmount, | ||
uint256 remainingMakingAmount, | ||
bytes calldata extraData | ||
) internal view virtual override returns (uint256) { | ||
unchecked { | ||
(, uint256 integratorFee, uint256 resolverFee, bytes calldata tail) = _parseFeeData(extraData, taker, _isWhitelistedGetterImpl); | ||
return Math.mulDiv( | ||
super._getMakingAmount(order, extension, orderHash, taker, takingAmount, remainingMakingAmount, tail), | ||
_FEE_BASE, | ||
_FEE_BASE + integratorFee + resolverFee | ||
); | ||
} | ||
} | ||
|
||
/** | ||
* @dev Calculates takingAmount with fee. | ||
*/ | ||
function _getTakingAmount( | ||
IOrderMixin.Order calldata order, | ||
bytes calldata extension, | ||
bytes32 orderHash, | ||
address taker, | ||
uint256 makingAmount, | ||
uint256 remainingMakingAmount, | ||
bytes calldata extraData | ||
) internal view virtual override returns (uint256) { | ||
unchecked { | ||
(, uint256 integratorFee, uint256 resolverFee, bytes calldata tail) = _parseFeeData(extraData, taker, _isWhitelistedGetterImpl); | ||
return Math.mulDiv( | ||
super._getTakingAmount(order, extension, orderHash, taker, makingAmount, remainingMakingAmount, tail), | ||
_FEE_BASE + integratorFee + resolverFee, | ||
_FEE_BASE, | ||
Math.Rounding.Ceil | ||
); | ||
} | ||
} | ||
|
||
/** | ||
* @dev `extraData` consists of: | ||
* 2 bytes — integrator fee percentage (in 1e5) | ||
* 2 bytes — resolver fee percentage (in 1e5) | ||
* 1 byte - whitelist discount numerator (in 1e2) | ||
* bytes — whitelist structure determined by `_isWhitelisted` implementation | ||
* bytes — custom data (optional) | ||
* @param _isWhitelisted internal function to parse and check whitelist | ||
*/ | ||
function _parseFeeData( | ||
bytes calldata extraData, | ||
address taker, | ||
function (bytes calldata, address) internal view returns (bool, bytes calldata) _isWhitelisted | ||
) internal view returns (bool isWhitelisted, uint256 integratorFee, uint256 resolverFee, bytes calldata tail) { | ||
unchecked { | ||
integratorFee = uint256(uint16(bytes2(extraData))); | ||
resolverFee = uint256(uint16(bytes2(extraData[2:]))); | ||
uint256 whitelistDiscountNumerator = uint256(uint8(bytes1(extraData[4:]))); | ||
(isWhitelisted, tail) = _isWhitelisted(extraData[5:], taker); | ||
if (isWhitelisted) { | ||
resolverFee = resolverFee * whitelistDiscountNumerator / _DISCOUNT_BASE; | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* @dev Validates whether the taker is whitelisted. | ||
* @param whitelistData Whitelist data is a tightly packed struct of the following format: | ||
* ``` | ||
* 1 byte - size of the whitelist | ||
* (bytes10)[N] whitelisted addresses; | ||
* ``` | ||
* Only 10 lowest bytes of the address are used for comparison. | ||
* @param taker The taker address to check. | ||
* @return isWhitelisted Whether the taker is whitelisted. | ||
* @return tail Remaining calldata. | ||
*/ | ||
function _isWhitelistedGetterImpl(bytes calldata whitelistData, address taker) internal pure returns (bool isWhitelisted, bytes calldata tail) { | ||
unchecked { | ||
uint80 maskedTakerAddress = uint80(uint160(taker)); | ||
uint256 size = uint8(whitelistData[0]); | ||
bytes calldata whitelist = whitelistData[1:1 + 10 * size]; | ||
tail = whitelistData[1 + 10 * size:]; | ||
for (uint256 i = 0; i < size; i++) { | ||
uint80 whitelistedAddress = uint80(bytes10(whitelist[:10])); | ||
if (maskedTakerAddress == whitelistedAddress) { | ||
return (true, tail); | ||
} | ||
whitelist = whitelist[10:]; | ||
} | ||
} | ||
} | ||
} |
Oops, something went wrong.