Skip to content

Commit ce5a320

Browse files
committedAug 26, 2024
adapt to changes in gix-revision
1 parent dfedf6a commit ce5a320

File tree

15 files changed

+96
-59
lines changed

15 files changed

+96
-59
lines changed
 

‎gix-negotiate/src/consecutive.rs

+6-6
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ impl Default for Algorithm {
1919

2020
impl Algorithm {
2121
/// Add `id` to our priority queue and *add* `flags` to it.
22-
fn add_to_queue(&mut self, id: ObjectId, mark: Flags, graph: &mut crate::Graph<'_>) -> Result<(), Error> {
22+
fn add_to_queue(&mut self, id: ObjectId, mark: Flags, graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
2323
let mut is_common = false;
2424
let mut has_mark = false;
2525
if let Some(commit) = graph
@@ -43,7 +43,7 @@ impl Algorithm {
4343
id: ObjectId,
4444
mode: Mark,
4545
ancestors: Ancestors,
46-
graph: &mut crate::Graph<'_>,
46+
graph: &mut crate::Graph<'_, '_>,
4747
) -> Result<(), Error> {
4848
let mut is_common = false;
4949
if let Some(commit) = graph
@@ -89,7 +89,7 @@ impl Algorithm {
8989
}
9090

9191
impl Negotiator for Algorithm {
92-
fn known_common(&mut self, id: ObjectId, graph: &mut crate::Graph<'_>) -> Result<(), Error> {
92+
fn known_common(&mut self, id: ObjectId, graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
9393
if graph
9494
.get(&id)
9595
.map_or(true, |commit| !commit.data.flags.contains(Flags::SEEN))
@@ -100,11 +100,11 @@ impl Negotiator for Algorithm {
100100
Ok(())
101101
}
102102

103-
fn add_tip(&mut self, id: ObjectId, graph: &mut crate::Graph<'_>) -> Result<(), Error> {
103+
fn add_tip(&mut self, id: ObjectId, graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
104104
self.add_to_queue(id, Flags::SEEN, graph)
105105
}
106106

107-
fn next_have(&mut self, graph: &mut crate::Graph<'_>) -> Option<Result<ObjectId, Error>> {
107+
fn next_have(&mut self, graph: &mut crate::Graph<'_, '_>) -> Option<Result<ObjectId, Error>> {
108108
loop {
109109
let id = self.revs.pop_value().filter(|_| self.non_common_revs != 0)?;
110110
let commit = graph.get_mut(&id).expect("it was added to the graph by now");
@@ -145,7 +145,7 @@ impl Negotiator for Algorithm {
145145
}
146146
}
147147

148-
fn in_common_with_remote(&mut self, id: ObjectId, graph: &mut crate::Graph<'_>) -> Result<bool, Error> {
148+
fn in_common_with_remote(&mut self, id: ObjectId, graph: &mut crate::Graph<'_, '_>) -> Result<bool, Error> {
149149
let known_to_be_common = graph
150150
.get(&id)
151151
.map_or(false, |commit| commit.data.flags.contains(Flags::COMMON));

‎gix-negotiate/src/lib.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,7 @@ pub struct Metadata {
5858
}
5959

6060
/// The graph our callers use to store traversal information, for (re-)use in the negotiation implementation.
61-
pub type Graph<'find> = gix_revwalk::Graph<'find, gix_revwalk::graph::Commit<Metadata>>;
61+
pub type Graph<'find, 'cache> = gix_revwalk::Graph<'find, 'cache, gix_revwalk::graph::Commit<Metadata>>;
6262

6363
/// A map associating an object id with its commit-metadata.
6464
pub type IdMap = gix_revwalk::graph::IdMap<gix_revwalk::graph::Commit<Metadata>>;
@@ -125,22 +125,22 @@ pub trait Negotiator {
125125
/// Mark `id` as common between the remote and us.
126126
///
127127
/// These ids are typically the local tips of remote tracking branches.
128-
fn known_common(&mut self, id: gix_hash::ObjectId, graph: &mut Graph<'_>) -> Result<(), Error>;
128+
fn known_common(&mut self, id: gix_hash::ObjectId, graph: &mut Graph<'_, '_>) -> Result<(), Error>;
129129

130130
/// Add `id` as starting point of a traversal across commits that aren't necessarily common between the remote and us.
131131
///
132132
/// These tips are usually the commits of local references whose tips should lead to objects that we have in common with the remote.
133-
fn add_tip(&mut self, id: gix_hash::ObjectId, graph: &mut Graph<'_>) -> Result<(), Error>;
133+
fn add_tip(&mut self, id: gix_hash::ObjectId, graph: &mut Graph<'_, '_>) -> Result<(), Error>;
134134

135135
/// Produce the next id of an object that we want the server to know we have. It's an object we don't know we have in common or not.
136136
///
137137
/// Returns `None` if we have exhausted all options, which might mean we have traversed the entire commit graph.
138-
fn next_have(&mut self, graph: &mut Graph<'_>) -> Option<Result<gix_hash::ObjectId, Error>>;
138+
fn next_have(&mut self, graph: &mut Graph<'_, '_>) -> Option<Result<gix_hash::ObjectId, Error>>;
139139

140140
/// Mark `id` as being common with the remote (as informed by the remote itself) and return `true` if we knew it was common already.
141141
///
142142
/// We can assume to have already seen `id` as we were the one to inform the remote in a prior `have`.
143-
fn in_common_with_remote(&mut self, id: gix_hash::ObjectId, graph: &mut Graph<'_>) -> Result<bool, Error>;
143+
fn in_common_with_remote(&mut self, id: gix_hash::ObjectId, graph: &mut Graph<'_, '_>) -> Result<bool, Error>;
144144
}
145145

146146
/// An error that happened during any of the methods on a [`Negotiator`].

‎gix-negotiate/src/noop.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -5,19 +5,19 @@ use crate::{Error, Negotiator};
55
pub(crate) struct Noop;
66

77
impl Negotiator for Noop {
8-
fn known_common(&mut self, _id: ObjectId, _graph: &mut crate::Graph<'_>) -> Result<(), Error> {
8+
fn known_common(&mut self, _id: ObjectId, _graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
99
Ok(())
1010
}
1111

12-
fn add_tip(&mut self, _id: ObjectId, _graph: &mut crate::Graph<'_>) -> Result<(), Error> {
12+
fn add_tip(&mut self, _id: ObjectId, _graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
1313
Ok(())
1414
}
1515

16-
fn next_have(&mut self, _graph: &mut crate::Graph<'_>) -> Option<Result<ObjectId, Error>> {
16+
fn next_have(&mut self, _graph: &mut crate::Graph<'_, '_>) -> Option<Result<ObjectId, Error>> {
1717
None
1818
}
1919

20-
fn in_common_with_remote(&mut self, _id: ObjectId, _graph: &mut crate::Graph<'_>) -> Result<bool, Error> {
20+
fn in_common_with_remote(&mut self, _id: ObjectId, _graph: &mut crate::Graph<'_, '_>) -> Result<bool, Error> {
2121
Ok(false)
2222
}
2323
}

‎gix-negotiate/src/skipping.rs

+7-7
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ impl Default for Algorithm {
1919

2020
impl Algorithm {
2121
/// Add `id` to our priority queue and *add* `flags` to it.
22-
fn add_to_queue(&mut self, id: ObjectId, mark: Flags, graph: &mut crate::Graph<'_>) -> Result<(), Error> {
22+
fn add_to_queue(&mut self, id: ObjectId, mark: Flags, graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
2323
let commit = graph.try_lookup_or_insert_commit(id, |entry| {
2424
entry.flags |= mark | Flags::SEEN;
2525
})?;
@@ -32,7 +32,7 @@ impl Algorithm {
3232
Ok(())
3333
}
3434

35-
fn mark_common(&mut self, id: ObjectId, graph: &mut crate::Graph<'_>) -> Result<(), Error> {
35+
fn mark_common(&mut self, id: ObjectId, graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
3636
let mut is_common = false;
3737
if let Some(commit) = graph
3838
.try_lookup_or_insert_commit(id, |entry| {
@@ -76,7 +76,7 @@ impl Algorithm {
7676
&mut self,
7777
entry: Metadata,
7878
parent_id: ObjectId,
79-
graph: &mut crate::Graph<'_>,
79+
graph: &mut crate::Graph<'_, '_>,
8080
) -> Result<bool, Error> {
8181
let mut was_seen = false;
8282
if let Some(parent) = graph
@@ -113,7 +113,7 @@ impl Algorithm {
113113
}
114114

115115
impl Negotiator for Algorithm {
116-
fn known_common(&mut self, id: ObjectId, graph: &mut crate::Graph<'_>) -> Result<(), Error> {
116+
fn known_common(&mut self, id: ObjectId, graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
117117
if graph
118118
.get(&id)
119119
.map_or(false, |commit| commit.data.flags.contains(Flags::SEEN))
@@ -123,7 +123,7 @@ impl Negotiator for Algorithm {
123123
self.add_to_queue(id, Flags::ADVERTISED, graph)
124124
}
125125

126-
fn add_tip(&mut self, id: ObjectId, graph: &mut crate::Graph<'_>) -> Result<(), Error> {
126+
fn add_tip(&mut self, id: ObjectId, graph: &mut crate::Graph<'_, '_>) -> Result<(), Error> {
127127
if graph
128128
.get(&id)
129129
.map_or(false, |commit| commit.data.flags.contains(Flags::SEEN))
@@ -133,7 +133,7 @@ impl Negotiator for Algorithm {
133133
self.add_to_queue(id, Flags::default(), graph)
134134
}
135135

136-
fn next_have(&mut self, graph: &mut crate::Graph<'_>) -> Option<Result<ObjectId, Error>> {
136+
fn next_have(&mut self, graph: &mut crate::Graph<'_, '_>) -> Option<Result<ObjectId, Error>> {
137137
loop {
138138
let id = self.revs.pop_value().filter(|_| self.non_common_revs != 0)?;
139139
let commit = graph.get_mut(&id).expect("it was added to the graph by now");
@@ -166,7 +166,7 @@ impl Negotiator for Algorithm {
166166
}
167167
}
168168

169-
fn in_common_with_remote(&mut self, id: ObjectId, graph: &mut crate::Graph<'_>) -> Result<bool, Error> {
169+
fn in_common_with_remote(&mut self, id: ObjectId, graph: &mut crate::Graph<'_, '_>) -> Result<bool, Error> {
170170
let mut was_seen = false;
171171
let known_to_be_common = graph.get(&id).map_or(false, |commit| {
172172
was_seen = commit.data.flags.contains(Flags::SEEN);

‎gix-negotiate/tests/baseline/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ fn run() -> crate::Result {
5757
let cache = use_cache
5858
.then(|| gix_commitgraph::at(store.store_ref().path().join("info")).ok())
5959
.flatten();
60-
let mut graph = gix_revwalk::Graph::new(&store, cache);
60+
let mut graph = gix_revwalk::Graph::new(&store, cache.as_ref());
6161
let mut negotiator = algo.into_negotiator();
6262
if debug {
6363
eprintln!("ALGO {algo_name} CASE {case}");

‎gix-revision/src/describe.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -160,7 +160,7 @@ pub(crate) mod function {
160160
/// candidate by setting `fallback_to_oid` to true.
161161
pub fn describe<'name>(
162162
commit: &oid,
163-
graph: &mut Graph<'_, Flags>,
163+
graph: &mut Graph<'_, '_, Flags>,
164164
Options {
165165
name_by_oid,
166166
mut max_candidates,
@@ -304,7 +304,7 @@ pub(crate) mod function {
304304
}
305305

306306
fn parents_by_date_onto_queue_and_track_names(
307-
graph: &mut Graph<'_, Flags>,
307+
graph: &mut Graph<'_, '_, Flags>,
308308
queue: &mut PriorityQueue<CommitTime, gix_hash::ObjectId>,
309309
commit: gix_hash::ObjectId,
310310
commit_flags: Flags,
@@ -326,7 +326,7 @@ pub(crate) mod function {
326326

327327
fn finish_depth_computation(
328328
mut queue: PriorityQueue<CommitTime, gix_hash::ObjectId>,
329-
graph: &mut Graph<'_, Flags>,
329+
graph: &mut Graph<'_, '_, Flags>,
330330
best_candidate: &mut Candidate<'_>,
331331
first_parent: bool,
332332
) -> Result<u32, Error> {

‎gix-revision/src/merge_base.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ pub(crate) mod function {
4242
pub fn merge_base(
4343
first: ObjectId,
4444
others: &[ObjectId],
45-
graph: &mut Graph<'_, Flags>,
45+
graph: &mut Graph<'_, '_, Flags>,
4646
) -> Result<Option<Vec<ObjectId>>, Error> {
4747
let _span = gix_trace::coarse!("gix_revision::merge_base()", ?first, ?others,);
4848
if others.is_empty() || others.contains(&first) {
@@ -60,7 +60,7 @@ pub(crate) mod function {
6060
/// That way, we return only the topologically most recent commits in `commits`.
6161
fn remove_redundant(
6262
commits: &[(ObjectId, GenThenTime)],
63-
graph: &mut Graph<'_, Flags>,
63+
graph: &mut Graph<'_, '_, Flags>,
6464
) -> Result<Vec<ObjectId>, Error> {
6565
if commits.is_empty() {
6666
return Ok(Vec::new());
@@ -151,7 +151,7 @@ pub(crate) mod function {
151151
fn paint_down_to_common(
152152
first: ObjectId,
153153
others: &[ObjectId],
154-
graph: &mut Graph<'_, Flags>,
154+
graph: &mut Graph<'_, '_, Flags>,
155155
) -> Result<Vec<(ObjectId, GenThenTime)>, Error> {
156156
let mut queue = PriorityQueue::<GenThenTime, ObjectId>::new();
157157
graph.insert_data(first, |commit| -> Result<_, Error> {
@@ -213,10 +213,10 @@ pub(crate) mod function {
213213
time: gix_date::SecondsSinceUnixEpoch,
214214
}
215215

216-
impl TryFrom<gix_revwalk::graph::LazyCommit<'_>> for GenThenTime {
216+
impl TryFrom<gix_revwalk::graph::LazyCommit<'_, '_>> for GenThenTime {
217217
type Error = gix_object::decode::Error;
218218

219-
fn try_from(commit: LazyCommit<'_>) -> Result<Self, Self::Error> {
219+
fn try_from(commit: LazyCommit<'_, '_>) -> Result<Self, Self::Error> {
220220
Ok(GenThenTime {
221221
generation: commit
222222
.generation()

‎gix-revision/tests/describe/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ fn run_test(
2222
let cache = use_commitgraph
2323
.then(|| gix_commitgraph::Graph::from_info_dir(&store.store_ref().path().join("info")).ok())
2424
.flatten();
25-
let mut graph = gix_revision::Graph::new(&store, cache);
25+
let mut graph = gix_revision::Graph::new(&store, cache.as_ref());
2626
run_assertions(
2727
gix_revision::describe(&commit_id, &mut graph, options(commit_id)),
2828
commit_id,

‎gix-revision/tests/merge_base/mod.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -13,10 +13,10 @@ mod baseline {
1313
for baseline_path in expectation_paths(&root)? {
1414
count += 1;
1515
for use_commitgraph in [false, true] {
16+
let cache = use_commitgraph
17+
.then(|| gix_commitgraph::Graph::from_info_dir(&odb.store_ref().path().join("info")).unwrap());
1618
for expected in parse_expectations(&baseline_path)? {
17-
let cache = use_commitgraph
18-
.then(|| gix_commitgraph::Graph::from_info_dir(&odb.store_ref().path().join("info")).unwrap());
19-
let mut graph = gix_revision::Graph::new(&odb, cache);
19+
let mut graph = gix_revision::Graph::new(&odb, cache.as_ref());
2020

2121
let actual = merge_base(expected.first, &expected.others, &mut graph)?;
2222
assert_eq!(

‎gix-revwalk/src/graph/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -207,7 +207,7 @@ impl<'find, 'cache, T> Graph<'find, 'cache, T> {
207207
pub fn new(objects: impl gix_object::Find + 'find, cache: Option<&'cache gix_commitgraph::Graph>) -> Self {
208208
Graph {
209209
find: Box::new(objects),
210-
cache: cache.into(),
210+
cache,
211211
map: gix_hashtable::HashMap::default(),
212212
buf: Vec::new(),
213213
parent_buf: Vec::new(),

‎gix/src/commit.rs

+17-5
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,8 @@ pub mod describe {
7373
#[derive(Debug, thiserror::Error)]
7474
#[allow(missing_docs)]
7575
pub enum Error {
76+
#[error(transparent)]
77+
OpenCache(#[from] crate::repository::commit_graph_if_enabled::Error),
7678
#[error(transparent)]
7779
Describe(#[from] gix_revision::describe::Error),
7880
#[error("Could not produce an unambiguous shortened id for formatting.")]
@@ -218,11 +220,11 @@ pub mod describe {
218220
///
219221
/// It is greatly recommended to [assure an object cache is set](crate::Repository::object_cache_size_if_unset())
220222
/// to save ~40% of time.
221-
pub fn try_resolve(&self) -> Result<Option<Resolution<'repo>>, Error> {
222-
let mut graph = gix_revwalk::Graph::new(
223-
&self.repo.objects,
224-
gix_commitgraph::Graph::from_info_dir(self.repo.objects.store_ref().path().join("info").as_ref()).ok(),
225-
);
223+
pub fn try_resolve_with_cache(
224+
&self,
225+
cache: Option<&'_ gix_commitgraph::Graph>,
226+
) -> Result<Option<Resolution<'repo>>, Error> {
227+
let mut graph = self.repo.revision_graph(cache);
226228
let outcome = gix_revision::describe(
227229
&self.id,
228230
&mut graph,
@@ -240,6 +242,16 @@ pub mod describe {
240242
}))
241243
}
242244

245+
/// Like [`Self::try_resolve_with_cache()`], but obtains the commitgraph-cache internally for a single use.
246+
///
247+
/// # Performance
248+
///
249+
/// Prefer to use the [`Self::try_resolve_with_cache()`] method when processing more than one commit at a time.
250+
pub fn try_resolve(&self) -> Result<Option<Resolution<'repo>>, Error> {
251+
let cache = self.repo.commit_graph_if_enabled()?;
252+
self.try_resolve_with_cache(cache.as_ref())
253+
}
254+
243255
/// Like [`try_format()`](Self::try_format()), but turns `id_as_fallback()` on to always produce a format.
244256
pub fn format(&mut self) -> Result<gix_revision::describe::Format<'static>, Error> {
245257
self.id_as_fallback = true;

‎gix/src/remote/connection/fetch/negotiate.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ pub(crate) enum Action {
6565
pub(crate) fn mark_complete_and_common_ref(
6666
repo: &crate::Repository,
6767
negotiator: &mut dyn gix_negotiate::Negotiator,
68-
graph: &mut gix_negotiate::Graph<'_>,
68+
graph: &mut gix_negotiate::Graph<'_, '_>,
6969
ref_map: &fetch::RefMap,
7070
shallow: &fetch::Shallow,
7171
mapping_is_ignored: impl Fn(&fetch::Mapping) -> bool,
@@ -258,7 +258,7 @@ pub(crate) fn add_wants(
258258
/// Remove all commits that are more recent than the cut-off, which is the commit time of the oldest common commit we have with the server.
259259
fn mark_recent_complete_commits(
260260
queue: &mut Queue,
261-
graph: &mut gix_negotiate::Graph<'_>,
261+
graph: &mut gix_negotiate::Graph<'_, '_>,
262262
cutoff: SecondsSinceUnixEpoch,
263263
) -> Result<(), Error> {
264264
let _span = gix_trace::detail!("mark_recent_complete", queue_len = queue.len());
@@ -286,7 +286,7 @@ fn mark_recent_complete_commits(
286286

287287
fn mark_all_refs_in_repo(
288288
repo: &crate::Repository,
289-
graph: &mut gix_negotiate::Graph<'_>,
289+
graph: &mut gix_negotiate::Graph<'_, '_>,
290290
queue: &mut Queue,
291291
mark: Flags,
292292
) -> Result<(), Error> {
@@ -310,7 +310,7 @@ fn mark_all_refs_in_repo(
310310

311311
fn mark_alternate_complete(
312312
repo: &crate::Repository,
313-
graph: &mut gix_negotiate::Graph<'_>,
313+
graph: &mut gix_negotiate::Graph<'_, '_>,
314314
queue: &mut Queue,
315315
) -> Result<(), Error> {
316316
let alternates = repo.objects.store_ref().alternate_db_paths()?;
@@ -332,7 +332,7 @@ fn mark_alternate_complete(
332332
/// Returns the amount of haves actually sent.
333333
pub(crate) fn one_round(
334334
negotiator: &mut dyn gix_negotiate::Negotiator,
335-
graph: &mut gix_negotiate::Graph<'_>,
335+
graph: &mut gix_negotiate::Graph<'_, '_>,
336336
haves_to_send: usize,
337337
arguments: &mut gix_protocol::fetch::Arguments,
338338
previous_response: Option<&gix_protocol::fetch::Response>,

‎gix/src/remote/connection/fetch/receive_pack.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -155,7 +155,8 @@ where
155155
r.objects.unset_object_cache();
156156
r
157157
};
158-
let mut graph = graph_repo.revision_graph();
158+
let cache = graph_repo.commit_graph_if_enabled().ok().flatten();
159+
let mut graph = graph_repo.revision_graph(cache.as_ref());
159160
let action = negotiate::mark_complete_and_common_ref(
160161
&graph_repo,
161162
negotiator.deref_mut(),

‎gix/src/repository/graph.rs

+23-12
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,8 @@ impl crate::Repository {
22
/// Create a graph data-structure capable of accelerating graph traversals and storing state of type `T` with each commit
33
/// it encountered.
44
///
5-
/// Note that the commitgraph will be used if it is present and readable, but it won't be an error if it is corrupted. In that case,
6-
/// it will just not be used.
5+
/// Note that the `cache` will be used if present, and it's best obtained with
6+
/// [`commit_graph_if_enabled()`](crate::Repository::commit_graph_if_enabled()).
77
///
88
/// Note that a commitgraph is only allowed to be used if `core.commitGraph` is true (the default), and that configuration errors are
99
/// ignored as well.
@@ -12,23 +12,34 @@ impl crate::Repository {
1212
///
1313
/// Note that the [Graph][gix_revwalk::Graph] can be sensitive to various object database settings that may affect the performance
1414
/// of the commit walk.
15-
pub fn revision_graph<T>(&self) -> gix_revwalk::Graph<'_, T> {
16-
gix_revwalk::Graph::new(
17-
&self.objects,
18-
self.config
19-
.may_use_commit_graph()
20-
.unwrap_or(true)
21-
.then(|| gix_commitgraph::at(self.objects.store_ref().path().join("info")).ok())
22-
.flatten(),
23-
)
15+
pub fn revision_graph<'cache, T>(
16+
&self,
17+
cache: Option<&'cache gix_commitgraph::Graph>,
18+
) -> gix_revwalk::Graph<'_, 'cache, T> {
19+
gix_revwalk::Graph::new(&self.objects, cache)
2420
}
2521

2622
/// Return a cache for commits and their graph structure, as managed by `git commit-graph`, for accelerating commit walks on
2723
/// a low level.
2824
///
2925
/// Note that [`revision_graph()`][crate::Repository::revision_graph()] should be preferred for general purpose walks that don't
30-
/// rely on the actual commit cache to be present, while leveraging it if possible.
26+
/// rely on the actual commit cache to be present, while leveraging the commit-graph if possible.
3127
pub fn commit_graph(&self) -> Result<gix_commitgraph::Graph, gix_commitgraph::init::Error> {
3228
gix_commitgraph::at(self.objects.store_ref().path().join("info"))
3329
}
30+
31+
/// Return a newly opened commit-graph if it is available *and* enabled in the Git configuration.
32+
pub fn commit_graph_if_enabled(
33+
&self,
34+
) -> Result<Option<gix_commitgraph::Graph>, super::commit_graph_if_enabled::Error> {
35+
Ok(self
36+
.config
37+
.may_use_commit_graph()?
38+
.then(|| gix_commitgraph::at(self.objects.store_ref().path().join("info")))
39+
.transpose()
40+
.or_else(|err| match err {
41+
gix_commitgraph::init::Error::Io { err, .. } if err.kind() == std::io::ErrorKind::NotFound => Ok(None),
42+
_ => Err(err),
43+
})?)
44+
}
3445
}

‎gix/src/repository/mod.rs

+13
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,19 @@ mod submodule;
7272
mod thread_safe;
7373
mod worktree;
7474

75+
///
76+
pub mod commit_graph_if_enabled {
77+
/// The error returned by [Repository::commit_graph_if_enabled()](crate::Repository::commit_graph_if_enabled()).
78+
#[derive(Debug, thiserror::Error)]
79+
#[allow(missing_docs)]
80+
pub enum Error {
81+
#[error(transparent)]
82+
ConfigBoolean(#[from] crate::config::boolean::Error),
83+
#[error(transparent)]
84+
OpenCommitGraph(#[from] gix_commitgraph::init::Error),
85+
}
86+
}
87+
7588
///
7689
#[cfg(feature = "index")]
7790
pub mod index_from_tree {

0 commit comments

Comments
 (0)
Please sign in to comment.