use nautilus_core::UnixNanos;
use serde::{Deserialize, Serialize};
use crate::{
enums::{OrderSide, PositionSide},
identifiers::{AccountId, ClientOrderId, InstrumentId, PositionId, StrategyId, TraderId},
position::Position,
types::{Currency, Money, Quantity},
};
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
#[cfg_attr(
feature = "python",
pyo3::pyclass(module = "nautilus_trader.core.nautilus_pyo3.model", from_py_object)
)]
#[cfg_attr(
feature = "python",
pyo3_stub_gen::derive::gen_stub_pyclass(module = "nautilus_trader.model")
)]
pub struct PositionSnapshot {
pub trader_id: TraderId,
pub strategy_id: StrategyId,
pub instrument_id: InstrumentId,
pub position_id: PositionId,
pub account_id: AccountId,
pub opening_order_id: ClientOrderId,
pub closing_order_id: Option<ClientOrderId>,
pub entry: OrderSide,
pub side: PositionSide,
pub signed_qty: f64,
pub quantity: Quantity,
pub peak_qty: Quantity,
pub quote_currency: Currency,
pub base_currency: Option<Currency>,
pub settlement_currency: Currency,
pub avg_px_open: f64,
pub avg_px_close: Option<f64>,
pub realized_return: Option<f64>,
pub realized_pnl: Option<Money>,
pub unrealized_pnl: Option<Money>,
pub commissions: Vec<Money>,
pub duration_ns: Option<u64>,
pub ts_opened: UnixNanos,
pub ts_closed: Option<UnixNanos>,
pub ts_init: UnixNanos,
pub ts_last: UnixNanos,
}
impl PositionSnapshot {
pub fn from(position: &Position, unrealized_pnl: Option<Money>) -> Self {
Self {
trader_id: position.trader_id,
strategy_id: position.strategy_id,
instrument_id: position.instrument_id,
position_id: position.id,
account_id: position.account_id,
opening_order_id: position.opening_order_id,
closing_order_id: position.closing_order_id,
entry: position.entry,
side: position.side,
signed_qty: position.signed_qty,
quantity: position.quantity,
peak_qty: position.peak_qty,
quote_currency: position.quote_currency,
base_currency: position.base_currency,
settlement_currency: position.settlement_currency,
avg_px_open: position.avg_px_open,
avg_px_close: position.avg_px_close,
realized_return: Some(position.realized_return), realized_pnl: position.realized_pnl,
unrealized_pnl,
commissions: position.commissions.values().copied().collect(), duration_ns: Some(position.duration_ns), ts_opened: position.ts_opened,
ts_closed: position.ts_closed,
ts_init: position.ts_init,
ts_last: position.ts_last,
}
}
}
#[cfg(test)]
mod tests {
use nautilus_core::{UUID4, UnixNanos};
use rstest::*;
use super::*;
use crate::{
enums::{LiquiditySide, OrderSide, OrderType, PositionSide},
events::OrderFilled,
identifiers::{
AccountId, ClientOrderId, InstrumentId, PositionId, StrategyId, TradeId, TraderId,
VenueOrderId,
},
instruments::{InstrumentAny, stubs::audusd_sim},
position::Position,
types::{Currency, Money, Price, Quantity},
};
fn create_test_position_snapshot() -> PositionSnapshot {
PositionSnapshot {
trader_id: TraderId::from("TRADER-001"),
strategy_id: StrategyId::from("EMA-CROSS"),
instrument_id: InstrumentId::from("EURUSD.SIM"),
position_id: PositionId::from("P-001"),
account_id: AccountId::from("SIM-001"),
opening_order_id: ClientOrderId::from("O-19700101-000000-001-001-1"),
closing_order_id: Some(ClientOrderId::from("O-19700101-000000-001-001-2")),
entry: OrderSide::Buy,
side: PositionSide::Long,
signed_qty: 100.0,
quantity: Quantity::from("100"),
peak_qty: Quantity::from("100"),
quote_currency: Currency::USD(),
base_currency: Some(Currency::EUR()),
settlement_currency: Currency::USD(),
avg_px_open: 1.0500,
avg_px_close: Some(1.0600),
realized_return: Some(0.0095),
realized_pnl: Some(Money::new(100.0, Currency::USD())),
unrealized_pnl: Some(Money::new(50.0, Currency::USD())),
commissions: vec![Money::new(2.0, Currency::USD())],
duration_ns: Some(3_600_000_000_000), ts_opened: UnixNanos::from(1_000_000_000),
ts_closed: Some(UnixNanos::from(4_600_000_000)),
ts_init: UnixNanos::from(2_000_000_000),
ts_last: UnixNanos::from(4_600_000_000),
}
}
fn create_test_order_filled() -> OrderFilled {
OrderFilled::new(
TraderId::from("TRADER-001"),
StrategyId::from("EMA-CROSS"),
InstrumentId::from("AUD/USD.SIM"),
ClientOrderId::from("O-19700101-000000-001-001-1"),
VenueOrderId::from("1"),
AccountId::from("SIM-001"),
TradeId::from("T-001"),
OrderSide::Buy,
OrderType::Market,
Quantity::from("100"),
Price::from("0.8000"),
Currency::USD(),
LiquiditySide::Taker,
UUID4::default(),
UnixNanos::from(1_000_000_000),
UnixNanos::from(2_000_000_000),
false,
Some(PositionId::from("P-001")),
Some(Money::new(2.0, Currency::USD())),
)
}
#[rstest]
fn test_position_snapshot_from() {
let instrument = audusd_sim();
let fill = create_test_order_filled();
let position = Position::new(&InstrumentAny::CurrencyPair(instrument), fill);
let unrealized_pnl = Some(Money::new(75.0, Currency::USD()));
let snapshot = PositionSnapshot::from(&position, unrealized_pnl);
assert_eq!(snapshot.trader_id, position.trader_id);
assert_eq!(snapshot.strategy_id, position.strategy_id);
assert_eq!(snapshot.instrument_id, position.instrument_id);
assert_eq!(snapshot.position_id, position.id);
assert_eq!(snapshot.account_id, position.account_id);
assert_eq!(snapshot.opening_order_id, position.opening_order_id);
assert_eq!(snapshot.closing_order_id, position.closing_order_id);
assert_eq!(snapshot.entry, position.entry);
assert_eq!(snapshot.side, position.side);
assert_eq!(snapshot.signed_qty, position.signed_qty);
assert_eq!(snapshot.quantity, position.quantity);
assert_eq!(snapshot.peak_qty, position.peak_qty);
assert_eq!(snapshot.quote_currency, position.quote_currency);
assert_eq!(snapshot.base_currency, position.base_currency);
assert_eq!(snapshot.settlement_currency, position.settlement_currency);
assert_eq!(snapshot.avg_px_open, position.avg_px_open);
assert_eq!(snapshot.avg_px_close, position.avg_px_close);
assert_eq!(snapshot.realized_return, Some(position.realized_return));
assert_eq!(snapshot.realized_pnl, position.realized_pnl);
assert_eq!(snapshot.unrealized_pnl, unrealized_pnl);
assert_eq!(snapshot.duration_ns, Some(position.duration_ns));
assert_eq!(snapshot.ts_opened, position.ts_opened);
assert_eq!(snapshot.ts_closed, position.ts_closed);
assert_eq!(snapshot.ts_init, position.ts_init);
assert_eq!(snapshot.ts_last, position.ts_last);
}
#[rstest]
fn test_position_snapshot_from_with_no_unrealized_pnl() {
let instrument = audusd_sim();
let fill = create_test_order_filled();
let position = Position::new(&InstrumentAny::CurrencyPair(instrument), fill);
let snapshot = PositionSnapshot::from(&position, None);
assert_eq!(snapshot.unrealized_pnl, None);
}
#[rstest]
fn test_position_snapshot_serialization() {
let original = create_test_position_snapshot();
let json = serde_json::to_string(&original).unwrap();
let deserialized: PositionSnapshot = serde_json::from_str(&json).unwrap();
assert_eq!(original, deserialized);
}
}