generated from PaulRBerg/foundry-template
-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathReplaceableRewardsToken.sol
92 lines (76 loc) · 4.14 KB
/
ReplaceableRewardsToken.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
// SPDX-License-Identifier: AGPL-3.0-only
pragma solidity ^0.8.0;
import { ERC20 } from "@solmate/tokens/ERC20.sol";
import { SafeTransferLib } from "@solmate/utils/SafeTransferLib.sol";
import { Owned } from "@solmate/auth/Owned.sol";
/**
* @dev This ERC20 tokenized vault overcomes the ERC20Rewards limitation of not being able to change the rewards token.
* A permissioned function allows to mint shares to an address.
* A permissioned function allows to set or change the asset.
* When changing the asset, the caller may consider keeping the price per share constant.
* A permissioned function allows to extract any tokens except the asset to any address.
* On transfer, the token is burned and the to receives his share of assets.
* Two helper functions allow to convert between assets and shares.
*/
contract ReplaceableRewardsToken is ERC20, Owned {
using SafeTransferLib for ERC20;
event AssetSet(ERC20 newAsset);
event Skimmed(ERC20 token, address to);
ERC20 public asset;
constructor (address owner_, string memory name_, string memory symbol_, uint8 decimals_)
ERC20(name_, symbol_, decimals_)
Owned(owner_)
{}
/// @dev Total assets held by the contract.
function totalAssets() external view virtual returns (uint256) {
return asset != ERC20(address(0)) ? asset.balanceOf(address(this)) : 0;
}
/// @dev Convert shares to assets.
function convertToAssets(uint256 shares) external view virtual returns (uint256) {
require(asset != ERC20(address(0)), "Asset not set");
return totalSupply > 0 ? asset.balanceOf(address(this)) * shares / totalSupply : 0;
}
/// @dev Convert assets to shares.
function convertToShares(uint256 assets) external view returns (uint256) {
require(asset != ERC20(address(0)), "Asset not set");
uint _totalAssets = asset.balanceOf(address(this));
return _totalAssets > 0 ? assets * totalSupply / _totalAssets : 0;
}
/// @dev A permissioned function that allows to set or change the asset.
/// @notice If changing the asset, the caller may consider keeping the price per share constant in USD or ETH terms.
/// After changing the asset, use `skim` to remove any old funds.
/// Be careful what address you pass on as an ERC20, it is not validated in any way.
/// Funds should be transferred before this call. Consider rounding.
function set(ERC20 asset_) external virtual onlyOwner {
asset = asset_;
emit AssetSet(asset_);
}
/// @dev A permissioned function that allows to extract any tokens except the asset to any address.
/// @notice Be careful what address you pass on as an ERC20, it is not validated in any way.
function skim(ERC20 token, address to) external virtual onlyOwner {
require(token != asset, "Cannot skim asset");
token.safeTransfer(to, token.balanceOf(address(this)));
emit Skimmed(token, to);
}
/// @dev A permissioned function that allows to mint shares to an address, if the asset is set.
/// @notice If minting with a non-zero supply, the price per share will change unless funds are added in the exact amount needed.
function mint(address to, uint256 amount) external virtual onlyOwner {
_mint(to, amount);
}
/// @dev On transfer, the token is burned and the recipient receives his share of assets.
/// @notice If the asset is not a valid ERC20, the transfer may fail.
function _transfer(address from, address to, uint256 shares) internal virtual returns (bool) {
uint assetAmount = asset.balanceOf(address(this)) * shares / totalSupply;
_burn(from, shares);
asset.safeTransfer(to, assetAmount);
return true;
}
/// @dev Transfer tokens, after updating rewards for source and destination.
function transfer(address to, uint amount) public virtual override returns (bool) {
return _transfer(msg.sender, to, amount);
}
/// @dev Transfer tokens, after updating rewards for source and destination.
function transferFrom(address from, address to, uint amount) public virtual override returns (bool) {
return _transfer(from, to, amount);
}
}