use std::collections::VecDeque;
use gix_hash::ObjectId;
#[derive(Debug, thiserror::Error)]
#[allow(missing_docs)]
pub enum Error {
#[error(transparent)]
Find(#[from] gix_object::find::existing_iter::Error),
#[error("The delegate cancelled the operation")]
Cancelled,
#[error(transparent)]
ObjectDecode(#[from] gix_object::decode::Error),
}
#[derive(Default, Clone)]
pub struct State {
next: VecDeque<ObjectId>,
buf: Vec<u8>,
}
impl State {
fn clear(&mut self) {
self.next.clear();
self.buf.clear();
}
}
pub(super) mod function {
use std::borrow::BorrowMut;
use gix_object::{FindExt, TreeRefIter};
use super::{Error, State};
use crate::tree::Visit;
pub fn breadthfirst<StateMut, Find, V>(
root: TreeRefIter<'_>,
mut state: StateMut,
objects: Find,
delegate: &mut V,
) -> Result<(), Error>
where
Find: gix_object::Find,
StateMut: BorrowMut<State>,
V: Visit,
{
let state = state.borrow_mut();
state.clear();
let mut tree = root;
loop {
for entry in tree {
let entry = entry?;
if entry.mode.is_tree() {
delegate.push_path_component(entry.filename);
let action = delegate.visit_tree(&entry);
match action {
std::ops::ControlFlow::Continue(false) => {}
std::ops::ControlFlow::Continue(true) => {
delegate.pop_path_component();
delegate.push_back_tracked_path_component(entry.filename);
state.next.push_back(entry.oid.to_owned());
}
std::ops::ControlFlow::Break(()) => {
return Err(Error::Cancelled);
}
}
} else {
delegate.push_path_component(entry.filename);
if delegate.visit_nontree(&entry).is_break() {
return Err(Error::Cancelled);
}
}
delegate.pop_path_component();
}
match state.next.pop_front() {
Some(oid) => {
delegate.pop_front_tracked_path_and_set_current();
tree = objects.find_tree_iter(&oid, &mut state.buf)?;
}
None => break Ok(()),
}
}
}
}