Created
February 28, 2025 14:22
Created using remix-ide: Realtime Ethereum Contract Compiler and Runtime. Load this file by pasting this gists URL or ID at https://remix.ethereum.org/#version=soljson-v0.8.28+commit.7893614a.js&optimize=false&runs=200&gist=
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
// File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v5.0.1/contracts/token/ERC20/IERC20.sol | |
// OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/IERC20.sol) | |
pragma solidity ^0.8.20; | |
/** | |
* @dev Interface of the ERC20 standard as defined in the EIP. | |
*/ | |
interface IERC20 { | |
/** | |
* @dev Emitted when `value` tokens are moved from one account (`from`) to | |
* another (`to`). | |
* | |
* Note that `value` may be zero. | |
*/ | |
event Transfer(address indexed from, address indexed to, uint256 value); | |
/** | |
* @dev Emitted when the allowance of a `spender` for an `owner` is set by | |
* a call to {approve}. `value` is the new allowance. | |
*/ | |
event Approval(address indexed owner, address indexed spender, uint256 value); | |
/** | |
* @dev Returns the value of tokens in existence. | |
*/ | |
function totalSupply() external view returns (uint256); | |
/** | |
* @dev Returns the value of tokens owned by `account`. | |
*/ | |
function balanceOf(address account) external view returns (uint256); | |
/** | |
* @dev Moves a `value` amount of tokens from the caller's account to `to`. | |
* | |
* Returns a boolean value indicating whether the operation succeeded. | |
* | |
* Emits a {Transfer} event. | |
*/ | |
function transfer(address to, uint256 value) external returns (bool); | |
/** | |
* @dev Returns the remaining number of tokens that `spender` will be | |
* allowed to spend on behalf of `owner` through {transferFrom}. This is | |
* zero by default. | |
* | |
* This value changes when {approve} or {transferFrom} are called. | |
*/ | |
function allowance(address owner, address spender) external view returns (uint256); | |
/** | |
* @dev Sets a `value` amount of tokens as the allowance of `spender` over the | |
* caller's tokens. | |
* | |
* Returns a boolean value indicating whether the operation succeeded. | |
* | |
* IMPORTANT: Beware that changing an allowance with this method brings the risk | |
* that someone may use both the old and the new allowance by unfortunate | |
* transaction ordering. One possible solution to mitigate this race | |
* condition is to first reduce the spender's allowance to 0 and set the | |
* desired value afterwards: | |
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 | |
* | |
* Emits an {Approval} event. | |
*/ | |
function approve(address spender, uint256 value) external returns (bool); | |
/** | |
* @dev Moves a `value` amount of tokens from `from` to `to` using the | |
* allowance mechanism. `value` is then deducted from the caller's | |
* allowance. | |
* | |
* Returns a boolean value indicating whether the operation succeeded. | |
* | |
* Emits a {Transfer} event. | |
*/ | |
function transferFrom(address from, address to, uint256 value) external returns (bool); | |
} | |
// File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v5.0.1/contracts/token/ERC20/extensions/IERC20Metadata.sol | |
// OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/IERC20Metadata.sol) | |
pragma solidity ^0.8.20; | |
/** | |
* @dev Interface for the optional metadata functions from the ERC20 standard. | |
*/ | |
interface IERC20Metadata is IERC20 { | |
/** | |
* @dev Returns the name of the token. | |
*/ | |
function name() external view returns (string memory); | |
/** | |
* @dev Returns the symbol of the token. | |
*/ | |
function symbol() external view returns (string memory); | |
/** | |
* @dev Returns the decimals places of the token. | |
*/ | |
function decimals() external view returns (uint8); | |
} | |
// File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v5.0.1/contracts/utils/Context.sol | |
// OpenZeppelin Contracts (last updated v5.0.1) (utils/Context.sol) | |
pragma solidity ^0.8.20; | |
/** | |
* @dev Provides information about the current execution context, including the | |
* sender of the transaction and its data. While these are generally available | |
* via msg.sender and msg.data, they should not be accessed in such a direct | |
* manner, since when dealing with meta-transactions the account sending and | |
* paying for execution may not be the actual sender (as far as an application | |
* is concerned). | |
* | |
* This contract is only required for intermediate, library-like contracts. | |
*/ | |
abstract contract Context { | |
function _msgSender() internal view virtual returns (address) { | |
return msg.sender; | |
} | |
function _msgData() internal view virtual returns (bytes calldata) { | |
return msg.data; | |
} | |
function _contextSuffixLength() internal view virtual returns (uint256) { | |
return 0; | |
} | |
} | |
// File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v5.0.1/contracts/interfaces/draft-IERC6093.sol | |
// OpenZeppelin Contracts (last updated v5.0.0) (interfaces/draft-IERC6093.sol) | |
pragma solidity ^0.8.20; | |
/** | |
* @dev Standard ERC20 Errors | |
* Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC20 tokens. | |
*/ | |
interface IERC20Errors { | |
/** | |
* @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers. | |
* @param sender Address whose tokens are being transferred. | |
* @param balance Current balance for the interacting account. | |
* @param needed Minimum amount required to perform a transfer. | |
*/ | |
error ERC20InsufficientBalance(address sender, uint256 balance, uint256 needed); | |
/** | |
* @dev Indicates a failure with the token `sender`. Used in transfers. | |
* @param sender Address whose tokens are being transferred. | |
*/ | |
error ERC20InvalidSender(address sender); | |
/** | |
* @dev Indicates a failure with the token `receiver`. Used in transfers. | |
* @param receiver Address to which tokens are being transferred. | |
*/ | |
error ERC20InvalidReceiver(address receiver); | |
/** | |
* @dev Indicates a failure with the `spender`’s `allowance`. Used in transfers. | |
* @param spender Address that may be allowed to operate on tokens without being their owner. | |
* @param allowance Amount of tokens a `spender` is allowed to operate with. | |
* @param needed Minimum amount required to perform a transfer. | |
*/ | |
error ERC20InsufficientAllowance(address spender, uint256 allowance, uint256 needed); | |
/** | |
* @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. | |
* @param approver Address initiating an approval operation. | |
*/ | |
error ERC20InvalidApprover(address approver); | |
/** | |
* @dev Indicates a failure with the `spender` to be approved. Used in approvals. | |
* @param spender Address that may be allowed to operate on tokens without being their owner. | |
*/ | |
error ERC20InvalidSpender(address spender); | |
} | |
/** | |
* @dev Standard ERC721 Errors | |
* Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC721 tokens. | |
*/ | |
interface IERC721Errors { | |
/** | |
* @dev Indicates that an address can't be an owner. For example, `address(0)` is a forbidden owner in EIP-20. | |
* Used in balance queries. | |
* @param owner Address of the current owner of a token. | |
*/ | |
error ERC721InvalidOwner(address owner); | |
/** | |
* @dev Indicates a `tokenId` whose `owner` is the zero address. | |
* @param tokenId Identifier number of a token. | |
*/ | |
error ERC721NonexistentToken(uint256 tokenId); | |
/** | |
* @dev Indicates an error related to the ownership over a particular token. Used in transfers. | |
* @param sender Address whose tokens are being transferred. | |
* @param tokenId Identifier number of a token. | |
* @param owner Address of the current owner of a token. | |
*/ | |
error ERC721IncorrectOwner(address sender, uint256 tokenId, address owner); | |
/** | |
* @dev Indicates a failure with the token `sender`. Used in transfers. | |
* @param sender Address whose tokens are being transferred. | |
*/ | |
error ERC721InvalidSender(address sender); | |
/** | |
* @dev Indicates a failure with the token `receiver`. Used in transfers. | |
* @param receiver Address to which tokens are being transferred. | |
*/ | |
error ERC721InvalidReceiver(address receiver); | |
/** | |
* @dev Indicates a failure with the `operator`’s approval. Used in transfers. | |
* @param operator Address that may be allowed to operate on tokens without being their owner. | |
* @param tokenId Identifier number of a token. | |
*/ | |
error ERC721InsufficientApproval(address operator, uint256 tokenId); | |
/** | |
* @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. | |
* @param approver Address initiating an approval operation. | |
*/ | |
error ERC721InvalidApprover(address approver); | |
/** | |
* @dev Indicates a failure with the `operator` to be approved. Used in approvals. | |
* @param operator Address that may be allowed to operate on tokens without being their owner. | |
*/ | |
error ERC721InvalidOperator(address operator); | |
} | |
/** | |
* @dev Standard ERC1155 Errors | |
* Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC1155 tokens. | |
*/ | |
interface IERC1155Errors { | |
/** | |
* @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers. | |
* @param sender Address whose tokens are being transferred. | |
* @param balance Current balance for the interacting account. | |
* @param needed Minimum amount required to perform a transfer. | |
* @param tokenId Identifier number of a token. | |
*/ | |
error ERC1155InsufficientBalance(address sender, uint256 balance, uint256 needed, uint256 tokenId); | |
/** | |
* @dev Indicates a failure with the token `sender`. Used in transfers. | |
* @param sender Address whose tokens are being transferred. | |
*/ | |
error ERC1155InvalidSender(address sender); | |
/** | |
* @dev Indicates a failure with the token `receiver`. Used in transfers. | |
* @param receiver Address to which tokens are being transferred. | |
*/ | |
error ERC1155InvalidReceiver(address receiver); | |
/** | |
* @dev Indicates a failure with the `operator`’s approval. Used in transfers. | |
* @param operator Address that may be allowed to operate on tokens without being their owner. | |
* @param owner Address of the current owner of a token. | |
*/ | |
error ERC1155MissingApprovalForAll(address operator, address owner); | |
/** | |
* @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. | |
* @param approver Address initiating an approval operation. | |
*/ | |
error ERC1155InvalidApprover(address approver); | |
/** | |
* @dev Indicates a failure with the `operator` to be approved. Used in approvals. | |
* @param operator Address that may be allowed to operate on tokens without being their owner. | |
*/ | |
error ERC1155InvalidOperator(address operator); | |
/** | |
* @dev Indicates an array length mismatch between ids and values in a safeBatchTransferFrom operation. | |
* Used in batch transfers. | |
* @param idsLength Length of the array of token identifiers | |
* @param valuesLength Length of the array of token amounts | |
*/ | |
error ERC1155InvalidArrayLength(uint256 idsLength, uint256 valuesLength); | |
} | |
// File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v5.0.1/contracts/token/ERC20/ERC20.sol | |
// OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/ERC20.sol) | |
pragma solidity ^0.8.20; | |
/** | |
* @dev Implementation of the {IERC20} interface. | |
* | |
* This implementation is agnostic to the way tokens are created. This means | |
* that a supply mechanism has to be added in a derived contract using {_mint}. | |
* | |
* TIP: For a detailed writeup see our guide | |
* https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How | |
* to implement supply mechanisms]. | |
* | |
* The default value of {decimals} is 18. To change this, you should override | |
* this function so it returns a different value. | |
* | |
* We have followed general OpenZeppelin Contracts guidelines: functions revert | |
* instead returning `false` on failure. This behavior is nonetheless | |
* conventional and does not conflict with the expectations of ERC20 | |
* applications. | |
* | |
* Additionally, an {Approval} event is emitted on calls to {transferFrom}. | |
* This allows applications to reconstruct the allowance for all accounts just | |
* by listening to said events. Other implementations of the EIP may not emit | |
* these events, as it isn't required by the specification. | |
*/ | |
abstract contract ERC20 is Context, IERC20, IERC20Metadata, IERC20Errors { | |
mapping(address account => uint256) private _balances; | |
mapping(address account => mapping(address spender => uint256)) private _allowances; | |
uint256 private _totalSupply; | |
string private _name; | |
string private _symbol; | |
/** | |
* @dev Sets the values for {name} and {symbol}. | |
* | |
* All two of these values are immutable: they can only be set once during | |
* construction. | |
*/ | |
constructor(string memory name_, string memory symbol_) { | |
_name = name_; | |
_symbol = symbol_; | |
} | |
/** | |
* @dev Returns the name of the token. | |
*/ | |
function name() public view virtual returns (string memory) { | |
return _name; | |
} | |
/** | |
* @dev Returns the symbol of the token, usually a shorter version of the | |
* name. | |
*/ | |
function symbol() public view virtual returns (string memory) { | |
return _symbol; | |
} | |
/** | |
* @dev Returns the number of decimals used to get its user representation. | |
* For example, if `decimals` equals `2`, a balance of `505` tokens should | |
* be displayed to a user as `5.05` (`505 / 10 ** 2`). | |
* | |
* Tokens usually opt for a value of 18, imitating the relationship between | |
* Ether and Wei. This is the default value returned by this function, unless | |
* it's overridden. | |
* | |
* NOTE: This information is only used for _display_ purposes: it in | |
* no way affects any of the arithmetic of the contract, including | |
* {IERC20-balanceOf} and {IERC20-transfer}. | |
*/ | |
function decimals() public view virtual returns (uint8) { | |
return 18; | |
} | |
/** | |
* @dev See {IERC20-totalSupply}. | |
*/ | |
function totalSupply() public view virtual returns (uint256) { | |
return _totalSupply; | |
} | |
/** | |
* @dev See {IERC20-balanceOf}. | |
*/ | |
function balanceOf(address account) public view virtual returns (uint256) { | |
return _balances[account]; | |
} | |
/** | |
* @dev See {IERC20-transfer}. | |
* | |
* Requirements: | |
* | |
* - `to` cannot be the zero address. | |
* - the caller must have a balance of at least `value`. | |
*/ | |
function transfer(address to, uint256 value) public virtual returns (bool) { | |
address owner = _msgSender(); | |
_transfer(owner, to, value); | |
return true; | |
} | |
/** | |
* @dev See {IERC20-allowance}. | |
*/ | |
function allowance(address owner, address spender) public view virtual returns (uint256) { | |
return _allowances[owner][spender]; | |
} | |
/** | |
* @dev See {IERC20-approve}. | |
* | |
* NOTE: If `value` is the maximum `uint256`, the allowance is not updated on | |
* `transferFrom`. This is semantically equivalent to an infinite approval. | |
* | |
* Requirements: | |
* | |
* - `spender` cannot be the zero address. | |
*/ | |
function approve(address spender, uint256 value) public virtual returns (bool) { | |
address owner = _msgSender(); | |
_approve(owner, spender, value); | |
return true; | |
} | |
/** | |
* @dev See {IERC20-transferFrom}. | |
* | |
* Emits an {Approval} event indicating the updated allowance. This is not | |
* required by the EIP. See the note at the beginning of {ERC20}. | |
* | |
* NOTE: Does not update the allowance if the current allowance | |
* is the maximum `uint256`. | |
* | |
* Requirements: | |
* | |
* - `from` and `to` cannot be the zero address. | |
* - `from` must have a balance of at least `value`. | |
* - the caller must have allowance for ``from``'s tokens of at least | |
* `value`. | |
*/ | |
function transferFrom(address from, address to, uint256 value) public virtual returns (bool) { | |
address spender = _msgSender(); | |
_spendAllowance(from, spender, value); | |
_transfer(from, to, value); | |
return true; | |
} | |
/** | |
* @dev Moves a `value` amount of tokens from `from` to `to`. | |
* | |
* This internal function is equivalent to {transfer}, and can be used to | |
* e.g. implement automatic token fees, slashing mechanisms, etc. | |
* | |
* Emits a {Transfer} event. | |
* | |
* NOTE: This function is not virtual, {_update} should be overridden instead. | |
*/ | |
function _transfer(address from, address to, uint256 value) internal { | |
if (from == address(0)) { | |
revert ERC20InvalidSender(address(0)); | |
} | |
if (to == address(0)) { | |
revert ERC20InvalidReceiver(address(0)); | |
} | |
_update(from, to, value); | |
} | |
/** | |
* @dev Transfers a `value` amount of tokens from `from` to `to`, or alternatively mints (or burns) if `from` | |
* (or `to`) is the zero address. All customizations to transfers, mints, and burns should be done by overriding | |
* this function. | |
* | |
* Emits a {Transfer} event. | |
*/ | |
function _update(address from, address to, uint256 value) internal virtual { | |
if (from == address(0)) { | |
// Overflow check required: The rest of the code assumes that totalSupply never overflows | |
_totalSupply += value; | |
} else { | |
uint256 fromBalance = _balances[from]; | |
if (fromBalance < value) { | |
revert ERC20InsufficientBalance(from, fromBalance, value); | |
} | |
unchecked { | |
// Overflow not possible: value <= fromBalance <= totalSupply. | |
_balances[from] = fromBalance - value; | |
} | |
} | |
if (to == address(0)) { | |
unchecked { | |
// Overflow not possible: value <= totalSupply or value <= fromBalance <= totalSupply. | |
_totalSupply -= value; | |
} | |
} else { | |
unchecked { | |
// Overflow not possible: balance + value is at most totalSupply, which we know fits into a uint256. | |
_balances[to] += value; | |
} | |
} | |
emit Transfer(from, to, value); | |
} | |
/** | |
* @dev Creates a `value` amount of tokens and assigns them to `account`, by transferring it from address(0). | |
* Relies on the `_update` mechanism | |
* | |
* Emits a {Transfer} event with `from` set to the zero address. | |
* | |
* NOTE: This function is not virtual, {_update} should be overridden instead. | |
*/ | |
function _mint(address account, uint256 value) internal { | |
if (account == address(0)) { | |
revert ERC20InvalidReceiver(address(0)); | |
} | |
_update(address(0), account, value); | |
} | |
/** | |
* @dev Destroys a `value` amount of tokens from `account`, lowering the total supply. | |
* Relies on the `_update` mechanism. | |
* | |
* Emits a {Transfer} event with `to` set to the zero address. | |
* | |
* NOTE: This function is not virtual, {_update} should be overridden instead | |
*/ | |
function _burn(address account, uint256 value) internal { | |
if (account == address(0)) { | |
revert ERC20InvalidSender(address(0)); | |
} | |
_update(account, address(0), value); | |
} | |
/** | |
* @dev Sets `value` as the allowance of `spender` over the `owner` s tokens. | |
* | |
* This internal function is equivalent to `approve`, and can be used to | |
* e.g. set automatic allowances for certain subsystems, etc. | |
* | |
* Emits an {Approval} event. | |
* | |
* Requirements: | |
* | |
* - `owner` cannot be the zero address. | |
* - `spender` cannot be the zero address. | |
* | |
* Overrides to this logic should be done to the variant with an additional `bool emitEvent` argument. | |
*/ | |
function _approve(address owner, address spender, uint256 value) internal { | |
_approve(owner, spender, value, true); | |
} | |
/** | |
* @dev Variant of {_approve} with an optional flag to enable or disable the {Approval} event. | |
* | |
* By default (when calling {_approve}) the flag is set to true. On the other hand, approval changes made by | |
* `_spendAllowance` during the `transferFrom` operation set the flag to false. This saves gas by not emitting any | |
* `Approval` event during `transferFrom` operations. | |
* | |
* Anyone who wishes to continue emitting `Approval` events on the`transferFrom` operation can force the flag to | |
* true using the following override: | |
* ``` | |
* function _approve(address owner, address spender, uint256 value, bool) internal virtual override { | |
* super._approve(owner, spender, value, true); | |
* } | |
* ``` | |
* | |
* Requirements are the same as {_approve}. | |
*/ | |
function _approve(address owner, address spender, uint256 value, bool emitEvent) internal virtual { | |
if (owner == address(0)) { | |
revert ERC20InvalidApprover(address(0)); | |
} | |
if (spender == address(0)) { | |
revert ERC20InvalidSpender(address(0)); | |
} | |
_allowances[owner][spender] = value; | |
if (emitEvent) { | |
emit Approval(owner, spender, value); | |
} | |
} | |
/** | |
* @dev Updates `owner` s allowance for `spender` based on spent `value`. | |
* | |
* Does not update the allowance value in case of infinite allowance. | |
* Revert if not enough allowance is available. | |
* | |
* Does not emit an {Approval} event. | |
*/ | |
function _spendAllowance(address owner, address spender, uint256 value) internal virtual { | |
uint256 currentAllowance = allowance(owner, spender); | |
if (currentAllowance != type(uint256).max) { | |
if (currentAllowance < value) { | |
revert ERC20InsufficientAllowance(spender, currentAllowance, value); | |
} | |
unchecked { | |
_approve(owner, spender, currentAllowance - value, false); | |
} | |
} | |
} | |
} | |
// File: contracts/Ownable.sol | |
/// @title Owner Base Class | |
/* | |
* __ __ _______ ____ | |
* / / / / / / ___// __ \ | |
* / / / / / /\__ \/ / / / | |
* / /___/ /_/ /___/ / /_/ / | |
* /_____/\____//____/_____/ | |
*/ | |
pragma solidity ^0.8.28; | |
interface IOwnable { | |
function transferOwnership(address newOwner) external; | |
} | |
contract Ownable is IOwnable { | |
address internal _owner; | |
event OwnershipRenounced(address indexed previousOwner); | |
event OwnershipTransferred( | |
address indexed previousOwner, | |
address indexed newOwner | |
); | |
constructor() { | |
_owner = msg.sender; | |
} | |
function owner() public view returns (address) { | |
return _owner; | |
} | |
modifier onlyOwner() { | |
require(isOwner()); | |
_; | |
} | |
function isOwner() public view returns (bool) { | |
return msg.sender == _owner; | |
} | |
function renounceOwnership() public onlyOwner { | |
emit OwnershipRenounced(_owner); | |
_owner = address(0); | |
} | |
function transferOwnership(address newOwner) public onlyOwner { | |
_transferOwnership(newOwner); | |
} | |
function _transferOwnership(address newOwner) internal { | |
require(newOwner != address(0)); | |
emit OwnershipTransferred(_owner, newOwner); | |
_owner = newOwner; | |
} | |
} | |
// File: contracts/Normalizer.sol | |
/// @title Base Decimal Normalizer based on work by Paul Razvan Berg | |
// Sources: https://github.com/PaulRBerg/prb-contracts/blob/main/src/token/erc20/ERC20Normalizer.sol | |
/* | |
* __ __ _______ ____ | |
* / / / / / / ___// __ \ | |
* / / / / / /\__ \/ / / / | |
* / /___/ /_/ /___/ / /_/ / | |
* /_____/\____//____/_____/ | |
*/ | |
pragma solidity ^0.8.28; | |
abstract contract Normalizer { | |
mapping(uint8 => uint256) private _scalars; | |
function _normalize( | |
uint256 amount, | |
uint8 decimals | |
) internal returns (uint256) { | |
if (decimals == 18) { | |
return amount; | |
} | |
uint256 scalar = _computeScalar(decimals); | |
return scalar == 1 ? amount : amount * _computeScalar(decimals); | |
} | |
function _denormalize( | |
uint256 amount, | |
uint8 decimals | |
) internal returns (uint256) { | |
if (decimals == 18) { | |
return amount; | |
} | |
uint256 scalar = _computeScalar(decimals); | |
return scalar == 1 ? amount : amount / _computeScalar(decimals); | |
} | |
function _computeScalar(uint8 decimals) internal returns (uint256 scalar) { | |
if (decimals > 18) { | |
revert DecimalsGreaterThan18(decimals); | |
} | |
scalar = _scalars[decimals]; | |
if (scalar == 0) { | |
unchecked { | |
_scalars[decimals] = scalar = 10 ** (18 - decimals); | |
} | |
} | |
} | |
error DecimalsGreaterThan18(uint256 decimals); | |
} | |
// File: contracts/interfaces/IAggregator.sol | |
/// @title IAggregator | |
/* | |
* __ __ _______ ____ | |
* / / / / / / ___// __ \ | |
* / / / / / /\__ \/ / / / | |
* / /___/ /_/ /___/ / /_/ / | |
* /_____/\____//____/_____/ | |
*/ | |
pragma solidity ^0.8.28; | |
interface IAggregator { | |
function decimals() external view returns (uint8); | |
function latestAnswer() external view returns (int256); | |
function latestTimestamp() external view returns (uint256); | |
function latestRound() external view returns (uint256); | |
function getAnswer(uint256 roundId) external view returns (int256); | |
function getTimestamp(uint256 roundId) external view returns (uint256); | |
event AnswerUpdated( | |
int256 indexed current, | |
uint256 indexed roundId, | |
uint256 updatedAt | |
); | |
event NewRound( | |
uint256 indexed roundId, | |
address indexed startedBy, | |
uint256 startedAt | |
); | |
} | |
// File: contracts/interfaces/ILiquidity.sol | |
/// @title Basic liqudity interface | |
// Sources: | |
// https://docs.uniswap.org/contracts/v2/reference/smart-contracts/common-errors | |
/* | |
* __ __ _______ ____ | |
* / / / / / / ___// __ \ | |
* / / / / / /\__ \/ / / / | |
* / /___/ /_/ /___/ / /_/ / | |
* /_____/\____//____/_____/ | |
*/ | |
pragma solidity ^0.8.28; | |
interface ILiquidity { | |
function sync() external; | |
} | |
// File: contracts/interfaces/IERC20.sol | |
/// @title Standard Interface for ERC20 tokens | |
// Sources: | |
// https://eips.ethereum.org/EIPS/eip-20 | |
// https://docs.google.com/document/d/1YLPtQxZu1UAvO9cZ1O2RPXBbT0mooh4DYKjA_jp-RLM | |
// https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC20/IERC20.sol | |
/* | |
* __ __ _______ ____ | |
* / / / / / / ___// __ \ | |
* / / / / / /\__ \/ / / / | |
* / /___/ /_/ /___/ / /_/ / | |
* /_____/\____//____/_____/ | |
*/ | |
pragma solidity ^0.8.28; | |
interface ERC20Interface { | |
/** | |
* @dev Emitted when `value` tokens are moved from one account (`from`) to | |
* another (`to`). | |
* | |
* Note that `value` may be zero. | |
*/ | |
event Transfer(address indexed from, address indexed to, uint256 value); | |
/** | |
* @dev Emitted when the allowance of a `spender` for an `owner` is set by | |
* a call to {approve}. `value` is the new allowance. | |
*/ | |
event Approval( | |
address indexed owner, | |
address indexed spender, | |
uint256 value | |
); | |
/** | |
* @dev OPTIONAL Returns the name of the token | |
*/ | |
function name() external view returns (string memory); | |
/** | |
* @dev OPTIONAL Returns the symbol of the token | |
*/ | |
function symbol() external view returns (string memory); | |
/** | |
* @dev OPTIONAL Returns the amount of decimals supported by the token | |
*/ | |
function decimals() external view returns (uint8); | |
/** | |
* @dev Returns the value of tokens in existence. | |
*/ | |
function totalSupply() external view returns (uint256); | |
/** | |
* @dev Returns the value of tokens owned by `account`. | |
*/ | |
function balanceOf(address account) external view returns (uint256); | |
/** | |
* @dev Moves a `value` amount of tokens from the caller's account to `to`. | |
* | |
* Returns a boolean value indicating whether the operation succeeded. | |
* | |
* Emits a {Transfer} event. | |
*/ | |
function transfer(address to, uint256 value) external returns (bool); | |
/** | |
* @dev Returns the remaining number of tokens that `spender` will be | |
* allowed to spend on behalf of `owner` through {transferFrom}. This is | |
* zero by default. | |
* | |
* This value changes when {approve} or {transferFrom} are called. | |
*/ | |
function allowance( | |
address owner, | |
address spender | |
) external view returns (uint256); | |
/** | |
* @dev Sets a `value` amount of tokens as the allowance of `spender` over the | |
* caller's tokens. | |
* | |
* Returns a boolean value indicating whether the operation succeeded. | |
* | |
* IMPORTANT: Beware that changing an allowance with this method brings the risk | |
* that someone may use both the old and the new allowance by unfortunate | |
* transaction ordering. One possible solution to mitigate this race | |
* condition is to first reduce the spender's allowance to 0 and set the | |
* desired value afterwards: | |
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 | |
* | |
* Emits an {Approval} event. | |
*/ | |
function approve(address spender, uint256 value) external returns (bool); | |
/** | |
* @dev Moves a `value` amount of tokens from `from` to `to` using the | |
* allowance mechanism. `value` is then deducted from the caller's | |
* allowance. | |
* | |
* Returns a boolean value indicating whether the operation succeeded. | |
* | |
* Emits a {Transfer} event. | |
*/ | |
function transferFrom( | |
address from, | |
address to, | |
uint256 value | |
) external returns (bool); | |
} | |
// File: contracts/libraries/PengMath.sol | |
/// @title Direct port of mulDiv functions from PRBMath by Paul Razvan Berg | |
// Sources: | |
// https://2π.com/21/muldiv/ | |
// https://github.com/PaulRBerg/prb-math/tree/main/src/ud60x18 | |
/* | |
* __ __ _______ ____ | |
* / / / / / / ___// __ \ | |
* / / / / / /\__ \/ / / / | |
* / /___/ /_/ /___/ / /_/ / | |
* /_____/\____//____/_____/ | |
*/ | |
pragma solidity ^0.8.22; | |
library PengMath { | |
/// @dev The unit number, which the decimal precision of the fixed-point types. | |
uint256 constant UNIT = 1e18; | |
/// @dev The the largest power of two that divides the decimal value of `UNIT`. The logarithm of this value is the least significant | |
/// bit in the binary representation of `UNIT`. | |
uint256 constant UNIT_LPOTD = 262144; | |
/// @dev The unit number inverted mod 2^256. | |
uint256 constant UNIT_INVERSE = 78156646155174841979727994598816262306175212592076161876661_508869554232690281; | |
error MulDiv18Overflow(uint256 x, uint256 y); | |
error MulDivOverflow(uint256 x, uint256 y, uint256 denominator); | |
function mul(uint256 x, uint256 y) internal pure returns(uint256 result) { | |
return _mulDiv18(x, y); | |
} | |
function div(uint256 x, uint256 y) internal pure returns(uint256 result) { | |
return _mulDiv(x, UNIT, y); | |
} | |
/// @notice Calculates x*y÷1e18 with 512-bit precision. | |
/// | |
/// @dev A variant of {mulDiv} with constant folding, i.e. in which the denominator is hard coded to 1e18. | |
/// | |
/// Notes: | |
/// - The body is purposely left uncommented; to understand how this works, see the documentation in {mulDiv}. | |
/// - The result is rounded toward zero. | |
/// - We take as an axiom that the result cannot be `MAX_UINT256` when x and y solve the following system of equations: | |
/// | |
/// $$ | |
/// \begin{cases} | |
/// x * y = MAX\_UINT256 * UNIT \\ | |
/// (x * y) \% UNIT \geq \frac{UNIT}{2} | |
/// \end{cases} | |
/// $$ | |
/// | |
/// Requirements: | |
/// - Refer to the requirements in {mulDiv}. | |
/// - The result must fit in uint256. | |
/// | |
/// @param x The multiplicand as an unsigned 60.18-decimal fixed-point number. | |
/// @param y The multiplier as an unsigned 60.18-decimal fixed-point number. | |
/// @return result The result as an unsigned 60.18-decimal fixed-point number. | |
/// @custom:smtchecker abstract-function-nondet | |
function _mulDiv18(uint256 x, uint256 y) private pure returns (uint256 result) { | |
uint256 prod0; | |
uint256 prod1; | |
assembly ("memory-safe") { | |
let mm := mulmod(x, y, not(0)) | |
prod0 := mul(x, y) | |
prod1 := sub(sub(mm, prod0), lt(mm, prod0)) | |
} | |
if (prod1 == 0) { | |
unchecked { | |
return prod0 / UNIT; | |
} | |
} | |
if (prod1 >= UNIT) { | |
revert MulDiv18Overflow(x, y); | |
} | |
uint256 remainder; | |
assembly ("memory-safe") { | |
remainder := mulmod(x, y, UNIT) | |
result := | |
mul( | |
or( | |
div(sub(prod0, remainder), UNIT_LPOTD), | |
mul(sub(prod1, gt(remainder, prod0)), add(div(sub(0, UNIT_LPOTD), UNIT_LPOTD), 1)) | |
), | |
UNIT_INVERSE | |
) | |
} | |
} | |
/// @notice Calculates x*y÷denominator with 512-bit precision. | |
/// | |
/// @dev Credits to Remco Bloemen under MIT license https://xn--2-umb.com/21/muldiv. | |
/// | |
/// Notes: | |
/// - The result is rounded toward zero. | |
/// | |
/// Requirements: | |
/// - The denominator must not be zero. | |
/// - The result must fit in uint256. | |
/// | |
/// @param x The multiplicand as a uint256. | |
/// @param y The multiplier as a uint256. | |
/// @param denominator The divisor as a uint256. | |
/// @return result The result as a uint256. | |
/// @custom:smtchecker abstract-function-nondet | |
function _mulDiv(uint256 x, uint256 y, uint256 denominator) private pure returns (uint256 result) { | |
// 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use | |
// use the Chinese Remainder Theorem to reconstruct the 512-bit result. The result is stored in two 256 | |
// variables such that product = prod1 * 2^256 + prod0. | |
uint256 prod0; // Least significant 256 bits of the product | |
uint256 prod1; // Most significant 256 bits of the product | |
assembly ("memory-safe") { | |
let mm := mulmod(x, y, not(0)) | |
prod0 := mul(x, y) | |
prod1 := sub(sub(mm, prod0), lt(mm, prod0)) | |
} | |
// Handle non-overflow cases, 256 by 256 division. | |
if (prod1 == 0) { | |
unchecked { | |
return prod0 / denominator; | |
} | |
} | |
// Make sure the result is less than 2^256. Also prevents denominator == 0. | |
if (prod1 >= denominator) { | |
revert MulDivOverflow(x, y, denominator); | |
} | |
//////////////////////////////////////////////////////////////////////////// | |
// 512 by 256 division | |
//////////////////////////////////////////////////////////////////////////// | |
// Make division exact by subtracting the remainder from [prod1 prod0]. | |
uint256 remainder; | |
assembly ("memory-safe") { | |
// Compute remainder using the mulmod Yul instruction. | |
remainder := mulmod(x, y, denominator) | |
// Subtract 256 bit number from 512-bit number. | |
prod1 := sub(prod1, gt(remainder, prod0)) | |
prod0 := sub(prod0, remainder) | |
} | |
unchecked { | |
// Calculate the largest power of two divisor of the denominator using the unary operator ~. This operation cannot overflow | |
// because the denominator cannot be zero at this point in the function execution. The result is always >= 1. | |
// For more detail, see https://cs.stackexchange.com/q/138556/92363. | |
uint256 lpotdod = denominator & (~denominator + 1); | |
uint256 flippedLpotdod; | |
assembly ("memory-safe") { | |
// Factor powers of two out of denominator. | |
denominator := div(denominator, lpotdod) | |
// Divide [prod1 prod0] by lpotdod. | |
prod0 := div(prod0, lpotdod) | |
// Get the flipped value `2^256 / lpotdod`. If the `lpotdod` is zero, the flipped value is one. | |
// `sub(0, lpotdod)` produces the two's complement version of `lpotdod`, which is equivalent to flipping all the bits. | |
// However, `div` interprets this value as an unsigned value: https://ethereum.stackexchange.com/q/147168/24693 | |
flippedLpotdod := add(div(sub(0, lpotdod), lpotdod), 1) | |
} | |
// Shift in bits from prod1 into prod0. | |
prod0 |= prod1 * flippedLpotdod; | |
// Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such | |
// that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for | |
// four bits. That is, denominator * inv = 1 mod 2^4. | |
uint256 inverse = (3 * denominator) ^ 2; | |
// Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works | |
// in modular arithmetic, doubling the correct bits in each step. | |
inverse *= 2 - denominator * inverse; // inverse mod 2^8 | |
inverse *= 2 - denominator * inverse; // inverse mod 2^16 | |
inverse *= 2 - denominator * inverse; // inverse mod 2^32 | |
inverse *= 2 - denominator * inverse; // inverse mod 2^64 | |
inverse *= 2 - denominator * inverse; // inverse mod 2^128 | |
inverse *= 2 - denominator * inverse; // inverse mod 2^256 | |
// Because the division is now exact we can divide by multiplying with the modular inverse of denominator. | |
// This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is | |
// less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 | |
// is no longer required. | |
result = prod0 * inverse; | |
} | |
} | |
} | |
// File: contracts/LUSD.sol | |
/// @title LUSD ERC20 Root Implementation | |
pragma solidity ^0.8.28; | |
contract LUSD is ERC20, Normalizer, Ownable { | |
address public taxCollector; | |
ERC20Interface public tokenZero; | |
IAggregator public oracle; | |
ILiquidity public liquidity; | |
uint8 internal _tokenZeroDec; | |
uint8 internal _oracleDec; | |
uint256 private constant INITIAL_SUPPLY = 4_000_000_000e18; | |
uint256 private constant FEE_PERC = 5; // 0.05% fee | |
event Taxed(address addr, uint256 amount); | |
constructor() ERC20("Link Dollar", "LUSD") { | |
_mint(_owner, INITIAL_SUPPLY); | |
} | |
function setLiquidity(address liquidityAddr) public onlyOwner { | |
liquidity = ILiquidity(liquidityAddr); | |
rebase(); | |
} | |
function setTaxCollector(address taxAddr) public onlyOwner { | |
taxCollector = taxAddr; | |
} | |
function setOracle(address oracleAddr) public onlyOwner { | |
oracle = IAggregator(oracleAddr); | |
try oracle.decimals() returns (uint8 dec) { | |
_oracleDec = dec; | |
} catch { | |
_oracleDec = 8; | |
} | |
} | |
function setTokenZero(address tokenZeroAddr) public onlyOwner { | |
tokenZero = ERC20Interface(tokenZeroAddr); | |
try tokenZero.decimals() returns (uint8 dec) { | |
_tokenZeroDec = dec; | |
} catch { | |
_tokenZeroDec = 18; | |
} | |
} | |
function initialize( | |
address liquidityAddr, | |
address oracleAddr, | |
address tokenZeroAddr, | |
address taxAddr | |
) external onlyOwner { | |
require(address(liquidity) == address(0)); | |
require(address(tokenZero) == address(0)); | |
require(address(oracle) == address(0)); | |
require(taxCollector == address(0)); | |
setTaxCollector(taxAddr); | |
setTokenZero(tokenZeroAddr); | |
setOracle(oracleAddr); | |
setLiquidity(liquidityAddr); | |
} | |
function getBalances() | |
external | |
view | |
returns (uint256 balanceZero, uint256 balanceOne) | |
{ | |
balanceZero = tokenZero.balanceOf(address(liquidity)); | |
balanceOne = balanceOf(address(liquidity)); | |
} | |
function transfer( | |
address to, | |
uint256 value | |
) public virtual override returns (bool) { | |
address from = _msgSender(); | |
value = _tax(from, value); | |
_transfer(from, to, value); | |
return true; // Removed `rebase();` | |
} | |
function transferFrom( | |
address from, | |
address to, | |
uint256 value | |
) public virtual override returns (bool) { | |
address spender = _msgSender(); | |
_spendAllowance(from, spender, value); | |
value = _tax(from, value); | |
_transfer(from, to, value); | |
return true; // Removed `rebase();` | |
} | |
function approve( | |
address spender, | |
uint256 amount | |
) public virtual override returns (bool) { | |
address owner = _msgSender(); | |
_approve(owner, spender, amount); | |
rebase(); // Now triggers rebase when approve is called | |
return true; | |
} | |
function _tax(address from, uint256 value) private returns (uint256) { | |
address taxAcc = taxCollector; | |
uint256 transferAmount = value; | |
if (taxAcc == address(0)) { | |
taxAcc = _owner; | |
} | |
if (taxAcc == address(0)) { | |
return transferAmount; | |
} | |
uint256 fee = (value * FEE_PERC) / 10_000; | |
(uint256 amount, bool negative) = _absDiff(value, fee); | |
if (!negative && fee > 0) { | |
transferAmount = amount; | |
if (taxAcc != from) { | |
_transfer(from, taxAcc, fee); | |
} | |
emit Taxed(from, fee); | |
} | |
return transferAmount; | |
} | |
function _absDiff( | |
uint256 v1, | |
uint256 v2 | |
) private pure returns (uint256 diff, bool negative) { | |
if (v2 > v1) { | |
uint256 tmp = v1; | |
v1 = v2; | |
v2 = tmp; | |
negative = true; | |
} | |
diff = v1 - v2; | |
} | |
function getPrice() public view returns (int256) { | |
return oracle.latestAnswer(); | |
} | |
function _trySync() private { | |
try liquidity.sync() {} catch {} | |
} | |
function rebase() public { | |
address liquidityAddress = address(liquidity); | |
if ( | |
liquidityAddress == address(0) || address(tokenZero) == address(0) | |
) { | |
return; | |
} | |
uint256 price = uint256(getPrice()); | |
uint256 balanceZero = tokenZero.balanceOf(liquidityAddress); | |
uint256 balanceOne = balanceOf(liquidityAddress); | |
uint256 nbalanceZero = _normalize(balanceZero, _tokenZeroDec); | |
uint256 nprice = _normalize(price, _oracleDec); | |
uint256 nlastRebase = PengMath.mul(nbalanceZero, nprice); | |
uint256 nBalanceOne = _normalize(balanceOne, decimals()); | |
(uint256 nrebaseFactor, bool negative) = _absDiff( | |
nlastRebase, | |
nBalanceOne | |
); | |
uint256 rebaseFactor = _denormalize(nrebaseFactor, decimals()); | |
if (negative) { | |
_burn(liquidityAddress, rebaseFactor); | |
} else if (rebaseFactor > 0) { | |
_mint(liquidityAddress, rebaseFactor); | |
} | |
_trySync(); // Ensure sync is always called after rebase | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment