use crate::ion_data::{IonEq, IonOrd};
use crate::result::decoding_error;
use crate::{IonResult, SymbolRef};
use std::borrow::Borrow;
use std::cmp::Ordering;
use std::fmt::{Display, Formatter};
use std::hash::{Hash, Hasher};
use std::sync::Arc;
#[derive(Debug, Eq)]
enum SymbolText {
Shared(Arc<str>),
Owned(String),
Unknown,
}
impl SymbolText {
fn text(&self) -> Option<&str> {
let text = match self {
SymbolText::Shared(s) => s.as_ref(),
SymbolText::Owned(s) => s.as_str(),
SymbolText::Unknown => return None,
};
Some(text)
}
}
impl Hash for SymbolText {
fn hash<H: Hasher>(&self, state: &mut H) {
match self {
SymbolText::Shared(text) => text.hash(state),
SymbolText::Owned(text) => text.hash(state),
SymbolText::Unknown => 0.hash(state),
}
}
}
impl Clone for SymbolText {
fn clone(&self) -> Self {
match self {
SymbolText::Owned(text) => SymbolText::Owned(text.to_owned()),
SymbolText::Shared(text) => SymbolText::Shared(Arc::clone(text)),
SymbolText::Unknown => SymbolText::Unknown,
}
}
}
impl PartialEq<Self> for SymbolText {
fn eq(&self, other: &Self) -> bool {
self.cmp(other) == Ordering::Equal
}
}
impl PartialOrd<Self> for SymbolText {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Some(self.cmp(other))
}
}
impl Ord for SymbolText {
fn cmp(&self, other: &Self) -> Ordering {
match (self.text(), other.text()) {
(Some(s1), Some(s2)) => s1.cmp(s2),
(Some(_), None) => Ordering::Greater,
(None, Some(_)) => Ordering::Less,
(None, None) => Ordering::Equal,
}
}
}
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Hash, Clone)]
pub struct Symbol {
text: SymbolText,
}
impl Symbol {
pub fn owned<I: Into<String>>(text: I) -> Symbol {
Symbol {
text: SymbolText::Owned(text.into()),
}
}
pub fn shared(text: Arc<str>) -> Symbol {
Symbol {
text: SymbolText::Shared(text),
}
}
pub fn unknown_text() -> Symbol {
Symbol {
text: SymbolText::Unknown,
}
}
pub(crate) fn into_shared(self) -> Symbol {
match self.text {
SymbolText::Shared(text) => Symbol::shared(text),
SymbolText::Owned(text) => Symbol::shared(text.into()),
SymbolText::Unknown => Symbol::unknown_text(),
}
}
pub fn text(&self) -> Option<&str> {
self.text.text()
}
pub fn text_or_error(&self) -> IonResult<&str> {
match self.text() {
Some(text) => Ok(text),
None => decoding_error("symbol has unknown text"),
}
}
}
impl IonEq for Symbol {
fn ion_eq(&self, other: &Self) -> bool {
self == other
}
}
impl IonOrd for Symbol {
fn ion_cmp(&self, other: &Self) -> Ordering {
self.cmp(other)
}
}
impl From<&str> for Symbol {
fn from(text: &str) -> Self {
Symbol::owned(text)
}
}
impl From<String> for Symbol {
fn from(text: String) -> Self {
Symbol::owned(text)
}
}
impl From<&String> for Symbol {
fn from(text: &String) -> Self {
text.as_str().into()
}
}
impl<'a> From<&'a Symbol> for Symbol {
fn from(text: &'a Symbol) -> Self {
text.clone()
}
}
impl<'a> From<SymbolRef<'a>> for Symbol {
fn from(symbol_ref: SymbolRef<'a>) -> Self {
symbol_ref.to_owned()
}
}
impl Display for Symbol {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
match self.text() {
None => write!(f, "$0"),
Some(text) => write!(f, "'{text}'"),
}
}
}
impl<A: AsRef<str>> PartialEq<A> for Symbol {
fn eq(&self, other: &A) -> bool {
self.text()
.map(|t| t == other.as_ref())
.unwrap_or(false)
}
}
impl PartialEq<Symbol> for String {
fn eq(&self, other: &Symbol) -> bool {
other.text().map(|t| t == self.as_str()).unwrap_or(false)
}
}
impl PartialEq<Symbol> for &str {
fn eq(&self, other: &Symbol) -> bool {
other.text().map(|t| &t == self).unwrap_or(false)
}
}
impl Borrow<str> for Symbol {
fn borrow(&self) -> &str {
self.text()
.expect("cannot borrow a &str from a Symbol with unknown text")
}
}
#[cfg(test)]
mod symbol_tests {
use super::*;
#[test]
fn ordering_and_eq() {
let mut symbols = vec![
Symbol::owned("foo"),
Symbol::shared(Arc::from("bar")),
Symbol::shared(Arc::from("baz")),
Symbol::owned("quux"),
];
symbols.as_mut_slice().sort();
let expected = vec![
Symbol::owned("bar"),
Symbol::owned("baz"),
Symbol::owned("foo"),
Symbol::owned("quux"),
];
assert_eq!(symbols, expected)
}
}