1use indexmap::IndexMap;
17use nautilus_core::{
18 UUID4, UnixNanos,
19 python::{IntoPyObjectNautilusExt, to_pyruntime_err},
20};
21use pyo3::{basic::CompareOp, prelude::*, types::PyDict};
22use ustr::Ustr;
23
24use crate::{
25 enums::{ContingencyType, OrderSide, OrderStatus, OrderType, TimeInForce, TriggerType},
26 events::order::initialized::OrderInitialized,
27 identifiers::{
28 ClientOrderId, ExecAlgorithmId, InstrumentId, OrderListId, StrategyId, TraderId,
29 },
30 orders::{
31 StopLimitOrder,
32 base::{Order, str_indexmap_to_ustr},
33 },
34 python::{
35 common::commissions_from_indexmap,
36 events::order::{order_event_to_pyobject, pyobject_to_order_event},
37 },
38 types::{Price, Quantity},
39};
40
41#[pymethods]
42impl StopLimitOrder {
43 #[new]
44 #[allow(clippy::too_many_arguments)]
45 #[pyo3(signature = (trader_id, strategy_id, instrument_id, client_order_id, order_side, quantity, price, trigger_price, trigger_type, time_in_force, post_only, reduce_only, quote_quantity, init_id, ts_init, expire_time=None, display_qty=None, emulation_trigger=None, trigger_instrument_id=None, contingency_type=None, order_list_id=None, linked_order_ids=None, parent_order_id=None, exec_algorithm_id=None, exec_algorithm_params=None, exec_spawn_id=None, tags=None))]
46 fn py_new(
47 trader_id: TraderId,
48 strategy_id: StrategyId,
49 instrument_id: InstrumentId,
50 client_order_id: ClientOrderId,
51 order_side: OrderSide,
52 quantity: Quantity,
53 price: Price,
54 trigger_price: Price,
55 trigger_type: TriggerType,
56 time_in_force: TimeInForce,
57 post_only: bool,
58 reduce_only: bool,
59 quote_quantity: bool,
60 init_id: UUID4,
61 ts_init: u64,
62 expire_time: Option<u64>,
63 display_qty: Option<Quantity>,
64 emulation_trigger: Option<TriggerType>,
65 trigger_instrument_id: Option<InstrumentId>,
66 contingency_type: Option<ContingencyType>,
67 order_list_id: Option<OrderListId>,
68 linked_order_ids: Option<Vec<ClientOrderId>>,
69 parent_order_id: Option<ClientOrderId>,
70 exec_algorithm_id: Option<ExecAlgorithmId>,
71 exec_algorithm_params: Option<IndexMap<String, String>>,
72 exec_spawn_id: Option<ClientOrderId>,
73 tags: Option<Vec<String>>,
74 ) -> Self {
75 let exec_algorithm_params = exec_algorithm_params.map(str_indexmap_to_ustr);
76 Self::new(
77 trader_id,
78 strategy_id,
79 instrument_id,
80 client_order_id,
81 order_side,
82 quantity,
83 price,
84 trigger_price,
85 trigger_type,
86 time_in_force,
87 expire_time.map(std::convert::Into::into),
88 post_only,
89 reduce_only,
90 quote_quantity,
91 display_qty,
92 emulation_trigger,
93 trigger_instrument_id,
94 contingency_type,
95 order_list_id,
96 linked_order_ids,
97 parent_order_id,
98 exec_algorithm_id,
99 exec_algorithm_params,
100 exec_spawn_id,
101 tags.map(|vec| vec.into_iter().map(|s| Ustr::from(s.as_str())).collect()),
102 init_id,
103 ts_init.into(),
104 )
105 }
106
107 fn __richcmp__(&self, other: &Self, op: CompareOp, py: Python<'_>) -> Py<PyAny> {
108 match op {
109 CompareOp::Eq => self.eq(other).into_py_any_unwrap(py),
110 CompareOp::Ne => self.ne(other).into_py_any_unwrap(py),
111 _ => py.NotImplemented(),
112 }
113 }
114
115 fn __repr__(&self) -> String {
116 self.to_string()
117 }
118
119 fn __str__(&self) -> String {
120 self.to_string()
121 }
122
123 #[staticmethod]
124 #[pyo3(name = "create")]
125 fn py_create(init: OrderInitialized) -> PyResult<Self> {
126 Ok(StopLimitOrder::from(init))
127 }
128
129 #[getter]
130 #[pyo3(name = "trader_id")]
131 fn py_trader_id(&self) -> TraderId {
132 self.trader_id
133 }
134
135 #[getter]
136 #[pyo3(name = "strategy_id")]
137 fn py_strategy_id(&self) -> StrategyId {
138 self.strategy_id
139 }
140
141 #[getter]
142 #[pyo3(name = "instrument_id")]
143 fn py_instrument_id(&self) -> InstrumentId {
144 self.instrument_id
145 }
146
147 #[getter]
148 #[pyo3(name = "client_order_id")]
149 fn py_client_order_id(&self) -> ClientOrderId {
150 self.client_order_id
151 }
152
153 #[getter]
154 #[pyo3(name = "side")]
155 fn py_order_side(&self) -> OrderSide {
156 self.side
157 }
158
159 #[getter]
160 #[pyo3(name = "quantity")]
161 fn py_quantity(&self) -> Quantity {
162 self.quantity
163 }
164
165 #[getter]
166 #[pyo3(name = "price")]
167 fn py_price(&self) -> Price {
168 self.price
169 }
170
171 #[getter]
172 #[pyo3(name = "trigger_price")]
173 fn py_trigger_price(&self) -> Price {
174 self.trigger_price
175 }
176
177 #[getter]
178 #[pyo3(name = "trigger_type")]
179 fn py_trigger_type(&self) -> TriggerType {
180 self.trigger_type
181 }
182
183 #[getter]
184 #[pyo3(name = "order_type")]
185 fn py_order_type(&self) -> OrderType {
186 self.order_type
187 }
188
189 #[getter]
190 #[pyo3(name = "time_in_force")]
191 fn py_time_in_force(&self) -> TimeInForce {
192 self.time_in_force
193 }
194
195 #[getter]
196 #[pyo3(name = "expire_time")]
197 fn py_expire_time(&self) -> Option<u64> {
198 self.expire_time.map(std::convert::Into::into)
199 }
200
201 #[getter]
202 #[pyo3(name = "status")]
203 fn py_order_status(&self) -> OrderStatus {
204 self.status
205 }
206
207 #[getter]
208 #[pyo3(name = "init_id")]
209 fn py_init_id(&self) -> UUID4 {
210 self.init_id
211 }
212
213 #[getter]
214 #[pyo3(name = "ts_init")]
215 fn py_ts_init(&self) -> u64 {
216 self.ts_init.as_u64()
217 }
218
219 #[getter]
220 #[pyo3(name = "init_event")]
221 fn py_init_event(&self, py: Python<'_>) -> PyResult<PyObject> {
222 match self.init_event() {
223 Some(event) => order_event_to_pyobject(py, event),
224 None => Ok(py.None()),
225 }
226 }
227
228 #[getter]
229 #[pyo3(name = "has_price")]
230 fn py_has_price(&self) -> bool {
231 true
232 }
233
234 #[getter]
235 #[pyo3(name = "is_passive")]
236 fn py_is_passive(&self) -> bool {
237 self.is_passive()
238 }
239
240 #[getter]
241 #[pyo3(name = "is_aggressive")]
242 fn py_is_aggressive(&self) -> bool {
243 self.is_aggressive()
244 }
245
246 #[getter]
247 #[pyo3(name = "is_closed")]
248 fn py_is_closed(&self) -> bool {
249 self.is_closed()
250 }
251
252 #[getter]
253 #[pyo3(name = "is_open")]
254 fn py_is_open(&self) -> bool {
255 self.is_open()
256 }
257
258 #[getter]
259 #[pyo3(name = "has_trigger_price")]
260 fn py_has_trigger_price(&self) -> bool {
261 true
262 }
263
264 #[getter]
265 #[pyo3(name = "is_post_only")]
266 fn py_post_only(&self) -> bool {
267 self.is_post_only
268 }
269
270 #[getter]
271 #[pyo3(name = "is_reduce_only")]
272 fn py_reduce_only(&self) -> bool {
273 self.is_reduce_only
274 }
275
276 #[getter]
277 #[pyo3(name = "is_quote_quantity")]
278 fn py_quote_quantity(&self) -> bool {
279 self.is_quote_quantity
280 }
281
282 #[getter]
283 #[pyo3(name = "display_qty")]
284 fn py_display_qty(&self) -> Option<Quantity> {
285 self.display_qty
286 }
287
288 #[getter]
289 #[pyo3(name = "emulation_trigger")]
290 fn py_emulation_trigger(&self) -> Option<TriggerType> {
291 self.emulation_trigger
292 }
293
294 #[getter]
295 #[pyo3(name = "trigger_instrument_id")]
296 fn py_trigger_instrument_id(&self) -> Option<InstrumentId> {
297 self.trigger_instrument_id
298 }
299
300 #[getter]
301 #[pyo3(name = "contingency_type")]
302 fn py_contingency_type(&self) -> Option<ContingencyType> {
303 self.contingency_type
304 }
305
306 #[getter]
307 #[pyo3(name = "order_list_id")]
308 fn py_order_list_id(&self) -> Option<OrderListId> {
309 self.order_list_id
310 }
311
312 #[getter]
313 #[pyo3(name = "linked_order_ids")]
314 fn py_linked_order_ids(&self) -> Option<Vec<ClientOrderId>> {
315 self.linked_order_ids.clone()
316 }
317
318 #[getter]
319 #[pyo3(name = "parent_order_id")]
320 fn py_parent_order_id(&self) -> Option<ClientOrderId> {
321 self.parent_order_id
322 }
323
324 #[getter]
325 #[pyo3(name = "exec_algorithm_id")]
326 fn py_exec_algorithm_id(&self) -> Option<ExecAlgorithmId> {
327 self.exec_algorithm_id
328 }
329
330 #[getter]
331 #[pyo3(name = "exec_algorithm_params")]
332 fn py_exec_algorithm_params(&self) -> Option<IndexMap<&str, &str>> {
333 self.exec_algorithm_params
334 .as_ref()
335 .map(|x| x.iter().map(|(k, v)| (k.as_str(), v.as_str())).collect())
336 }
337
338 #[getter]
339 #[pyo3(name = "exec_spawn_id")]
340 fn py_exec_spawn_id(&self) -> Option<ClientOrderId> {
341 self.exec_spawn_id
342 }
343
344 #[getter]
345 #[pyo3(name = "tags")]
346 fn py_tags(&self) -> Option<Vec<&str>> {
347 self.tags
348 .as_ref()
349 .map(|vec| vec.iter().map(|s| s.as_str()).collect())
350 }
351
352 #[getter]
353 #[pyo3(name = "events")]
354 fn py_events(&self, py: Python<'_>) -> PyResult<Vec<PyObject>> {
355 self.events()
356 .into_iter()
357 .map(|event| order_event_to_pyobject(py, event.clone()))
358 .collect()
359 }
360
361 #[pyo3(name = "to_dict")]
362 fn to_dict(&self, py: Python<'_>) -> PyResult<PyObject> {
363 let dict = PyDict::new(py);
364 dict.set_item("trader_id", self.trader_id.to_string())?;
365 dict.set_item("strategy_id", self.strategy_id.to_string())?;
366 dict.set_item("instrument_id", self.instrument_id.to_string())?;
367 dict.set_item("client_order_id", self.client_order_id.to_string())?;
368 dict.set_item("side", self.side.to_string())?;
369 dict.set_item("type", self.order_type.to_string())?;
370 dict.set_item("side", self.side.to_string())?;
371 dict.set_item("quantity", self.quantity.to_string())?;
372 dict.set_item("status", self.status.to_string())?;
373 dict.set_item("price", self.price.to_string())?;
374 dict.set_item("trigger_price", self.trigger_price.to_string())?;
375 dict.set_item("trigger_type", self.trigger_type.to_string())?;
376 dict.set_item("filled_qty", self.filled_qty.to_string())?;
377 dict.set_item("time_in_force", self.time_in_force.to_string())?;
378 dict.set_item("is_post_only", self.is_post_only)?;
379 dict.set_item("is_reduce_only", self.is_reduce_only)?;
380 dict.set_item("is_quote_quantity", self.is_quote_quantity)?;
381 dict.set_item("init_id", self.init_id.to_string())?;
382 dict.set_item(
383 "expire_time_ns",
384 self.expire_time.filter(|&t| t != 0).map(|t| t.as_u64()),
385 )?;
386 dict.set_item("ts_init", self.ts_init.as_u64())?;
387 dict.set_item("ts_last", self.ts_last.as_u64())?;
388 dict.set_item(
389 "commissions",
390 commissions_from_indexmap(py, self.commissions())?,
391 )?;
392 self.last_trade_id.map_or_else(
393 || dict.set_item("last_trade_id", py.None()),
394 |x| dict.set_item("last_trade_id", x.to_string()),
395 )?;
396 self.avg_px.map_or_else(
397 || dict.set_item("avg_px", py.None()),
398 |x| dict.set_item("avg_px", x),
399 )?;
400 self.position_id.map_or_else(
401 || dict.set_item("position_id", py.None()),
402 |x| dict.set_item("position_id", x.to_string()),
403 )?;
404 self.liquidity_side.map_or_else(
405 || dict.set_item("liquidity_side", py.None()),
406 |x| dict.set_item("liquidity_side", x.to_string()),
407 )?;
408 self.slippage.map_or_else(
409 || dict.set_item("slippage", py.None()),
410 |x| dict.set_item("slippage", x),
411 )?;
412 self.account_id.map_or_else(
413 || dict.set_item("account_id", py.None()),
414 |x| dict.set_item("account_id", x.to_string()),
415 )?;
416 self.venue_order_id.map_or_else(
417 || dict.set_item("venue_order_id", py.None()),
418 |x| dict.set_item("venue_order_id", x.to_string()),
419 )?;
420 self.display_qty.map_or_else(
421 || dict.set_item("display_qty", py.None()),
422 |x| dict.set_item("display_qty", x.to_string()),
423 )?;
424 self.emulation_trigger.map_or_else(
425 || dict.set_item("emulation_trigger", py.None()),
426 |x| dict.set_item("emulation_trigger", x.to_string()),
427 )?;
428 dict.set_item("trigger_instrument_id", self.trigger_instrument_id)?;
429 self.contingency_type.map_or_else(
430 || dict.set_item("contingency_type", py.None()),
431 |x| dict.set_item("contingency_type", x.to_string()),
432 )?;
433 self.order_list_id.map_or_else(
434 || dict.set_item("order_list_id", py.None()),
435 |x| dict.set_item("order_list_id", x.to_string()),
436 )?;
437 dict.set_item(
438 "linked_order_ids",
439 self.linked_order_ids.as_ref().map(|x| {
440 x.iter()
441 .map(std::string::ToString::to_string)
442 .collect::<Vec<String>>()
443 }),
444 )?;
445 self.parent_order_id.map_or_else(
446 || dict.set_item("parent_order_id", py.None()),
447 |x| dict.set_item("parent_order_id", x.to_string()),
448 )?;
449 self.exec_algorithm_id.map_or_else(
450 || dict.set_item("exec_algorithm_id", py.None()),
451 |x| dict.set_item("exec_algorithm_id", x.to_string()),
452 )?;
453 dict.set_item(
454 "exec_algorithm_params",
455 self.exec_algorithm_params.as_ref().map(|x| {
456 x.iter()
457 .map(|(k, v)| (k.to_string(), v.to_string()))
458 .collect::<IndexMap<String, String>>()
459 }),
460 )?;
461 self.exec_spawn_id.map_or_else(
462 || dict.set_item("exec_spawn_id", py.None()),
463 |x| dict.set_item("exec_spawn_id", x.to_string()),
464 )?;
465 dict.set_item(
466 "tags",
467 self.tags
468 .as_ref()
469 .map(|vec| vec.iter().map(|s| s.to_string()).collect::<Vec<String>>()),
470 )?;
471 Ok(dict.into())
472 }
473
474 #[staticmethod]
475 #[pyo3(name = "from_dict")]
476 fn py_from_dict(values: &Bound<'_, PyDict>) -> PyResult<Self> {
477 let dict = values.as_ref();
478 let trader_id = TraderId::from(dict.get_item("trader_id")?.extract::<&str>()?);
479 let strategy_id = StrategyId::from(dict.get_item("strategy_id")?.extract::<&str>()?);
480 let instrument_id = InstrumentId::from(dict.get_item("instrument_id")?.extract::<&str>()?);
481 let client_order_id =
482 ClientOrderId::from(dict.get_item("client_order_id")?.extract::<&str>()?);
483 let order_side = dict
484 .get_item("side")?
485 .extract::<&str>()?
486 .parse::<OrderSide>()
487 .unwrap();
488 let quantity = Quantity::from(dict.get_item("quantity")?.extract::<&str>()?);
489 let price = Price::from(dict.get_item("price")?.extract::<&str>()?);
490 let trigger_price = Price::from(dict.get_item("trigger_price")?.extract::<&str>()?);
491 let trigger_type = dict
492 .get_item("trigger_type")?
493 .extract::<&str>()?
494 .parse::<TriggerType>()
495 .unwrap();
496 let time_in_force = dict
497 .get_item("time_in_force")?
498 .extract::<&str>()?
499 .parse::<TimeInForce>()
500 .unwrap();
501 let post_only = dict.get_item("is_post_only")?.extract::<bool>()?;
502 let reduce_only = dict.get_item("is_reduce_only")?.extract::<bool>()?;
503 let quote_quantity = dict.get_item("is_quote_quantity")?.extract::<bool>()?;
504 let expire_time = dict
505 .get_item("expire_time_ns")
506 .map(|x| {
507 let extracted = x.extract::<u64>();
508 match extracted {
509 Ok(item) => Some(UnixNanos::from(item)),
510 Err(_) => None,
511 }
512 })
513 .unwrap();
514 let display_quantity = dict
515 .get_item("display_qty")
516 .map(|x| x.extract::<Quantity>().ok())
517 .unwrap();
518 let emulation_trigger = dict
519 .get_item("emulation_trigger")
520 .map(|x| x.extract::<&str>().unwrap().parse::<TriggerType>().ok())
521 .unwrap();
522 let trigger_instrument_id = dict
523 .get_item("trigger_instrument_id")
524 .map(|x| {
525 let extracted = x.extract::<&str>();
526 match extracted {
527 Ok(item) => Some(item.parse::<InstrumentId>().unwrap()),
528 Err(_) => None,
529 }
530 })
531 .unwrap();
532 let contingency_type = dict
533 .get_item("contingency_type")
534 .map(|x| {
535 let extracted = x.extract::<&str>();
536 match extracted {
537 Ok(item) => Some(item.parse::<ContingencyType>().unwrap()),
538 Err(_) => None,
539 }
540 })
541 .unwrap();
542 let order_list_id = dict
543 .get_item("order_list_id")
544 .map(|x| {
545 let extracted = x.extract::<&str>();
546 match extracted {
547 Ok(item) => Some(OrderListId::from(item)),
548 Err(_) => None,
549 }
550 })
551 .unwrap();
552 let linked_order_ids = dict.get_item("linked_order_ids").map(|x| {
553 let extracted_str = x.extract::<Vec<String>>();
554 match extracted_str {
555 Ok(item) => Some(
556 item.iter()
557 .map(|x| ClientOrderId::from(x.as_str()))
558 .collect(),
559 ),
560 Err(_) => None,
561 }
562 })?;
563 let parent_order_id = dict
564 .get_item("parent_order_id")
565 .map(|x| {
566 let extracted = x.extract::<&str>();
567 match extracted {
568 Ok(item) => Some(ClientOrderId::from(item)),
569 Err(_) => None,
570 }
571 })
572 .unwrap();
573 let exec_algorithm_id = dict
574 .get_item("exec_algorithm_id")
575 .map(|x| {
576 let extracted = x.extract::<&str>();
577 match extracted {
578 Ok(item) => Some(ExecAlgorithmId::from(item)),
579 Err(_) => None,
580 }
581 })
582 .unwrap();
583 let exec_algorithm_params = dict.get_item("exec_algorithm_params").map(|x| {
584 let extracted_str = x.extract::<IndexMap<String, String>>();
585 match extracted_str {
586 Ok(item) => Some(str_indexmap_to_ustr(item)),
587 Err(_) => None,
588 }
589 })?;
590 let exec_spawn_id = dict
591 .get_item("exec_spawn_id")
592 .map(|x| {
593 let extracted = x.extract::<&str>();
594 match extracted {
595 Ok(item) => Some(ClientOrderId::from(item)),
596 Err(_) => None,
597 }
598 })
599 .unwrap();
600 let tags = dict.get_item("tags").map(|x| {
601 let extracted_str = x.extract::<Vec<String>>();
602 match extracted_str {
603 Ok(item) => Some(item.iter().map(|s| Ustr::from(s)).collect()),
604 Err(_) => None,
605 }
606 })?;
607 let init_id = dict
608 .get_item("init_id")
609 .map(|x| x.extract::<&str>().unwrap().parse::<UUID4>().ok())?
610 .unwrap();
611 let ts_init = dict.get_item("ts_init")?.extract::<u64>()?;
612 let stop_limit_order = Self::new(
613 trader_id,
614 strategy_id,
615 instrument_id,
616 client_order_id,
617 order_side,
618 quantity,
619 price,
620 trigger_price,
621 trigger_type,
622 time_in_force,
623 expire_time,
624 post_only,
625 reduce_only,
626 quote_quantity,
627 display_quantity,
628 emulation_trigger,
629 trigger_instrument_id,
630 contingency_type,
631 order_list_id,
632 linked_order_ids,
633 parent_order_id,
634 exec_algorithm_id,
635 exec_algorithm_params,
636 exec_spawn_id,
637 tags,
638 init_id,
639 ts_init.into(),
640 );
641 Ok(stop_limit_order)
642 }
643
644 #[pyo3(name = "apply")]
645 fn py_apply(&mut self, event: PyObject, py: Python<'_>) -> PyResult<()> {
646 let event_any = pyobject_to_order_event(py, event).unwrap();
647 self.apply(event_any).map(|_| ()).map_err(to_pyruntime_err)
648 }
649}