use std::collections::{HashMap, HashSet, hash_map::Entry};
use std::default::Default;
use std::fmt::{self, Display};
use chrono::Datelike;
use cusip::CUSIP;
use itertools::Itertools;
use isin::ISIN;
use log::debug;
use maybe_owned::MaybeOwned;
use serde::Deserialize;
use serde::de::Deserializer;
use crate::core::{GenericResult, EmptyResult};
use crate::exchanges::Exchanges;
use crate::localities::Jurisdiction;
use crate::time::Date;
#[derive(Clone, PartialEq, Eq, Hash, Debug)]
pub enum InstrumentId {
Symbol(String),
Isin(ISIN),
Name(String),
InternalId(String), }
impl Display for InstrumentId {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let id: &dyn Display = match self {
InstrumentId::Symbol(symbol) => symbol,
InstrumentId::Isin(isin) => isin,
InstrumentId::Name(name) => name,
InstrumentId::InternalId(id) => id,
};
id.fmt(f)
}
}
#[derive(Default, Clone)]
pub struct InstrumentInternalIds(HashMap<String, String>);
impl InstrumentInternalIds {
fn get_symbol(&self, id: &str) -> GenericResult<&str> {
Ok(self.0.get(id).ok_or_else(|| format!(concat!(
"Unable to determine stock symbol by its broker-specific internal ID ({}). ",
"Please specify the mapping via `instrument_internal_ids` configuration option"
), id))?)
}
pub fn deserialize<'de, D>(deserializer: D) -> Result<InstrumentInternalIds, D::Error>
where D: Deserializer<'de>
{
Ok(InstrumentInternalIds(Deserialize::deserialize(deserializer)?))
}
}
pub struct InstrumentInfo {
instruments: HashMap<String, Instrument>,
internal_ids: Option<InstrumentInternalIds>,
}
impl InstrumentInfo {
pub fn new() -> InstrumentInfo {
InstrumentInfo {
instruments: HashMap::new(),
internal_ids: None,
}
}
#[cfg(test)]
pub fn is_empty(&self) -> bool {
self.instruments.is_empty()
}
pub fn set_internal_ids(&mut self, ids: InstrumentInternalIds) {
self.internal_ids.replace(ids);
}
pub fn get_name(&self, symbol: &str) -> String {
if let Some(name) = self.instruments.get(symbol).and_then(|info| info.name.as_ref()) {
format!("{} ({})", name, symbol)
} else {
symbol.to_owned()
}
}
pub fn add(&mut self, symbol: &str) -> GenericResult<&mut Instrument> {
match self.instruments.entry(symbol.to_owned()) {
Entry::Vacant(entry) => Ok(entry.insert(Instrument::new(symbol))),
Entry::Occupied(_) => Err!("Duplicated security symbol: {}", symbol),
}
}
pub fn get(&self, symbol: &str) -> Option<&Instrument> {
self.instruments.get(symbol)
}
pub fn get_or_empty(&self, symbol: &str) -> MaybeOwned<Instrument> {
match self.instruments.get(symbol) {
Some(instrument) => MaybeOwned::Borrowed(instrument),
None => MaybeOwned::Owned(Instrument::new(symbol))
}
}
pub fn get_or_add(&mut self, symbol: &str) -> &mut Instrument {
match self.instruments.entry(symbol.to_owned()) {
Entry::Vacant(entry) => entry.insert(Instrument::new(symbol)),
Entry::Occupied(entry) => entry.into_mut(),
}
}
pub fn get_by_id(&self, instrument_id: &InstrumentId) -> GenericResult<&Instrument> {
Ok(match instrument_id {
InstrumentId::Symbol(symbol) => {
self.get(symbol).ok_or_else(|| format!(
"Unable to find instrument information by its symbol ({})", symbol))?
},
InstrumentId::Isin(isin) => {
let mut results = Vec::with_capacity(1);
for instrument in self.instruments.values() {
if instrument.isin.contains(isin) {
results.push(instrument);
}
}
match results.len() {
1 => results.first().unwrap(),
0 => return Err!("Unable to find instrument information by its ISIN ({})", isin),
_ => return Err!(
"Unable to map {} ISIN to instrument symbol: it maps into several symbols: {}",
isin, results.iter().map(|result| &result.symbol).join(", ")),
}
},
InstrumentId::Name(name) => {
let mut results = Vec::with_capacity(1);
for instrument in self.instruments.values() {
if matches!(instrument.name, Some(ref other) if other == name) {
results.push(instrument);
}
}
match results.len() {
1 => results.first().unwrap(),
0 => return Err!("Unable to find instrument information by its name ({})", name),
_ => return Err!(
"Unable to map {:?} to instrument symbol: it maps into several symbols: {}",
name, results.iter().map(|result| &result.symbol).join(", ")),
}
},
InstrumentId::InternalId(id) => {
let symbol = self.internal_ids.as_ref().unwrap().get_symbol(id)?.to_owned();
self.get_by_id(&InstrumentId::Symbol(symbol))?
},
})
}
pub fn get_or_add_by_id(&mut self, instrument_id: &InstrumentId) -> GenericResult<&Instrument> {
Ok(match instrument_id {
InstrumentId::Symbol(symbol) => {
self.get_or_add(symbol)
},
InstrumentId::InternalId(id) => {
let symbol = self.internal_ids.as_ref().unwrap().get_symbol(id)?.to_owned();
self.get_or_add(&symbol)
},
InstrumentId::Isin(..) | InstrumentId::Name(..) => {
self.get_by_id(instrument_id)?
},
})
}
pub fn remove(&mut self, symbol: &str) -> Option<Instrument> {
self.instruments.remove(symbol)
}
pub fn suggest_remapping(&self) -> Vec<(String, String)> {
let mut rules = Vec::new();
'symbol_loop: for symbol in self.instruments.keys() {
let Ok(isin) = parse_isin(symbol) else {
continue;
};
let mut real_symbol = None;
for instrument in self.instruments.values() {
if instrument.isin.contains(&isin) && parse_isin(&instrument.symbol).is_err() {
if let Some(other_symbol) = real_symbol.replace(instrument.symbol.clone()) {
debug!(concat!(
"Do not provide {isin} -> {other_symbol} automatic symbol remapping: ",
"{current_symbol} also points to {isin} ISIN"
), isin=isin, other_symbol=other_symbol, current_symbol=instrument.symbol);
continue 'symbol_loop;
}
}
}
if let Some(real_symbol) = real_symbol {
debug!("Got automatic symbol remapping rule: {symbol} -> {real_symbol}.");
rules.push((symbol.clone(), real_symbol));
}
}
rules
}
pub fn remap(&mut self, old_symbol: &str, new_symbol: &str) -> EmptyResult {
let Some(mut old_info) = self.instruments.remove(old_symbol) else {
return Ok(());
};
match self.instruments.entry(new_symbol.to_owned()) {
Entry::Occupied(mut entry) => {
let new_info = entry.get_mut();
match parse_isin(old_symbol) {
Ok(isin) if new_info.isin.contains(&isin) => {
new_info.merge(old_info, false)
},
_ => {
self.instruments.insert(old_symbol.to_owned(), old_info);
return Err!("The portfolio already has {} symbol", new_symbol);
}
}
},
Entry::Vacant(entry) => {
new_symbol.clone_into(&mut old_info.symbol);
entry.insert(old_info);
},
}
Ok(())
}
pub fn merge(&mut self, other: InstrumentInfo) {
assert!(other.internal_ids.is_none());
for (symbol, info) in other.instruments {
match self.instruments.entry(symbol) {
Entry::Occupied(mut entry) => {
entry.get_mut().merge(info, true);
},
Entry::Vacant(entry) => {
entry.insert(info);
},
};
}
}
}
pub struct Instrument {
pub symbol: String,
name: Option<String>,
pub isin: HashSet<ISIN>,
cusip: HashSet<CUSIP>,
pub exchanges: Exchanges,
}
impl Instrument {
fn new(symbol: &str) -> Instrument {
Instrument {
symbol: symbol.to_owned(),
name: None,
isin: HashSet::new(),
cusip: HashSet::new(),
exchanges: Exchanges::new_empty(),
}
}
pub fn set_name(&mut self, name: &str) {
self.name.replace(name.to_owned());
}
pub fn add_isin(&mut self, isin: ISIN) {
self.isin.insert(isin);
}
pub fn add_cusip(&mut self, cusip: CUSIP) {
self.cusip.insert(cusip);
}
pub fn get_taxation_type(&self, date: Date, broker_jurisdiction: Jurisdiction) -> GenericResult<IssuerTaxationType> {
let russian_country_code = Jurisdiction::Russia.traits().code;
let russian_brokers_are_full_tax_agents = date.year() >= 2024;
let get_taxation_type = |issuer_country_code: &str| -> IssuerTaxationType {
if broker_jurisdiction == Jurisdiction::Russia && (
russian_brokers_are_full_tax_agents || issuer_country_code == russian_country_code
) {
return IssuerTaxationType::TaxAgent {
auto_detected: !russian_brokers_are_full_tax_agents,
foreign: issuer_country_code != russian_country_code,
};
}
IssuerTaxationType::Manual {
country_code: Some(issuer_country_code.to_owned()),
}
};
let mut result_taxation_type = if self.cusip.is_empty() {
None
} else {
Some(get_taxation_type(Jurisdiction::Usa.traits().code))
};
for isin in &self.isin {
let taxation_type = get_taxation_type(isin.prefix());
if let Some(prev) = result_taxation_type.as_ref() {
if *prev != taxation_type {
let ids = self.isin.iter().map(ToString::to_string)
.chain(self.cusip.iter().map(|id| format!("CUSIP:{}", id)))
.join(", ");
return Err!(
"Unable to determine {} taxation type: it has several ISIN with different jurisdictions: {}",
self.symbol, ids)
}
} else {
result_taxation_type.replace(taxation_type);
}
}
Ok(if let Some(taxation_type) = result_taxation_type {
taxation_type
} else if broker_jurisdiction == Jurisdiction::Russia {
return Err!(
"Unable to determine {} taxation type: there is no ISIN information for it in the broker statement",
self.symbol);
} else {
IssuerTaxationType::Manual {
country_code: None,
}
})
}
pub fn merge(&mut self, other: Instrument, newer: bool) {
if let Some(name) = other.name {
if self.name.is_none() || newer {
self.name.replace(name);
}
}
self.isin.extend(other.isin);
self.cusip.extend(other.cusip);
self.exchanges.merge(other.exchanges);
}
}
#[derive(Clone, PartialEq)]
pub enum IssuerTaxationType {
Manual {country_code: Option<String>},
TaxAgent {
foreign: bool,
auto_detected: bool,
},
}
pub const ISIN_REGEX: &str = r"[A-Z]{2}[A-Z0-9]{9}[0-9]";
pub fn parse_isin(value: &str) -> GenericResult<ISIN> {
Ok(value.parse().map_err(|_| format!("Invalid ISIN: {}", value))?)
}