blob: 6a1b4b89acc50792137f8268cc0c99f7ff337cbb [file] [log] [blame]
use std::{borrow::Cow, fmt};
use crate::KStringBase;
use crate::KStringRef;
use crate::KStringRefInner;
type StdString = std::string::String;
type BoxedStr = Box<str>;
/// A reference to a UTF-8 encoded, immutable string.
pub type KStringCow<'s> = KStringCowBase<'s, crate::backend::DefaultStr>;
/// A reference to a UTF-8 encoded, immutable string.
#[derive(Clone)]
#[repr(transparent)]
pub struct KStringCowBase<'s, B = crate::backend::DefaultStr> {
pub(crate) inner: KStringCowInner<'s, B>,
}
#[derive(Clone)]
pub(crate) enum KStringCowInner<'s, B> {
Borrowed(&'s str),
Owned(KStringBase<B>),
}
impl<'s, B> KStringCowBase<'s, B> {
/// Create a new empty `KStringCowBase`.
#[inline]
#[must_use]
pub const fn new() -> Self {
Self::from_static("")
}
/// Create a reference to a `'static` data.
#[inline]
#[must_use]
pub const fn from_static(other: &'static str) -> Self {
Self {
inner: KStringCowInner::Owned(KStringBase::from_static(other)),
}
}
}
impl<'s, B: crate::backend::HeapStr> KStringCowBase<'s, B> {
/// Create an owned `KStringCowBase`.
#[inline]
#[must_use]
pub fn from_boxed(other: BoxedStr) -> Self {
Self {
inner: KStringCowInner::Owned(KStringBase::from_boxed(other)),
}
}
/// Create an owned `KStringCowBase`.
#[inline]
#[must_use]
pub fn from_string(other: StdString) -> Self {
Self {
inner: KStringCowInner::Owned(KStringBase::from_string(other)),
}
}
/// Create a reference to a borrowed data.
#[inline]
#[must_use]
pub fn from_ref(other: &'s str) -> Self {
Self {
inner: KStringCowInner::Borrowed(other),
}
}
/// Get a reference to the `KStringBase`.
#[inline]
#[must_use]
pub fn as_ref(&self) -> KStringRef<'_> {
self.inner.as_ref()
}
/// Clone the data into an owned-type.
#[inline]
#[must_use]
pub fn into_owned(self) -> KStringBase<B> {
self.inner.into_owned()
}
/// Extracts a string slice containing the entire `KStringCowBase`.
#[inline]
#[must_use]
pub fn as_str(&self) -> &str {
self.inner.as_str()
}
/// Convert to a mutable string type, cloning the data if necessary.
#[inline]
#[must_use]
pub fn into_string(self) -> StdString {
String::from(self.into_boxed_str())
}
/// Convert to a mutable string type, cloning the data if necessary.
#[inline]
#[must_use]
pub fn into_boxed_str(self) -> BoxedStr {
self.inner.into_boxed_str()
}
/// Convert to a Cow str
#[inline]
#[must_use]
pub fn into_cow_str(self) -> Cow<'s, str> {
self.inner.into_cow_str()
}
}
impl<'s, B: crate::backend::HeapStr> KStringCowInner<'s, B> {
#[inline]
fn as_ref(&self) -> KStringRef<'_> {
match self {
Self::Borrowed(s) => KStringRef::from_ref(s),
Self::Owned(s) => s.as_ref(),
}
}
#[inline]
fn into_owned(self) -> KStringBase<B> {
match self {
Self::Borrowed(s) => KStringBase::from_ref(s),
Self::Owned(s) => s,
}
}
#[inline]
fn as_str(&self) -> &str {
match self {
Self::Borrowed(s) => s,
Self::Owned(s) => s.as_str(),
}
}
#[inline]
fn into_boxed_str(self) -> BoxedStr {
match self {
Self::Borrowed(s) => BoxedStr::from(s),
Self::Owned(s) => s.into_boxed_str(),
}
}
/// Convert to a Cow str
#[inline]
fn into_cow_str(self) -> Cow<'s, str> {
match self {
Self::Borrowed(s) => Cow::Borrowed(s),
Self::Owned(s) => s.into_cow_str(),
}
}
}
impl<'s, B: crate::backend::HeapStr> std::ops::Deref for KStringCowBase<'s, B> {
type Target = str;
#[inline]
fn deref(&self) -> &str {
self.as_str()
}
}
impl<'s, B: crate::backend::HeapStr> Eq for KStringCowBase<'s, B> {}
impl<'s, B: crate::backend::HeapStr> PartialEq<KStringCowBase<'s, B>> for KStringCowBase<'s, B> {
#[inline]
fn eq(&self, other: &KStringCowBase<'s, B>) -> bool {
PartialEq::eq(self.as_str(), other.as_str())
}
}
impl<'s, B: crate::backend::HeapStr> PartialEq<str> for KStringCowBase<'s, B> {
#[inline]
fn eq(&self, other: &str) -> bool {
PartialEq::eq(self.as_str(), other)
}
}
impl<'s, B: crate::backend::HeapStr> PartialEq<&'s str> for KStringCowBase<'s, B> {
#[inline]
fn eq(&self, other: &&str) -> bool {
PartialEq::eq(self.as_str(), *other)
}
}
impl<'s, B: crate::backend::HeapStr> PartialEq<String> for KStringCowBase<'s, B> {
#[inline]
fn eq(&self, other: &StdString) -> bool {
PartialEq::eq(self.as_str(), other.as_str())
}
}
impl<'s, B: crate::backend::HeapStr> Ord for KStringCowBase<'s, B> {
#[inline]
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
self.as_str().cmp(other.as_str())
}
}
impl<'s, B: crate::backend::HeapStr> PartialOrd for KStringCowBase<'s, B> {
#[inline]
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
self.as_str().partial_cmp(other.as_str())
}
}
impl<'s, B: crate::backend::HeapStr> std::hash::Hash for KStringCowBase<'s, B> {
#[inline]
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
self.as_str().hash(state);
}
}
impl<'s, B: crate::backend::HeapStr> fmt::Debug for KStringCowBase<'s, B> {
#[inline]
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.as_str().fmt(f)
}
}
impl<'s, B: crate::backend::HeapStr> fmt::Display for KStringCowBase<'s, B> {
#[inline]
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Display::fmt(self.as_str(), f)
}
}
impl<'s, B: crate::backend::HeapStr> AsRef<str> for KStringCowBase<'s, B> {
#[inline]
fn as_ref(&self) -> &str {
self.as_str()
}
}
impl<'s, B: crate::backend::HeapStr> AsRef<[u8]> for KStringCowBase<'s, B> {
#[inline]
fn as_ref(&self) -> &[u8] {
self.as_bytes()
}
}
impl<'s, B: crate::backend::HeapStr> AsRef<std::ffi::OsStr> for KStringCowBase<'s, B> {
#[inline]
fn as_ref(&self) -> &std::ffi::OsStr {
(&**self).as_ref()
}
}
impl<'s, B: crate::backend::HeapStr> AsRef<std::path::Path> for KStringCowBase<'s, B> {
#[inline]
fn as_ref(&self) -> &std::path::Path {
std::path::Path::new(self)
}
}
impl<'s, B: crate::backend::HeapStr> std::borrow::Borrow<str> for KStringCowBase<'s, B> {
#[inline]
fn borrow(&self) -> &str {
self.as_str()
}
}
impl<'s, B> Default for KStringCowBase<'s, B> {
#[inline]
fn default() -> Self {
Self::new()
}
}
impl<'s, B: crate::backend::HeapStr> From<KStringBase<B>> for KStringCowBase<'s, B> {
#[inline]
fn from(other: KStringBase<B>) -> Self {
let inner = KStringCowInner::Owned(other);
Self { inner }
}
}
impl<'s, B: crate::backend::HeapStr> From<&'s KStringBase<B>> for KStringCowBase<'s, B> {
#[inline]
fn from(other: &'s KStringBase<B>) -> Self {
let other = other.as_ref();
other.into()
}
}
impl<'s, B: crate::backend::HeapStr> From<KStringRef<'s>> for KStringCowBase<'s, B> {
#[inline]
fn from(other: KStringRef<'s>) -> Self {
match other.inner {
KStringRefInner::Borrowed(s) => Self::from_ref(s),
KStringRefInner::Singleton(s) => Self::from_static(s),
}
}
}
impl<'s, B: crate::backend::HeapStr> From<&'s KStringRef<'s>> for KStringCowBase<'s, B> {
#[inline]
fn from(other: &'s KStringRef<'s>) -> Self {
match other.inner {
KStringRefInner::Borrowed(s) => Self::from_ref(s),
KStringRefInner::Singleton(s) => Self::from_static(s),
}
}
}
impl<'s, B: crate::backend::HeapStr> From<StdString> for KStringCowBase<'s, B> {
#[inline]
fn from(other: StdString) -> Self {
Self::from_string(other)
}
}
impl<'s, B: crate::backend::HeapStr> From<&'s StdString> for KStringCowBase<'s, B> {
#[inline]
fn from(other: &'s StdString) -> Self {
Self::from_ref(other.as_str())
}
}
impl<'s, B: crate::backend::HeapStr> From<BoxedStr> for KStringCowBase<'s, B> {
#[inline]
fn from(other: BoxedStr) -> Self {
// Since the memory is already allocated, don't bother moving it into a FixedString
Self::from_boxed(other)
}
}
impl<'s, B: crate::backend::HeapStr> From<&'s BoxedStr> for KStringCowBase<'s, B> {
#[inline]
fn from(other: &'s BoxedStr) -> Self {
Self::from_ref(other)
}
}
impl<'s, B: crate::backend::HeapStr> From<&'s str> for KStringCowBase<'s, B> {
#[inline]
fn from(other: &'s str) -> Self {
Self::from_ref(other)
}
}
impl<B: crate::backend::HeapStr> std::str::FromStr for KStringCowBase<'_, B> {
type Err = std::convert::Infallible;
#[inline]
fn from_str(s: &str) -> Result<Self, Self::Err> {
Ok(Self::from_string(s.into()))
}
}
#[cfg(feature = "serde")]
impl<'s, B: crate::backend::HeapStr> serde::Serialize for KStringCowBase<'s, B> {
#[inline]
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_str(self.as_str())
}
}
#[cfg(feature = "serde")]
impl<'de, 's, B: crate::backend::HeapStr> serde::Deserialize<'de> for KStringCowBase<'s, B> {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
KStringBase::deserialize(deserializer).map(|s| s.into())
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_size() {
println!("KStringCow: {}", std::mem::size_of::<KStringCow<'static>>());
}
}