Skip to content

Commit e0969c3

Browse files
committed
fix!: simplify ancestor module..
We remove ability to reuse `State` as ultimately,this capability wasn't all that useful.
1 parent 7f6bee5 commit e0969c3

File tree

3 files changed

+69
-93
lines changed

3 files changed

+69
-93
lines changed

gix-traverse/src/commit/ancestors.rs

+50-78
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,8 @@
1+
use gix_date::SecondsSinceUnixEpoch;
2+
use gix_hash::ObjectId;
3+
use gix_hashtable::HashSet;
14
use smallvec::SmallVec;
2-
3-
/// An iterator over the ancestors one or more starting commits
4-
pub struct Ancestors<Find, Predicate, StateMut> {
5-
objects: Find,
6-
cache: Option<gix_commitgraph::Graph>,
7-
predicate: Predicate,
8-
state: StateMut,
9-
parents: super::Parents,
10-
sorting: Sorting,
11-
}
5+
use std::collections::VecDeque;
126

137
/// Specify how to sort commits during traversal.
148
///
@@ -58,46 +52,39 @@ pub enum Sorting {
5852
},
5953
}
6054

55+
/// The error is part of the item returned by the [Ancestors](super::Ancestors) iterator.
56+
#[derive(Debug, thiserror::Error)]
57+
#[allow(missing_docs)]
58+
pub enum Error {
59+
#[error(transparent)]
60+
Find(#[from] gix_object::find::existing_iter::Error),
61+
#[error(transparent)]
62+
ObjectDecode(#[from] gix_object::decode::Error),
63+
}
64+
65+
/// The state used and potentially shared by multiple graph traversals.
66+
#[derive(Clone)]
67+
pub(super) struct State {
68+
next: VecDeque<ObjectId>,
69+
queue: gix_revwalk::PriorityQueue<SecondsSinceUnixEpoch, ObjectId>,
70+
buf: Vec<u8>,
71+
seen: HashSet<ObjectId>,
72+
parents_buf: Vec<u8>,
73+
parent_ids: SmallVec<[(ObjectId, SecondsSinceUnixEpoch); 2]>,
74+
}
75+
6176
///
6277
#[allow(clippy::empty_docs)]
63-
pub mod ancestors {
64-
use std::{
65-
borrow::{Borrow, BorrowMut},
66-
collections::VecDeque,
67-
};
68-
78+
mod init {
6979
use gix_date::SecondsSinceUnixEpoch;
7080
use gix_hash::{oid, ObjectId};
71-
use gix_hashtable::HashSet;
7281
use gix_object::{CommitRefIter, FindExt};
73-
use smallvec::SmallVec;
7482

7583
use super::{
76-
super::{Either, Info, ParentIds, Parents},
77-
collect_parents, Ancestors, Sorting,
84+
super::{Ancestors, Either, Info, ParentIds, Parents},
85+
collect_parents, Error, Sorting, State,
7886
};
7987

80-
/// The error is part of the item returned by the [Ancestors] iterator.
81-
#[derive(Debug, thiserror::Error)]
82-
#[allow(missing_docs)]
83-
pub enum Error {
84-
#[error(transparent)]
85-
Find(#[from] gix_object::find::existing_iter::Error),
86-
#[error(transparent)]
87-
ObjectDecode(#[from] gix_object::decode::Error),
88-
}
89-
90-
/// The state used and potentially shared by multiple graph traversals.
91-
#[derive(Clone)]
92-
pub struct State {
93-
next: VecDeque<ObjectId>,
94-
queue: gix_revwalk::PriorityQueue<SecondsSinceUnixEpoch, ObjectId>,
95-
buf: Vec<u8>,
96-
seen: HashSet<ObjectId>,
97-
parents_buf: Vec<u8>,
98-
parent_ids: SmallVec<[(ObjectId, SecondsSinceUnixEpoch); 2]>,
99-
}
100-
10188
impl Default for State {
10289
fn default() -> Self {
10390
State {
@@ -121,12 +108,11 @@ pub mod ancestors {
121108
}
122109

123110
/// Builder
124-
impl<Find, Predicate, StateMut> Ancestors<Find, Predicate, StateMut>
111+
impl<Find, Predicate> Ancestors<Find, Predicate>
125112
where
126113
Find: gix_object::Find,
127-
StateMut: BorrowMut<State>,
128114
{
129-
/// Set the sorting method, either topological or by author date
115+
/// Set the `sorting` method.
130116
pub fn sorting(mut self, sorting: Sorting) -> Result<Self, Error> {
131117
self.sorting = sorting;
132118
match self.sorting {
@@ -135,7 +121,7 @@ pub mod ancestors {
135121
}
136122
Sorting::ByCommitTimeNewestFirst | Sorting::ByCommitTimeNewestFirstCutoffOlderThan { .. } => {
137123
let cutoff_time = self.sorting.cutoff_time();
138-
let state = self.state.borrow_mut();
124+
let state = &mut self.state;
139125
for commit_id in state.next.drain(..) {
140126
let commit_iter = self.objects.find_commit_iter(&commit_id, &mut state.buf)?;
141127
let time = commit_iter.committer()?.time.seconds;
@@ -173,7 +159,7 @@ pub mod ancestors {
173159
}
174160

175161
fn queue_to_vecdeque(&mut self) {
176-
let state = self.state.borrow_mut();
162+
let state = &mut self.state;
177163
state.next.extend(
178164
std::mem::replace(&mut state.queue, gix_revwalk::PriorityQueue::new())
179165
.into_iter_unordered()
@@ -182,55 +168,48 @@ pub mod ancestors {
182168
}
183169
}
184170

185-
/// Initialization
186-
impl<Find, StateMut> Ancestors<Find, fn(&oid) -> bool, StateMut>
171+
/// Lifecyle
172+
impl<Find> Ancestors<Find, fn(&oid) -> bool>
187173
where
188174
Find: gix_object::Find,
189-
StateMut: BorrowMut<State>,
190175
{
191176
/// Create a new instance.
192177
///
193178
/// * `find` - a way to lookup new object data during traversal by their `ObjectId`, writing their data into buffer and returning
194179
/// an iterator over commit tokens if the object is present and is a commit. Caching should be implemented within this function
195180
/// as needed.
196-
/// * `state` - all state used for the traversal. If multiple traversals are performed, allocations can be minimized by reusing
197-
/// this state.
198181
/// * `tips`
199182
/// * the starting points of the iteration, usually commits
200183
/// * each commit they lead to will only be returned once, including the tip that started it
201-
pub fn new(tips: impl IntoIterator<Item = impl Into<ObjectId>>, state: StateMut, find: Find) -> Self {
202-
Self::filtered(tips, state, find, |_| true)
184+
pub fn new(tips: impl IntoIterator<Item = impl Into<ObjectId>>, find: Find) -> Self {
185+
Self::filtered(tips, find, |_| true)
203186
}
204187
}
205188

206-
/// Initialization
207-
impl<Find, Predicate, StateMut> Ancestors<Find, Predicate, StateMut>
189+
/// Lifecyle
190+
impl<Find, Predicate> Ancestors<Find, Predicate>
208191
where
209192
Find: gix_object::Find,
210193
Predicate: FnMut(&oid) -> bool,
211-
StateMut: BorrowMut<State>,
212194
{
213195
/// Create a new instance with commit filtering enabled.
214196
///
215197
/// * `find` - a way to lookup new object data during traversal by their `ObjectId`, writing their data into buffer and returning
216198
/// an iterator over commit tokens if the object is present and is a commit. Caching should be implemented within this function
217199
/// as needed.
218-
/// * `state` - all state used for the traversal. If multiple traversals are performed, allocations can be minimized by reusing
219-
/// this state.
220200
/// * `tips`
221201
/// * the starting points of the iteration, usually commits
222202
/// * each commit they lead to will only be returned once, including the tip that started it
223203
/// * `predicate` - indicate whether a given commit should be included in the result as well
224204
/// as whether its parent commits should be traversed.
225205
pub fn filtered(
226206
tips: impl IntoIterator<Item = impl Into<ObjectId>>,
227-
mut state: StateMut,
228207
find: Find,
229208
mut predicate: Predicate,
230209
) -> Self {
231210
let tips = tips.into_iter();
211+
let mut state = State::default();
232212
{
233-
let state = state.borrow_mut();
234213
state.clear();
235214
state.next.reserve(tips.size_hint().0);
236215
for tip in tips.map(Into::into) {
@@ -250,27 +229,24 @@ pub mod ancestors {
250229
}
251230
}
252231
}
232+
253233
/// Access
254-
impl<Find, Predicate, StateMut> Ancestors<Find, Predicate, StateMut>
255-
where
256-
StateMut: Borrow<State>,
257-
{
258-
/// Return an iterator for accessing more of the current commits data.
234+
impl<Find, Predicate> Ancestors<Find, Predicate> {
235+
/// Return an iterator for accessing data of the current commit, parsed lazily.
259236
pub fn commit_iter(&self) -> CommitRefIter<'_> {
260-
CommitRefIter::from_bytes(&self.state.borrow().buf)
237+
CommitRefIter::from_bytes(&self.state.buf)
261238
}
262239

263-
/// Return the current commits data.
240+
/// Return the current commits' raw data, which can be parsed using [`gix_object::CommitRef::from_bytes()`].
264241
pub fn commit_data(&self) -> &[u8] {
265-
&self.state.borrow().buf
242+
&self.state.buf
266243
}
267244
}
268245

269-
impl<Find, Predicate, StateMut> Iterator for Ancestors<Find, Predicate, StateMut>
246+
impl<Find, Predicate> Iterator for Ancestors<Find, Predicate>
270247
where
271248
Find: gix_object::Find,
272249
Predicate: FnMut(&oid) -> bool,
273-
StateMut: BorrowMut<State>,
274250
{
275251
type Item = Result<Info, Error>;
276252

@@ -300,17 +276,16 @@ pub mod ancestors {
300276
}
301277

302278
/// Utilities
303-
impl<Find, Predicate, StateMut> Ancestors<Find, Predicate, StateMut>
279+
impl<Find, Predicate> Ancestors<Find, Predicate>
304280
where
305281
Find: gix_object::Find,
306282
Predicate: FnMut(&oid) -> bool,
307-
StateMut: BorrowMut<State>,
308283
{
309284
fn next_by_commit_date(
310285
&mut self,
311286
cutoff_older_than: Option<SecondsSinceUnixEpoch>,
312287
) -> Option<Result<Info, Error>> {
313-
let state = self.state.borrow_mut();
288+
let state = &mut self.state;
314289

315290
let (commit_time, oid) = state.queue.pop()?;
316291
let mut parents: ParentIds = Default::default();
@@ -371,14 +346,13 @@ pub mod ancestors {
371346
}
372347

373348
/// Utilities
374-
impl<Find, Predicate, StateMut> Ancestors<Find, Predicate, StateMut>
349+
impl<Find, Predicate> Ancestors<Find, Predicate>
375350
where
376351
Find: gix_object::Find,
377352
Predicate: FnMut(&oid) -> bool,
378-
StateMut: BorrowMut<State>,
379353
{
380354
fn next_by_topology(&mut self) -> Option<Result<Info, Error>> {
381-
let state = self.state.borrow_mut();
355+
let state = &mut self.state;
382356
let oid = state.next.pop_front()?;
383357
let mut parents: ParentIds = Default::default();
384358
match super::super::find(self.cache.as_ref(), &self.objects, &oid, &mut state.buf) {
@@ -430,8 +404,6 @@ pub mod ancestors {
430404
}
431405
}
432406

433-
pub use ancestors::{Error, State};
434-
435407
fn collect_parents(
436408
dest: &mut SmallVec<[(gix_hash::ObjectId, gix_date::SecondsSinceUnixEpoch); 2]>,
437409
cache: Option<&gix_commitgraph::Graph>,

gix-traverse/src/commit/mod.rs

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,19 @@
11
use gix_object::FindExt;
22
use smallvec::SmallVec;
33

4+
/// An iterator over the ancestors one or more starting commits
5+
pub struct Ancestors<Find, Predicate> {
6+
objects: Find,
7+
cache: Option<gix_commitgraph::Graph>,
8+
predicate: Predicate,
9+
state: ancestors::State,
10+
parents: Parents,
11+
sorting: Sorting,
12+
}
13+
414
/// Simple ancestors traversal
515
pub mod ancestors;
6-
pub use ancestors::{Ancestors, Sorting};
16+
pub use ancestors::Sorting;
717

818
// Topological traversal
919
pub mod topo;

gix-traverse/tests/commit/mod.rs

+8-14
Original file line numberDiff line numberDiff line change
@@ -64,17 +64,12 @@ mod ancestor {
6464
let (store, tips, expected) = self.setup()?;
6565

6666
for use_commitgraph in [false, true] {
67-
let oids = commit::Ancestors::filtered(
68-
tips.clone(),
69-
commit::ancestors::State::default(),
70-
&store,
71-
predicate.clone(),
72-
)
73-
.sorting(self.sorting)?
74-
.parents(self.mode)
75-
.commit_graph(self.setup_commitgraph(store.store_ref(), use_commitgraph))
76-
.map(|res| res.map(|info| info.id))
77-
.collect::<Result<Vec<_>, _>>()?;
67+
let oids = commit::Ancestors::filtered(tips.clone(), &store, predicate.clone())
68+
.sorting(self.sorting)?
69+
.parents(self.mode)
70+
.commit_graph(self.setup_commitgraph(store.store_ref(), use_commitgraph))
71+
.map(|res| res.map(|info| info.id))
72+
.collect::<Result<Vec<_>, _>>()?;
7873

7974
assert_eq!(oids, expected);
8075
}
@@ -85,7 +80,7 @@ mod ancestor {
8580
let (store, tips, expected) = self.setup()?;
8681

8782
for use_commitgraph in [false, true] {
88-
let oids = commit::Ancestors::new(tips.clone(), commit::ancestors::State::default(), &store)
83+
let oids = commit::Ancestors::new(tips.clone(), &store)
8984
.sorting(self.sorting)?
9085
.parents(self.mode)
9186
.commit_graph(self.setup_commitgraph(store.store_ref(), use_commitgraph))
@@ -343,7 +338,7 @@ mod ancestor {
343338

344339
/// Some dates adjusted to be a year apart, but still 'c1' and 'c2' with the same date.
345340
mod adjusted_dates {
346-
use gix_traverse::commit::{ancestors, Ancestors, Parents, Sorting};
341+
use gix_traverse::commit::{Ancestors, Parents, Sorting};
347342

348343
use crate::{commit::ancestor::TraversalAssertion, hex_to_id};
349344

@@ -398,7 +393,6 @@ mod ancestor {
398393
let store = gix_odb::at(dir.join(".git").join("objects"))?;
399394
let iter = Ancestors::new(
400395
Some(hex_to_id("9902e3c3e8f0c569b4ab295ddf473e6de763e1e7" /* c2 */)),
401-
ancestors::State::default(),
402396
&store,
403397
)
404398
.sorting(Sorting::ByCommitTimeNewestFirstCutoffOlderThan {

0 commit comments

Comments
 (0)