use std::hash::{Hash, Hasher};
use crate::{
subtree_path_builder::{SubtreePathBuilder, SubtreePathRelative},
util::CowLike,
SubtreePathIter,
};
#[derive(Debug)]
pub struct SubtreePath<'b, B> {
pub(crate) ref_variant: SubtreePathInner<'b, B>,
}
#[derive(Debug)]
pub(crate) enum SubtreePathInner<'b, B> {
Slice(&'b [B]),
SubtreePath(&'b SubtreePathBuilder<'b, B>),
SubtreePathIter(SubtreePathIter<'b, B>),
}
impl<'bl, 'br, BL, BR> PartialEq<SubtreePath<'br, BR>> for SubtreePath<'bl, BL>
where
BL: AsRef<[u8]>,
BR: AsRef<[u8]>,
{
fn eq(&self, other: &SubtreePath<'br, BR>) -> bool {
self.clone()
.into_reverse_iter()
.eq(other.clone().into_reverse_iter())
}
}
impl<'b, B: AsRef<[u8]>> Eq for SubtreePath<'b, B> {}
impl<'b, B> From<SubtreePathInner<'b, B>> for SubtreePath<'b, B> {
fn from(ref_variant: SubtreePathInner<'b, B>) -> Self {
Self { ref_variant }
}
}
impl<'b, B> From<&'b [B]> for SubtreePath<'b, B> {
fn from(value: &'b [B]) -> Self {
SubtreePathInner::Slice(value).into()
}
}
impl<'b, B, const N: usize> From<&'b [B; N]> for SubtreePath<'b, B> {
fn from(value: &'b [B; N]) -> Self {
SubtreePathInner::Slice(value).into()
}
}
impl<'s, 'b, B> From<&'s SubtreePathBuilder<'b, B>> for SubtreePath<'s, B> {
fn from(value: &'s SubtreePathBuilder<'b, B>) -> Self {
SubtreePathInner::SubtreePath(value).into()
}
}
impl<'b, B: AsRef<[u8]>> Hash for SubtreePath<'b, B> {
fn hash<H: Hasher>(&self, state: &mut H) {
match &self.ref_variant {
SubtreePathInner::Slice(slice) => slice
.iter()
.map(AsRef::as_ref)
.rev()
.for_each(|s| s.hash(state)),
SubtreePathInner::SubtreePath(path) => path.hash(state),
SubtreePathInner::SubtreePathIter(path_iter) => {
path_iter.clone().for_each(|s| s.hash(state))
}
}
}
}
impl<B> Clone for SubtreePath<'_, B> {
fn clone(&self) -> Self {
match &self.ref_variant {
SubtreePathInner::Slice(x) => SubtreePathInner::Slice(x),
SubtreePathInner::SubtreePath(x) => SubtreePathInner::SubtreePath(x),
SubtreePathInner::SubtreePathIter(x) => SubtreePathInner::SubtreePathIter(x.clone()),
}
.into()
}
}
impl SubtreePath<'static, [u8; 0]> {
pub const fn empty() -> Self {
SubtreePath {
ref_variant: SubtreePathInner::Slice(&[]),
}
}
}
impl<B> SubtreePath<'_, B> {
pub fn len(&self) -> usize {
match &self.ref_variant {
SubtreePathInner::Slice(s) => s.len(),
SubtreePathInner::SubtreePath(path) => path.len(),
SubtreePathInner::SubtreePathIter(path_iter) => path_iter.len(),
}
}
pub fn is_empty(&self) -> bool {
match &self.ref_variant {
SubtreePathInner::Slice(s) => s.is_empty(),
SubtreePathInner::SubtreePath(path) => path.is_empty(),
SubtreePathInner::SubtreePathIter(path_iter) => path_iter.is_empty(),
}
}
}
impl<'b, B: AsRef<[u8]>> SubtreePath<'b, B> {
pub fn derive_owned(&self) -> SubtreePathBuilder<'b, B> {
self.into()
}
pub fn derive_owned_with_child<'s, S>(&'b self, segment: S) -> SubtreePathBuilder<'b, B>
where
S: Into<CowLike<'s>>,
's: 'b,
{
SubtreePathBuilder {
base: self.clone(),
relative: SubtreePathRelative::Single(segment.into()),
}
}
pub fn derive_parent(&self) -> Option<(SubtreePath<'b, B>, &'b [u8])> {
match &self.ref_variant {
SubtreePathInner::Slice(path) => path
.split_last()
.map(|(tail, rest)| (SubtreePathInner::Slice(rest).into(), tail.as_ref())),
SubtreePathInner::SubtreePath(path) => path.derive_parent(),
SubtreePathInner::SubtreePathIter(iter) => {
let mut derived_iter = iter.clone();
derived_iter.next().map(|segment| {
(
SubtreePathInner::SubtreePathIter(derived_iter).into(),
segment,
)
})
}
}
}
pub fn into_reverse_iter(self) -> SubtreePathIter<'b, B> {
match self.ref_variant {
SubtreePathInner::Slice(slice) => SubtreePathIter::new(slice.iter()),
SubtreePathInner::SubtreePath(path) => path.reverse_iter(),
SubtreePathInner::SubtreePathIter(iter) => iter,
}
}
pub fn is_root(&self) -> bool {
match &self.ref_variant {
SubtreePathInner::Slice(s) => s.is_empty(),
SubtreePathInner::SubtreePath(path) => path.is_root(),
SubtreePathInner::SubtreePathIter(iter) => iter.is_empty(),
}
}
pub fn to_vec(&self) -> Vec<Vec<u8>> {
match &self.ref_variant {
SubtreePathInner::Slice(slice) => slice.iter().map(|x| x.as_ref().to_vec()).collect(),
SubtreePathInner::SubtreePath(path) => path.to_vec(),
SubtreePathInner::SubtreePathIter(iter) => {
let mut path = iter
.clone()
.map(|x| x.as_ref().to_vec())
.collect::<Vec<Vec<u8>>>();
path.reverse();
path
}
}
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn to_vec() {
let base: SubtreePath<_> = (&[b"one" as &[u8], b"two", b"three"]).into();
let mut builder = base.derive_owned_with_child(b"four");
builder.push_segment(b"five");
builder.push_segment(b"six");
builder.push_segment(b"seven");
builder.push_segment(b"eight");
let parent = builder.derive_parent().unwrap().0;
let as_vec = parent.to_vec();
let reference_vec = vec![
b"one".to_vec(),
b"two".to_vec(),
b"three".to_vec(),
b"four".to_vec(),
b"five".to_vec(),
b"six".to_vec(),
b"seven".to_vec(),
];
assert_eq!(as_vec, reference_vec);
assert_eq!(parent.len(), reference_vec.len());
}
}