1use std::{
17 fmt::Display,
18 ops::{Deref, DerefMut},
19};
20
21use indexmap::IndexMap;
22use nautilus_core::{UUID4, UnixNanos, correctness::FAILED};
23use rust_decimal::Decimal;
24use serde::{Deserialize, Serialize};
25use ustr::Ustr;
26
27use super::{Order, OrderAny, OrderCore, check_display_qty, check_time_in_force};
28use crate::{
29 enums::{
30 ContingencyType, LiquiditySide, OrderSide, OrderStatus, OrderType, PositionSide,
31 TimeInForce, TrailingOffsetType, TriggerType,
32 },
33 events::{OrderEventAny, OrderInitialized, OrderUpdated},
34 identifiers::{
35 AccountId, ClientOrderId, ExecAlgorithmId, InstrumentId, OrderListId, PositionId,
36 StrategyId, Symbol, TradeId, TraderId, Venue, VenueOrderId,
37 },
38 orders::OrderError,
39 types::{Currency, Money, Price, Quantity, quantity::check_positive_quantity},
40};
41
42#[derive(Clone, Debug, Serialize, Deserialize)]
43#[cfg_attr(
44 feature = "python",
45 pyo3::pyclass(module = "nautilus_trader.core.nautilus_pyo3.model", from_py_object)
46)]
47#[cfg_attr(
48 feature = "python",
49 pyo3_stub_gen::derive::gen_stub_pyclass(module = "nautilus_trader.model")
50)]
51pub struct LimitOrder {
52 core: OrderCore,
53 pub price: Price,
54 pub expire_time: Option<UnixNanos>,
55 pub is_post_only: bool,
56 pub display_qty: Option<Quantity>,
57 pub trigger_instrument_id: Option<InstrumentId>,
58}
59
60impl LimitOrder {
61 #[expect(clippy::too_many_arguments)]
70 pub fn new_checked(
71 trader_id: TraderId,
72 strategy_id: StrategyId,
73 instrument_id: InstrumentId,
74 client_order_id: ClientOrderId,
75 order_side: OrderSide,
76 quantity: Quantity,
77 price: Price,
78 time_in_force: TimeInForce,
79 expire_time: Option<UnixNanos>,
80 post_only: bool,
81 reduce_only: bool,
82 quote_quantity: bool,
83 display_qty: Option<Quantity>,
84 emulation_trigger: Option<TriggerType>,
85 trigger_instrument_id: Option<InstrumentId>,
86 contingency_type: Option<ContingencyType>,
87 order_list_id: Option<OrderListId>,
88 linked_order_ids: Option<Vec<ClientOrderId>>,
89 parent_order_id: Option<ClientOrderId>,
90 exec_algorithm_id: Option<ExecAlgorithmId>,
91 exec_algorithm_params: Option<IndexMap<Ustr, Ustr>>,
92 exec_spawn_id: Option<ClientOrderId>,
93 tags: Option<Vec<Ustr>>,
94 init_id: UUID4,
95 ts_init: UnixNanos,
96 ) -> Result<Self, OrderError> {
97 check_positive_quantity(quantity, stringify!(quantity))?;
98 check_display_qty(display_qty, quantity)?;
99 check_time_in_force(time_in_force, expire_time)?;
100
101 let init_order = OrderInitialized::new(
102 trader_id,
103 strategy_id,
104 instrument_id,
105 client_order_id,
106 order_side,
107 OrderType::Limit,
108 quantity,
109 time_in_force,
110 post_only,
111 reduce_only,
112 quote_quantity,
113 false,
114 init_id,
115 ts_init, ts_init,
117 Some(price),
118 None,
119 None,
120 None,
121 None,
122 None,
123 expire_time,
124 display_qty,
125 emulation_trigger,
126 trigger_instrument_id,
127 contingency_type,
128 order_list_id,
129 linked_order_ids,
130 parent_order_id,
131 exec_algorithm_id,
132 exec_algorithm_params,
133 exec_spawn_id,
134 tags,
135 );
136
137 Ok(Self {
138 core: OrderCore::new(init_order),
139 price,
140 expire_time,
141 is_post_only: post_only,
142 display_qty,
143 trigger_instrument_id,
144 })
145 }
146
147 #[expect(clippy::too_many_arguments)]
153 #[must_use]
154 pub fn new(
155 trader_id: TraderId,
156 strategy_id: StrategyId,
157 instrument_id: InstrumentId,
158 client_order_id: ClientOrderId,
159 order_side: OrderSide,
160 quantity: Quantity,
161 price: Price,
162 time_in_force: TimeInForce,
163 expire_time: Option<UnixNanos>,
164 post_only: bool,
165 reduce_only: bool,
166 quote_quantity: bool,
167 display_qty: Option<Quantity>,
168 emulation_trigger: Option<TriggerType>,
169 trigger_instrument_id: Option<InstrumentId>,
170 contingency_type: Option<ContingencyType>,
171 order_list_id: Option<OrderListId>,
172 linked_order_ids: Option<Vec<ClientOrderId>>,
173 parent_order_id: Option<ClientOrderId>,
174 exec_algorithm_id: Option<ExecAlgorithmId>,
175 exec_algorithm_params: Option<IndexMap<Ustr, Ustr>>,
176 exec_spawn_id: Option<ClientOrderId>,
177 tags: Option<Vec<Ustr>>,
178 init_id: UUID4,
179 ts_init: UnixNanos,
180 ) -> Self {
181 Self::new_checked(
182 trader_id,
183 strategy_id,
184 instrument_id,
185 client_order_id,
186 order_side,
187 quantity,
188 price,
189 time_in_force,
190 expire_time,
191 post_only,
192 reduce_only,
193 quote_quantity,
194 display_qty,
195 emulation_trigger,
196 trigger_instrument_id,
197 contingency_type,
198 order_list_id,
199 linked_order_ids,
200 parent_order_id,
201 exec_algorithm_id,
202 exec_algorithm_params,
203 exec_spawn_id,
204 tags,
205 init_id,
206 ts_init,
207 )
208 .unwrap_or_else(|e| panic!("{FAILED}: {e}"))
209 }
210}
211
212impl Deref for LimitOrder {
213 type Target = OrderCore;
214
215 fn deref(&self) -> &Self::Target {
216 &self.core
217 }
218}
219
220impl DerefMut for LimitOrder {
221 fn deref_mut(&mut self) -> &mut Self::Target {
222 &mut self.core
223 }
224}
225
226impl PartialEq for LimitOrder {
227 fn eq(&self, other: &Self) -> bool {
228 self.client_order_id == other.client_order_id
229 }
230}
231
232impl Order for LimitOrder {
233 fn into_any(self) -> OrderAny {
234 OrderAny::Limit(self)
235 }
236
237 fn status(&self) -> OrderStatus {
238 self.status
239 }
240
241 fn trader_id(&self) -> TraderId {
242 self.trader_id
243 }
244
245 fn strategy_id(&self) -> StrategyId {
246 self.strategy_id
247 }
248
249 fn instrument_id(&self) -> InstrumentId {
250 self.instrument_id
251 }
252
253 fn symbol(&self) -> Symbol {
254 self.instrument_id.symbol
255 }
256
257 fn venue(&self) -> Venue {
258 self.instrument_id.venue
259 }
260
261 fn client_order_id(&self) -> ClientOrderId {
262 self.client_order_id
263 }
264
265 fn venue_order_id(&self) -> Option<VenueOrderId> {
266 self.venue_order_id
267 }
268
269 fn position_id(&self) -> Option<PositionId> {
270 self.position_id
271 }
272
273 fn account_id(&self) -> Option<AccountId> {
274 self.account_id
275 }
276
277 fn last_trade_id(&self) -> Option<TradeId> {
278 self.last_trade_id
279 }
280
281 fn order_side(&self) -> OrderSide {
282 self.side
283 }
284
285 fn order_type(&self) -> OrderType {
286 self.order_type
287 }
288
289 fn quantity(&self) -> Quantity {
290 self.quantity
291 }
292
293 fn time_in_force(&self) -> TimeInForce {
294 self.time_in_force
295 }
296
297 fn expire_time(&self) -> Option<UnixNanos> {
298 self.expire_time
299 }
300
301 fn price(&self) -> Option<Price> {
302 Some(self.price)
303 }
304
305 fn trigger_price(&self) -> Option<Price> {
306 None
307 }
308
309 fn trigger_type(&self) -> Option<TriggerType> {
310 None
311 }
312
313 fn liquidity_side(&self) -> Option<LiquiditySide> {
314 self.liquidity_side
315 }
316
317 fn is_post_only(&self) -> bool {
318 self.is_post_only
319 }
320
321 fn is_reduce_only(&self) -> bool {
322 self.is_reduce_only
323 }
324
325 fn is_quote_quantity(&self) -> bool {
326 self.is_quote_quantity
327 }
328
329 fn has_price(&self) -> bool {
330 true
331 }
332
333 fn display_qty(&self) -> Option<Quantity> {
334 self.display_qty
335 }
336
337 fn limit_offset(&self) -> Option<Decimal> {
338 None
339 }
340
341 fn trailing_offset(&self) -> Option<Decimal> {
342 None
343 }
344
345 fn trailing_offset_type(&self) -> Option<TrailingOffsetType> {
346 None
347 }
348
349 fn emulation_trigger(&self) -> Option<TriggerType> {
350 self.emulation_trigger
351 }
352
353 fn trigger_instrument_id(&self) -> Option<InstrumentId> {
354 self.trigger_instrument_id
355 }
356
357 fn contingency_type(&self) -> Option<ContingencyType> {
358 self.contingency_type
359 }
360
361 fn order_list_id(&self) -> Option<OrderListId> {
362 self.order_list_id
363 }
364
365 fn linked_order_ids(&self) -> Option<&[ClientOrderId]> {
366 self.linked_order_ids.as_deref()
367 }
368
369 fn parent_order_id(&self) -> Option<ClientOrderId> {
370 self.parent_order_id
371 }
372
373 fn exec_algorithm_id(&self) -> Option<ExecAlgorithmId> {
374 self.exec_algorithm_id
375 }
376
377 fn exec_algorithm_params(&self) -> Option<&IndexMap<Ustr, Ustr>> {
378 self.exec_algorithm_params.as_ref()
379 }
380
381 fn exec_spawn_id(&self) -> Option<ClientOrderId> {
382 self.exec_spawn_id
383 }
384
385 fn tags(&self) -> Option<&[Ustr]> {
386 self.tags.as_deref()
387 }
388
389 fn filled_qty(&self) -> Quantity {
390 self.filled_qty
391 }
392
393 fn leaves_qty(&self) -> Quantity {
394 self.leaves_qty
395 }
396
397 fn overfill_qty(&self) -> Quantity {
398 self.overfill_qty
399 }
400
401 fn avg_px(&self) -> Option<f64> {
402 self.avg_px
403 }
404
405 fn slippage(&self) -> Option<f64> {
406 self.slippage
407 }
408
409 fn init_id(&self) -> UUID4 {
410 self.init_id
411 }
412
413 fn ts_init(&self) -> UnixNanos {
414 self.ts_init
415 }
416
417 fn ts_submitted(&self) -> Option<UnixNanos> {
418 self.ts_submitted
419 }
420
421 fn ts_accepted(&self) -> Option<UnixNanos> {
422 self.ts_accepted
423 }
424
425 fn ts_closed(&self) -> Option<UnixNanos> {
426 self.ts_closed
427 }
428
429 fn ts_last(&self) -> UnixNanos {
430 self.ts_last
431 }
432
433 fn events(&self) -> Vec<&OrderEventAny> {
434 self.events.iter().collect()
435 }
436
437 fn venue_order_ids(&self) -> Vec<&VenueOrderId> {
438 self.venue_order_ids.iter().collect()
439 }
440
441 fn trade_ids(&self) -> Vec<&TradeId> {
442 self.trade_ids.iter().collect()
443 }
444
445 fn commissions(&self) -> &IndexMap<Currency, Money> {
446 &self.commissions
447 }
448
449 fn apply(&mut self, event: OrderEventAny) -> Result<(), OrderError> {
450 let is_order_filled = matches!(event, OrderEventAny::Filled(_));
451
452 self.core.apply(event.clone())?;
453
454 if let OrderEventAny::Updated(ref event) = event {
455 self.update(event);
456 }
457
458 if is_order_filled {
459 self.core.set_slippage(self.price);
460 }
461
462 Ok(())
463 }
464
465 fn update(&mut self, event: &OrderUpdated) {
466 assert!(
467 event.trigger_price.is_none(),
468 "{}",
469 OrderError::InvalidOrderEvent
470 );
471
472 if let Some(price) = event.price {
473 self.price = price;
474 }
475
476 self.quantity = event.quantity;
477 self.leaves_qty = self.quantity.saturating_sub(self.filled_qty);
478 }
479
480 fn is_triggered(&self) -> Option<bool> {
481 None
482 }
483
484 fn set_position_id(&mut self, position_id: Option<PositionId>) {
485 self.position_id = position_id;
486 }
487
488 fn set_quantity(&mut self, quantity: Quantity) {
489 self.quantity = quantity;
490 }
491
492 fn set_leaves_qty(&mut self, leaves_qty: Quantity) {
493 self.leaves_qty = leaves_qty;
494 }
495
496 fn set_emulation_trigger(&mut self, emulation_trigger: Option<TriggerType>) {
497 self.emulation_trigger = emulation_trigger;
498 }
499
500 fn set_is_quote_quantity(&mut self, is_quote_quantity: bool) {
501 self.is_quote_quantity = is_quote_quantity;
502 }
503
504 fn set_liquidity_side(&mut self, liquidity_side: LiquiditySide) {
505 self.liquidity_side = Some(liquidity_side);
506 }
507
508 fn would_reduce_only(&self, side: PositionSide, position_qty: Quantity) -> bool {
509 self.core.would_reduce_only(side, position_qty)
510 }
511
512 fn previous_status(&self) -> Option<OrderStatus> {
513 self.core.previous_status
514 }
515}
516
517impl Display for LimitOrder {
518 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
519 write!(
520 f,
521 "LimitOrder(\
522 {} {} {} {} @ {} {}, \
523 status={}, \
524 client_order_id={}, \
525 venue_order_id={}, \
526 position_id={}, \
527 exec_algorithm_id={}, \
528 exec_spawn_id={}, \
529 tags={:?}\
530 )",
531 self.side,
532 self.quantity.to_formatted_string(),
533 self.instrument_id,
534 self.order_type,
535 self.price,
536 self.time_in_force,
537 self.status,
538 self.client_order_id,
539 self.venue_order_id.map_or_else(
540 || "None".to_string(),
541 |venue_order_id| format!("{venue_order_id}")
542 ),
543 self.position_id.map_or_else(
544 || "None".to_string(),
545 |position_id| format!("{position_id}")
546 ),
547 self.exec_algorithm_id
548 .map_or_else(|| "None".to_string(), |id| format!("{id}")),
549 self.exec_spawn_id
550 .map_or_else(|| "None".to_string(), |id| format!("{id}")),
551 self.tags
552 )
553 }
554}
555
556impl From<OrderInitialized> for LimitOrder {
557 fn from(event: OrderInitialized) -> Self {
558 Self::new(
559 event.trader_id,
560 event.strategy_id,
561 event.instrument_id,
562 event.client_order_id,
563 event.order_side,
564 event.quantity,
565 event
566 .price .expect("Error initializing order: `price` was `None` for `LimitOrder"),
568 event.time_in_force,
569 event.expire_time,
570 event.post_only,
571 event.reduce_only,
572 event.quote_quantity,
573 event.display_qty,
574 event.emulation_trigger,
575 event.trigger_instrument_id,
576 event.contingency_type,
577 event.order_list_id,
578 event.linked_order_ids,
579 event.parent_order_id,
580 event.exec_algorithm_id,
581 event.exec_algorithm_params,
582 event.exec_spawn_id,
583 event.tags,
584 event.event_id,
585 event.ts_event,
586 )
587 }
588}
589
590#[cfg(test)]
591mod tests {
592 use nautilus_core::UnixNanos;
593 use rstest::rstest;
594
595 use crate::{
596 enums::{OrderSide, OrderType, TimeInForce},
597 events::{OrderEventAny, OrderUpdated},
598 identifiers::InstrumentId,
599 instruments::{CurrencyPair, stubs::*},
600 orders::{Order, OrderTestBuilder, stubs::TestOrderStubs},
601 types::{Price, Quantity},
602 };
603
604 #[rstest]
605 fn test_initialize(audusd_sim: CurrencyPair) {
606 let order = OrderTestBuilder::new(OrderType::Limit)
607 .instrument_id(audusd_sim.id)
608 .side(OrderSide::Buy)
609 .price(Price::from("0.68000"))
610 .quantity(Quantity::from(1))
611 .build();
612
613 assert_eq!(order.time_in_force(), TimeInForce::Gtc);
614
615 assert_eq!(order.filled_qty(), Quantity::from(0));
616 assert_eq!(order.leaves_qty(), Quantity::from(1));
617
618 assert_eq!(order.display_qty(), None);
619 assert_eq!(order.trigger_instrument_id(), None);
620 assert_eq!(order.order_list_id(), None);
621 }
622
623 #[rstest]
624 fn test_display(audusd_sim: CurrencyPair) {
625 let order = OrderTestBuilder::new(OrderType::Limit)
626 .instrument_id(audusd_sim.id)
627 .side(OrderSide::Buy)
628 .price(Price::from("1.00000"))
629 .quantity(Quantity::from(100_000))
630 .build();
631
632 assert_eq!(
633 order.to_string(),
634 "LimitOrder(BUY 100_000 AUD/USD.SIM LIMIT @ 1.00000 GTC, \
635 status=INITIALIZED, client_order_id=O-19700101-000000-001-001-1, \
636 venue_order_id=None, position_id=None, exec_algorithm_id=None, \
637 exec_spawn_id=None, tags=None)"
638 );
639 }
640
641 #[rstest]
642 #[should_panic(
643 expected = "Condition failed: invalid `Quantity` for 'quantity' not positive, was 0"
644 )]
645 fn test_positive_quantity_condition(audusd_sim: CurrencyPair) {
646 let _ = OrderTestBuilder::new(OrderType::Limit)
647 .instrument_id(audusd_sim.id)
648 .side(OrderSide::Buy)
649 .price(Price::from("0.8"))
650 .quantity(Quantity::from(0))
651 .build();
652 }
653
654 #[rstest]
655 #[should_panic(expected = "Condition failed: `expire_time` is required for `GTD` order")]
656 fn test_correct_expiration_with_time_in_force_gtd(audusd_sim: CurrencyPair) {
657 let _ = OrderTestBuilder::new(OrderType::Limit)
658 .instrument_id(audusd_sim.id)
659 .side(OrderSide::Buy)
660 .price(Price::from("0.8"))
661 .quantity(Quantity::from(1))
662 .time_in_force(TimeInForce::Gtd)
663 .build();
664 }
665
666 #[rstest]
667 fn test_limit_order_creation() {
668 let order = OrderTestBuilder::new(OrderType::Limit)
669 .instrument_id(InstrumentId::from("BTC-USDT.BINANCE"))
670 .quantity(Quantity::from(10))
671 .price(Price::new(100.0, 2))
672 .side(OrderSide::Buy)
673 .time_in_force(TimeInForce::Gtc)
674 .build();
675
676 assert_eq!(order.price(), Some(Price::new(100.0, 2)));
677 assert_eq!(order.quantity(), Quantity::from(10));
678 assert_eq!(order.time_in_force(), TimeInForce::Gtc);
679 assert_eq!(order.order_side(), OrderSide::Buy);
680 }
681
682 #[rstest]
683 fn test_limit_order_with_expire_time() {
684 let expire_time = UnixNanos::from(1_700_000_000_000_000);
685 let order = OrderTestBuilder::new(OrderType::Limit)
686 .instrument_id(InstrumentId::from("BTC-USDT.BINANCE"))
687 .quantity(Quantity::from(10))
688 .price(Price::new(100.0, 2))
689 .time_in_force(TimeInForce::Gtd)
690 .expire_time(expire_time)
691 .build();
692
693 assert_eq!(order.expire_time(), Some(expire_time));
694 assert_eq!(order.time_in_force(), TimeInForce::Gtd);
695 }
696
697 #[rstest]
698 #[should_panic(expected = "Condition failed: `expire_time` is required for `GTD` order")]
699 fn test_limit_order_missing_expire_time() {
700 let _ = OrderTestBuilder::new(OrderType::Limit)
701 .instrument_id(InstrumentId::from("BTC-USDT.BINANCE"))
702 .quantity(Quantity::from(10))
703 .price(Price::new(100.0, 2))
704 .time_in_force(TimeInForce::Gtd)
705 .build();
706 }
707
708 #[rstest]
709 fn test_limit_order_post_only() {
710 let order = OrderTestBuilder::new(OrderType::Limit)
711 .instrument_id(InstrumentId::from("BTC-USDT.BINANCE"))
712 .quantity(Quantity::from(10))
713 .price(Price::new(100.0, 2))
714 .post_only(true)
715 .build();
716
717 assert!(order.is_post_only());
718 }
719
720 #[rstest]
721 fn test_limit_order_display_quantity() {
722 let display_qty = Quantity::from(5);
723 let order = OrderTestBuilder::new(OrderType::Limit)
724 .instrument_id(InstrumentId::from("BTC-USDT.BINANCE"))
725 .quantity(Quantity::from(10))
726 .price(Price::new(100.0, 2))
727 .display_qty(display_qty)
728 .build();
729
730 assert_eq!(order.display_qty(), Some(display_qty));
731 }
732
733 #[rstest]
734 fn test_limit_order_update() {
735 let order = OrderTestBuilder::new(OrderType::Limit)
736 .instrument_id(InstrumentId::from("BTC-USDT.BINANCE"))
737 .quantity(Quantity::from(10))
738 .price(Price::new(100.0, 2))
739 .build();
740
741 let mut accepted_order = TestOrderStubs::make_accepted_order(&order);
742
743 let updated_price = Price::new(105.0, 2);
744 let updated_quantity = Quantity::from(5);
745
746 let event = OrderUpdated {
747 client_order_id: accepted_order.client_order_id(),
748 strategy_id: accepted_order.strategy_id(),
749 price: Some(updated_price),
750 quantity: updated_quantity,
751 ..Default::default()
752 };
753
754 accepted_order.apply(OrderEventAny::Updated(event)).unwrap();
755
756 assert_eq!(accepted_order.quantity(), updated_quantity);
757 assert_eq!(accepted_order.price(), Some(updated_price));
758 }
759
760 #[rstest]
761 fn test_limit_order_expire_time() {
762 let expire_time = UnixNanos::from(1_700_000_000_000_000);
763 let order = OrderTestBuilder::new(OrderType::Limit)
764 .instrument_id(InstrumentId::from("BTC-USDT.BINANCE"))
765 .quantity(Quantity::from(10))
766 .price(Price::new(100.0, 2))
767 .time_in_force(TimeInForce::Gtd)
768 .expire_time(expire_time)
769 .build();
770
771 assert_eq!(order.expire_time(), Some(expire_time));
772 }
773}