nautilus_model/python/orderbook/
book.rsuse indexmap::IndexMap;
use nautilus_core::python::{to_pyruntime_err, to_pyvalue_err};
use pyo3::prelude::*;
use rust_decimal::Decimal;
use crate::{
data::{BookOrder, OrderBookDelta, OrderBookDeltas, OrderBookDepth10, QuoteTick, TradeTick},
enums::{BookType, OrderSide},
identifiers::InstrumentId,
orderbook::{analysis::book_check_integrity, BookLevel, OrderBook},
types::{Price, Quantity},
};
#[pymethods]
impl OrderBook {
#[new]
fn py_new(instrument_id: InstrumentId, book_type: BookType) -> Self {
Self::new(instrument_id, book_type)
}
fn __repr__(&self) -> String {
format!("{self:?}")
}
fn __str__(&self) -> String {
format!("{self:?}")
}
#[getter]
#[pyo3(name = "instrument_id")]
fn py_instrument_id(&self) -> InstrumentId {
self.instrument_id
}
#[getter]
#[pyo3(name = "book_type")]
fn py_book_type(&self) -> BookType {
self.book_type
}
#[getter]
#[pyo3(name = "sequence")]
fn py_sequence(&self) -> u64 {
self.sequence
}
#[getter]
#[pyo3(name = "ts_event")]
fn py_ts_event(&self) -> u64 {
self.ts_last.as_u64()
}
#[getter]
#[pyo3(name = "ts_init")]
fn py_ts_init(&self) -> u64 {
self.ts_last.as_u64()
}
#[getter]
#[pyo3(name = "ts_last")]
fn py_ts_last(&self) -> u64 {
self.ts_last.as_u64()
}
#[getter]
#[pyo3(name = "count")]
fn py_count(&self) -> u64 {
self.count
}
#[pyo3(name = "reset")]
fn py_reset(&mut self) {
self.reset();
}
#[pyo3(signature = (order, flags, sequence, ts_event))]
#[pyo3(name = "add")]
fn py_add(&mut self, order: BookOrder, flags: u8, sequence: u64, ts_event: u64) {
self.add(order, flags, sequence, ts_event.into());
}
#[pyo3(signature = (order, flags, sequence, ts_event))]
#[pyo3(name = "update")]
fn py_update(&mut self, order: BookOrder, flags: u8, sequence: u64, ts_event: u64) {
self.update(order, flags, sequence, ts_event.into());
}
#[pyo3(signature = (order, flags, sequence, ts_event))]
#[pyo3(name = "delete")]
fn py_delete(&mut self, order: BookOrder, flags: u8, sequence: u64, ts_event: u64) {
self.delete(order, flags, sequence, ts_event.into());
}
#[pyo3(signature = (sequence, ts_event))]
#[pyo3(name = "clear")]
fn py_clear(&mut self, sequence: u64, ts_event: u64) {
self.clear(sequence, ts_event.into());
}
#[pyo3(signature = (sequence, ts_event))]
#[pyo3(name = "clear_bids")]
fn py_clear_bids(&mut self, sequence: u64, ts_event: u64) {
self.clear_bids(sequence, ts_event.into());
}
#[pyo3(signature = (sequence, ts_event))]
#[pyo3(name = "clear_asks")]
fn py_clear_asks(&mut self, sequence: u64, ts_event: u64) {
self.clear_asks(sequence, ts_event.into());
}
#[pyo3(name = "apply_delta")]
fn py_apply_delta(&mut self, delta: &OrderBookDelta) {
self.apply_delta(delta);
}
#[pyo3(name = "apply_deltas")]
fn py_apply_deltas(&mut self, deltas: &OrderBookDeltas) {
self.apply_deltas(deltas);
}
#[pyo3(name = "apply_depth")]
fn py_apply_depth(&mut self, depth: &OrderBookDepth10) {
self.apply_depth(depth);
}
#[pyo3(name = "check_integrity")]
fn py_check_integrity(&mut self) -> PyResult<()> {
book_check_integrity(self).map_err(to_pyruntime_err)
}
#[pyo3(signature = (depth=None))]
#[pyo3(name = "bids")]
fn py_bids(&self, depth: Option<usize>) -> Vec<BookLevel> {
self.bids(depth)
.map(|level_ref| (*level_ref).clone())
.collect()
}
#[pyo3(signature = (depth=None))]
#[pyo3(name = "asks")]
fn py_asks(&self, depth: Option<usize>) -> Vec<BookLevel> {
self.asks(depth)
.map(|level_ref| (*level_ref).clone())
.collect()
}
#[pyo3(signature = (depth=None))]
#[pyo3(name = "bids_to_dict")]
fn py_bids_to_dict(&self, depth: Option<usize>) -> IndexMap<Decimal, Decimal> {
self.bids_as_map(depth)
}
#[pyo3(signature = (depth=None))]
#[pyo3(name = "asks_to_dict")]
fn py_asks_to_dict(&self, depth: Option<usize>) -> IndexMap<Decimal, Decimal> {
self.asks_as_map(depth)
}
#[pyo3(signature = (group_size, depth=None))]
#[pyo3(name = "group_bids")]
pub fn py_group_bids(
&self,
group_size: Decimal,
depth: Option<usize>,
) -> IndexMap<Decimal, Decimal> {
self.group_bids(group_size, depth)
}
#[pyo3(signature = (group_size, depth=None))]
#[pyo3(name = "group_asks")]
pub fn py_group_asks(
&self,
group_size: Decimal,
depth: Option<usize>,
) -> IndexMap<Decimal, Decimal> {
self.group_asks(group_size, depth)
}
#[pyo3(name = "best_bid_price")]
fn py_best_bid_price(&self) -> Option<Price> {
self.best_bid_price()
}
#[pyo3(name = "best_ask_price")]
fn py_best_ask_price(&self) -> Option<Price> {
self.best_ask_price()
}
#[pyo3(name = "best_bid_size")]
fn py_best_bid_size(&self) -> Option<Quantity> {
self.best_bid_size()
}
#[pyo3(name = "best_ask_size")]
fn py_best_ask_size(&self) -> Option<Quantity> {
self.best_ask_size()
}
#[pyo3(name = "spread")]
fn py_spread(&self) -> Option<f64> {
self.spread()
}
#[pyo3(name = "midpoint")]
fn py_midpoint(&self) -> Option<f64> {
self.midpoint()
}
#[pyo3(name = "get_avg_px_for_quantity")]
fn py_get_avg_px_for_quantity(&self, qty: Quantity, order_side: OrderSide) -> f64 {
self.get_avg_px_for_quantity(qty, order_side)
}
#[pyo3(name = "get_avg_px_qty_for_exposure")]
fn py_get_avg_px_qty_for_exposure(
&self,
qty: Quantity,
order_side: OrderSide,
) -> (f64, f64, f64) {
self.get_avg_px_qty_for_exposure(qty, order_side)
}
#[pyo3(name = "get_quantity_for_price")]
fn py_get_quantity_for_price(&self, price: Price, order_side: OrderSide) -> f64 {
self.get_quantity_for_price(price, order_side)
}
#[pyo3(name = "simulate_fills")]
fn py_simulate_fills(&self, order: &BookOrder) -> Vec<(Price, Quantity)> {
self.simulate_fills(order)
}
#[pyo3(name = "pprint")]
fn py_pprint(&self, num_levels: usize) -> String {
self.pprint(num_levels)
}
}
#[pyfunction()]
#[pyo3(name = "update_book_with_quote_tick")]
pub fn py_update_book_with_quote_tick(book: &mut OrderBook, quote: &QuoteTick) -> PyResult<()> {
book.update_quote_tick(quote).map_err(to_pyvalue_err)
}
#[pyfunction()]
#[pyo3(name = "update_book_with_trade_tick")]
pub fn py_update_book_with_trade_tick(book: &mut OrderBook, trade: &TradeTick) -> PyResult<()> {
book.update_trade_tick(trade).map_err(to_pyvalue_err)
}