use self::{
book::l1::BinanceOrderBookL1, channel::BinanceChannel, market::BinanceMarket,
subscription::BinanceSubResponse, trade::BinanceTrade,
};
use crate::{
ExchangeWsStream, NoInitialSnapshots,
exchange::{Connector, ExchangeServer, ExchangeSub, StreamSelector},
instrument::InstrumentData,
subscriber::{WebSocketSubscriber, validator::WebSocketSubValidator},
subscription::{Map, book::OrderBooksL1, trade::PublicTrades},
transformer::stateless::StatelessTransformer,
};
use barter_instrument::exchange::ExchangeId;
use barter_integration::protocol::websocket::{WebSocketSerdeParser, WsMessage};
use std::{fmt::Debug, marker::PhantomData};
use url::Url;
pub mod book;
pub mod channel;
pub mod futures;
pub mod market;
pub mod spot;
pub mod subscription;
pub mod trade;
pub type BinanceWsStream<Transformer> = ExchangeWsStream<WebSocketSerdeParser, Transformer>;
#[derive(Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash, Debug, Default)]
pub struct Binance<Server> {
server: PhantomData<Server>,
}
impl<Server> Connector for Binance<Server>
where
Server: ExchangeServer,
{
const ID: ExchangeId = Server::ID;
type Channel = BinanceChannel;
type Market = BinanceMarket;
type Subscriber = WebSocketSubscriber;
type SubValidator = WebSocketSubValidator;
type SubResponse = BinanceSubResponse;
fn url() -> Result<Url, url::ParseError> {
Url::parse(Server::websocket_url())
}
fn requests(exchange_subs: Vec<ExchangeSub<Self::Channel, Self::Market>>) -> Vec<WsMessage> {
let stream_names = exchange_subs
.into_iter()
.map(|sub| {
format!(
"{}{}",
sub.market.as_ref().to_lowercase(),
sub.channel.as_ref()
)
})
.collect::<Vec<String>>();
vec![WsMessage::text(
serde_json::json!({
"method": "SUBSCRIBE",
"params": stream_names,
"id": 1
})
.to_string(),
)]
}
fn expected_responses<InstrumentKey>(_: &Map<InstrumentKey>) -> usize {
1
}
}
impl<Instrument, Server> StreamSelector<Instrument, PublicTrades> for Binance<Server>
where
Instrument: InstrumentData,
Server: ExchangeServer + Debug + Send + Sync,
{
type SnapFetcher = NoInitialSnapshots;
type Stream =
BinanceWsStream<StatelessTransformer<Self, Instrument::Key, PublicTrades, BinanceTrade>>;
}
impl<Instrument, Server> StreamSelector<Instrument, OrderBooksL1> for Binance<Server>
where
Instrument: InstrumentData,
Server: ExchangeServer + Debug + Send + Sync,
{
type SnapFetcher = NoInitialSnapshots;
type Stream = BinanceWsStream<
StatelessTransformer<Self, Instrument::Key, OrderBooksL1, BinanceOrderBookL1>,
>;
}
impl<'de, Server> serde::Deserialize<'de> for Binance<Server>
where
Server: ExchangeServer,
{
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::de::Deserializer<'de>,
{
let input = <String as serde::Deserialize>::deserialize(deserializer)?;
if input.as_str() == Self::ID.as_str() {
Ok(Self::default())
} else {
Err(serde::de::Error::invalid_value(
serde::de::Unexpected::Str(input.as_str()),
&Self::ID.as_str(),
))
}
}
}
impl<Server> serde::Serialize for Binance<Server>
where
Server: ExchangeServer,
{
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::ser::Serializer,
{
serializer.serialize_str(Self::ID.as_str())
}
}