use crate::{
event::MarketIter,
exchange::{
bybit::{channel::BybitChannel, subscription::BybitResponse, trade::BybitTrade},
ExchangeId,
},
subscription::trade::PublicTrade,
Identifier,
};
use barter_integration::model::{instrument::Instrument, SubscriptionId};
use chrono::{DateTime, Utc};
use serde::{
de::{Error, Unexpected},
Deserialize, Serialize,
};
#[derive(Debug, Serialize, Deserialize)]
#[serde(untagged)]
pub enum BybitMessage {
Response(BybitResponse),
Trade(BybitTrade),
}
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Debug, Deserialize, Serialize)]
pub struct BybitPayload<T> {
#[serde(alias = "topic", deserialize_with = "de_message_subscription_id")]
pub subscription_id: SubscriptionId,
#[serde(rename = "type")]
pub r#type: String,
#[serde(
alias = "ts",
deserialize_with = "barter_integration::de::de_u64_epoch_ms_as_datetime_utc"
)]
pub time: DateTime<Utc>,
pub data: T,
}
pub fn de_message_subscription_id<'de, D>(deserializer: D) -> Result<SubscriptionId, D::Error>
where
D: serde::de::Deserializer<'de>,
{
let input = <&str as serde::Deserialize>::deserialize(deserializer)?;
let mut tokens = input.split('.');
match (tokens.next(), tokens.next(), tokens.next()) {
(Some("publicTrade"), Some(market), None) => Ok(SubscriptionId::from(format!(
"{}|{market}",
BybitChannel::TRADES.0
))),
_ => Err(Error::invalid_value(
Unexpected::Str(input),
&"invalid message type expected pattern: <type>.<symbol>",
)),
}
}
impl Identifier<Option<SubscriptionId>> for BybitMessage {
fn id(&self) -> Option<SubscriptionId> {
match self {
BybitMessage::Trade(trade) => Some(trade.subscription_id.clone()),
_ => None,
}
}
}
impl From<(ExchangeId, Instrument, BybitMessage)> for MarketIter<PublicTrade> {
fn from((exchange_id, instrument, message): (ExchangeId, Instrument, BybitMessage)) -> Self {
match message {
BybitMessage::Response(_) => Self(vec![]),
BybitMessage::Trade(trade) => Self::from((exchange_id, instrument, trade)),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
mod de {
use super::*;
use crate::exchange::bybit::subscription::BybitReturnMessage;
use barter_integration::error::SocketError;
#[test]
fn test_bybit_pong() {
struct TestCase {
input: &'static str,
expected: Result<BybitResponse, SocketError>,
}
let tests = vec![
TestCase {
input: r#"
{
"success": true,
"ret_msg": "pong",
"conn_id": "0970e817-426e-429a-a679-ff7f55e0b16a",
"op": "ping"
}
"#,
expected: Ok(BybitResponse {
success: true,
ret_msg: BybitReturnMessage::Pong,
}),
},
];
for (index, test) in tests.into_iter().enumerate() {
let actual = serde_json::from_str::<BybitResponse>(test.input);
match (actual, test.expected) {
(Ok(actual), Ok(expected)) => {
assert_eq!(actual, expected, "TC{} failed", index)
}
(Err(_), Err(_)) => {
}
(actual, expected) => {
panic!("TC{index} failed because actual != expected. \nActual: {actual:?}\nExpected: {expected:?}\n");
}
}
}
}
}
}