use crypto_market_type::MarketType;
use crate::{MessageType, TradeMsg, TradeSide};
use serde::{Deserialize, Serialize};
use serde_json::{Result, Value};
use std::collections::HashMap;
const EXCHANGE_NAME: &str = "deribit";
#[derive(Serialize, Deserialize)]
struct RawTradeMsg {
trade_seq: i64,
trade_id: String,
timestamp: i64,
price: f64,
instrument_name: String,
direction: String, amount: f64,
#[serde(flatten)]
extra: HashMap<String, Value>,
}
#[derive(Serialize, Deserialize)]
struct Params<T: Sized> {
channel: String,
data: Vec<T>,
}
#[derive(Serialize, Deserialize)]
struct WebsocketMsg<T: Sized> {
jsonrpc: String,
method: String,
params: Params<T>,
}
fn calc_quantity_and_volume(market_type: MarketType, price: f64, amount: f64) -> (f64, f64) {
match market_type {
MarketType::InverseSwap | MarketType::InverseFuture => {
let volume = amount;
(volume / price, volume)
}
MarketType::Option => (amount, amount * price),
_ => panic!("Unknown market_type {}", market_type),
}
}
pub(crate) fn parse_trade(market_type: MarketType, msg: &str) -> Result<Vec<TradeMsg>> {
let ws_msg = serde_json::from_str::<WebsocketMsg<RawTradeMsg>>(msg)?;
let trades: Vec<TradeMsg> = ws_msg
.params
.data
.into_iter()
.map(|raw_trade| {
let (quantity, volume) =
calc_quantity_and_volume(market_type, raw_trade.price, raw_trade.amount);
TradeMsg {
exchange: EXCHANGE_NAME.to_string(),
market_type,
symbol: raw_trade.instrument_name.clone(),
pair: crypto_pair::normalize_pair(&raw_trade.instrument_name, EXCHANGE_NAME)
.unwrap(),
msg_type: MessageType::Trade,
timestamp: raw_trade.timestamp,
price: raw_trade.price,
quantity,
volume,
side: if raw_trade.direction == "sell" {
TradeSide::Sell
} else {
TradeSide::Buy
},
trade_id: raw_trade.trade_id.to_string(),
raw: serde_json::to_value(&raw_trade).unwrap(),
}
})
.collect();
Ok(trades)
}