nautilus_common/generators/
position_id.rs1use std::{cell::RefCell, collections::HashMap, fmt::Debug, rc::Rc};
17
18use nautilus_model::identifiers::{PositionId, StrategyId, TraderId};
19
20use super::get_datetime_tag;
21use crate::clock::Clock;
22
23#[repr(C)]
24pub struct PositionIdGenerator {
25 clock: Rc<RefCell<dyn Clock>>,
26 trader_id: TraderId,
27 counts: HashMap<StrategyId, usize>,
28}
29
30impl Debug for PositionIdGenerator {
31 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
32 f.debug_struct(stringify!(PositionIdGenerator))
33 .field("trader_id", &self.trader_id)
34 .field("counts", &self.counts)
35 .finish()
36 }
37}
38
39impl PositionIdGenerator {
40 #[must_use]
42 pub fn new(trader_id: TraderId, clock: Rc<RefCell<dyn Clock>>) -> Self {
43 Self {
44 clock,
45 trader_id,
46 counts: HashMap::new(),
47 }
48 }
49
50 pub fn set_count(&mut self, count: usize, strategy_id: StrategyId) {
51 self.counts.insert(strategy_id, count);
52 }
53
54 pub fn reset(&mut self) {
55 self.counts.clear();
56 }
57
58 #[must_use]
59 pub fn count(&self, strategy_id: StrategyId) -> usize {
60 *self.counts.get(&strategy_id).unwrap_or(&0)
61 }
62
63 pub fn generate(&mut self, strategy_id: StrategyId, flipped: bool) -> PositionId {
64 let strategy = strategy_id;
65 let next_count = self.count(strategy_id) + 1;
66 self.set_count(next_count, strategy_id);
67 let datetime_tag = get_datetime_tag(self.clock.borrow().timestamp_ms());
68 let trader_tag = self.trader_id.get_tag();
69 let strategy_tag = strategy.get_tag();
70 let flipped = if flipped { "F" } else { "" };
71 let value = format!("P-{datetime_tag}-{trader_tag}-{strategy_tag}-{next_count}{flipped}");
72 PositionId::from(value)
73 }
74}
75
76#[cfg(test)]
77mod tests {
78 use std::{cell::RefCell, rc::Rc};
79
80 use nautilus_model::{
81 identifiers::{PositionId, StrategyId, TraderId},
82 stubs::TestDefault,
83 };
84 use rstest::rstest;
85
86 use crate::{clock::TestClock, generators::position_id::PositionIdGenerator};
87
88 fn get_position_id_generator() -> PositionIdGenerator {
89 PositionIdGenerator::new(
90 TraderId::test_default(),
91 Rc::new(RefCell::new(TestClock::new())),
92 )
93 }
94
95 #[rstest]
96 fn test_generate_position_id_one_strategy() {
97 let mut generator = get_position_id_generator();
98 let result1 = generator.generate(StrategyId::from("S-001"), false);
99 let result2 = generator.generate(StrategyId::from("S-001"), false);
100
101 assert_eq!(result1, PositionId::from("P-19700101-000000-001-001-1"));
102 assert_eq!(result2, PositionId::from("P-19700101-000000-001-001-2"));
103 }
104
105 #[rstest]
106 fn test_generate_position_id_multiple_strategies() {
107 let mut generator = get_position_id_generator();
108 let result1 = generator.generate(StrategyId::from("S-001"), false);
109 let result2 = generator.generate(StrategyId::from("S-002"), false);
110 let result3 = generator.generate(StrategyId::from("S-002"), false);
111
112 assert_eq!(result1, PositionId::from("P-19700101-000000-001-001-1"));
113 assert_eq!(result2, PositionId::from("P-19700101-000000-001-002-1"));
114 assert_eq!(result3, PositionId::from("P-19700101-000000-001-002-2"));
115 }
116
117 #[rstest]
118 fn test_generate_position_id_with_flipped_appends_correctly() {
119 let mut generator = get_position_id_generator();
120 let result1 = generator.generate(StrategyId::from("S-001"), false);
121 let result2 = generator.generate(StrategyId::from("S-002"), true);
122 let result3 = generator.generate(StrategyId::from("S-001"), true);
123
124 assert_eq!(result1, PositionId::from("P-19700101-000000-001-001-1"));
125 assert_eq!(result2, PositionId::from("P-19700101-000000-001-002-1F"));
126 assert_eq!(result3, PositionId::from("P-19700101-000000-001-001-2F"));
127 }
128
129 #[rstest]
130 fn test_get_count_when_strategy_id_has_not_been_used() {
131 let generator = get_position_id_generator();
132 let result = generator.count(StrategyId::from("S-001"));
133
134 assert_eq!(result, 0);
135 }
136
137 #[rstest]
138 fn set_count_with_valid_strategy() {
139 let mut generator = get_position_id_generator();
140 generator.set_count(7, StrategyId::from("S-001"));
141 let result = generator.count(StrategyId::from("S-001"));
142
143 assert_eq!(result, 7);
144 }
145
146 #[rstest]
147 fn test_reset() {
148 let mut generator = get_position_id_generator();
149 generator.generate(StrategyId::from("S-001"), false);
150 generator.generate(StrategyId::from("S-001"), false);
151 generator.reset();
152 let result = generator.generate(StrategyId::from("S-001"), false);
153
154 assert_eq!(result, PositionId::from("P-19700101-000000-001-001-1"));
155 }
156}