use crate::Hash;
use serde::{Deserialize, Serialize};
use std::cmp;
use std::fmt;
use std::hash;
use std::ops;
#[derive(Clone, Serialize, Deserialize)]
pub struct StaticString {
inner: String,
hash: Hash,
}
impl cmp::PartialEq for StaticString {
fn eq(&self, other: &Self) -> bool {
self.hash == other.hash && self.inner == other.inner
}
}
impl cmp::Eq for StaticString {}
impl cmp::PartialOrd for StaticString {
fn partial_cmp(&self, other: &Self) -> Option<cmp::Ordering> {
self.inner.partial_cmp(&other.inner)
}
}
impl cmp::Ord for StaticString {
fn cmp(&self, other: &Self) -> cmp::Ordering {
self.inner.cmp(&other.inner)
}
}
impl hash::Hash for StaticString {
fn hash<H: hash::Hasher>(&self, state: &mut H) {
self.hash.hash(state)
}
}
impl StaticString {
pub fn new<S>(s: S) -> Self
where
S: AsRef<str>,
{
let inner = s.as_ref().to_owned();
let hash = Hash::of(&inner);
Self { inner, hash }
}
pub fn hash(&self) -> Hash {
self.hash
}
}
impl AsRef<String> for StaticString {
fn as_ref(&self) -> &String {
&self.inner
}
}
impl fmt::Debug for StaticString {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{:?}", &self.inner)?;
Ok(())
}
}
impl ops::Deref for StaticString {
type Target = String;
fn deref(&self) -> &Self::Target {
&self.inner
}
}
impl From<String> for StaticString {
fn from(inner: String) -> Self {
let hash = Hash::of(inner.as_str());
Self { inner, hash }
}
}