use reqwest::header::HeaderMap;
use reqwest::{self, Client};
use serde::{Deserialize, Serialize};
use sha2::{Digest, Sha256};
use std::error::Error as StdError;
use std::fmt::{self, Display};
#[derive(Deserialize, Debug)]
pub struct CommonErrorResponse {
pub message: Option<String>,
}
#[derive(Deserialize, Debug)]
pub struct TickerResponse {
pub symbol: Option<String>,
pub price: Option<String>,
}
#[derive(Deserialize, Debug)]
pub struct BalanceResponse {
pub equity: Option<String>,
pub balance: Option<String>,
}
#[derive(Serialize)]
struct CreateOrderPayload {
symbol: String,
size: String,
side: String,
price: Option<String>,
}
#[derive(Deserialize, Debug)]
pub struct CreateOrderResponse {
pub price: Option<String>,
pub size: Option<String>,
pub fee: Option<String>,
}
#[derive(Serialize)]
struct CloseAllPositionsPayload {
symbol: Option<String>,
}
#[derive(Deserialize, Debug)]
pub struct CloseAllPositionsResponse {}
#[derive(Clone, Debug)]
pub struct DexClient {
client: Client,
base_url: String,
}
#[derive(Debug)]
pub enum DexError {
Serde(serde_json::Error),
Reqwest(reqwest::Error),
ServerResponse(String),
}
impl Display for DexError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match *self {
DexError::Serde(ref e) => write!(f, "Serde JSON error: {}", e),
DexError::Reqwest(ref e) => write!(f, "Reqwest error: {}", e),
DexError::ServerResponse(ref e) => write!(f, "Server response error: {}", e),
}
}
}
impl StdError for DexError {
fn source(&self) -> Option<&(dyn StdError + 'static)> {
match *self {
DexError::Serde(ref e) => Some(e),
DexError::Reqwest(ref e) => Some(e),
DexError::ServerResponse(_) => None,
}
}
}
impl From<serde_json::Error> for DexError {
fn from(err: serde_json::Error) -> DexError {
DexError::Serde(err)
}
}
impl From<reqwest::Error> for DexError {
fn from(err: reqwest::Error) -> DexError {
DexError::Reqwest(err)
}
}
impl DexClient {
pub async fn new(api_key: String, base_url: String) -> Result<Self, reqwest::Error> {
let client = Client::builder()
.default_headers(Self::headers_with_hashed_api_key(api_key))
.build()?;
Ok(DexClient { client, base_url })
}
fn headers_with_hashed_api_key(api_key: String) -> HeaderMap {
let mut hasher = Sha256::new();
hasher.update(api_key);
let hashed_api_key = format!("{:x}", hasher.finalize());
let mut headers = HeaderMap::new();
headers.insert("Authorization", hashed_api_key.parse().unwrap());
headers
}
async fn handle_request<T: serde::de::DeserializeOwned>(
&self,
result: Result<reqwest::Response, reqwest::Error>,
url: &str,
) -> Result<T, DexError> {
let response = result.map_err(DexError::from)?;
let status = response.status();
if status.is_success() {
let headers = response.headers().clone();
let body = response.text().await.map_err(DexError::from)?;
log::debug!("Response body: {}", body);
serde_json::from_str(&body).map_err(|e| {
log::warn!("Response header: {:?}", headers);
log::error!("Failed to deserialize response: {}", e);
DexError::Serde(e)
})
} else {
let error_response: CommonErrorResponse = response
.json()
.await
.unwrap_or(CommonErrorResponse { message: None });
let error_message = format!(
"Server returned error: {}. Requested url: {}, message: {:?}",
status, url, error_response.message,
);
log::error!("{}", &error_message);
Err(DexError::ServerResponse(error_message))
}
}
pub async fn get_ticker(&self, dex: &str, symbol: &str) -> Result<TickerResponse, DexError> {
let url = format!("{}/ticker?dex={}&symbol={}", self.base_url, dex, symbol);
log::trace!("{:?}", url);
self.handle_request(self.client.get(&url).send().await, &url)
.await
}
pub async fn get_balance(&self, dex: &str) -> Result<BalanceResponse, DexError> {
let url = format!("{}/get-balance?dex={}", self.base_url, dex);
log::trace!("{:?}", url);
self.handle_request(self.client.get(&url).send().await, &url)
.await
}
pub async fn create_order(
&self,
dex: &str,
symbol: &str,
size: &str,
side: &str,
price: Option<String>,
) -> Result<CreateOrderResponse, DexError> {
let url = format!("{}/create-order?dex={}", self.base_url, dex);
log::trace!("{:?}", url);
let payload = CreateOrderPayload {
symbol: symbol.to_string(),
size: size.to_string(),
side: side.to_string(),
price,
};
self.handle_request(self.client.post(&url).json(&payload).send().await, &url)
.await
}
pub async fn close_all_positions(
&self,
dex: &str,
symbol: Option<String>,
) -> Result<CloseAllPositionsResponse, DexError> {
let url = format!("{}/close_all_positions?dex={}", self.base_url, dex);
log::trace!("{:?}", url);
let payload = CloseAllPositionsPayload { symbol };
self.handle_request(self.client.post(&url).json(&payload).send().await, &url)
.await
}
}