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
use crate::{client::Web3, jsonrpc::error::Web3Error};
use clarity::{
abi::Token,
constants::{TT160M1, TT24M1},
Address, Uint256,
};
lazy_static! {
pub static ref UNISWAP_QUOTER_ADDRESS: Address =
Address::parse_and_validate("0xb27308f9F90D607463bb33eA1BeBb41C27CE5AB6").unwrap();
pub static ref DAI_CONTRACT_ADDRESS: Address =
Address::parse_and_validate("0x6B175474E89094C44Da98b954EedeAC495271d0F").unwrap();
pub static ref WETH_CONTRACT_ADDRESS: Address =
Address::parse_and_validate("0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2").unwrap();
}
impl Web3 {
#[allow(clippy::too_many_arguments)]
pub async fn get_uniswap_price(
&self,
caller_address: Address,
token_in: Address,
token_out: Address,
fee_uint24: Uint256,
amount: Uint256,
sqrt_price_limit_x96_uint160: Uint256,
uniswap_quoter: Option<Address>,
) -> Result<Uint256, Web3Error> {
let quoter = match uniswap_quoter {
Some(val) => val,
None => *UNISWAP_QUOTER_ADDRESS,
};
if fee_uint24 > *TT24M1 {
return Err(Web3Error::BadInput(
"Bad fee input to swap price - value too large for uint24".to_string(),
));
}
if sqrt_price_limit_x96_uint160 > *TT160M1 {
return Err(Web3Error::BadInput(
"Bad sqrt_price_limit_x96 input to swap price - value too large for uint160"
.to_string(),
));
}
let tokens: &[Token] = &[
Token::Address(token_in),
Token::Address(token_out),
Token::Uint(fee_uint24),
Token::Uint(amount),
Token::Uint(sqrt_price_limit_x96_uint160),
];
debug!("tokens is {:?}", tokens);
let result = self
.contract_call(
quoter,
"quoteExactInputSingle(address,address,uint24,uint256,uint160)",
&tokens,
caller_address,
None,
)
.await?;
debug!("result is {:?}", result);
Ok(Uint256::from_bytes_be(match result.get(0..32) {
Some(val) => val,
None => {
return Err(Web3Error::ContractCallError(
"Bad response from swap price".to_string(),
))
}
}))
}
}
#[test]
fn get_uniswap_price_test() {
use actix::System;
use env_logger::{Builder, Env};
use std::time::Duration;
Builder::from_env(Env::default().default_filter_or("warn")).init();
let runner = System::new();
let web3 = Web3::new("https://eth.althea.net", Duration::from_secs(5));
let caller_address =
Address::parse_and_validate("0x5A0b54D5dc17e0AadC383d2db43B0a0D3E029c4c").unwrap();
let amount = Uint256::from(1_000_000_000_000_000_000u64);
let fee = Uint256::from(500u16);
let sqrt_price_limit_x96_uint160 = Uint256::from(0u16);
runner.block_on(async move {
let price = web3
.get_uniswap_price(
caller_address,
*WETH_CONTRACT_ADDRESS,
*DAI_CONTRACT_ADDRESS,
fee.clone(),
amount.clone(),
sqrt_price_limit_x96_uint160.clone(),
None,
)
.await;
let weth2dai = price.unwrap();
debug!("weth->dai price is {}", weth2dai);
assert!(weth2dai > 0u32.into());
let price = web3
.get_uniswap_price(
caller_address,
*DAI_CONTRACT_ADDRESS,
*WETH_CONTRACT_ADDRESS,
fee.clone(),
weth2dai,
sqrt_price_limit_x96_uint160,
None,
)
.await;
let dai2weth = price.unwrap();
debug!("dai->weth price is {}", &dai2weth);
let amount_float: f64 = (amount.to_string()).parse().unwrap();
let dai2weth_float: f64 = (dai2weth.to_string()).parse().unwrap();
assert!((0.95 * amount_float) < dai2weth_float && dai2weth_float < (1.05 * amount_float));
});
}