From f33a835b5aeb5a3dbdd840b7f0732cc6d76b1e2c Mon Sep 17 00:00:00 2001 From: Jacob Date: Wed, 25 Sep 2024 21:58:55 +0900 Subject: [PATCH 01/14] feat: adds cmc api --- core/lib/external_price_api/src/cmc_api.rs | 531 +++++++++++++++++++++ core/lib/external_price_api/src/lib.rs | 1 + 2 files changed, 532 insertions(+) create mode 100644 core/lib/external_price_api/src/cmc_api.rs diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs new file mode 100644 index 000000000000..99b09e78c154 --- /dev/null +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -0,0 +1,531 @@ +use std::{collections::HashMap, str::FromStr}; + +use async_trait::async_trait; +use chrono::Utc; +use serde::Deserialize; +use tokio::sync::RwLock; +use url::Url; +use zksync_config::configs::ExternalPriceApiClientConfig; +use zksync_types::{base_token_ratio::BaseTokenAPIRatio, Address}; + +use crate::{utils::get_fraction, PriceAPIClient}; + +const CMC_AUTH_HEADER: &str = "x-cmc_pro_api_key"; +const DEFAULT_CMC_API_URL: &str = "https://pro-api.coinmarketcap.com"; +// it's safe to have id hardcoded as they are stable as claimed by CMC +// const CMC_ETH_ID: i32 = 1027; +const SUPPORT_TOKENS_ONLY_ON_CMC_PLATFORM_ID: i32 = 1; // 1 = Ethereum + +#[derive(Debug)] +pub struct CMCPriceAPIClient { + base_url: Url, + client: reqwest::Client, + cache_token_id_by_address: RwLock>, +} + +impl CMCPriceAPIClient { + pub fn new(config: ExternalPriceApiClientConfig) -> Self { + let client = if let Some(api_key) = &config.api_key { + use reqwest::header::{HeaderMap, HeaderName, HeaderValue}; + + let mut headers = HeaderMap::new(); + headers.insert( + HeaderName::from_static(CMC_AUTH_HEADER), + HeaderValue::from_str(api_key).expect("Failed to create header value"), + ); + + reqwest::Client::builder().default_headers(headers) + } else { + reqwest::Client::builder() + } + .timeout(config.client_timeout()) + .build() + .expect("Failed to build reqwest client"); + + let base_url = config.base_url.unwrap_or(DEFAULT_CMC_API_URL.to_string()); + + Self { + base_url: Url::parse(&base_url).expect("Failed to parse CoinMarketCap API URL"), + client, + cache_token_id_by_address: RwLock::new(HashMap::new()), + } + } + + fn get(&self, path: &str) -> reqwest::RequestBuilder { + self.client + .get(self.base_url.join(path).expect("Failed to join URL path")) + } + + async fn get_token_id(&self, address: Address) -> anyhow::Result { + if let Some(x) = self.cache_token_id_by_address.read().await.get(&address) { + return Ok(*x); + } + // drop read lock + + let response = self.get("/v1/cryptocurrency/map").send().await?; + let status = response.status(); + if !status.is_success() { + return Err(anyhow::anyhow!( + "Http error while fetching token id. Status: {status}, token: {address}, msg: {}", + response.text().await.unwrap_or_default(), + )); + } + + let parsed = response.json::().await?; + for token_info in parsed.data { + if let Some(platform) = token_info.platform { + if platform.id == SUPPORT_TOKENS_ONLY_ON_CMC_PLATFORM_ID + && Address::from_str(&platform.token_address).is_ok_and(|a| a == address) + { + self.cache_token_id_by_address + .write() + .await + .insert(address, token_info.id); + return Ok(token_info.id); + } + } + } + + Err(anyhow::anyhow!("Token ID not found for address {address}")) + } + + async fn get_token_price_by_address(&self, address: Address) -> anyhow::Result { + let id = self.get_token_id(address).await?; + self.get_token_price_by_id(id).await + } + + async fn get_token_price_by_id(&self, id: i32) -> anyhow::Result { + let response = self + .get("/v2/cryptocurrency/quotes/latest") + .query(&[("id", id)]) + .send() + .await?; + + let status = response.status(); + if !status.is_success() { + return Err(anyhow::anyhow!( + "Http error while fetching token price. Status: {status}, token: {id}, msg: {}", + response.text().await.unwrap_or("".to_string()) + )); + } + + response + .json::() + .await? + .data + .get(&id) + .and_then(|data| data.quote.get("USD")) + .map(|mq| mq.price) + .ok_or_else(|| anyhow::anyhow!("Price not found for token: {id}")) + } +} + +#[derive(Debug, Deserialize)] +struct V2CryptocurrencyQuotesLatestResponse { + data: HashMap, +} + +#[derive(Debug, Deserialize)] +struct CryptocurrencyQuoteObject { + // #[serde(flatten)] + // cryptocurrency_object: CryptocurrencyObject, + quote: HashMap, +} + +#[derive(Debug, Deserialize)] +struct MarketQuote { + price: f64, + // last_updated: chrono::DateTime, // TODO: Recency? +} + +#[derive(Debug, Deserialize)] +struct V1CryptocurrencyMapResponse { + data: Vec, +} + +#[derive(Debug, Deserialize)] +struct CryptocurrencyObject { + id: i32, + // name: String, + // symbol: String, + // slug: String, + // is_active: u8, // TODO: This field is available, should we at least emit a warning if the listing is not marked as active? + platform: Option, +} + +#[derive(Debug, Deserialize)] +struct CryptocurrencyPlatform { + id: i32, + // name: String, + token_address: String, +} + +#[async_trait] +impl PriceAPIClient for CMCPriceAPIClient { + async fn fetch_ratio(&self, token_address: Address) -> anyhow::Result { + let base_token_in_eth = self.get_token_price_by_address(token_address).await?; + let (numerator, denominator) = get_fraction(base_token_in_eth); + + return Ok(BaseTokenAPIRatio { + numerator, + denominator, + ratio_timestamp: Utc::now(), // TODO: Should this be now (as written), or should it be the time returned by the API? + }); + } +} + +#[cfg(test)] +mod irl_tests { + use super::*; + + #[tokio::test] + #[ignore = "run manually; specify CoinMarketCap API key in env var CMC_API_KEY"] + async fn test() { + let client = CMCPriceAPIClient::new(ExternalPriceApiClientConfig { + api_key: Some(std::env::var("CMC_API_KEY").unwrap()), + base_url: None, + client_timeout_ms: 5000, + source: String::new(), + forced: None, + }); + + let tether: Address = "0xdac17f958d2ee523a2206206994597c13d831ec7" + .parse() + .unwrap(); + + let r = client.get_token_price_by_address(tether).await.unwrap(); + + assert!((r - 1f64).abs() < 0.001, "USDT lost its peg"); + + println!("{r}"); + } +} + +/* +#[cfg(test)] +mod tests { + use std::{collections::HashMap, str::FromStr}; + + use bigdecimal::BigDecimal; + use httpmock::{Mock, MockServer}; + use zksync_types::{base_token_price::BaseTokenAPIPrice, Address}; + + use crate::{ + address_to_string, + cmc_api::{CMCPriceAPIClient, CMC_AUTH_HEADER, CMC_ETH_ID}, + tests::tests::{ + add_mock, base_token_price_not_found_test, eth_price_not_found_test, happy_day_test, + no_base_token_price_404_test, no_eth_price_404_test, server_url, + }, + PriceAPIClient, + }; + + const TEST_API_KEY: &str = "test"; + + fn mock_crypto_map<'a>( + server: &'a MockServer, + address: &'a Address, + mock_id: &'a String, + ) -> Mock<'a> { + let address_str = address_to_string(address); + let body = format!( + r#"{{ + "data": [ + {{ + "id": 9999, + "platform": {{ + "name": "Ethereum2", + "token_address": "{}" + }} + }}, + {{ + "id": {}, + "platform": {{ + "name": "Ethereum", + "token_address": "{}" + }} + }} + ] + }}"#, + address_str, mock_id, address_str + ); + add_mock( + server, + httpmock::Method::GET, + "/v1/cryptocurrency/map".to_string(), + HashMap::new(), + 200, + body, + CMC_AUTH_HEADER.to_string(), + Some(TEST_API_KEY.to_string()), + ) + } + + fn add_mock_by_id<'a>( + server: &'a MockServer, + id: &'a String, + price: &'a String, + currency: &'a String, + ) -> Mock<'a> { + let body = format!( + r#"{{ + "data": {{ + "{}": {{ + "quote": {{ + "{}": {{ + "price": {} + }} + }} + }} + }} + }}"#, + id, currency, price + ); + let mut params = HashMap::new(); + params.insert("id".to_string(), id.clone()); + add_mock( + server, + httpmock::Method::GET, + "/v1/cryptocurrency/quotes/latest".to_string(), + params, + 200, + body, + CMC_AUTH_HEADER.to_string(), + Some(TEST_API_KEY.to_string()), + ) + } + + fn happy_day_setup( + server: &MockServer, + api_key: Option, + address: Address, + base_token_price: f64, + eth_price: f64, + ) -> Box { + let id = "50".to_string(); + let currency = "USD".to_string(); + mock_crypto_map(server, &address, &id); + add_mock_by_id(server, &id, &base_token_price.to_string(), ¤cy); + add_mock_by_id( + server, + &CMC_ETH_ID.to_string(), + ð_price.to_string(), + ¤cy, + ); + Box::new(CMCPriceAPIClient::new( + server_url(&server), + api_key.unwrap(), + reqwest::Client::new(), + )) + } + + #[tokio::test] + async fn test_happy_day() { + happy_day_test(Some(TEST_API_KEY.to_string()), happy_day_setup).await + } + + #[tokio::test] + async fn test_no_token_id() { + let server = MockServer::start(); + let address_str = "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984"; + let address = Address::from_str(address_str).unwrap(); + let id = "50".to_string(); + let base_token_price = 198.9; + let eth_price = 3000.0; + let currency = "USD".to_string(); + + // the response will be missing the token that we are seeking for + mock_crypto_map( + &server, + &Address::from_str("0x3Bad7800d9149B53Cba5da927E6449e4A3487a1F").unwrap(), + &"123".to_string(), + ); + add_mock_by_id(&server, &id, &base_token_price.to_string(), ¤cy); + add_mock_by_id( + &server, + &CMC_ETH_ID.to_string(), + ð_price.to_string(), + ¤cy, + ); + + let mut client = CMCPriceAPIClient::new( + server_url(&server), + TEST_API_KEY.to_string(), + reqwest::Client::new(), + ); + let api_price = client.fetch_price(address).await; + + assert!(api_price.is_err()); + let msg = api_price.err().unwrap().to_string(); + assert_eq!( + "Token id not found for address 0x1f9840a85d5af5bf1d1762f925bdaddc4201f984".to_string(), + msg + ) + } + + #[tokio::test] + async fn should_reuse_token_id_from_map() { + let server = MockServer::start(); + let address_str = "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984"; + let address = Address::from_str(address_str).unwrap(); + let base_token_price = 198.9; + let eth_price = 3000.0; + let id = "50".to_string(); + let currency = "USD".to_string(); + + let cm_mock = mock_crypto_map(&server, &address, &id); + add_mock_by_id(&server, &id, &base_token_price.to_string(), ¤cy); + add_mock_by_id( + &server, + &CMC_ETH_ID.to_string(), + ð_price.to_string(), + ¤cy, + ); + let mut client = CMCPriceAPIClient::new( + server_url(&server), + TEST_API_KEY.to_string(), + reqwest::Client::new(), + ); + + client.fetch_price(address).await.unwrap(); + let api_price = client.fetch_price(address).await.unwrap(); + + assert_eq!( + BaseTokenAPIPrice { + base_token_price: BigDecimal::from_str(&base_token_price.to_string()).unwrap(), + eth_price: BigDecimal::from_str(ð_price.to_string()).unwrap(), + ratio_timestamp: api_price.ratio_timestamp, + }, + api_price + ); + // crypto map should be fetched only once + assert_eq!(1, cm_mock.hits()); + } + + #[tokio::test] + async fn test_no_eth_price_404() { + no_eth_price_404_test( + Some(TEST_API_KEY.to_string()), + |server: &MockServer, + api_key: Option, + address: Address, + _base_token_price: f64, + _eth_price: f64| + -> Box { + let id = "50".to_string(); + mock_crypto_map(&server, &address, &id); + add_mock_by_id(&server, &id, &"123".to_string(), &"USD".to_string()); + + Box::new(CMCPriceAPIClient::new( + server_url(&server), + api_key.unwrap(), + reqwest::Client::new(), + )) + }, + ) + .await; + } + + #[tokio::test] + async fn test_eth_price_not_found() { + eth_price_not_found_test( + Some(TEST_API_KEY.to_string()), + |server: &MockServer, + api_key: Option, + address: Address, + _base_token_price: f64, + _eth_price: f64| + -> Box { + let id = "50".to_string(); + mock_crypto_map(&server, &address, &id); + add_mock_by_id(&server, &id, &"123".to_string(), &"USD".to_string()); + let mut params = HashMap::new(); + params.insert("id".to_string(), CMC_ETH_ID.to_string()); + add_mock( + server, + httpmock::Method::GET, + "/v1/cryptocurrency/quotes/latest".to_string(), + params, + 200, + "{}".to_string(), + CMC_AUTH_HEADER.to_string(), + Some(TEST_API_KEY.to_string()), + ); + Box::new(CMCPriceAPIClient::new( + server_url(&server), + api_key.unwrap(), + reqwest::Client::new(), + )) + }, + ) + .await; + } + + #[tokio::test] + async fn test_no_base_token_price_404() { + no_base_token_price_404_test( + Some(TEST_API_KEY.to_string()), + |server: &MockServer, + api_key: Option, + address: Address, + _base_token_price: f64, + _eth_price: f64| + -> Box { + mock_crypto_map(&server, &address, &"55".to_string()); + add_mock_by_id( + &server, + &CMC_ETH_ID.to_string(), + &"3900.12".to_string(), + &"USD".to_string(), + ); + Box::new(CMCPriceAPIClient::new( + server_url(&server), + api_key.unwrap(), + reqwest::Client::new(), + )) + }, + ) + .await; + } + + #[tokio::test] + async fn test_base_token_price_not_found() { + base_token_price_not_found_test( + Some(TEST_API_KEY.to_string()), + |server: &MockServer, + api_key: Option, + address: Address, + _base_token_price: f64, + _eth_price: f64| + -> Box { + let id = "55".to_string(); + mock_crypto_map(&server, &address, &id); + add_mock_by_id( + &server, + &CMC_ETH_ID.to_string(), + &"3900.12".to_string(), + &"USD".to_string(), + ); + let mut params = HashMap::new(); + params.insert("id".to_string(), id); + add_mock( + server, + httpmock::Method::GET, + "/v1/cryptocurrency/quotes/latest".to_string(), + params, + 200, + "{}".to_string(), + CMC_AUTH_HEADER.to_string(), + Some(TEST_API_KEY.to_string()), + ); + Box::new(CMCPriceAPIClient::new( + server_url(&server), + api_key.unwrap(), + reqwest::Client::new(), + )) + }, + ) + .await; + } +} + */ diff --git a/core/lib/external_price_api/src/lib.rs b/core/lib/external_price_api/src/lib.rs index e86279dbe850..f923d65baaad 100644 --- a/core/lib/external_price_api/src/lib.rs +++ b/core/lib/external_price_api/src/lib.rs @@ -1,3 +1,4 @@ +pub mod cmc_api; pub mod coingecko_api; pub mod forced_price_client; mod utils; From bc73e39050dd1e5f004b30796fb470b2d6c26a0b Mon Sep 17 00:00:00 2001 From: Jacob Date: Thu, 26 Sep 2024 22:04:54 +0900 Subject: [PATCH 02/14] chore: small cleanup --- core/lib/external_price_api/src/cmc_api.rs | 354 +-------------------- 1 file changed, 11 insertions(+), 343 deletions(-) diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index 99b09e78c154..2abfe41c550b 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -12,9 +12,7 @@ use crate::{utils::get_fraction, PriceAPIClient}; const CMC_AUTH_HEADER: &str = "x-cmc_pro_api_key"; const DEFAULT_CMC_API_URL: &str = "https://pro-api.coinmarketcap.com"; -// it's safe to have id hardcoded as they are stable as claimed by CMC -// const CMC_ETH_ID: i32 = 1027; -const SUPPORT_TOKENS_ONLY_ON_CMC_PLATFORM_ID: i32 = 1; // 1 = Ethereum +const ALLOW_TOKENS_ONLY_ON_CMC_PLATFORM_ID: i32 = 1; // 1 = Ethereum #[derive(Debug)] pub struct CMCPriceAPIClient { @@ -28,13 +26,12 @@ impl CMCPriceAPIClient { let client = if let Some(api_key) = &config.api_key { use reqwest::header::{HeaderMap, HeaderName, HeaderValue}; - let mut headers = HeaderMap::new(); - headers.insert( + let default_headers = HeaderMap::from_iter([( HeaderName::from_static(CMC_AUTH_HEADER), HeaderValue::from_str(api_key).expect("Failed to create header value"), - ); + )]); - reqwest::Client::builder().default_headers(headers) + reqwest::Client::builder().default_headers(default_headers) } else { reqwest::Client::builder() } @@ -43,11 +40,12 @@ impl CMCPriceAPIClient { .expect("Failed to build reqwest client"); let base_url = config.base_url.unwrap_or(DEFAULT_CMC_API_URL.to_string()); + let base_url = Url::parse(&base_url).expect("Failed to parse CoinMarketCap API URL"); Self { - base_url: Url::parse(&base_url).expect("Failed to parse CoinMarketCap API URL"), + base_url, client, - cache_token_id_by_address: RwLock::new(HashMap::new()), + cache_token_id_by_address: RwLock::default(), } } @@ -74,7 +72,7 @@ impl CMCPriceAPIClient { let parsed = response.json::().await?; for token_info in parsed.data { if let Some(platform) = token_info.platform { - if platform.id == SUPPORT_TOKENS_ONLY_ON_CMC_PLATFORM_ID + if platform.id == ALLOW_TOKENS_ONLY_ON_CMC_PLATFORM_ID && Address::from_str(&platform.token_address).is_ok_and(|a| a == address) { self.cache_token_id_by_address @@ -135,7 +133,6 @@ struct CryptocurrencyQuoteObject { #[derive(Debug, Deserialize)] struct MarketQuote { price: f64, - // last_updated: chrono::DateTime, // TODO: Recency? } #[derive(Debug, Deserialize)] @@ -169,17 +166,17 @@ impl PriceAPIClient for CMCPriceAPIClient { return Ok(BaseTokenAPIRatio { numerator, denominator, - ratio_timestamp: Utc::now(), // TODO: Should this be now (as written), or should it be the time returned by the API? + ratio_timestamp: Utc::now(), }); } } #[cfg(test)] -mod irl_tests { +mod tests { use super::*; #[tokio::test] - #[ignore = "run manually; specify CoinMarketCap API key in env var CMC_API_KEY"] + #[ignore = "run manually (accesses network); specify CoinMarketCap API key in env var CMC_API_KEY"] async fn test() { let client = CMCPriceAPIClient::new(ExternalPriceApiClientConfig { api_key: Some(std::env::var("CMC_API_KEY").unwrap()), @@ -200,332 +197,3 @@ mod irl_tests { println!("{r}"); } } - -/* -#[cfg(test)] -mod tests { - use std::{collections::HashMap, str::FromStr}; - - use bigdecimal::BigDecimal; - use httpmock::{Mock, MockServer}; - use zksync_types::{base_token_price::BaseTokenAPIPrice, Address}; - - use crate::{ - address_to_string, - cmc_api::{CMCPriceAPIClient, CMC_AUTH_HEADER, CMC_ETH_ID}, - tests::tests::{ - add_mock, base_token_price_not_found_test, eth_price_not_found_test, happy_day_test, - no_base_token_price_404_test, no_eth_price_404_test, server_url, - }, - PriceAPIClient, - }; - - const TEST_API_KEY: &str = "test"; - - fn mock_crypto_map<'a>( - server: &'a MockServer, - address: &'a Address, - mock_id: &'a String, - ) -> Mock<'a> { - let address_str = address_to_string(address); - let body = format!( - r#"{{ - "data": [ - {{ - "id": 9999, - "platform": {{ - "name": "Ethereum2", - "token_address": "{}" - }} - }}, - {{ - "id": {}, - "platform": {{ - "name": "Ethereum", - "token_address": "{}" - }} - }} - ] - }}"#, - address_str, mock_id, address_str - ); - add_mock( - server, - httpmock::Method::GET, - "/v1/cryptocurrency/map".to_string(), - HashMap::new(), - 200, - body, - CMC_AUTH_HEADER.to_string(), - Some(TEST_API_KEY.to_string()), - ) - } - - fn add_mock_by_id<'a>( - server: &'a MockServer, - id: &'a String, - price: &'a String, - currency: &'a String, - ) -> Mock<'a> { - let body = format!( - r#"{{ - "data": {{ - "{}": {{ - "quote": {{ - "{}": {{ - "price": {} - }} - }} - }} - }} - }}"#, - id, currency, price - ); - let mut params = HashMap::new(); - params.insert("id".to_string(), id.clone()); - add_mock( - server, - httpmock::Method::GET, - "/v1/cryptocurrency/quotes/latest".to_string(), - params, - 200, - body, - CMC_AUTH_HEADER.to_string(), - Some(TEST_API_KEY.to_string()), - ) - } - - fn happy_day_setup( - server: &MockServer, - api_key: Option, - address: Address, - base_token_price: f64, - eth_price: f64, - ) -> Box { - let id = "50".to_string(); - let currency = "USD".to_string(); - mock_crypto_map(server, &address, &id); - add_mock_by_id(server, &id, &base_token_price.to_string(), ¤cy); - add_mock_by_id( - server, - &CMC_ETH_ID.to_string(), - ð_price.to_string(), - ¤cy, - ); - Box::new(CMCPriceAPIClient::new( - server_url(&server), - api_key.unwrap(), - reqwest::Client::new(), - )) - } - - #[tokio::test] - async fn test_happy_day() { - happy_day_test(Some(TEST_API_KEY.to_string()), happy_day_setup).await - } - - #[tokio::test] - async fn test_no_token_id() { - let server = MockServer::start(); - let address_str = "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984"; - let address = Address::from_str(address_str).unwrap(); - let id = "50".to_string(); - let base_token_price = 198.9; - let eth_price = 3000.0; - let currency = "USD".to_string(); - - // the response will be missing the token that we are seeking for - mock_crypto_map( - &server, - &Address::from_str("0x3Bad7800d9149B53Cba5da927E6449e4A3487a1F").unwrap(), - &"123".to_string(), - ); - add_mock_by_id(&server, &id, &base_token_price.to_string(), ¤cy); - add_mock_by_id( - &server, - &CMC_ETH_ID.to_string(), - ð_price.to_string(), - ¤cy, - ); - - let mut client = CMCPriceAPIClient::new( - server_url(&server), - TEST_API_KEY.to_string(), - reqwest::Client::new(), - ); - let api_price = client.fetch_price(address).await; - - assert!(api_price.is_err()); - let msg = api_price.err().unwrap().to_string(); - assert_eq!( - "Token id not found for address 0x1f9840a85d5af5bf1d1762f925bdaddc4201f984".to_string(), - msg - ) - } - - #[tokio::test] - async fn should_reuse_token_id_from_map() { - let server = MockServer::start(); - let address_str = "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984"; - let address = Address::from_str(address_str).unwrap(); - let base_token_price = 198.9; - let eth_price = 3000.0; - let id = "50".to_string(); - let currency = "USD".to_string(); - - let cm_mock = mock_crypto_map(&server, &address, &id); - add_mock_by_id(&server, &id, &base_token_price.to_string(), ¤cy); - add_mock_by_id( - &server, - &CMC_ETH_ID.to_string(), - ð_price.to_string(), - ¤cy, - ); - let mut client = CMCPriceAPIClient::new( - server_url(&server), - TEST_API_KEY.to_string(), - reqwest::Client::new(), - ); - - client.fetch_price(address).await.unwrap(); - let api_price = client.fetch_price(address).await.unwrap(); - - assert_eq!( - BaseTokenAPIPrice { - base_token_price: BigDecimal::from_str(&base_token_price.to_string()).unwrap(), - eth_price: BigDecimal::from_str(ð_price.to_string()).unwrap(), - ratio_timestamp: api_price.ratio_timestamp, - }, - api_price - ); - // crypto map should be fetched only once - assert_eq!(1, cm_mock.hits()); - } - - #[tokio::test] - async fn test_no_eth_price_404() { - no_eth_price_404_test( - Some(TEST_API_KEY.to_string()), - |server: &MockServer, - api_key: Option, - address: Address, - _base_token_price: f64, - _eth_price: f64| - -> Box { - let id = "50".to_string(); - mock_crypto_map(&server, &address, &id); - add_mock_by_id(&server, &id, &"123".to_string(), &"USD".to_string()); - - Box::new(CMCPriceAPIClient::new( - server_url(&server), - api_key.unwrap(), - reqwest::Client::new(), - )) - }, - ) - .await; - } - - #[tokio::test] - async fn test_eth_price_not_found() { - eth_price_not_found_test( - Some(TEST_API_KEY.to_string()), - |server: &MockServer, - api_key: Option, - address: Address, - _base_token_price: f64, - _eth_price: f64| - -> Box { - let id = "50".to_string(); - mock_crypto_map(&server, &address, &id); - add_mock_by_id(&server, &id, &"123".to_string(), &"USD".to_string()); - let mut params = HashMap::new(); - params.insert("id".to_string(), CMC_ETH_ID.to_string()); - add_mock( - server, - httpmock::Method::GET, - "/v1/cryptocurrency/quotes/latest".to_string(), - params, - 200, - "{}".to_string(), - CMC_AUTH_HEADER.to_string(), - Some(TEST_API_KEY.to_string()), - ); - Box::new(CMCPriceAPIClient::new( - server_url(&server), - api_key.unwrap(), - reqwest::Client::new(), - )) - }, - ) - .await; - } - - #[tokio::test] - async fn test_no_base_token_price_404() { - no_base_token_price_404_test( - Some(TEST_API_KEY.to_string()), - |server: &MockServer, - api_key: Option, - address: Address, - _base_token_price: f64, - _eth_price: f64| - -> Box { - mock_crypto_map(&server, &address, &"55".to_string()); - add_mock_by_id( - &server, - &CMC_ETH_ID.to_string(), - &"3900.12".to_string(), - &"USD".to_string(), - ); - Box::new(CMCPriceAPIClient::new( - server_url(&server), - api_key.unwrap(), - reqwest::Client::new(), - )) - }, - ) - .await; - } - - #[tokio::test] - async fn test_base_token_price_not_found() { - base_token_price_not_found_test( - Some(TEST_API_KEY.to_string()), - |server: &MockServer, - api_key: Option, - address: Address, - _base_token_price: f64, - _eth_price: f64| - -> Box { - let id = "55".to_string(); - mock_crypto_map(&server, &address, &id); - add_mock_by_id( - &server, - &CMC_ETH_ID.to_string(), - &"3900.12".to_string(), - &"USD".to_string(), - ); - let mut params = HashMap::new(); - params.insert("id".to_string(), id); - add_mock( - server, - httpmock::Method::GET, - "/v1/cryptocurrency/quotes/latest".to_string(), - params, - 200, - "{}".to_string(), - CMC_AUTH_HEADER.to_string(), - Some(TEST_API_KEY.to_string()), - ); - Box::new(CMCPriceAPIClient::new( - server_url(&server), - api_key.unwrap(), - reqwest::Client::new(), - )) - }, - ) - .await; - } -} - */ From fb31d1f136acb5e7a1d88584cb66d3bf34ca9a00 Mon Sep 17 00:00:00 2001 From: Jacob Date: Thu, 26 Sep 2024 23:32:14 +0900 Subject: [PATCH 03/14] chore: rename to proper CamelCase --- core/lib/external_price_api/src/cmc_api.rs | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index 2abfe41c550b..4ff605f2fa39 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -15,13 +15,13 @@ const DEFAULT_CMC_API_URL: &str = "https://pro-api.coinmarketcap.com"; const ALLOW_TOKENS_ONLY_ON_CMC_PLATFORM_ID: i32 = 1; // 1 = Ethereum #[derive(Debug)] -pub struct CMCPriceAPIClient { +pub struct CmcPriceApiClient { base_url: Url, client: reqwest::Client, cache_token_id_by_address: RwLock>, } -impl CMCPriceAPIClient { +impl CmcPriceApiClient { pub fn new(config: ExternalPriceApiClientConfig) -> Self { let client = if let Some(api_key) = &config.api_key { use reqwest::header::{HeaderMap, HeaderName, HeaderValue}; @@ -158,7 +158,7 @@ struct CryptocurrencyPlatform { } #[async_trait] -impl PriceAPIClient for CMCPriceAPIClient { +impl PriceAPIClient for CmcPriceApiClient { async fn fetch_ratio(&self, token_address: Address) -> anyhow::Result { let base_token_in_eth = self.get_token_price_by_address(token_address).await?; let (numerator, denominator) = get_fraction(base_token_in_eth); @@ -178,7 +178,7 @@ mod tests { #[tokio::test] #[ignore = "run manually (accesses network); specify CoinMarketCap API key in env var CMC_API_KEY"] async fn test() { - let client = CMCPriceAPIClient::new(ExternalPriceApiClientConfig { + let client = CmcPriceApiClient::new(ExternalPriceApiClientConfig { api_key: Some(std::env::var("CMC_API_KEY").unwrap()), base_url: None, client_timeout_ms: 5000, From 4f4dfb8715910c78b3c050b5f7736140ba4cabc5 Mon Sep 17 00:00:00 2001 From: Jacob Date: Thu, 26 Sep 2024 23:32:34 +0900 Subject: [PATCH 04/14] feat: adds cmc to node builder --- core/bin/zksync_server/src/node_builder.rs | 5 +- .../layers/base_token/cmc_client.rs | 55 +++++++++++++++++++ .../implementations/layers/base_token/mod.rs | 1 + 3 files changed, 60 insertions(+), 1 deletion(-) create mode 100644 core/node/node_framework/src/implementations/layers/base_token/cmc_client.rs diff --git a/core/bin/zksync_server/src/node_builder.rs b/core/bin/zksync_server/src/node_builder.rs index 14db83b9f25a..86d6fb9c3b5f 100644 --- a/core/bin/zksync_server/src/node_builder.rs +++ b/core/bin/zksync_server/src/node_builder.rs @@ -19,7 +19,7 @@ use zksync_node_framework::{ implementations::layers::{ base_token::{ base_token_ratio_persister::BaseTokenRatioPersisterLayer, - base_token_ratio_provider::BaseTokenRatioProviderLayer, + base_token_ratio_provider::BaseTokenRatioProviderLayer, cmc_client::CmcClientLayer, coingecko_client::CoingeckoClientLayer, forced_price_client::ForcedPriceClientLayer, no_op_external_price_api_client::NoOpExternalPriceApiClientLayer, }, @@ -555,6 +555,9 @@ impl MainNodeBuilder { fn add_external_api_client_layer(mut self) -> anyhow::Result { let config = try_load_config!(self.configs.external_price_api_client_config); match config.source.as_str() { + CmcClientLayer::CLIENT_NAME => { + self.node.add_layer(CmcClientLayer::new(config)); + } CoingeckoClientLayer::CLIENT_NAME => { self.node.add_layer(CoingeckoClientLayer::new(config)); } diff --git a/core/node/node_framework/src/implementations/layers/base_token/cmc_client.rs b/core/node/node_framework/src/implementations/layers/base_token/cmc_client.rs new file mode 100644 index 000000000000..b1a2305f1d32 --- /dev/null +++ b/core/node/node_framework/src/implementations/layers/base_token/cmc_client.rs @@ -0,0 +1,55 @@ +use std::sync::Arc; + +use zksync_config::configs::ExternalPriceApiClientConfig; +use zksync_external_price_api::cmc_api::CmcPriceApiClient; + +use crate::{ + implementations::resources::price_api_client::PriceAPIClientResource, + wiring_layer::{WiringError, WiringLayer}, + IntoContext, +}; + +/// Wiring layer for `CmcPriceApiClient`. +/// +/// Responsible for inserting a resource with a client to get base token prices +/// from CoinMarketCap to be used by the `BaseTokenRatioPersister`. +#[derive(Debug)] +pub struct CmcClientLayer { + config: ExternalPriceApiClientConfig, +} + +impl CmcClientLayer { + /// Identifier of used client type. + /// Can be used to choose the layer for the client based on configuration variables. + pub const CLIENT_NAME: &'static str = "coinmarketcap"; +} + +#[derive(Debug, IntoContext)] +#[context(crate = crate)] +pub struct Output { + pub price_api_client: PriceAPIClientResource, +} + +impl CmcClientLayer { + pub fn new(config: ExternalPriceApiClientConfig) -> Self { + Self { config } + } +} + +#[async_trait::async_trait] +impl WiringLayer for CmcClientLayer { + type Input = (); + type Output = Output; + + fn layer_name(&self) -> &'static str { + "coinmarketcap_api_client" + } + + async fn wire(self, _input: Self::Input) -> Result { + let client = Arc::new(CmcPriceApiClient::new(self.config)); + + Ok(Output { + price_api_client: client.into(), + }) + } +} diff --git a/core/node/node_framework/src/implementations/layers/base_token/mod.rs b/core/node/node_framework/src/implementations/layers/base_token/mod.rs index 5b58527a3d82..1e58f8d84452 100644 --- a/core/node/node_framework/src/implementations/layers/base_token/mod.rs +++ b/core/node/node_framework/src/implementations/layers/base_token/mod.rs @@ -1,5 +1,6 @@ pub mod base_token_ratio_persister; pub mod base_token_ratio_provider; +pub mod cmc_client; pub mod coingecko_client; pub mod forced_price_client; pub mod no_op_external_price_api_client; From 8bb5c7ad655d9b030eff6eff79adee901584bb9c Mon Sep 17 00:00:00 2001 From: Jacob Date: Fri, 27 Sep 2024 00:36:24 +0900 Subject: [PATCH 05/14] chore: trying enums instead of inline string matching --- core/bin/zksync_server/src/node_builder.rs | 27 +---- .../layers/base_token/cmc_client.rs | 55 ---------- .../layers/base_token/coingecko_client.rs | 55 ---------- .../layers/base_token/forced_price_client.rs | 52 --------- .../implementations/layers/base_token/mod.rs | 103 +++++++++++++++++- .../no_op_external_price_api_client.rs | 45 -------- 6 files changed, 102 insertions(+), 235 deletions(-) delete mode 100644 core/node/node_framework/src/implementations/layers/base_token/cmc_client.rs delete mode 100644 core/node/node_framework/src/implementations/layers/base_token/coingecko_client.rs delete mode 100644 core/node/node_framework/src/implementations/layers/base_token/forced_price_client.rs delete mode 100644 core/node/node_framework/src/implementations/layers/base_token/no_op_external_price_api_client.rs diff --git a/core/bin/zksync_server/src/node_builder.rs b/core/bin/zksync_server/src/node_builder.rs index 86d6fb9c3b5f..87a2b4076465 100644 --- a/core/bin/zksync_server/src/node_builder.rs +++ b/core/bin/zksync_server/src/node_builder.rs @@ -19,9 +19,7 @@ use zksync_node_framework::{ implementations::layers::{ base_token::{ base_token_ratio_persister::BaseTokenRatioPersisterLayer, - base_token_ratio_provider::BaseTokenRatioProviderLayer, cmc_client::CmcClientLayer, - coingecko_client::CoingeckoClientLayer, forced_price_client::ForcedPriceClientLayer, - no_op_external_price_api_client::NoOpExternalPriceApiClientLayer, + base_token_ratio_provider::BaseTokenRatioProviderLayer, ExternalPriceApiLayer, }, circuit_breaker_checker::CircuitBreakerCheckerLayer, commitment_generator::CommitmentGeneratorLayer, @@ -554,27 +552,8 @@ impl MainNodeBuilder { fn add_external_api_client_layer(mut self) -> anyhow::Result { let config = try_load_config!(self.configs.external_price_api_client_config); - match config.source.as_str() { - CmcClientLayer::CLIENT_NAME => { - self.node.add_layer(CmcClientLayer::new(config)); - } - CoingeckoClientLayer::CLIENT_NAME => { - self.node.add_layer(CoingeckoClientLayer::new(config)); - } - NoOpExternalPriceApiClientLayer::CLIENT_NAME => { - self.node.add_layer(NoOpExternalPriceApiClientLayer); - } - ForcedPriceClientLayer::CLIENT_NAME => { - self.node.add_layer(ForcedPriceClientLayer::new(config)); - } - _ => { - anyhow::bail!( - "Unknown external price API client source: {}", - config.source - ); - } - } - + self.node + .add_layer(ExternalPriceApiLayer::try_from(config)?); Ok(self) } diff --git a/core/node/node_framework/src/implementations/layers/base_token/cmc_client.rs b/core/node/node_framework/src/implementations/layers/base_token/cmc_client.rs deleted file mode 100644 index b1a2305f1d32..000000000000 --- a/core/node/node_framework/src/implementations/layers/base_token/cmc_client.rs +++ /dev/null @@ -1,55 +0,0 @@ -use std::sync::Arc; - -use zksync_config::configs::ExternalPriceApiClientConfig; -use zksync_external_price_api::cmc_api::CmcPriceApiClient; - -use crate::{ - implementations::resources::price_api_client::PriceAPIClientResource, - wiring_layer::{WiringError, WiringLayer}, - IntoContext, -}; - -/// Wiring layer for `CmcPriceApiClient`. -/// -/// Responsible for inserting a resource with a client to get base token prices -/// from CoinMarketCap to be used by the `BaseTokenRatioPersister`. -#[derive(Debug)] -pub struct CmcClientLayer { - config: ExternalPriceApiClientConfig, -} - -impl CmcClientLayer { - /// Identifier of used client type. - /// Can be used to choose the layer for the client based on configuration variables. - pub const CLIENT_NAME: &'static str = "coinmarketcap"; -} - -#[derive(Debug, IntoContext)] -#[context(crate = crate)] -pub struct Output { - pub price_api_client: PriceAPIClientResource, -} - -impl CmcClientLayer { - pub fn new(config: ExternalPriceApiClientConfig) -> Self { - Self { config } - } -} - -#[async_trait::async_trait] -impl WiringLayer for CmcClientLayer { - type Input = (); - type Output = Output; - - fn layer_name(&self) -> &'static str { - "coinmarketcap_api_client" - } - - async fn wire(self, _input: Self::Input) -> Result { - let client = Arc::new(CmcPriceApiClient::new(self.config)); - - Ok(Output { - price_api_client: client.into(), - }) - } -} diff --git a/core/node/node_framework/src/implementations/layers/base_token/coingecko_client.rs b/core/node/node_framework/src/implementations/layers/base_token/coingecko_client.rs deleted file mode 100644 index 14ab568c2f3a..000000000000 --- a/core/node/node_framework/src/implementations/layers/base_token/coingecko_client.rs +++ /dev/null @@ -1,55 +0,0 @@ -use std::sync::Arc; - -use zksync_config::configs::ExternalPriceApiClientConfig; -use zksync_external_price_api::coingecko_api::CoinGeckoPriceAPIClient; - -use crate::{ - implementations::resources::price_api_client::PriceAPIClientResource, - wiring_layer::{WiringError, WiringLayer}, - IntoContext, -}; - -/// Wiring layer for `CoingeckoApiClient` -/// -/// Responsible for inserting a resource with a client to get base token prices from CoinGecko to be -/// used by the `BaseTokenRatioPersister`. -#[derive(Debug)] -pub struct CoingeckoClientLayer { - config: ExternalPriceApiClientConfig, -} - -impl CoingeckoClientLayer { - /// Identifier of used client type. - /// Can be used to choose the layer for the client based on configuration variables. - pub const CLIENT_NAME: &'static str = "coingecko"; -} - -#[derive(Debug, IntoContext)] -#[context(crate = crate)] -pub struct Output { - pub price_api_client: PriceAPIClientResource, -} - -impl CoingeckoClientLayer { - pub fn new(config: ExternalPriceApiClientConfig) -> Self { - Self { config } - } -} - -#[async_trait::async_trait] -impl WiringLayer for CoingeckoClientLayer { - type Input = (); - type Output = Output; - - fn layer_name(&self) -> &'static str { - "coingecko_api_client" - } - - async fn wire(self, _input: Self::Input) -> Result { - let cg_client = Arc::new(CoinGeckoPriceAPIClient::new(self.config)); - - Ok(Output { - price_api_client: cg_client.into(), - }) - } -} diff --git a/core/node/node_framework/src/implementations/layers/base_token/forced_price_client.rs b/core/node/node_framework/src/implementations/layers/base_token/forced_price_client.rs deleted file mode 100644 index 67785dc26ed4..000000000000 --- a/core/node/node_framework/src/implementations/layers/base_token/forced_price_client.rs +++ /dev/null @@ -1,52 +0,0 @@ -use std::sync::Arc; - -use zksync_config::configs::ExternalPriceApiClientConfig; -use zksync_external_price_api::forced_price_client::ForcedPriceClient; - -use crate::{ - implementations::resources::price_api_client::PriceAPIClientResource, - wiring_layer::{WiringError, WiringLayer}, - IntoContext, -}; - -/// Wiring layer for `ForcedPriceClient` -/// -/// Inserts a resource with a forced configured price to be used by the `BaseTokenRatioPersister`. -#[derive(Debug)] -pub struct ForcedPriceClientLayer { - config: ExternalPriceApiClientConfig, -} - -impl ForcedPriceClientLayer { - pub fn new(config: ExternalPriceApiClientConfig) -> Self { - Self { config } - } - - /// Identifier of used client type. - /// Can be used to choose the layer for the client based on configuration variables. - pub const CLIENT_NAME: &'static str = "forced"; -} - -#[derive(Debug, IntoContext)] -#[context(crate = crate)] -pub struct Output { - pub price_api_client: PriceAPIClientResource, -} - -#[async_trait::async_trait] -impl WiringLayer for ForcedPriceClientLayer { - type Input = (); - type Output = Output; - - fn layer_name(&self) -> &'static str { - "forced_price_client" - } - - async fn wire(self, _input: Self::Input) -> Result { - let forced_client = Arc::new(ForcedPriceClient::new(self.config)); - - Ok(Output { - price_api_client: forced_client.into(), - }) - } -} diff --git a/core/node/node_framework/src/implementations/layers/base_token/mod.rs b/core/node/node_framework/src/implementations/layers/base_token/mod.rs index 1e58f8d84452..f2212c03fde1 100644 --- a/core/node/node_framework/src/implementations/layers/base_token/mod.rs +++ b/core/node/node_framework/src/implementations/layers/base_token/mod.rs @@ -1,6 +1,101 @@ +use std::{str::FromStr, sync::Arc}; + +use zksync_config::configs::ExternalPriceApiClientConfig; +use zksync_external_price_api::{ + cmc_api::CmcPriceApiClient, coingecko_api::CoinGeckoPriceAPIClient, + forced_price_client::ForcedPriceClient, NoOpPriceAPIClient, +}; + +use crate::{ + implementations::resources::price_api_client::PriceAPIClientResource, IntoContext, WiringError, + WiringLayer, +}; + pub mod base_token_ratio_persister; pub mod base_token_ratio_provider; -pub mod cmc_client; -pub mod coingecko_client; -pub mod forced_price_client; -pub mod no_op_external_price_api_client; + +#[derive(Debug, Clone, Copy, Hash, PartialEq, Eq, PartialOrd, Ord, Default)] +enum ExternalPriceApiKind { + #[default] + NoOp, + Forced, + CoinGecko, + CoinMarketCap, +} + +impl ExternalPriceApiKind { + fn layer_name(&self) -> &'static str { + match self { + Self::NoOp => "no_op_external_price_api_client", + Self::Forced => "forced_price_client", + Self::CoinGecko => "coingecko_api_client", + Self::CoinMarketCap => "coinmarketcap_api_client", + } + } + + fn instantiate(&self, config: ExternalPriceApiClientConfig) -> PriceAPIClientResource { + PriceAPIClientResource(match self { + Self::NoOp => Arc::new(NoOpPriceAPIClient {}), + Self::Forced => Arc::new(ForcedPriceClient::new(config)), + Self::CoinGecko => Arc::new(CoinGeckoPriceAPIClient::new(config)), + Self::CoinMarketCap => Arc::new(CmcPriceApiClient::new(config)), + }) + } +} + +#[derive(Debug, thiserror::Error)] +#[error("Unknown external price API client source: {0}")] +pub struct UnknownExternalPriceApiClientSourceError(String); + +impl FromStr for ExternalPriceApiKind { + type Err = UnknownExternalPriceApiClientSourceError; + + fn from_str(s: &str) -> Result { + Ok(match &s.to_lowercase()[..] { + "no-op" => Self::NoOp, + "forced" => Self::Forced, + "coingecko" => Self::CoinGecko, + "coinmarketcap" => Self::CoinMarketCap, + _ => return Err(UnknownExternalPriceApiClientSourceError(s.to_owned())), + }) + } +} + +#[derive(Debug)] +pub struct ExternalPriceApiLayer { + kind: ExternalPriceApiKind, + config: ExternalPriceApiClientConfig, +} + +impl TryFrom for ExternalPriceApiLayer { + type Error = UnknownExternalPriceApiClientSourceError; + + fn try_from(config: ExternalPriceApiClientConfig) -> Result { + Ok(Self { + kind: config.source.parse()?, + config, + }) + } +} + +#[derive(Debug, IntoContext)] +#[context(crate = crate)] +pub struct Output { + pub price_api_client: PriceAPIClientResource, +} + +#[async_trait::async_trait] +impl WiringLayer for ExternalPriceApiLayer { + type Input = (); + type Output = Output; + + fn layer_name(&self) -> &'static str { + self.kind.layer_name() + } + + async fn wire(self, _input: Self::Input) -> Result { + Ok(Output { + price_api_client: self.kind.instantiate(self.config).into(), + }) + } +} diff --git a/core/node/node_framework/src/implementations/layers/base_token/no_op_external_price_api_client.rs b/core/node/node_framework/src/implementations/layers/base_token/no_op_external_price_api_client.rs deleted file mode 100644 index 2bf5eda798fa..000000000000 --- a/core/node/node_framework/src/implementations/layers/base_token/no_op_external_price_api_client.rs +++ /dev/null @@ -1,45 +0,0 @@ -use std::sync::Arc; - -use zksync_external_price_api::NoOpPriceAPIClient; - -use crate::{ - implementations::resources::price_api_client::PriceAPIClientResource, - wiring_layer::{WiringError, WiringLayer}, - IntoContext, -}; - -/// Wiring layer for `NoOpExternalPriceApiClient` -/// -/// Inserts a resource with a no-op client to get base token prices to be used by the `BaseTokenRatioPersister`. -#[derive(Debug)] -pub struct NoOpExternalPriceApiClientLayer; - -impl NoOpExternalPriceApiClientLayer { - /// Identifier of used client type. - /// Can be used to choose the layer for the client based on configuration variables. - pub const CLIENT_NAME: &'static str = "no-op"; -} - -#[derive(Debug, IntoContext)] -#[context(crate = crate)] -pub struct Output { - pub price_api_client: PriceAPIClientResource, -} - -#[async_trait::async_trait] -impl WiringLayer for NoOpExternalPriceApiClientLayer { - type Input = (); - type Output = Output; - - fn layer_name(&self) -> &'static str { - "no_op_external_price_api_client" - } - - async fn wire(self, _input: Self::Input) -> Result { - let no_op_client = Arc::new(NoOpPriceAPIClient {}); - - Ok(Output { - price_api_client: no_op_client.into(), - }) - } -} From a6b02ded61dee9cfbc57a059071d677c98e1cafb Mon Sep 17 00:00:00 2001 From: Jacob Date: Fri, 27 Sep 2024 15:43:29 +0900 Subject: [PATCH 06/14] chore: remove dynamic layer name --- .../implementations/layers/base_token/mod.rs | 37 +++++++------------ 1 file changed, 14 insertions(+), 23 deletions(-) diff --git a/core/node/node_framework/src/implementations/layers/base_token/mod.rs b/core/node/node_framework/src/implementations/layers/base_token/mod.rs index f2212c03fde1..1146d9e9f005 100644 --- a/core/node/node_framework/src/implementations/layers/base_token/mod.rs +++ b/core/node/node_framework/src/implementations/layers/base_token/mod.rs @@ -23,28 +23,8 @@ enum ExternalPriceApiKind { CoinMarketCap, } -impl ExternalPriceApiKind { - fn layer_name(&self) -> &'static str { - match self { - Self::NoOp => "no_op_external_price_api_client", - Self::Forced => "forced_price_client", - Self::CoinGecko => "coingecko_api_client", - Self::CoinMarketCap => "coinmarketcap_api_client", - } - } - - fn instantiate(&self, config: ExternalPriceApiClientConfig) -> PriceAPIClientResource { - PriceAPIClientResource(match self { - Self::NoOp => Arc::new(NoOpPriceAPIClient {}), - Self::Forced => Arc::new(ForcedPriceClient::new(config)), - Self::CoinGecko => Arc::new(CoinGeckoPriceAPIClient::new(config)), - Self::CoinMarketCap => Arc::new(CmcPriceApiClient::new(config)), - }) - } -} - #[derive(Debug, thiserror::Error)] -#[error("Unknown external price API client source: {0}")] +#[error("Unknown external price API client source: \"{0}\"")] pub struct UnknownExternalPriceApiClientSourceError(String); impl FromStr for ExternalPriceApiKind { @@ -52,7 +32,7 @@ impl FromStr for ExternalPriceApiKind { fn from_str(s: &str) -> Result { Ok(match &s.to_lowercase()[..] { - "no-op" => Self::NoOp, + "no-op" | "noop" => Self::NoOp, "forced" => Self::Forced, "coingecko" => Self::CoinGecko, "coinmarketcap" => Self::CoinMarketCap, @@ -61,6 +41,17 @@ impl FromStr for ExternalPriceApiKind { } } +impl ExternalPriceApiKind { + fn instantiate(&self, config: ExternalPriceApiClientConfig) -> PriceAPIClientResource { + PriceAPIClientResource(match self { + Self::NoOp => Arc::new(NoOpPriceAPIClient {}), + Self::Forced => Arc::new(ForcedPriceClient::new(config)), + Self::CoinGecko => Arc::new(CoinGeckoPriceAPIClient::new(config)), + Self::CoinMarketCap => Arc::new(CmcPriceApiClient::new(config)), + }) + } +} + #[derive(Debug)] pub struct ExternalPriceApiLayer { kind: ExternalPriceApiKind, @@ -90,7 +81,7 @@ impl WiringLayer for ExternalPriceApiLayer { type Output = Output; fn layer_name(&self) -> &'static str { - self.kind.layer_name() + "external_price_api" } async fn wire(self, _input: Self::Input) -> Result { From 1042be07b1d14fea6e3acadf5c8b8564d18e0b22 Mon Sep 17 00:00:00 2001 From: Jacob Date: Fri, 27 Sep 2024 15:49:52 +0900 Subject: [PATCH 07/14] fix: removes superfluous .into() call --- .../node_framework/src/implementations/layers/base_token/mod.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/node/node_framework/src/implementations/layers/base_token/mod.rs b/core/node/node_framework/src/implementations/layers/base_token/mod.rs index 1146d9e9f005..7a63b573d78e 100644 --- a/core/node/node_framework/src/implementations/layers/base_token/mod.rs +++ b/core/node/node_framework/src/implementations/layers/base_token/mod.rs @@ -86,7 +86,7 @@ impl WiringLayer for ExternalPriceApiLayer { async fn wire(self, _input: Self::Input) -> Result { Ok(Output { - price_api_client: self.kind.instantiate(self.config).into(), + price_api_client: self.kind.instantiate(self.config), }) } } From 7c5a72392db5ba29ed0fa950f6c226fca4891ecf Mon Sep 17 00:00:00 2001 From: Jacob Date: Mon, 30 Sep 2024 13:52:06 +0900 Subject: [PATCH 08/14] fix: tidy up names --- core/lib/external_price_api/src/cmc_api.rs | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index 4ff605f2fa39..f06d20acc913 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -10,9 +10,9 @@ use zksync_types::{base_token_ratio::BaseTokenAPIRatio, Address}; use crate::{utils::get_fraction, PriceAPIClient}; -const CMC_AUTH_HEADER: &str = "x-cmc_pro_api_key"; -const DEFAULT_CMC_API_URL: &str = "https://pro-api.coinmarketcap.com"; -const ALLOW_TOKENS_ONLY_ON_CMC_PLATFORM_ID: i32 = 1; // 1 = Ethereum +const AUTH_HEADER: &str = "x-cmc_pro_api_key"; +const DEFAULT_API_URL: &str = "https://pro-api.coinmarketcap.com"; +const ALLOW_TOKENS_ONLY_ON_PLATFORM_ID: i32 = 1; // 1 = Ethereum #[derive(Debug)] pub struct CmcPriceApiClient { @@ -27,7 +27,7 @@ impl CmcPriceApiClient { use reqwest::header::{HeaderMap, HeaderName, HeaderValue}; let default_headers = HeaderMap::from_iter([( - HeaderName::from_static(CMC_AUTH_HEADER), + HeaderName::from_static(AUTH_HEADER), HeaderValue::from_str(api_key).expect("Failed to create header value"), )]); @@ -39,7 +39,7 @@ impl CmcPriceApiClient { .build() .expect("Failed to build reqwest client"); - let base_url = config.base_url.unwrap_or(DEFAULT_CMC_API_URL.to_string()); + let base_url = config.base_url.unwrap_or(DEFAULT_API_URL.to_string()); let base_url = Url::parse(&base_url).expect("Failed to parse CoinMarketCap API URL"); Self { @@ -72,7 +72,7 @@ impl CmcPriceApiClient { let parsed = response.json::().await?; for token_info in parsed.data { if let Some(platform) = token_info.platform { - if platform.id == ALLOW_TOKENS_ONLY_ON_CMC_PLATFORM_ID + if platform.id == ALLOW_TOKENS_ONLY_ON_PLATFORM_ID && Address::from_str(&platform.token_address).is_ok_and(|a| a == address) { self.cache_token_id_by_address From 03be5e76175d366fcd909c451b1001c84b4bee41 Mon Sep 17 00:00:00 2001 From: Jacob Date: Mon, 30 Sep 2024 14:30:37 +0900 Subject: [PATCH 09/14] feat: warn for inactive token listing on cmc --- Cargo.lock | 1 + core/lib/external_price_api/Cargo.toml | 1 + core/lib/external_price_api/src/cmc_api.rs | 28 +++++++++++++--------- 3 files changed, 19 insertions(+), 11 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 37e4569cbef6..060810d129e1 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -9923,6 +9923,7 @@ dependencies = [ "reqwest 0.12.5", "serde", "tokio", + "tracing", "url", "zksync_config", "zksync_types", diff --git a/core/lib/external_price_api/Cargo.toml b/core/lib/external_price_api/Cargo.toml index 9539aa3fdc3c..26a17f1a629c 100644 --- a/core/lib/external_price_api/Cargo.toml +++ b/core/lib/external_price_api/Cargo.toml @@ -20,6 +20,7 @@ serde.workspace = true reqwest = { workspace = true, features = ["json"] } fraction.workspace = true rand.workspace = true +tracing.workspace = true zksync_config.workspace = true zksync_types.workspace = true diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index f06d20acc913..e805b6e90a20 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -8,7 +8,7 @@ use url::Url; use zksync_config::configs::ExternalPriceApiClientConfig; use zksync_types::{base_token_ratio::BaseTokenAPIRatio, Address}; -use crate::{utils::get_fraction, PriceAPIClient}; +use crate::{address_to_string, utils::get_fraction, PriceAPIClient}; const AUTH_HEADER: &str = "x-cmc_pro_api_key"; const DEFAULT_API_URL: &str = "https://pro-api.coinmarketcap.com"; @@ -75,6 +75,16 @@ impl CmcPriceApiClient { if platform.id == ALLOW_TOKENS_ONLY_ON_PLATFORM_ID && Address::from_str(&platform.token_address).is_ok_and(|a| a == address) { + if token_info.is_active != 1 { + tracing::warn!( + "CoinMarketCap API reports token {} ({}) on platform {} ({}) is not active", + address_to_string(&address), + token_info.name, + platform.id, + platform.name, + ); + } + self.cache_token_id_by_address .write() .await @@ -103,7 +113,7 @@ impl CmcPriceApiClient { if !status.is_success() { return Err(anyhow::anyhow!( "Http error while fetching token price. Status: {status}, token: {id}, msg: {}", - response.text().await.unwrap_or("".to_string()) + response.text().await.unwrap_or_default(), )); } @@ -125,8 +135,6 @@ struct V2CryptocurrencyQuotesLatestResponse { #[derive(Debug, Deserialize)] struct CryptocurrencyQuoteObject { - // #[serde(flatten)] - // cryptocurrency_object: CryptocurrencyObject, quote: HashMap, } @@ -143,17 +151,15 @@ struct V1CryptocurrencyMapResponse { #[derive(Debug, Deserialize)] struct CryptocurrencyObject { id: i32, - // name: String, - // symbol: String, - // slug: String, - // is_active: u8, // TODO: This field is available, should we at least emit a warning if the listing is not marked as active? + name: String, + is_active: u8, platform: Option, } #[derive(Debug, Deserialize)] struct CryptocurrencyPlatform { id: i32, - // name: String, + name: String, token_address: String, } @@ -177,12 +183,12 @@ mod tests { #[tokio::test] #[ignore = "run manually (accesses network); specify CoinMarketCap API key in env var CMC_API_KEY"] - async fn test() { + async fn cmc_tether_peg() { let client = CmcPriceApiClient::new(ExternalPriceApiClientConfig { api_key: Some(std::env::var("CMC_API_KEY").unwrap()), base_url: None, client_timeout_ms: 5000, - source: String::new(), + source: "coinmarketcap".to_string(), forced: None, }); From be01905ddb8af89646fe498d89a2def05ecec736 Mon Sep 17 00:00:00 2001 From: Jacob Date: Mon, 30 Sep 2024 14:46:01 +0900 Subject: [PATCH 10/14] fix: report prices in terms of eth --- core/lib/external_price_api/src/cmc_api.rs | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index e805b6e90a20..a1bc20321966 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -13,6 +13,7 @@ use crate::{address_to_string, utils::get_fraction, PriceAPIClient}; const AUTH_HEADER: &str = "x-cmc_pro_api_key"; const DEFAULT_API_URL: &str = "https://pro-api.coinmarketcap.com"; const ALLOW_TOKENS_ONLY_ON_PLATFORM_ID: i32 = 1; // 1 = Ethereum +const REQUEST_QUOTE_IN_CURRENCY_ID: &str = "1027"; // 1027 = ETH #[derive(Debug)] pub struct CmcPriceApiClient { @@ -106,6 +107,7 @@ impl CmcPriceApiClient { let response = self .get("/v2/cryptocurrency/quotes/latest") .query(&[("id", id)]) + .query(&[("convert_id", REQUEST_QUOTE_IN_CURRENCY_ID)]) .send() .await?; @@ -122,7 +124,7 @@ impl CmcPriceApiClient { .await? .data .get(&id) - .and_then(|data| data.quote.get("USD")) + .and_then(|data| data.quote.get(REQUEST_QUOTE_IN_CURRENCY_ID)) .map(|mq| mq.price) .ok_or_else(|| anyhow::anyhow!("Price not found for token: {id}")) } @@ -167,11 +169,11 @@ struct CryptocurrencyPlatform { impl PriceAPIClient for CmcPriceApiClient { async fn fetch_ratio(&self, token_address: Address) -> anyhow::Result { let base_token_in_eth = self.get_token_price_by_address(token_address).await?; - let (numerator, denominator) = get_fraction(base_token_in_eth); + let (term_ether, term_base_token) = get_fraction(base_token_in_eth); return Ok(BaseTokenAPIRatio { - numerator, - denominator, + numerator: term_base_token, + denominator: term_ether, ratio_timestamp: Utc::now(), }); } From 6a74c120755fa5801e0a16e0e6c7806e4e1a42d7 Mon Sep 17 00:00:00 2001 From: Jacob Date: Thu, 3 Oct 2024 00:16:25 +0900 Subject: [PATCH 11/14] chore: adds initial testing --- Cargo.lock | 1 + core/lib/external_price_api/Cargo.toml | 3 + core/lib/external_price_api/src/cmc_api.rs | 125 ++++++++++++++++++++- core/lib/external_price_api/src/tests.rs | 6 +- 4 files changed, 132 insertions(+), 3 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 5fc0ef3f2c13..445cd951dd22 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -10245,6 +10245,7 @@ dependencies = [ "rand 0.8.5", "reqwest 0.12.7", "serde", + "serde_json", "tokio", "tracing", "url", diff --git a/core/lib/external_price_api/Cargo.toml b/core/lib/external_price_api/Cargo.toml index f52e7e98dfdb..1e849f600060 100644 --- a/core/lib/external_price_api/Cargo.toml +++ b/core/lib/external_price_api/Cargo.toml @@ -25,4 +25,7 @@ tracing.workspace = true zksync_config.workspace = true zksync_types.workspace = true tokio.workspace = true + +[dev-dependencies] httpmock.workspace = true +serde_json.workspace = true diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index a1bc20321966..36a95173dd1a 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -169,7 +169,7 @@ struct CryptocurrencyPlatform { impl PriceAPIClient for CmcPriceApiClient { async fn fetch_ratio(&self, token_address: Address) -> anyhow::Result { let base_token_in_eth = self.get_token_price_by_address(token_address).await?; - let (term_ether, term_base_token) = get_fraction(base_token_in_eth); + let (term_ether, term_base_token) = get_fraction(base_token_in_eth)?; return Ok(BaseTokenAPIRatio { numerator: term_base_token, @@ -181,11 +181,132 @@ impl PriceAPIClient for CmcPriceApiClient { #[cfg(test)] mod tests { + use httpmock::prelude::*; + use serde_json::json; + + use crate::tests::*; + use super::*; + #[tokio::test] + async fn mock_happy() { + let server = make_mock_server(); + let client: &dyn PriceAPIClient = &CmcPriceApiClient::new(ExternalPriceApiClientConfig { + source: "coinmarketcap".to_string(), + base_url: Some(server.base_url()), + api_key: Some("00000000-0000-0000-0000-000000000000".to_string()), + client_timeout_ms: 5000, + forced: None, + }); + + let token_address = "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984" + .parse() + .unwrap(); + + let api_price = client.fetch_ratio(token_address).await.unwrap(); + + const REPORTED_PRICE: f64 = 1_f64 / 0.0028306661720164175_f64; + const EPSILON: f64 = 0.000001_f64 * REPORTED_PRICE; + + assert!((approximate_value(&api_price) - REPORTED_PRICE).abs() < EPSILON); + } + + fn make_mock_server() -> MockServer { + let mock_server = MockServer::start(); + // cryptocurrency map + mock_server.mock(|when, then| { + when.method(GET).path("/v1/cryptocurrency/map"); + then.status(200) + .header("content-type", "application/json") + .json_body(json!({ + "status": { + "timestamp": "2024-09-25T11:29:38.440Z", + "error_code": 0, + "error_message": null, + "elapsed": 351, + "credit_count": 1, + "notice": null + }, + "data": [ + { + "id": 7083, + "rank": 26, + "name": "Uniswap", + "symbol": "UNI", + "slug": "uniswap", + "is_active": 1, + "first_historical_data": "2020-09-17T01:10:00.000Z", + "last_historical_data": "2024-09-25T11:25:00.000Z", + "platform": { + "id": 1, + "name": "Ethereum", + "symbol": "ETH", + "slug": "ethereum", + "token_address": "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984" + } + } + ] + })); + }); + + // cryptocurrency quote + mock_server.mock(|when, then| { + // TODO: check for api authentication header + when.method(GET) + .path("/v2/cryptocurrency/quotes/latest") + .query_param("id", "7083") // Uniswap + .query_param("convert_id", "1027"); // Ether + then.status(200) + .header("content-type", "application/json") + .json_body(json!({ + "status": { + "timestamp": "2024-10-02T14:15:07.189Z", + "error_code": 0, + "error_message": null, + "elapsed": 39, + "credit_count": 1, + "notice": null + }, + "data": { + "7083": { + "id": 7083, + "name": "Uniswap", + "symbol": "UNI", + "slug": "uniswap", + "date_added": "2020-09-17T00:00:00.000Z", + "tags": [], + "max_supply": null, + "circulating_supply": 600294743.71, + "total_supply": 1000000000, + "platform": { + "id": 1027, + "name": "Ethereum", + "symbol": "ETH", + "slug": "ethereum", + "token_address": "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984" + }, + "is_active": 1, + "infinite_supply": false, + "cmc_rank": 22, + "is_fiat": 0, + "last_updated": "2024-10-02T14:13:00.000Z", + "quote": { + "1027": { + "price": 0.0028306661720164175, + "last_updated": "2024-10-02T14:12:00.000Z" + } + } + } + } + })); + }); + + mock_server + } + #[tokio::test] #[ignore = "run manually (accesses network); specify CoinMarketCap API key in env var CMC_API_KEY"] - async fn cmc_tether_peg() { + async fn real_cmc_tether_peg() { let client = CmcPriceApiClient::new(ExternalPriceApiClientConfig { api_key: Some(std::env::var("CMC_API_KEY").unwrap()), base_url: None, diff --git a/core/lib/external_price_api/src/tests.rs b/core/lib/external_price_api/src/tests.rs index bb2af866cf5f..a50a406acf2f 100644 --- a/core/lib/external_price_api/src/tests.rs +++ b/core/lib/external_price_api/src/tests.rs @@ -2,7 +2,7 @@ use std::str::FromStr; use chrono::Utc; use httpmock::MockServer; -use zksync_types::Address; +use zksync_types::{base_token_ratio::BaseTokenAPIRatio, Address}; use crate::PriceAPIClient; @@ -16,6 +16,10 @@ const TEST_TOKEN_PRICE_ETH: f64 = 0.00269; const TEST_BASE_PRICE: f64 = 371.74; const PRICE_FLOAT_COMPARE_TOLERANCE: f64 = 0.1; +pub(crate) fn approximate_value(api_price: &BaseTokenAPIRatio) -> f64 { + api_price.numerator.get() as f64 / api_price.denominator.get() as f64 +} + pub(crate) struct SetupResult { pub(crate) client: Box, } From 9a3be4cb2b39424ac61cc50f1eb63b3db420de47 Mon Sep 17 00:00:00 2001 From: Jacob Date: Fri, 4 Oct 2024 12:13:53 +0900 Subject: [PATCH 12/14] chore: test failure cases --- core/lib/external_price_api/src/cmc_api.rs | 73 +++++++++++++++------- core/lib/external_price_api/src/tests.rs | 2 +- 2 files changed, 53 insertions(+), 22 deletions(-) diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index 36a95173dd1a..c2d266729fbf 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -188,34 +188,23 @@ mod tests { use super::*; - #[tokio::test] - async fn mock_happy() { - let server = make_mock_server(); - let client: &dyn PriceAPIClient = &CmcPriceApiClient::new(ExternalPriceApiClientConfig { + fn make_client(server: &MockServer, api_key: Option) -> Box { + Box::new(CmcPriceApiClient::new(ExternalPriceApiClientConfig { source: "coinmarketcap".to_string(), base_url: Some(server.base_url()), - api_key: Some("00000000-0000-0000-0000-000000000000".to_string()), + api_key, client_timeout_ms: 5000, forced: None, - }); - - let token_address = "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984" - .parse() - .unwrap(); - - let api_price = client.fetch_ratio(token_address).await.unwrap(); - - const REPORTED_PRICE: f64 = 1_f64 / 0.0028306661720164175_f64; - const EPSILON: f64 = 0.000001_f64 * REPORTED_PRICE; - - assert!((approximate_value(&api_price) - REPORTED_PRICE).abs() < EPSILON); + })) } fn make_mock_server() -> MockServer { let mock_server = MockServer::start(); // cryptocurrency map mock_server.mock(|when, then| { - when.method(GET).path("/v1/cryptocurrency/map"); + when.method(GET) + .header_exists(AUTH_HEADER) + .path("/v1/cryptocurrency/map"); then.status(200) .header("content-type", "application/json") .json_body(json!({ @@ -253,6 +242,7 @@ mod tests { mock_server.mock(|when, then| { // TODO: check for api authentication header when.method(GET) + .header_exists(AUTH_HEADER) .path("/v2/cryptocurrency/quotes/latest") .query_param("id", "7083") // Uniswap .query_param("convert_id", "1027"); // Ether @@ -304,9 +294,52 @@ mod tests { mock_server } + #[tokio::test] + async fn mock_happy() { + let server = make_mock_server(); + let client = make_client( + &server, + Some("00000000-0000-0000-0000-000000000000".to_string()), + ); + + let token_address: Address = TEST_TOKEN_ADDRESS.parse().unwrap(); + + let api_price = client.fetch_ratio(token_address).await.unwrap(); + + const REPORTED_PRICE: f64 = 1_f64 / 0.0028306661720164175_f64; + const EPSILON: f64 = 0.000001_f64 * REPORTED_PRICE; + + assert!((approximate_value(&api_price) - REPORTED_PRICE).abs() < EPSILON); + } + + #[tokio::test] + #[should_panic = "Request did not match any route or mock"] + async fn mock_fail_no_api_key() { + let server = make_mock_server(); + let client = make_client(&server, None); + + let token_address: Address = TEST_TOKEN_ADDRESS.parse().unwrap(); + + client.fetch_ratio(token_address).await.unwrap(); + } + + #[tokio::test] + #[should_panic = "Token ID not found for address"] + async fn mock_fail_not_found() { + let server = make_mock_server(); + let client = make_client( + &server, + Some("00000000-0000-0000-0000-000000000000".to_string()), + ); + + let token_address: Address = Address::random(); + + client.fetch_ratio(token_address).await.unwrap(); + } + #[tokio::test] #[ignore = "run manually (accesses network); specify CoinMarketCap API key in env var CMC_API_KEY"] - async fn real_cmc_tether_peg() { + async fn real_cmc_tether() { let client = CmcPriceApiClient::new(ExternalPriceApiClientConfig { api_key: Some(std::env::var("CMC_API_KEY").unwrap()), base_url: None, @@ -321,8 +354,6 @@ mod tests { let r = client.get_token_price_by_address(tether).await.unwrap(); - assert!((r - 1f64).abs() < 0.001, "USDT lost its peg"); - println!("{r}"); } } diff --git a/core/lib/external_price_api/src/tests.rs b/core/lib/external_price_api/src/tests.rs index a50a406acf2f..fd6a8b9928f1 100644 --- a/core/lib/external_price_api/src/tests.rs +++ b/core/lib/external_price_api/src/tests.rs @@ -8,7 +8,7 @@ use crate::PriceAPIClient; const TIME_TOLERANCE_MS: i64 = 100; /// Uniswap (UNI) -const TEST_TOKEN_ADDRESS: &str = "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984"; +pub const TEST_TOKEN_ADDRESS: &str = "0x1f9840a85d5af5bf1d1762f925bdaddc4201f984"; /// 1UNI = 0.00269ETH const TEST_TOKEN_PRICE_ETH: f64 = 0.00269; /// 1ETH = 371.74UNI; When converting gas price from ETH to UNI From 51e83ad02aacc153c7370d25d63a48ae3a98ce40 Mon Sep 17 00:00:00 2001 From: Jacob Date: Fri, 4 Oct 2024 12:15:04 +0900 Subject: [PATCH 13/14] chore: fmt --- core/lib/external_price_api/src/cmc_api.rs | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index c2d266729fbf..07c26e043e44 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -184,9 +184,8 @@ mod tests { use httpmock::prelude::*; use serde_json::json; - use crate::tests::*; - use super::*; + use crate::tests::*; fn make_client(server: &MockServer, api_key: Option) -> Box { Box::new(CmcPriceApiClient::new(ExternalPriceApiClientConfig { From 97a01bb87208a935cf1fc778581739469d5d6954 Mon Sep 17 00:00:00 2001 From: Jacob Date: Mon, 14 Oct 2024 18:36:44 +0900 Subject: [PATCH 14/14] chore: remove superfluous comment --- core/lib/external_price_api/src/cmc_api.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/core/lib/external_price_api/src/cmc_api.rs b/core/lib/external_price_api/src/cmc_api.rs index 07c26e043e44..05cb5e4d7285 100644 --- a/core/lib/external_price_api/src/cmc_api.rs +++ b/core/lib/external_price_api/src/cmc_api.rs @@ -59,7 +59,6 @@ impl CmcPriceApiClient { if let Some(x) = self.cache_token_id_by_address.read().await.get(&address) { return Ok(*x); } - // drop read lock let response = self.get("/v1/cryptocurrency/map").send().await?; let status = response.status();