Skip to main content

nautilus_model/orders/
limit.rs

1// -------------------------------------------------------------------------------------------------
2//  Copyright (C) 2015-2026 Nautech Systems Pty Ltd. All rights reserved.
3//  https://nautechsystems.io
4//
5//  Licensed under the GNU Lesser General Public License Version 3.0 (the "License");
6//  You may not use this file except in compliance with the License.
7//  You may obtain a copy of the License at https://www.gnu.org/licenses/lgpl-3.0.en.html
8//
9//  Unless required by applicable law or agreed to in writing, software
10//  distributed under the License is distributed on an "AS IS" BASIS,
11//  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12//  See the License for the specific language governing permissions and
13//  limitations under the License.
14// -------------------------------------------------------------------------------------------------
15
16use 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    /// Creates a new [`LimitOrder`] instance.
62    ///
63    /// # Errors
64    ///
65    /// Returns an error if:
66    /// - The `quantity` is not positive.
67    /// - The `display_qty` (when provided) exceeds `quantity`.
68    /// - The `time_in_force` is GTD and the `expire_time` is `None` or zero.
69    #[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_event timestamp identical to ts_init
116            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    /// Creates a new [`LimitOrder`] instance.
148    ///
149    /// # Panics
150    ///
151    /// Panics if any order validation fails (see [`LimitOrder::new_checked`]).
152    #[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 // TODO: Improve this error, model order domain errors
567                .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}