nautilus_common/generators/
position_id.rsuse std::{cell::RefCell, collections::HashMap, rc::Rc};
use nautilus_model::identifiers::{PositionId, StrategyId, TraderId};
use super::get_datetime_tag;
use crate::clock::Clock;
#[repr(C)]
pub struct PositionIdGenerator {
clock: Rc<RefCell<dyn Clock>>,
trader_id: TraderId,
counts: HashMap<StrategyId, usize>,
}
impl PositionIdGenerator {
#[must_use]
pub fn new(trader_id: TraderId, clock: Rc<RefCell<dyn Clock>>) -> Self {
Self {
clock,
trader_id,
counts: HashMap::new(),
}
}
pub fn set_count(&mut self, count: usize, strategy_id: StrategyId) {
self.counts.insert(strategy_id, count);
}
pub fn reset(&mut self) {
self.counts.clear();
}
#[must_use]
pub fn count(&self, strategy_id: StrategyId) -> usize {
*self.counts.get(&strategy_id).unwrap_or(&0)
}
pub fn generate(&mut self, strategy_id: StrategyId, flipped: bool) -> PositionId {
let strategy = strategy_id;
let next_count = self.count(strategy_id) + 1;
self.set_count(next_count, strategy_id);
let datetime_tag = get_datetime_tag(self.clock.borrow().timestamp_ms());
let trader_tag = self.trader_id.get_tag();
let strategy_tag = strategy.get_tag();
let flipped = if flipped { "F" } else { "" };
let value = format!("P-{datetime_tag}-{trader_tag}-{strategy_tag}-{next_count}{flipped}");
PositionId::from(value)
}
}
#[cfg(test)]
mod tests {
use std::{cell::RefCell, rc::Rc};
use nautilus_model::identifiers::{PositionId, StrategyId, TraderId};
use rstest::rstest;
use crate::{clock::TestClock, generators::position_id::PositionIdGenerator};
fn get_position_id_generator() -> PositionIdGenerator {
PositionIdGenerator::new(TraderId::default(), Rc::new(RefCell::new(TestClock::new())))
}
#[rstest]
fn test_generate_position_id_one_strategy() {
let mut generator = get_position_id_generator();
let result1 = generator.generate(StrategyId::from("S-001"), false);
let result2 = generator.generate(StrategyId::from("S-001"), false);
assert_eq!(result1, PositionId::from("P-19700101-000000-001-001-1"));
assert_eq!(result2, PositionId::from("P-19700101-000000-001-001-2"));
}
#[rstest]
fn test_generate_position_id_multiple_strategies() {
let mut generator = get_position_id_generator();
let result1 = generator.generate(StrategyId::from("S-001"), false);
let result2 = generator.generate(StrategyId::from("S-002"), false);
let result3 = generator.generate(StrategyId::from("S-002"), false);
assert_eq!(result1, PositionId::from("P-19700101-000000-001-001-1"));
assert_eq!(result2, PositionId::from("P-19700101-000000-001-002-1"));
assert_eq!(result3, PositionId::from("P-19700101-000000-001-002-2"));
}
#[rstest]
fn test_generate_position_id_with_flipped_appends_correctly() {
let mut generator = get_position_id_generator();
let result1 = generator.generate(StrategyId::from("S-001"), false);
let result2 = generator.generate(StrategyId::from("S-002"), true);
let result3 = generator.generate(StrategyId::from("S-001"), true);
assert_eq!(result1, PositionId::from("P-19700101-000000-001-001-1"));
assert_eq!(result2, PositionId::from("P-19700101-000000-001-002-1F"));
assert_eq!(result3, PositionId::from("P-19700101-000000-001-001-2F"));
}
#[rstest]
fn test_get_count_when_strategy_id_has_not_been_used() {
let generator = get_position_id_generator();
let result = generator.count(StrategyId::from("S-001"));
assert_eq!(result, 0);
}
#[rstest]
fn set_count_with_valid_strategy() {
let mut generator = get_position_id_generator();
generator.set_count(7, StrategyId::from("S-001"));
let result = generator.count(StrategyId::from("S-001"));
assert_eq!(result, 7);
}
#[rstest]
fn test_reset() {
let mut generator = get_position_id_generator();
generator.generate(StrategyId::from("S-001"), false);
generator.generate(StrategyId::from("S-001"), false);
generator.reset();
let result = generator.generate(StrategyId::from("S-001"), false);
assert_eq!(result, PositionId::from("P-19700101-000000-001-001-1"));
}
}