use git_hash::ObjectId;
use git_object::{bstr::BStr, TreeRefIter};
use crate::{object::find, Id, Tree};
impl<'repo> Tree<'repo> {
pub fn from_data(id: impl Into<ObjectId>, data: Vec<u8>, repo: &'repo crate::Repository) -> Self {
Tree {
id: id.into(),
data,
repo,
}
}
}
impl<'repo> Tree<'repo> {
pub fn id(&self) -> Id<'repo> {
Id::from_id(self.id, self.repo)
}
pub fn lookup_entry<I, P>(mut self, path: I) -> Result<Option<Entry<'repo>>, find::existing::Error>
where
I: IntoIterator<Item = P>,
P: PartialEq<BStr>,
{
let mut path = path.into_iter().peekable();
while let Some(component) = path.next() {
match TreeRefIter::from_bytes(&self.data)
.filter_map(Result::ok)
.find(|entry| component.eq(entry.filename))
{
Some(entry) => {
if path.peek().is_none() {
return Ok(Some(Entry {
inner: entry.into(),
repo: self.repo,
}));
} else {
let next_id = entry.oid.to_owned();
let repo = self.repo;
drop(self);
self = match repo.find_object(next_id)?.try_into_tree() {
Ok(tree) => tree,
Err(_) => return Ok(None),
};
}
}
None => return Ok(None),
}
}
Ok(None)
}
pub fn lookup_entry_by_path(
self,
relative_path: impl AsRef<std::path::Path>,
) -> Result<Option<Entry<'repo>>, find::existing::Error> {
self.lookup_entry(relative_path.as_ref().components().map(|c| {
git_path::os_str_into_bstr(c.as_os_str())
.unwrap_or_else(|_| "".into())
.as_ref()
}))
}
}
pub mod diff;
pub mod traverse;
mod iter;
pub use iter::EntryRef;
impl<'r> std::fmt::Debug for Tree<'r> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "Tree({})", self.id)
}
}
#[derive(PartialEq, Debug, Clone)]
pub struct Entry<'repo> {
inner: git_object::tree::Entry,
repo: &'repo crate::Repository,
}
mod entry {
use crate::{bstr::BStr, ext::ObjectIdExt, object::tree::Entry};
impl<'repo> Entry<'repo> {
pub fn mode(&self) -> git_object::tree::EntryMode {
self.inner.mode
}
pub fn filename(&self) -> &BStr {
self.inner.filename.as_ref()
}
pub fn id(&self) -> crate::Id<'repo> {
self.inner.oid.attach(self.repo)
}
pub fn object(&self) -> Result<crate::Object<'repo>, crate::object::find::existing::Error> {
self.id().object()
}
pub fn oid(&self) -> &git_hash::oid {
&self.inner.oid
}
pub fn object_id(&self) -> git_hash::ObjectId {
self.inner.oid
}
}
impl Entry<'_> {
pub fn detach(self) -> git_object::tree::Entry {
self.inner
}
}
}