123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109 |
- // SPDX-License-Identifier: MIT
- pragma solidity ^0.8.20;
- import {IERC20} from "../../token/ERC20/IERC20.sol";
- import {ERC4626} from "../../token/ERC20/extensions/ERC4626.sol";
- import {SafeERC20} from "../../token/ERC20/utils/SafeERC20.sol";
- import {Math} from "../../utils/math/Math.sol";
- /// @dev ERC-4626 vault with entry/exit fees expressed in https://en.wikipedia.org/wiki/Basis_point[basis point (bp)].
- ///
- /// NOTE: The contract charges fees in terms of assets, not shares. This means that the fees are calculated based on the
- /// amount of assets that are being deposited or withdrawn, and not based on the amount of shares that are being minted or
- /// redeemed. This is an opinionated design decision that should be taken into account when integrating this contract.
- ///
- /// WARNING: This contract has not been audited and shouldn't be considered production ready. Consider using it with caution.
- abstract contract ERC4626Fees is ERC4626 {
- using Math for uint256;
- uint256 private constant _BASIS_POINT_SCALE = 1e4;
- // === Overrides ===
- /// @dev Preview taking an entry fee on deposit. See {IERC4626-previewDeposit}.
- function previewDeposit(uint256 assets) public view virtual override returns (uint256) {
- uint256 fee = _feeOnTotal(assets, _entryFeeBasisPoints());
- return super.previewDeposit(assets - fee);
- }
- /// @dev Preview adding an entry fee on mint. See {IERC4626-previewMint}.
- function previewMint(uint256 shares) public view virtual override returns (uint256) {
- uint256 assets = super.previewMint(shares);
- return assets + _feeOnRaw(assets, _entryFeeBasisPoints());
- }
- /// @dev Preview adding an exit fee on withdraw. See {IERC4626-previewWithdraw}.
- function previewWithdraw(uint256 assets) public view virtual override returns (uint256) {
- uint256 fee = _feeOnRaw(assets, _exitFeeBasisPoints());
- return super.previewWithdraw(assets + fee);
- }
- /// @dev Preview taking an exit fee on redeem. See {IERC4626-previewRedeem}.
- function previewRedeem(uint256 shares) public view virtual override returns (uint256) {
- uint256 assets = super.previewRedeem(shares);
- return assets - _feeOnTotal(assets, _exitFeeBasisPoints());
- }
- /// @dev Send entry fee to {_entryFeeRecipient}. See {IERC4626-_deposit}.
- function _deposit(address caller, address receiver, uint256 assets, uint256 shares) internal virtual override {
- uint256 fee = _feeOnTotal(assets, _entryFeeBasisPoints());
- address recipient = _entryFeeRecipient();
- super._deposit(caller, receiver, assets, shares);
- if (fee > 0 && recipient != address(this)) {
- SafeERC20.safeTransfer(IERC20(asset()), recipient, fee);
- }
- }
- /// @dev Send exit fee to {_exitFeeRecipient}. See {IERC4626-_deposit}.
- function _withdraw(
- address caller,
- address receiver,
- address owner,
- uint256 assets,
- uint256 shares
- ) internal virtual override {
- uint256 fee = _feeOnRaw(assets, _exitFeeBasisPoints());
- address recipient = _exitFeeRecipient();
- super._withdraw(caller, receiver, owner, assets, shares);
- if (fee > 0 && recipient != address(this)) {
- SafeERC20.safeTransfer(IERC20(asset()), recipient, fee);
- }
- }
- // === Fee configuration ===
- function _entryFeeBasisPoints() internal view virtual returns (uint256) {
- return 0; // replace with e.g. 100 for 1%
- }
- function _exitFeeBasisPoints() internal view virtual returns (uint256) {
- return 0; // replace with e.g. 100 for 1%
- }
- function _entryFeeRecipient() internal view virtual returns (address) {
- return address(0); // replace with e.g. a treasury address
- }
- function _exitFeeRecipient() internal view virtual returns (address) {
- return address(0); // replace with e.g. a treasury address
- }
- // === Fee operations ===
- /// @dev Calculates the fees that should be added to an amount `assets` that does not already include fees.
- /// Used in {IERC4626-mint} and {IERC4626-withdraw} operations.
- function _feeOnRaw(uint256 assets, uint256 feeBasisPoints) private pure returns (uint256) {
- return assets.mulDiv(feeBasisPoints, _BASIS_POINT_SCALE, Math.Rounding.Ceil);
- }
- /// @dev Calculates the fee part of an amount `assets` that already includes fees.
- /// Used in {IERC4626-deposit} and {IERC4626-redeem} operations.
- function _feeOnTotal(uint256 assets, uint256 feeBasisPoints) private pure returns (uint256) {
- return assets.mulDiv(feeBasisPoints, feeBasisPoints + _BASIS_POINT_SCALE, Math.Rounding.Ceil);
- }
- }
|