pub mod structural_variant;
pub use self::structural_variant::StructuralVariant;
use std::{error, fmt, str::FromStr};
#[derive(Clone, Debug, Eq, Hash, PartialEq)]
pub enum Symbol {
StructuralVariant(StructuralVariant),
NonstructuralVariant(String),
Unspecified,
}
impl fmt::Display for Symbol {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::StructuralVariant(sv) => write!(f, "{sv}"),
Self::NonstructuralVariant(nsv) => f.write_str(nsv),
Self::Unspecified => f.write_str("*"),
}
}
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub enum ParseError {
Empty,
InvalidNonstructuralVariant,
}
impl error::Error for ParseError {}
impl fmt::Display for ParseError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Empty => f.write_str("empty input"),
Self::InvalidNonstructuralVariant => f.write_str("invalid nonstructural variant"),
}
}
}
impl FromStr for Symbol {
type Err = ParseError;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"" => Err(ParseError::Empty),
"*" | "NON_REF" => Ok(Self::Unspecified),
_ => s
.parse::<StructuralVariant>()
.map(Self::StructuralVariant)
.or_else(|_| {
if is_valid_id(s) {
Ok(Self::NonstructuralVariant(s.into()))
} else {
Err(ParseError::InvalidNonstructuralVariant)
}
}),
}
}
}
fn is_valid_id_char(c: char) -> bool {
!c.is_ascii_whitespace() && !matches!(c, ',' | '<' | '>')
}
fn is_valid_id(s: &str) -> bool {
s.chars().all(is_valid_id_char)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_fmt() {
let symbol =
Symbol::StructuralVariant(StructuralVariant::from(structural_variant::Type::Deletion));
assert_eq!(symbol.to_string(), "DEL");
let symbol = Symbol::NonstructuralVariant(String::from("CN:0"));
assert_eq!(symbol.to_string(), "CN:0");
let symbol = Symbol::Unspecified;
assert_eq!(symbol.to_string(), "*");
}
#[test]
fn test_from_str() {
assert_eq!(
"DEL".parse(),
Ok(Symbol::StructuralVariant(StructuralVariant::from(
structural_variant::Type::Deletion
)))
);
assert_eq!(
"CN:0".parse(),
Ok(Symbol::NonstructuralVariant(String::from("CN:0")))
);
assert_eq!("NON_REF".parse(), Ok(Symbol::Unspecified));
assert_eq!("*".parse(), Ok(Symbol::Unspecified));
assert_eq!("".parse::<Symbol>(), Err(ParseError::Empty));
assert_eq!(
"CN 0".parse::<Symbol>(),
Err(ParseError::InvalidNonstructuralVariant)
);
assert_eq!(
"CN,0".parse::<Symbol>(),
Err(ParseError::InvalidNonstructuralVariant)
);
assert_eq!(
"CN>0".parse::<Symbol>(),
Err(ParseError::InvalidNonstructuralVariant)
);
assert_eq!(
"CN<0".parse::<Symbol>(),
Err(ParseError::InvalidNonstructuralVariant)
);
}
}