use self::{
book::HyperliquidL2Book, channel::HyperliquidChannel, market::HyperliquidMarket,
subscription::HyperliquidSubResponse, trade::HyperliquidTrade,
};
use crate::{
ExchangeWsStream, NoInitialSnapshots,
exchange::{Connector, ExchangeSub, PingInterval, StreamSelector},
instrument::InstrumentData,
subscriber::{WebSocketSubscriber, validator::WebSocketSubValidator},
subscription::{book::OrderBooksL2, trade::PublicTrades},
transformer::stateless::StatelessTransformer,
};
use derive_more::Display;
use rustrade_instrument::exchange::ExchangeId;
use rustrade_integration::protocol::websocket::{WebSocketSerdeParser, WsMessage};
use rustrade_macro::{DeExchange, SerExchange};
use serde_json::json;
use std::time::Duration;
use url::Url;
pub mod book;
pub mod channel;
pub mod historical;
pub mod market;
pub mod spot_meta;
pub mod subscription;
pub mod trade;
pub use spot_meta::{SpotMetaResolver, resolve_spot_pair};
pub const BASE_URL_HYPERLIQUID: &str = "wss://api.hyperliquid.xyz/ws";
const PING_INTERVAL_SECS: u64 = 50;
pub type HyperliquidWsStream<Transformer> = ExchangeWsStream<WebSocketSerdeParser, Transformer>;
fn build_subscribe_messages(
exchange_subs: Vec<ExchangeSub<HyperliquidChannel, HyperliquidMarket>>,
) -> Vec<WsMessage> {
exchange_subs
.into_iter()
.map(|ExchangeSub { channel, market }| {
WsMessage::text(
json!({
"method": "subscribe",
"subscription": channel.subscription_payload(market.as_ref())
})
.to_string(),
)
})
.collect()
}
#[derive(
Copy,
Clone,
Eq,
PartialEq,
Ord,
PartialOrd,
Hash,
Debug,
Default,
Display,
DeExchange,
SerExchange,
)]
pub struct Hyperliquid;
impl Connector for Hyperliquid {
const ID: ExchangeId = ExchangeId::HyperliquidPerp;
type Channel = HyperliquidChannel;
type Market = HyperliquidMarket;
type Subscriber = WebSocketSubscriber;
type SubValidator = WebSocketSubValidator;
type SubResponse = HyperliquidSubResponse;
fn url() -> Result<Url, url::ParseError> {
Url::parse(BASE_URL_HYPERLIQUID)
}
fn ping_interval() -> Option<PingInterval> {
Some(PingInterval {
interval: tokio::time::interval(Duration::from_secs(PING_INTERVAL_SECS)),
ping: || WsMessage::text(r#"{"method":"ping"}"#),
})
}
fn requests(exchange_subs: Vec<ExchangeSub<Self::Channel, Self::Market>>) -> Vec<WsMessage> {
build_subscribe_messages(exchange_subs)
}
}
impl<Instrument> StreamSelector<Instrument, PublicTrades> for Hyperliquid
where
Instrument: InstrumentData,
{
type SnapFetcher = NoInitialSnapshots;
type Stream = HyperliquidWsStream<
StatelessTransformer<Self, Instrument::Key, PublicTrades, HyperliquidTrade>,
>;
}
impl<Instrument> StreamSelector<Instrument, OrderBooksL2> for Hyperliquid
where
Instrument: InstrumentData,
{
type SnapFetcher = NoInitialSnapshots;
type Stream = HyperliquidWsStream<
StatelessTransformer<Self, Instrument::Key, OrderBooksL2, HyperliquidL2Book>,
>;
}
#[derive(
Copy,
Clone,
Eq,
PartialEq,
Ord,
PartialOrd,
Hash,
Debug,
Default,
Display,
DeExchange,
SerExchange,
)]
pub struct HyperliquidSpot;
impl Connector for HyperliquidSpot {
const ID: ExchangeId = ExchangeId::HyperliquidSpot;
type Channel = HyperliquidChannel;
type Market = HyperliquidMarket;
type Subscriber = WebSocketSubscriber;
type SubValidator = WebSocketSubValidator;
type SubResponse = HyperliquidSubResponse;
fn url() -> Result<Url, url::ParseError> {
Url::parse(BASE_URL_HYPERLIQUID)
}
fn ping_interval() -> Option<PingInterval> {
Some(PingInterval {
interval: tokio::time::interval(Duration::from_secs(PING_INTERVAL_SECS)),
ping: || WsMessage::text(r#"{"method":"ping"}"#),
})
}
fn requests(exchange_subs: Vec<ExchangeSub<Self::Channel, Self::Market>>) -> Vec<WsMessage> {
build_subscribe_messages(exchange_subs)
}
}
impl<Instrument> StreamSelector<Instrument, PublicTrades> for HyperliquidSpot
where
Instrument: InstrumentData,
{
type SnapFetcher = NoInitialSnapshots;
type Stream = HyperliquidWsStream<
StatelessTransformer<Self, Instrument::Key, PublicTrades, HyperliquidTrade>,
>;
}
impl<Instrument> StreamSelector<Instrument, OrderBooksL2> for HyperliquidSpot
where
Instrument: InstrumentData,
{
type SnapFetcher = NoInitialSnapshots;
type Stream = HyperliquidWsStream<
StatelessTransformer<Self, Instrument::Key, OrderBooksL2, HyperliquidL2Book>,
>;
}