1use std::{fmt::Formatter, ops::Index};
2
3use gix_hash::oid;
4use smallvec::SmallVec;
5
6use crate::Graph;
7
8pub type IdMap<T> = gix_hashtable::HashMap<gix_hash::ObjectId, T>;
10
11pub mod commit;
13
14mod errors {
15 pub mod insert_parents {
17 use crate::graph::commit::iter_parents;
18
19 #[derive(Debug, thiserror::Error)]
21 #[allow(missing_docs)]
22 pub enum Error {
23 #[error(transparent)]
24 Lookup(#[from] gix_object::find::existing_iter::Error),
25 #[error("A commit could not be decoded during traversal")]
26 Decode(#[from] gix_object::decode::Error),
27 #[error(transparent)]
28 Parent(#[from] iter_parents::Error),
29 }
30 }
31
32 pub mod get_or_insert_default {
34 use crate::graph::commit::to_owned;
35
36 #[derive(Debug, thiserror::Error)]
38 #[allow(missing_docs)]
39 pub enum Error {
40 #[error(transparent)]
41 Lookup(#[from] gix_object::find::existing_iter::Error),
42 #[error(transparent)]
43 ToOwned(#[from] to_owned::Error),
44 }
45 }
46}
47pub use errors::{get_or_insert_default, insert_parents};
48use gix_date::SecondsSinceUnixEpoch;
49
50pub type Generation = u32;
55
56impl<T: std::fmt::Debug> std::fmt::Debug for Graph<'_, '_, T> {
57 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
58 std::fmt::Debug::fmt(&self.map, f)
59 }
60}
61
62impl<'cache, T: Default> Graph<'_, 'cache, T> {
63 pub fn try_lookup_or_insert(
68 &mut self,
69 id: gix_hash::ObjectId,
70 update_data: impl FnOnce(&mut T),
71 ) -> Result<Option<LazyCommit<'_, 'cache>>, get_or_insert_default::Error> {
72 self.try_lookup_or_insert_default(id, T::default, update_data)
73 }
74}
75
76impl<'cache, T> Graph<'_, 'cache, T> {
78 pub fn len(&self) -> usize {
80 self.map.len()
81 }
82
83 pub fn is_empty(&self) -> bool {
85 self.map.is_empty()
86 }
87
88 pub fn contains(&self, id: &gix_hash::oid) -> bool {
90 self.map.contains_key(id.as_ref())
91 }
92
93 pub fn get(&self, id: &gix_hash::oid) -> Option<&T> {
95 self.map.get(id)
96 }
97
98 pub fn get_mut(&mut self, id: &gix_hash::oid) -> Option<&mut T> {
100 self.map.get_mut(id)
101 }
102
103 pub fn insert(&mut self, id: gix_hash::ObjectId, value: T) -> Option<T> {
105 self.map.insert(id, value)
106 }
107
108 pub fn insert_data<E>(
112 &mut self,
113 id: gix_hash::ObjectId,
114 mut make_data: impl FnMut(LazyCommit<'_, 'cache>) -> Result<T, E>,
115 ) -> Result<Option<T>, E>
116 where
117 E: From<gix_object::find::existing_iter::Error>,
118 {
119 let value = make_data(self.lookup(&id).map_err(E::from)?)?;
120 Ok(self.map.insert(id, value))
121 }
122
123 pub fn clear(&mut self) {
125 self.map.clear();
126 }
127
128 pub fn insert_parents(
133 &mut self,
134 id: &gix_hash::oid,
135 new_parent_data: &mut dyn FnMut(gix_hash::ObjectId, SecondsSinceUnixEpoch) -> T,
136 update_existing: &mut dyn FnMut(gix_hash::ObjectId, &mut T),
137 first_parent: bool,
138 ) -> Result<(), insert_parents::Error> {
139 let commit = self.lookup(id)?;
140 let parents: SmallVec<[_; 2]> = commit.iter_parents().collect();
141 for parent_id in parents {
142 let parent_id = parent_id?;
143 match self.map.entry(parent_id) {
144 gix_hashtable::hash_map::Entry::Vacant(entry) => {
145 let parent = match try_lookup(&parent_id, &*self.find, self.cache, &mut self.parent_buf)? {
146 Some(p) => p,
147 None => continue, };
149
150 let parent_commit_date = parent.committer_timestamp().unwrap_or_default();
151 entry.insert(new_parent_data(parent_id, parent_commit_date));
152 }
153 gix_hashtable::hash_map::Entry::Occupied(mut entry) => {
154 update_existing(parent_id, entry.get_mut());
155 }
156 }
157 if first_parent {
158 break;
159 }
160 }
161 Ok(())
162 }
163
164 #[allow(clippy::type_complexity)]
170 pub fn insert_parents_with_lookup<E>(
171 &mut self,
172 id: &gix_hash::oid,
173 parent_data: &mut dyn FnMut(gix_hash::ObjectId, LazyCommit<'_, 'cache>, Option<&mut T>) -> Result<T, E>,
174 ) -> Result<(), E>
175 where
176 E: From<gix_object::find::existing_iter::Error>
177 + From<gix_object::decode::Error>
178 + From<commit::iter_parents::Error>,
179 {
180 let commit = self.lookup(id).map_err(E::from)?;
181 let parents: SmallVec<[_; 2]> = commit.iter_parents().collect();
182 for parent_id in parents {
183 let parent_id = parent_id.map_err(E::from)?;
184 let parent = match try_lookup(&parent_id, &*self.find, self.cache, &mut self.parent_buf).map_err(E::from)? {
185 Some(p) => p,
186 None => continue, };
188
189 match self.map.entry(parent_id) {
190 gix_hashtable::hash_map::Entry::Vacant(entry) => {
191 entry.insert(parent_data(parent_id, parent, None)?);
192 }
193 gix_hashtable::hash_map::Entry::Occupied(mut entry) => {
194 parent_data(parent_id, parent, Some(entry.get_mut()))?;
195 }
196 }
197 }
198 Ok(())
199 }
200
201 pub fn detach(self) -> IdMap<T> {
203 self.map
204 }
205}
206
207impl<'find, 'cache, T> Graph<'find, 'cache, T> {
209 pub fn new(objects: impl gix_object::Find + 'find, cache: Option<&'cache gix_commitgraph::Graph>) -> Self {
218 Graph {
219 find: Box::new(objects),
220 cache,
221 map: gix_hashtable::HashMap::default(),
222 buf: Vec::new(),
223 parent_buf: Vec::new(),
224 }
225 }
226}
227
228impl<T> Graph<'_, '_, Commit<T>> {
230 pub fn get_or_insert_commit_default(
236 &mut self,
237 id: gix_hash::ObjectId,
238 new_data: impl FnOnce() -> T,
239 update_data: impl FnOnce(&mut T),
240 ) -> Result<Option<&mut Commit<T>>, get_or_insert_default::Error> {
241 match self.map.entry(id) {
242 gix_hashtable::hash_map::Entry::Vacant(entry) => {
243 let res = try_lookup(&id, &*self.find, self.cache, &mut self.buf)?;
244 let commit = match res {
245 None => return Ok(None),
246 Some(commit) => commit,
247 };
248 let mut commit = commit.to_owned(new_data)?;
249 update_data(&mut commit.data);
250 entry.insert(commit);
251 }
252 gix_hashtable::hash_map::Entry::Occupied(mut entry) => {
253 update_data(&mut entry.get_mut().data);
254 }
255 };
256 Ok(self.map.get_mut(&id))
257 }
258
259 pub fn clear_commit_data(&mut self, mut clear: impl FnMut(&mut T)) {
261 self.map.values_mut().for_each(|c| clear(&mut c.data));
262 }
263}
264
265impl<T: Default> Graph<'_, '_, Commit<T>> {
267 pub fn get_or_insert_commit(
276 &mut self,
277 id: gix_hash::ObjectId,
278 update_data: impl FnOnce(&mut T),
279 ) -> Result<Option<&mut Commit<T>>, get_or_insert_default::Error> {
280 self.get_or_insert_commit_default(id, T::default, update_data)
281 }
282
283 pub fn get_or_insert_full_commit(
288 &mut self,
289 id: gix_hash::ObjectId,
290 update_commit: impl FnOnce(&mut Commit<T>),
291 ) -> Result<Option<&mut Commit<T>>, get_or_insert_default::Error> {
292 match self.map.entry(id) {
293 gix_hashtable::hash_map::Entry::Vacant(entry) => {
294 let res = try_lookup(&id, &*self.find, self.cache, &mut self.buf)?;
295 let commit = match res {
296 None => return Ok(None),
297 Some(commit) => commit,
298 };
299 let mut commit = commit.to_owned(T::default)?;
300 update_commit(&mut commit);
301 entry.insert(commit);
302 }
303 gix_hashtable::hash_map::Entry::Occupied(mut entry) => {
304 update_commit(entry.get_mut());
305 }
306 };
307 Ok(self.map.get_mut(&id))
308 }
309}
310
311impl<'cache, T> Graph<'_, 'cache, T> {
313 pub fn try_lookup_or_insert_default(
324 &mut self,
325 id: gix_hash::ObjectId,
326 default: impl FnOnce() -> T,
327 update_data: impl FnOnce(&mut T),
328 ) -> Result<Option<LazyCommit<'_, 'cache>>, get_or_insert_default::Error> {
329 let res = try_lookup(&id, &*self.find, self.cache, &mut self.buf)?;
330 Ok(res.map(|commit| {
331 match self.map.entry(id) {
332 gix_hashtable::hash_map::Entry::Vacant(entry) => {
333 let mut data = default();
334 update_data(&mut data);
335 entry.insert(data);
336 }
337 gix_hashtable::hash_map::Entry::Occupied(mut entry) => {
338 update_data(entry.get_mut());
339 }
340 };
341 commit
342 }))
343 }
344
345 pub fn try_lookup(
350 &mut self,
351 id: &gix_hash::oid,
352 ) -> Result<Option<LazyCommit<'_, 'cache>>, gix_object::find::existing_iter::Error> {
353 try_lookup(id, &*self.find, self.cache, &mut self.buf)
354 }
355
356 pub fn lookup(
358 &mut self,
359 id: &gix_hash::oid,
360 ) -> Result<LazyCommit<'_, 'cache>, gix_object::find::existing_iter::Error> {
361 self.try_lookup(id)?
362 .ok_or(gix_object::find::existing_iter::Error::NotFound { oid: id.to_owned() })
363 }
364}
365
366fn try_lookup<'graph, 'cache>(
367 id: &gix_hash::oid,
368 objects: &dyn gix_object::Find,
369 cache: Option<&'cache gix_commitgraph::Graph>,
370 buf: &'graph mut Vec<u8>,
371) -> Result<Option<LazyCommit<'graph, 'cache>>, gix_object::find::existing_iter::Error> {
372 if let Some(cache) = cache {
373 if let Some(pos) = cache.lookup(id) {
374 return Ok(Some(LazyCommit {
375 backing: Either::Right((cache, pos)),
376 }));
377 }
378 }
379 #[allow(clippy::manual_map)]
380 Ok(
381 match objects
382 .try_find(id, buf)
383 .map_err(gix_object::find::existing_iter::Error::Find)?
384 {
385 Some(data) => data.kind.is_commit().then_some(LazyCommit {
386 backing: Either::Left(buf),
387 }),
388 None => None,
389 },
390 )
391}
392
393impl<'a, T> Index<&'a gix_hash::oid> for Graph<'_, '_, T> {
394 type Output = T;
395
396 fn index(&self, index: &'a oid) -> &Self::Output {
397 &self.map[index]
398 }
399}
400
401pub struct Commit<T> {
403 pub parents: SmallVec<[gix_hash::ObjectId; 1]>,
405 pub commit_time: SecondsSinceUnixEpoch,
407 pub generation: Option<u32>,
409 pub data: T,
411}
412
413impl<T> std::fmt::Debug for Commit<T>
414where
415 T: std::fmt::Debug,
416{
417 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
418 f.debug_struct("Commit")
419 .field("parents", &self.parents)
420 .field("commit_time", &self.commit_time)
421 .field("generation", &self.generation)
422 .field("data", &self.data)
423 .finish()
424 }
425}
426
427impl<T> Clone for Commit<T>
428where
429 T: Clone,
430{
431 fn clone(&self) -> Self {
432 Commit {
433 parents: self.parents.clone(),
434 commit_time: self.commit_time,
435 generation: self.generation,
436 data: self.data.clone(),
437 }
438 }
439}
440
441pub struct LazyCommit<'graph, 'cache> {
445 backing: Either<&'graph [u8], (&'cache gix_commitgraph::Graph, gix_commitgraph::Position)>,
446}
447
448enum Either<T, U> {
449 Left(T),
450 Right(U),
451}