use std::{
fmt::{Debug, Display},
hash::Hash,
};
use nautilus_core::correctness::{
CorrectnessResult, CorrectnessResultExt, FAILED, check_valid_string_utf8,
};
use ustr::Ustr;
#[repr(C)]
#[derive(Clone, Copy, Hash, PartialEq, Eq, PartialOrd, Ord)]
#[cfg_attr(
feature = "python",
pyo3::pyclass(module = "nautilus_trader.core.nautilus_pyo3.model", from_py_object)
)]
#[cfg_attr(
feature = "python",
pyo3_stub_gen::derive::gen_stub_pyclass(module = "nautilus_trader.model")
)]
pub struct Symbol(Ustr);
impl Symbol {
pub fn new_checked<T: AsRef<str>>(value: T) -> CorrectnessResult<Self> {
let value = value.as_ref();
check_valid_string_utf8(value, stringify!(value))?;
Ok(Self(Ustr::from(value)))
}
pub fn new<T: AsRef<str>>(value: T) -> Self {
Self::new_checked(value).expect_display(FAILED)
}
#[cfg_attr(not(feature = "python"), allow(dead_code))]
pub(crate) fn set_inner(&mut self, value: &str) {
self.0 = Ustr::from(value);
}
#[must_use]
pub fn from_str_unchecked<T: AsRef<str>>(s: T) -> Self {
Self(Ustr::from(s.as_ref()))
}
#[must_use]
pub const fn from_ustr_unchecked(s: Ustr) -> Self {
Self(s)
}
#[must_use]
pub fn inner(&self) -> Ustr {
self.0
}
#[must_use]
pub fn as_str(&self) -> &str {
self.0.as_str()
}
#[must_use]
pub fn is_composite(&self) -> bool {
self.as_str().contains('.')
}
#[must_use]
pub fn root(&self) -> &str {
let symbol_str = self.as_str();
if let Some(index) = symbol_str.find('.') {
&symbol_str[..index]
} else {
symbol_str
}
}
#[must_use]
pub fn topic(&self) -> String {
let root_str = self.root();
if root_str == self.as_str() {
root_str.to_string()
} else {
format!("{root_str}*")
}
}
}
impl Debug for Symbol {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "\"{}\"", self.0)
}
}
impl Display for Symbol {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.0)
}
}
impl From<Ustr> for Symbol {
fn from(value: Ustr) -> Self {
Self(value)
}
}
#[cfg(test)]
mod tests {
use nautilus_core::correctness::CorrectnessError;
use rstest::rstest;
use crate::identifiers::{Symbol, stubs::*};
#[rstest]
fn test_string_reprs(symbol_eth_perp: Symbol) {
assert_eq!(symbol_eth_perp.as_str(), "ETH-PERP");
assert_eq!(format!("{symbol_eth_perp}"), "ETH-PERP");
}
#[rstest]
#[case("AUDUSD", false)]
#[case("AUD/USD", false)]
#[case("CL.FUT", true)]
#[case("LO.OPT", true)]
#[case("ES.c.0", true)]
fn test_symbol_is_composite(#[case] input: &str, #[case] expected: bool) {
let symbol = Symbol::new(input);
assert_eq!(symbol.is_composite(), expected);
}
#[rstest]
#[case("AUDUSD", "AUDUSD")]
#[case("AUD/USD", "AUD/USD")]
#[case("CL.FUT", "CL")]
#[case("LO.OPT", "LO")]
#[case("ES.c.0", "ES")]
fn test_symbol_root(#[case] input: &str, #[case] expected_root: &str) {
let symbol = Symbol::new(input);
assert_eq!(symbol.root(), expected_root);
}
#[rstest]
#[case("AUDUSD", "AUDUSD")]
#[case("AUD/USD", "AUD/USD")]
#[case("CL.FUT", "CL*")]
#[case("LO.OPT", "LO*")]
#[case("ES.c.0", "ES*")]
fn test_symbol_topic(#[case] input: &str, #[case] expected_topic: &str) {
let symbol = Symbol::new(input);
assert_eq!(symbol.topic(), expected_topic);
}
#[rstest]
#[case("")] #[case(" ")] fn test_symbol_with_invalid_values(#[case] input: &str) {
assert!(Symbol::new_checked(input).is_err());
}
#[rstest]
fn test_symbol_new_checked_returns_typed_error_with_stable_display() {
let error = Symbol::new_checked("").unwrap_err();
assert_eq!(
error,
CorrectnessError::EmptyString {
param: "value".to_string(),
}
);
assert_eq!(error.to_string(), "invalid string for 'value', was empty");
}
#[rstest]
#[should_panic(expected = "Condition failed: invalid string for 'value', was empty")]
fn test_symbol_new_with_empty_string_panics_with_display_format() {
let _ = Symbol::new("");
}
#[rstest]
fn test_symbol_deserialize_json_with_unicode_escapes() {
let symbol: Symbol = serde_json::from_str(r#""\u9f99\u867eUSDT""#).unwrap();
assert_eq!(symbol.as_str(), "\u{9f99}\u{867e}USDT");
}
#[rstest]
fn test_symbol_deserialize_from_owned_value_with_non_ascii() {
let value = serde_json::Value::String("\u{9f99}\u{867e}USDT".to_string());
let symbol: Symbol = serde_json::from_value(value).unwrap();
assert_eq!(symbol.as_str(), "\u{9f99}\u{867e}USDT");
}
#[rstest]
fn test_symbol_serialization_roundtrip_non_ascii() {
let symbol = Symbol::new("\u{9f99}\u{867e}USDT");
let json = serde_json::to_string(&symbol).unwrap();
assert_eq!(json, "\"\u{9f99}\u{867e}USDT\"");
let deserialized: Symbol = serde_json::from_str(&json).unwrap();
assert_eq!(deserialized, symbol);
}
#[rstest]
fn test_symbol_deserialize_rejects_empty_string() {
let result: Result<Symbol, _> = serde_json::from_str(r#""""#);
assert!(result.is_err());
}
}