-
Notifications
You must be signed in to change notification settings - Fork 19
/
Oracle.sol
220 lines (182 loc) · 9.78 KB
/
Oracle.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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
// SPDX-License-Identifier: Apache-2.0
pragma solidity 0.8.24;
import { UFixed6 } from "@equilibria/root/number/types/UFixed6.sol";
import { UFixed18Lib } from "@equilibria/root/number/types/UFixed18.sol";
import { Token18 } from "@equilibria/root/token/types/Token18.sol";
import { Instance } from "@equilibria/root/attribute/Instance.sol";
import { IOracleProvider} from "@perennial/v2-core/contracts/interfaces/IOracleProvider.sol";
import { IMarket } from "@perennial/v2-core/contracts/interfaces/IMarket.sol";
import { OracleVersion } from "@perennial/v2-core/contracts/types/OracleVersion.sol";
import { OracleReceipt } from "@perennial/v2-core/contracts/types/OracleReceipt.sol";
import { IOracle } from "./interfaces/IOracle.sol";
/// @title Oracle
/// @notice The top-level oracle contract that implements an oracle provider interface.
/// @dev Manages swapping between different underlying oracle provider interfaces over time.
contract Oracle is IOracle, Instance {
/// @notice A historical mapping of underlying oracle providers
mapping(uint256 => Epoch) private _oracles;
/// @notice The global state of the oracle
OracleGlobal private _global;
/// @notice The market associated with this oracle
IMarket public market;
/// @notice The beneficiary of the oracle fee
address public beneficiary;
/// @notice The name of the oracle
string public name;
/// @notice Initializes the contract state
/// @param initialProvider The initial oracle provider
/// @param name_ The name of the oracle
function initialize(IOracleProvider initialProvider, string calldata name_) external initializer(1) {
__Instance__initialize();
_updateCurrent(initialProvider);
_updateLatest(initialProvider.latest());
name = name_;
}
/// @notice Updates the current oracle provider
/// @dev Both the current and new oracle provider must have the same current
/// @param newProvider The new oracle provider
function update(IOracleProvider newProvider) external onlyFactory {
_updateCurrent(newProvider);
_updateLatest(newProvider.latest());
}
/// @notice Returns the global state of the oracle
/// @return Returns current and latest epoch of the oracle
function global() external view returns (OracleGlobal memory) {
return _global;
}
/// @notice Registers the market associated with this oracle
/// @param newMarket The market to register
function register(IMarket newMarket) external onlyOwner {
market = newMarket;
emit MarketUpdated(newMarket);
}
/// @notice Updates the beneficiary of the oracle fee
/// @param newBeneficiary The new beneficiary
function updateBeneficiary(address newBeneficiary) external onlyOwner {
beneficiary = newBeneficiary;
emit BeneficiaryUpdated(newBeneficiary);
}
/// @notice Requests a new version at the current timestamp
/// @param account Original sender to optionally use for callbacks
function request(IMarket, address account) external onlyMarket {
(OracleVersion memory latestVersion, uint256 currentTimestamp) = _oracles[_global.current].provider.status();
_oracles[
(currentTimestamp > _oracles[_global.latest].timestamp) ? _global.current : _global.latest
].provider.request(market, account);
_oracles[_global.current].timestamp = uint96(currentTimestamp);
_updateLatest(latestVersion);
}
/// @notice Returns the latest committed version as well as the current timestamp
/// @return latestVersion The latest committed version
/// @return currentTimestamp The current timestamp
function status() external view returns (OracleVersion memory latestVersion, uint256 currentTimestamp) {
(latestVersion, currentTimestamp) = _oracles[_global.current].provider.status();
latestVersion = _handleLatest(latestVersion);
}
/// @notice Returns the latest committed version
function latest() public view returns (OracleVersion memory) {
return _handleLatest(_oracles[_global.current].provider.latest());
}
/// @notice Returns the current value
function current() public view returns (uint256) {
return _oracles[_global.current].provider.current();
}
/// @notice Returns the oracle version at a given timestamp
/// @param timestamp The timestamp to query
/// @return atVersion The oracle version at the given timestamp
/// @return atReceipt The oracle receipt at the given timestamp
function at(uint256 timestamp) public view returns (OracleVersion memory atVersion, OracleReceipt memory atReceipt) {
if (timestamp == 0) return (atVersion, atReceipt);
IOracleProvider provider = _oracles[_global.current].provider;
for (uint256 i = _global.current - 1; i > 0; i--) {
if (timestamp > uint256(_oracles[i].timestamp)) break;
provider = _oracles[i].provider;
}
(atVersion, atReceipt) = provider.at(timestamp);
}
/// @notice Withdraws the accrued oracle fees to the beneficiary
/// @param token The token to withdraw
function withdraw(Token18 token) external onlyBeneficiary {
token.push(beneficiary);
}
/// @notice Claims an amount of incentive tokens, to be paid out as a fee to the keeper
/// @dev Will claim all outstanding oracle fees in the underlying market and leave unrequested fees for the beneficiary.
/// Can only be called by a registered underlying oracle provider factory.
/// @param settlementFeeRequested The fixed settmentment fee requested by the oracle
function claimFee(UFixed6 settlementFeeRequested) external onlySubOracle {
// claim the fee from the market
UFixed6 feeReceived = market.claimFee(address(this));
// return the settlement fee portion to the sub oracle's factory
market.token().push(msg.sender, UFixed18Lib.from(settlementFeeRequested));
emit FeeReceived(settlementFeeRequested, feeReceived.sub(settlementFeeRequested));
}
/// @notice Returns the oracle data for a given epoch
/// @param epoch The epoch to query
/// @return The oracle data for the given epoch
function oracles(uint256 epoch) external view returns (Epoch memory) {
return _oracles[epoch];
}
/// @notice Handles update the oracle to the new provider
/// @param newProvider The new oracle provider
function _updateCurrent(IOracleProvider newProvider) private {
// oracle must not already be updating
if (_global.current != _global.latest) revert OracleOutOfSyncError();
// if the latest version of the underlying oracle is further ahead than its latest request update its timestamp
if (_global.current != 0) {
OracleVersion memory latestVersion = _oracles[_global.current].provider.latest();
if (latestVersion.timestamp > _oracles[_global.current].timestamp)
_oracles[_global.current].timestamp = uint96(latestVersion.timestamp);
}
// add the new oracle registration
_oracles[++_global.current] = Epoch(newProvider, uint96(newProvider.current()));
emit OracleUpdated(newProvider);
}
/// @notice Handles updating the latest oracle to the current if it is ready
/// @param currentOracleLatestVersion The latest version from the current oracle
function _updateLatest(OracleVersion memory currentOracleLatestVersion) private {
if (_latestStale(currentOracleLatestVersion)) _global.latest = _global.current;
}
/// @notice Handles overriding the latest version
/// @dev Applicable if we haven't yet switched over to the current oracle from the latest oracle
/// @param currentOracleLatestVersion The latest version from the current oracle
/// @return latestVersion The latest version
function _handleLatest(
OracleVersion memory currentOracleLatestVersion
) private view returns (OracleVersion memory latestVersion) {
if (_global.current == _global.latest) return currentOracleLatestVersion;
bool isLatestStale = _latestStale(currentOracleLatestVersion);
latestVersion = isLatestStale ? currentOracleLatestVersion : _oracles[_global.latest].provider.latest();
uint256 latestOracleTimestamp =
uint256(isLatestStale ? _oracles[_global.current].timestamp : _oracles[_global.latest].timestamp);
if (!isLatestStale && latestVersion.timestamp > latestOracleTimestamp)
(latestVersion, ) = at(latestOracleTimestamp);
}
/// @notice Returns whether the latest oracle is ready to be updated
/// @param currentOracleLatestVersion The latest version from the current oracle
/// @return Whether the latest oracle is ready to be updated
function _latestStale(OracleVersion memory currentOracleLatestVersion) private view returns (bool) {
if (_global.current == _global.latest) return false;
if (_global.latest == 0) return true;
if (uint256(_oracles[_global.latest].timestamp) > _oracles[_global.latest].provider.latest().timestamp) return false;
if (uint256(_oracles[_global.latest].timestamp) >= currentOracleLatestVersion.timestamp) return false;
return true;
}
/// @dev Only if the caller is the beneficiary
modifier onlyBeneficiary {
if (msg.sender != beneficiary) revert OracleNotBeneficiaryError();
_;
}
/// @dev Only if the caller is the registered market
modifier onlyMarket {
if (msg.sender != address(market)) revert OracleNotMarketError();
_;
}
/// @dev Only if the caller is the registered sub oracle
modifier onlySubOracle {
if (
msg.sender != address(_oracles[_global.current].provider) &&
msg.sender != address(_oracles[_global.latest].provider)
) revert OracleNotSubOracleError();
_;
}
}