use std::{cmp::Ordering, collections::BTreeMap};
use nautilus_core::UnixNanos;
use rust_decimal::Decimal;
use crate::{
data::order::{BookOrder, OrderId},
orderbook::{BookIntegrityError, BookPrice},
types::fixed::FIXED_SCALAR,
};
#[derive(Clone, Debug, Eq)]
#[cfg_attr(
feature = "python",
pyo3::pyclass(module = "nautilus_trader.core.nautilus_pyo3.model")
)]
pub struct BookLevel {
pub price: BookPrice,
pub orders: BTreeMap<OrderId, BookOrder>,
insertion_order: Vec<OrderId>,
}
impl BookLevel {
#[must_use]
pub fn new(price: BookPrice) -> Self {
Self {
price,
orders: BTreeMap::new(),
insertion_order: Vec::new(),
}
}
#[must_use]
pub fn from_order(order: BookOrder) -> Self {
let mut level = Self {
price: order.to_book_price(),
orders: BTreeMap::new(),
insertion_order: Vec::new(),
};
level.add(order);
level
}
#[must_use]
pub fn len(&self) -> usize {
self.orders.len()
}
#[must_use]
pub fn is_empty(&self) -> bool {
self.orders.is_empty()
}
#[must_use]
pub fn first(&self) -> Option<&BookOrder> {
self.insertion_order
.first()
.and_then(|&id| self.orders.get(&id))
}
#[must_use]
pub fn get_orders(&self) -> Vec<BookOrder> {
self.insertion_order
.iter()
.filter_map(|id| self.orders.get(id))
.copied()
.collect()
}
#[must_use]
pub fn size(&self) -> f64 {
self.orders.values().map(|o| o.size.as_f64()).sum()
}
#[must_use]
pub fn size_raw(&self) -> u64 {
self.orders.values().map(|o| o.size.raw).sum()
}
#[must_use]
pub fn size_decimal(&self) -> Decimal {
self.orders.values().map(|o| o.size.as_decimal()).sum()
}
#[must_use]
pub fn exposure(&self) -> f64 {
self.orders
.values()
.map(|o| o.price.as_f64() * o.size.as_f64())
.sum()
}
#[must_use]
pub fn exposure_raw(&self) -> u64 {
self.orders
.values()
.map(|o| ((o.price.as_f64() * o.size.as_f64()) * FIXED_SCALAR) as u64)
.sum()
}
pub fn add_bulk(&mut self, orders: Vec<BookOrder>) {
self.insertion_order
.extend(orders.iter().map(|o| o.order_id));
for order in orders {
self.check_order_for_this_level(&order);
self.orders.insert(order.order_id, order);
}
}
pub fn add(&mut self, order: BookOrder) {
self.check_order_for_this_level(&order);
self.orders.insert(order.order_id, order);
self.insertion_order.push(order.order_id);
}
pub fn update(&mut self, order: BookOrder) {
self.check_order_for_this_level(&order);
if order.size.raw == 0 {
self.orders.remove(&order.order_id);
self.update_insertion_order();
} else {
self.orders.insert(order.order_id, order);
}
}
pub fn delete(&mut self, order: &BookOrder) {
self.orders.remove(&order.order_id);
self.update_insertion_order();
}
pub fn remove_by_id(&mut self, order_id: OrderId, sequence: u64, ts_event: UnixNanos) {
assert!(
self.orders.remove(&order_id).is_some(),
"{}",
&BookIntegrityError::OrderNotFound(order_id, sequence, ts_event)
);
self.update_insertion_order();
}
fn check_order_for_this_level(&self, order: &BookOrder) {
assert_eq!(order.price, self.price.value);
}
fn update_insertion_order(&mut self) {
if self
.insertion_order
.iter()
.any(|id| !self.orders.contains_key(id))
{
self.insertion_order
.retain(|&id| self.orders.contains_key(&id));
}
}
}
impl PartialEq for BookLevel {
fn eq(&self, other: &Self) -> bool {
self.price == other.price
}
}
impl PartialOrd for BookLevel {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Some(self.cmp(other))
}
fn lt(&self, other: &Self) -> bool {
self.price.lt(&other.price)
}
fn le(&self, other: &Self) -> bool {
self.price.le(&other.price)
}
fn gt(&self, other: &Self) -> bool {
self.price.gt(&other.price)
}
fn ge(&self, other: &Self) -> bool {
self.price.ge(&other.price)
}
}
impl Ord for BookLevel {
fn cmp(&self, other: &Self) -> Ordering {
self.price.cmp(&other.price)
}
}
#[cfg(test)]
mod tests {
use rstest::rstest;
use rust_decimal_macros::dec;
use crate::{
data::order::BookOrder,
enums::OrderSide,
orderbook::{BookLevel, BookPrice},
types::{Price, Quantity},
};
#[rstest]
fn test_empty_level() {
let level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
assert!(level.first().is_none());
}
#[rstest]
fn test_level_from_order() {
let order = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 1);
let level = BookLevel::from_order(order);
assert_eq!(level.price.value, Price::from("1.00"));
assert_eq!(level.price.side, OrderSide::Buy);
assert_eq!(level.len(), 1);
assert_eq!(level.first().unwrap(), &order);
assert_eq!(level.size(), 10.0);
}
#[rstest]
#[should_panic(expected = "assertion `left == right` failed")]
fn test_add_order_incorrect_price_level() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let incorrect_price_order =
BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(10), 1);
level.add(incorrect_price_order);
}
#[rstest]
#[should_panic(expected = "assertion `left == right` failed")]
fn test_add_bulk_orders_incorrect_price() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let orders = vec![
BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 1),
BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(20), 2), ];
level.add_bulk(orders);
}
#[rstest]
fn test_comparisons_bid_side() {
let level0 = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let level1 = BookLevel::new(BookPrice::new(Price::from("1.01"), OrderSide::Buy));
assert_eq!(level0, level0);
assert!(level0 > level1);
}
#[rstest]
fn test_comparisons_ask_side() {
let level0 = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Sell));
let level1 = BookLevel::new(BookPrice::new(Price::from("1.01"), OrderSide::Sell));
assert_eq!(level0, level0);
assert!(level0 < level1);
}
#[rstest]
fn test_add_single_order() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 0);
level.add(order);
assert!(!level.is_empty());
assert_eq!(level.len(), 1);
assert_eq!(level.size(), 10.0);
assert_eq!(level.first().unwrap(), &order);
}
#[rstest]
fn test_add_multiple_orders() {
let mut level = BookLevel::new(BookPrice::new(Price::from("2.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(10), 0);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(20), 1);
level.add(order1);
level.add(order2);
assert_eq!(level.len(), 2);
assert_eq!(level.size(), 30.0);
assert_eq!(level.exposure(), 60.0);
assert_eq!(level.first().unwrap(), &order1);
}
#[rstest]
fn test_get_orders() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 1);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(20), 2);
level.add(order1);
level.add(order2);
let orders = level.get_orders();
assert_eq!(orders.len(), 2);
assert_eq!(orders[0], order1); assert_eq!(orders[1], order2);
}
#[rstest]
fn test_update_order() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 0);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(20), 0);
level.add(order1);
level.update(order2);
assert_eq!(level.len(), 1);
assert_eq!(level.size(), 20.0);
assert_eq!(level.exposure(), 20.0);
}
#[rstest]
fn test_fifo_order_after_updates() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 1);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(20), 2);
level.add(order1.clone());
level.add(order2.clone());
let updated_order1 =
BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(15), 1);
level.update(updated_order1.clone());
let orders = level.get_orders();
assert_eq!(orders.len(), 2);
assert_eq!(orders[0], updated_order1); assert_eq!(orders[1], order2); }
#[rstest]
#[should_panic(expected = "assertion `left == right` failed")]
fn test_update_order_incorrect_price() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let initial_order =
BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 1);
level.add(initial_order);
let updated_order =
BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(20), 1);
level.update(updated_order);
}
#[rstest]
fn test_update_order_with_zero_size() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 0);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::zero(0), 0);
level.add(order1);
level.update(order2);
assert_eq!(level.len(), 0);
assert_eq!(level.size(), 0.0);
assert_eq!(level.exposure(), 0.0);
}
#[rstest]
fn test_update_insertion_order_optimization() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 1);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(20), 2);
level.add(order1);
level.add(order2);
let initial_insertion_order = level.insertion_order.clone();
level.update_insertion_order();
assert_eq!(level.insertion_order, initial_insertion_order);
level.orders.remove(&1);
level.update_insertion_order();
assert_eq!(level.insertion_order, vec![2]);
}
#[rstest]
fn test_delete_order() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order1_id = 0;
let order1 = BookOrder::new(
OrderSide::Buy,
Price::from("1.00"),
Quantity::from(10),
order1_id,
);
let order2_id = 1;
let order2 = BookOrder::new(
OrderSide::Buy,
Price::from("1.00"),
Quantity::from(20),
order2_id,
);
level.add(order1);
level.add(order2);
level.delete(&order1);
assert_eq!(level.len(), 1);
assert_eq!(level.size(), 20.0);
assert!(level.orders.contains_key(&order2_id));
assert_eq!(level.exposure(), 20.0);
}
#[rstest]
fn test_remove_order_by_id() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order1_id = 0;
let order1 = BookOrder::new(
OrderSide::Buy,
Price::from("1.00"),
Quantity::from(10),
order1_id,
);
let order2_id = 1;
let order2 = BookOrder::new(
OrderSide::Buy,
Price::from("1.00"),
Quantity::from(20),
order2_id,
);
level.add(order1);
level.add(order2);
level.remove_by_id(order2_id, 0, 0.into());
assert_eq!(level.len(), 1);
assert!(level.orders.contains_key(&order1_id));
assert_eq!(level.size(), 10.0);
assert_eq!(level.exposure(), 10.0);
}
#[rstest]
fn test_add_bulk_orders() {
let mut level = BookLevel::new(BookPrice::new(Price::from("2.00"), OrderSide::Buy));
let order1_id = 0;
let order1 = BookOrder::new(
OrderSide::Buy,
Price::from("2.00"),
Quantity::from(10),
order1_id,
);
let order2_id = 1;
let order2 = BookOrder::new(
OrderSide::Buy,
Price::from("2.00"),
Quantity::from(20),
order2_id,
);
let orders = vec![order1, order2];
level.add_bulk(orders);
assert_eq!(level.len(), 2);
assert_eq!(level.size(), 30.0);
assert_eq!(level.exposure(), 60.0);
}
#[rstest]
fn test_maximum_order_id() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order = BookOrder::new(
OrderSide::Buy,
Price::from("1.00"),
Quantity::from(10),
u64::MAX,
);
level.add(order.clone());
assert_eq!(level.len(), 1);
assert_eq!(level.first().unwrap(), &order);
}
#[rstest]
#[should_panic(
expected = "Integrity error: order not found: order_id=1, sequence=2, ts_event=3"
)]
fn test_remove_nonexistent_order() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
level.remove_by_id(1, 2, 3.into());
}
#[rstest]
fn test_size() {
let mut level = BookLevel::new(BookPrice::new(Price::from("1.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(10), 0);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("1.00"), Quantity::from(15), 1);
level.add(order1);
level.add(order2);
assert_eq!(level.size(), 25.0);
}
#[rstest]
fn test_size_raw() {
let mut level = BookLevel::new(BookPrice::new(Price::from("2.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(10), 0);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(20), 1);
level.add(order1);
level.add(order2);
assert_eq!(level.size_raw(), 30_000_000_000);
}
#[rstest]
fn test_size_decimal() {
let mut level = BookLevel::new(BookPrice::new(Price::from("2.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(10), 0);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(20), 1);
level.add(order1);
level.add(order2);
assert_eq!(level.size_decimal(), dec!(30.0));
}
#[rstest]
fn test_exposure() {
let mut level = BookLevel::new(BookPrice::new(Price::from("2.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(10), 0);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(20), 1);
level.add(order1);
level.add(order2);
assert_eq!(level.exposure(), 60.0);
}
#[rstest]
fn test_exposure_raw() {
let mut level = BookLevel::new(BookPrice::new(Price::from("2.00"), OrderSide::Buy));
let order1 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(10), 0);
let order2 = BookOrder::new(OrderSide::Buy, Price::from("2.00"), Quantity::from(20), 1);
level.add(order1);
level.add(order2);
assert_eq!(level.exposure_raw(), 60_000_000_000);
}
}