-
Notifications
You must be signed in to change notification settings - Fork 30
/
Copy pathUniswapV2Twap.sol
160 lines (144 loc) · 6.47 KB
/
UniswapV2Twap.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
// SPDX-License-Identifier: MIT
pragma solidity >= 0.4 < 0.9;
import {IUniswapV2Pair} from
"../../../src/interfaces/uniswap-v2/IUniswapV2Pair.sol";
import {FixedPoint} from "../../../src/uniswap-v2/FixedPoint.sol";
// Modified from https://github.com/Uniswap/v2-periphery/blob/master/contracts/examples/ExampleOracleSimple.sol
// Do not use this contract in production
error InsufficientTimeElapsed();
error InvalidToken();
contract UniswapV2Twap {
using FixedPoint for *;
// Minimum wait time in seconds before the function update can be called again
// TWAP of time > MIN_WAIT
uint256 private constant MIN_WAIT = 300;
IUniswapV2Pair public immutable pair;
address public immutable token0;
address public immutable token1;
// Cumulative prices are uq112x112 price * seconds
uint256 public price0CumulativeLast;
uint256 public price1CumulativeLast;
// Last timestamp the cumulative prices were updated
uint32 public updatedAt;
// TWAP of token0 and token1
// range: [0, 2**112 - 1]
// resolution: 1 / 2**112
// TWAP of token0 in terms of token1
FixedPoint.uq112x112 public price0Avg;
// TWAP of token1 in terms of token0
FixedPoint.uq112x112 public price1Avg;
// Exercise 1
constructor(address _pair) {
// 1. Set pair contract from constructor input
pair = IUniswapV2Pair(_pair);
// 2. Set token0 and token1 from pair contract
token0 = pair.token0();
token1 = pair.token1();
// 3. Store price0CumulativeLast and price1CumulativeLast from pair contract
price0CumulativeLast = pair.price0CumulativeLast();
price1CumulativeLast = pair.price1CumulativeLast();
// 4. Call pair.getReserve to get last timestamp the reserves were updated
// and store it into the state variable updatedAt
(,, updatedAt) = pair.getReserves();
}
// Exercise 2
// Calculates cumulative prices up to current timestamp
function _getCurrentCumulativePrices()
internal
view
returns (uint256 price0Cumulative, uint256 price1Cumulative)
{
// 1. Get latest cumulative prices from the pair contract
price0Cumulative = pair.price0CumulativeLast();
price1Cumulative = pair.price1CumulativeLast();
// If current block timestamp > last timestamp reserves were updated,
// calculate cumulative prices until current time.
// Otherwise return latest cumulative prices retrieved from the pair contract.
// 2. Get reserves and last timestamp the reserves were updated from
// the pair contract
(uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast) =
pair.getReserves();
// 3. Cast block.timestamp to uint32
uint32 blockTimestamp = uint32(block.timestamp);
if (blockTimestampLast != blockTimestamp) {
// 4. Calculate elapsed time
uint32 dt = blockTimestamp - blockTimestampLast;
// Addition overflow is desired
unchecked {
// 5. Add spot price * elapsed time to cumulative prices.
// - Use FixedPoint.fraction to calculate spot price.
// - FixedPoint.fraction returns UQ112x112, so cast it into uint256.
// - Multiply spot price by time elapsed
price0Cumulative +=
uint256(FixedPoint.fraction(reserve1, reserve0)._x) * dt;
price1Cumulative +=
uint256(FixedPoint.fraction(reserve0, reserve1)._x) * dt;
}
}
}
// Exercise 3
// Updates cumulative prices
function update() external {
// 1. Cast block.timestamp to uint32
uint32 blockTimestamp = uint32(block.timestamp);
// 2. Calculate elapsed time since last time cumulative prices were
// updated in this contract
uint32 dt = blockTimestamp - updatedAt;
// 3. Require time elapsed >= MIN_WAIT
if (dt < MIN_WAIT) {
revert InsufficientTimeElapsed();
}
// 4. Call the internal function _getCurrentCumulativePrices to get
// current cumulative prices
(uint256 price0Cumulative, uint256 price1Cumulative) =
_getCurrentCumulativePrices();
// Overflow is desired, casting never truncates
// https://docs.uniswap.org/contracts/v2/guides/smart-contract-integration/building-an-oracle
// Subtracting between two cumulative price values will result in
// a number that fits within the range of uint256 as long as the
// observations are made for periods of max 2^32 seconds, or ~136 years
unchecked {
// 5. Calculate TWAP price0Avg and price1Avg
// - TWAP = (current cumulative price - last cumulative price) / dt
// - Cast TWAP into uint224 and then into FixedPoint.uq112x112
price0Avg = FixedPoint.uq112x112(
uint224((price0Cumulative - price0CumulativeLast) / dt)
);
price1Avg = FixedPoint.uq112x112(
uint224((price1Cumulative - price1CumulativeLast) / dt)
);
}
// 6. Update state variables price0CumulativeLast, price1CumulativeLast and updatedAt
price0CumulativeLast = price0Cumulative;
price1CumulativeLast = price1Cumulative;
updatedAt = blockTimestamp;
}
// Exercise 4
// Returns the amount out corresponding to the amount in for a given token
function consult(address tokenIn, uint256 amountIn)
external
view
returns (uint256 amountOut)
{
// 1. Require tokenIn is either token0 or token1
if (tokenIn != token0 && tokenIn != token1) {
revert InvalidToken();
}
// 2. Calculate amountOut
// - amountOut = TWAP of tokenIn * amountIn
// - Use FixePoint.mul to multiply TWAP of tokenIn with amountIn
// - FixedPoint.mul returns uq144x112, use FixedPoint.decode144 to return uint144
if (tokenIn == token0) {
// Example
// token0 = WETH
// token1 = USDC
// price0Avg = avg price of WETH in terms of USDC = 2000 USDC / 1 WETH
// tokenIn = WETH
// amountIn = 2
// amountOut = price0Avg * amountIn = 4000 USDC
amountOut = FixedPoint.mul(price0Avg, amountIn).decode144();
} else {
amountOut = FixedPoint.mul(price1Avg, amountIn).decode144();
}
}
}