use std::{
cmp,
collections::HashMap,
fmt::{Display, Formatter},
hash::Hash,
};
use derive_builder::Builder;
use indexmap::IndexMap;
use nautilus_core::{
correctness::{check_equal_u8, FAILED},
serialization::Serializable,
UnixNanos,
};
use serde::{Deserialize, Serialize};
use super::GetTsInit;
use crate::{
enums::PriceType,
identifiers::InstrumentId,
types::{fixed::FIXED_PRECISION, Price, Quantity},
};
#[repr(C)]
#[derive(Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize, Builder)]
#[serde(tag = "type")]
#[cfg_attr(
feature = "python",
pyo3::pyclass(module = "nautilus_trader.core.nautilus_pyo3.model")
)]
#[cfg_attr(feature = "trivial_copy", derive(Copy))]
pub struct QuoteTick {
pub instrument_id: InstrumentId,
pub bid_price: Price,
pub ask_price: Price,
pub bid_size: Quantity,
pub ask_size: Quantity,
pub ts_event: UnixNanos,
pub ts_init: UnixNanos,
}
impl QuoteTick {
pub fn new_checked(
instrument_id: InstrumentId,
bid_price: Price,
ask_price: Price,
bid_size: Quantity,
ask_size: Quantity,
ts_event: UnixNanos,
ts_init: UnixNanos,
) -> anyhow::Result<Self> {
check_equal_u8(
bid_price.precision,
ask_price.precision,
"bid_price.precision",
"ask_price.precision",
)?;
check_equal_u8(
bid_size.precision,
ask_size.precision,
"bid_size.precision",
"ask_size.precision",
)?;
Ok(Self {
instrument_id,
bid_price,
ask_price,
bid_size,
ask_size,
ts_event,
ts_init,
})
}
pub fn new(
instrument_id: InstrumentId,
bid_price: Price,
ask_price: Price,
bid_size: Quantity,
ask_size: Quantity,
ts_event: UnixNanos,
ts_init: UnixNanos,
) -> Self {
Self::new_checked(
instrument_id,
bid_price,
ask_price,
bid_size,
ask_size,
ts_event,
ts_init,
)
.expect(FAILED)
}
#[must_use]
pub fn get_metadata(
instrument_id: &InstrumentId,
price_precision: u8,
size_precision: u8,
) -> HashMap<String, String> {
let mut metadata = HashMap::new();
metadata.insert("instrument_id".to_string(), instrument_id.to_string());
metadata.insert("price_precision".to_string(), price_precision.to_string());
metadata.insert("size_precision".to_string(), size_precision.to_string());
metadata
}
#[must_use]
pub fn get_fields() -> IndexMap<String, String> {
let mut metadata = IndexMap::new();
metadata.insert("bid_price".to_string(), "Int64".to_string());
metadata.insert("ask_price".to_string(), "Int64".to_string());
metadata.insert("bid_size".to_string(), "UInt64".to_string());
metadata.insert("ask_size".to_string(), "UInt64".to_string());
metadata.insert("ts_event".to_string(), "UInt64".to_string());
metadata.insert("ts_init".to_string(), "UInt64".to_string());
metadata
}
#[must_use]
pub fn extract_price(&self, price_type: PriceType) -> Price {
match price_type {
PriceType::Bid => self.bid_price,
PriceType::Ask => self.ask_price,
PriceType::Mid => Price::from_raw(
(self.bid_price.raw + self.ask_price.raw) / 2,
cmp::min(self.bid_price.precision + 1, FIXED_PRECISION),
),
_ => panic!("Cannot extract with price type {price_type}"),
}
}
#[must_use]
pub fn extract_size(&self, price_type: PriceType) -> Quantity {
match price_type {
PriceType::Bid => self.bid_size,
PriceType::Ask => self.ask_size,
PriceType::Mid => Quantity::from_raw(
(self.bid_size.raw + self.ask_size.raw) / 2,
cmp::min(self.bid_size.precision + 1, FIXED_PRECISION),
),
_ => panic!("Cannot extract with price type {price_type}"),
}
}
}
impl Display for QuoteTick {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
write!(
f,
"{},{},{},{},{},{}",
self.instrument_id,
self.bid_price,
self.ask_price,
self.bid_size,
self.ask_size,
self.ts_event,
)
}
}
impl Serializable for QuoteTick {}
impl GetTsInit for QuoteTick {
fn ts_init(&self) -> UnixNanos {
self.ts_init
}
}
#[cfg(test)]
mod tests {
use nautilus_core::{serialization::Serializable, UnixNanos};
use rstest::rstest;
use crate::{
data::{stubs::quote_ethusdt_binance, QuoteTick},
enums::PriceType,
identifiers::InstrumentId,
types::{Price, Quantity},
};
#[rstest]
#[should_panic(
expected = "'bid_price.precision' u8 of 4 was not equal to 'ask_price.precision' u8 of 5"
)]
fn test_quote_tick_new_with_precision_mismatch_panics() {
let instrument_id = InstrumentId::from("ETH-USDT-SWAP.OKX");
let bid_price = Price::from("10000.0000"); let ask_price = Price::from("10000.00100"); let bid_size = Quantity::from("1.000000");
let ask_size = Quantity::from("1.000000");
let ts_event = UnixNanos::from(0);
let ts_init = UnixNanos::from(1);
let _ = QuoteTick::new(
instrument_id,
bid_price,
ask_price,
bid_size,
ask_size,
ts_event,
ts_init,
);
}
#[rstest]
fn test_quote_tick_new_checked_with_precision_mismatch_error() {
let instrument_id = InstrumentId::from("ETH-USDT-SWAP.OKX");
let bid_price = Price::from("10000.0000");
let ask_price = Price::from("10000.0010");
let bid_size = Quantity::from("10.000000"); let ask_size = Quantity::from("10.0000000"); let ts_event = UnixNanos::from(0);
let ts_init = UnixNanos::from(1);
let result = QuoteTick::new_checked(
instrument_id,
bid_price,
ask_price,
bid_size,
ask_size,
ts_event,
ts_init,
);
assert!(result.is_err());
}
#[rstest]
fn test_to_string(quote_ethusdt_binance: QuoteTick) {
let quote = quote_ethusdt_binance;
assert_eq!(
quote.to_string(),
"ETHUSDT-PERP.BINANCE,10000.0000,10001.0000,1.00000000,1.00000000,0"
);
}
#[rstest]
#[case(PriceType::Bid, 10_000_000_000_000)]
#[case(PriceType::Ask, 10_001_000_000_000)]
#[case(PriceType::Mid, 10_000_500_000_000)]
fn test_extract_price(
#[case] input: PriceType,
#[case] expected: i64,
quote_ethusdt_binance: QuoteTick,
) {
let quote = quote_ethusdt_binance;
let result = quote.extract_price(input).raw;
assert_eq!(result, expected);
}
#[rstest]
fn test_json_serialization(quote_ethusdt_binance: QuoteTick) {
let quote = quote_ethusdt_binance;
let serialized = quote.as_json_bytes().unwrap();
let deserialized = QuoteTick::from_json_bytes(serialized.as_ref()).unwrap();
assert_eq!(deserialized, quote);
}
#[rstest]
fn test_msgpack_serialization(quote_ethusdt_binance: QuoteTick) {
let quote = quote_ethusdt_binance;
let serialized = quote.as_msgpack_bytes().unwrap();
let deserialized = QuoteTick::from_msgpack_bytes(serialized.as_ref()).unwrap();
assert_eq!(deserialized, quote);
}
}