use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)]
pub struct NodeId {
segments: Vec<String>,
}
impl NodeId {
pub fn from_segments<I, S>(segments: I) -> Self
where
I: IntoIterator<Item = S>,
S: Into<String>,
{
let segments: Vec<String> = segments.into_iter().map(Into::into).collect();
assert!(!segments.is_empty(), "NodeId requires at least one segment");
for seg in &segments {
assert!(!seg.is_empty(), "NodeId segment must not be empty");
assert!(!seg.contains('/'), "NodeId segment must not contain '/'");
}
Self { segments }
}
pub fn root(name: impl Into<String>) -> Self {
Self::from_segments([name.into()])
}
pub fn parse(canonical: &str) -> Self {
Self::from_segments(canonical.split('/'))
}
pub fn canonical(&self) -> String {
self.segments.join("/")
}
pub fn name(&self) -> &str {
self.segments
.last()
.expect("segments is non-empty by invariant")
}
pub fn depth(&self) -> usize {
self.segments.len()
}
pub fn parent(&self) -> Option<Self> {
if self.segments.len() == 1 {
return None;
}
Some(Self {
segments: self.segments[..self.segments.len() - 1].to_vec(),
})
}
#[must_use]
pub fn child(&self, segment: impl Into<String>) -> Self {
let mut new = self.segments.clone();
new.push(segment.into());
Self { segments: new }
}
pub fn segments(&self) -> &[String] {
&self.segments
}
}
impl std::fmt::Display for NodeId {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.write_str(&self.canonical())
}
}