use bytes::{Bytes, BytesMut};
use core::str;
use std::fmt::Display;
#[derive(Debug, Default, Clone, Eq, PartialEq)]
pub struct Utf8Bytes(Bytes);
impl Utf8Bytes {
#[inline]
pub const fn from_static(str: &'static str) -> Self {
Self(Bytes::from_static(str.as_bytes()))
}
#[inline]
pub fn as_str(&self) -> &str {
unsafe { str::from_utf8_unchecked(&self.0) }
}
pub unsafe fn from_bytes_unchecked(bytes: Bytes) -> Self {
Self(bytes)
}
}
impl std::ops::Deref for Utf8Bytes {
type Target = str;
#[inline]
fn deref(&self) -> &Self::Target {
self.as_str()
}
}
impl AsRef<[u8]> for Utf8Bytes {
#[inline]
fn as_ref(&self) -> &[u8] {
&self.0
}
}
impl AsRef<str> for Utf8Bytes {
#[inline]
fn as_ref(&self) -> &str {
self.as_str()
}
}
impl AsRef<Bytes> for Utf8Bytes {
#[inline]
fn as_ref(&self) -> &Bytes {
&self.0
}
}
impl std::borrow::Borrow<str> for Utf8Bytes {
fn borrow(&self) -> &str {
self.as_str()
}
}
impl core::hash::Hash for Utf8Bytes {
fn hash<H: core::hash::Hasher>(&self, state: &mut H) {
self.as_str().hash(state)
}
}
impl PartialOrd for Utf8Bytes {
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
Some(self.cmp(other))
}
}
impl Ord for Utf8Bytes {
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
self.as_str().cmp(other.as_str())
}
}
impl<T> PartialEq<T> for Utf8Bytes
where
for<'a> &'a str: PartialEq<T>,
{
#[inline]
fn eq(&self, other: &T) -> bool {
self.as_str() == *other
}
}
impl Display for Utf8Bytes {
#[inline]
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.write_str(self.as_str())
}
}
impl TryFrom<Bytes> for Utf8Bytes {
type Error = str::Utf8Error;
#[inline]
fn try_from(bytes: Bytes) -> Result<Self, Self::Error> {
str::from_utf8(&bytes)?;
Ok(Self(bytes))
}
}
impl TryFrom<BytesMut> for Utf8Bytes {
type Error = str::Utf8Error;
#[inline]
fn try_from(bytes: BytesMut) -> Result<Self, Self::Error> {
bytes.freeze().try_into()
}
}
impl TryFrom<Vec<u8>> for Utf8Bytes {
type Error = str::Utf8Error;
#[inline]
fn try_from(v: Vec<u8>) -> Result<Self, Self::Error> {
Bytes::from(v).try_into()
}
}
impl From<String> for Utf8Bytes {
#[inline]
fn from(s: String) -> Self {
Self(s.into())
}
}
impl From<&str> for Utf8Bytes {
#[inline]
fn from(s: &str) -> Self {
Self(Bytes::copy_from_slice(s.as_bytes()))
}
}
impl From<&String> for Utf8Bytes {
#[inline]
fn from(s: &String) -> Self {
s.as_str().into()
}
}
impl From<Utf8Bytes> for Bytes {
#[inline]
fn from(Utf8Bytes(bytes): Utf8Bytes) -> Self {
bytes
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::{
borrow::Borrow,
hash::{BuildHasher, RandomState},
};
#[test]
fn hash_consistency() {
let bytes = Utf8Bytes::from_static("hash_consistency");
let hasher = RandomState::new();
assert_eq!(hasher.hash_one::<&str>(bytes.borrow()), hasher.hash_one(bytes));
}
}