Skip to content

Commit d986b2b

Browse files
committedAug 21, 2024
feat: add Reference::follow_to_object()
It's an equivalent to `git2::Reference::resolve()`.`
1 parent 6c6f946 commit d986b2b

File tree

3 files changed

+64
-1
lines changed

3 files changed

+64
-1
lines changed
 

‎gix/src/reference/errors.rs

+18
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,24 @@ pub mod peel {
5454
}
5555
}
5656

57+
///
58+
#[allow(clippy::empty_docs)]
59+
pub mod follow {
60+
///
61+
#[allow(clippy::empty_docs)]
62+
pub mod to_object {
63+
/// The error returned by [`Reference::follow_to_object(…)`](crate::Reference::follow_to_object()).
64+
#[derive(Debug, thiserror::Error)]
65+
#[allow(missing_docs)]
66+
pub enum Error {
67+
#[error(transparent)]
68+
FollowToObject(#[from] gix_ref::peel::to_object::Error),
69+
#[error(transparent)]
70+
PackedRefsOpen(#[from] gix_ref::packed::buffer::open::Error),
71+
}
72+
}
73+
}
74+
5775
///
5876
#[allow(clippy::empty_docs)]
5977
pub mod head_id {

‎gix/src/reference/mod.rs

+28-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ pub mod iter;
1111
pub mod remote;
1212

1313
mod errors;
14-
pub use errors::{edit, find, head_commit, head_id, head_tree_id, peel};
14+
pub use errors::{edit, find, follow, head_commit, head_id, head_tree_id, peel};
1515

1616
use crate::ext::ObjectIdExt;
1717

@@ -153,6 +153,33 @@ impl<'repo> Reference<'repo> {
153153
Ok(target.object()?.peel_to_kind(kind)?)
154154
}
155155

156+
/// Follow all symbolic references we point to up to the first object, which is typically (but not always) a tag,
157+
/// returning its id.
158+
/// After this call, this ref will be pointing to an object directly, but may still not consider itself 'peeled' unless
159+
/// a symbolic target ref was looked up from packed-refs.
160+
#[doc(alias = "resolve", alias = "git2")]
161+
pub fn follow_to_object(&mut self) -> Result<Id<'repo>, follow::to_object::Error> {
162+
let packed = self.repo.refs.cached_packed_buffer().map_err(|err| {
163+
follow::to_object::Error::FollowToObject(gix_ref::peel::to_object::Error::Follow(
164+
file::find::existing::Error::Find(file::find::Error::PackedOpen(err)),
165+
))
166+
})?;
167+
self.follow_to_object_packed(packed.as_ref().map(|p| &***p))
168+
}
169+
170+
/// Like [`follow_to_object`](Self::follow_to_object), but can be used for repeated calls as it won't
171+
/// look up `packed` each time, but can reuse it instead.
172+
#[doc(alias = "resolve", alias = "git2")]
173+
pub fn follow_to_object_packed(
174+
&mut self,
175+
packed: Option<&gix_ref::packed::Buffer>,
176+
) -> Result<Id<'repo>, follow::to_object::Error> {
177+
Ok(self
178+
.inner
179+
.follow_to_object_in_place_packed(&self.repo.refs, packed)?
180+
.attach(self.repo))
181+
}
182+
156183
/// Follow this symbolic reference one level and return the ref it refers to.
157184
///
158185
/// Returns `None` if this is not a symbolic reference, hence the leaf of the chain.

‎gix/tests/reference/mod.rs

+18
Original file line numberDiff line numberDiff line change
@@ -153,6 +153,24 @@ mod find {
153153
);
154154
assert_eq!(tag_ref.peel_to_commit()?.id, obj.id);
155155

156+
let mut tag_ref = repo.find_reference("dt3")?;
157+
assert_eq!(
158+
tag_ref.follow_to_object()?,
159+
first_tag_id,
160+
"it's similar to peel_to_kind(), but provides the id instead"
161+
);
162+
assert_eq!(tag_ref.follow_to_object()?, first_tag_id, "it's idempotent");
163+
assert_eq!(
164+
tag_ref.target().id(),
165+
first_tag_id,
166+
"it now points to the first tag as well"
167+
);
168+
assert_eq!(
169+
tag_ref.inner.peeled,
170+
Some(target_commit_id),
171+
"as it was read from a packed-ref, it contains peeling information nonetheless"
172+
);
173+
156174
Ok(())
157175
}
158176

0 commit comments

Comments
 (0)