Skip to content

Commit

Permalink
Merge pull request #342 from 1inch/feature/fee-taker-upgrade
Browse files Browse the repository at this point in the history
[SC-1198] Upgrade FeeTaker
  • Loading branch information
ZumZoom authored Dec 10, 2024
2 parents 38cbfd6 + 994e529 commit c5e3424
Show file tree
Hide file tree
Showing 6 changed files with 349 additions and 242 deletions.
71 changes: 71 additions & 0 deletions contracts/extensions/AmountGetterBase.sol
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);
}
}
}
113 changes: 113 additions & 0 deletions contracts/extensions/AmountGetterWithFee.sol
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:];
}
}
}
}
Loading

0 comments on commit c5e3424

Please sign in to comment.