Skip to content

Commit bc991ff

Browse files
committed
Make it possible to ignore specs that don't match when iterating mappings. (#450)
This is an issue if `HEAD` as refspec is added after all other specs, but isn't serialized. Then at a later stage, the remote is restored from disk but won't contain one of the refspecs that was used for the mapping.
1 parent 700cc2d commit bc991ff

File tree

4 files changed

+17
-12
lines changed

4 files changed

+17
-12
lines changed

README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -54,8 +54,8 @@ Please see _'Development Status'_ for a listing of all crates and their capabili
5454
* [x] **ref-map** - show how remote references relate to their local tracking branches as mapped by refspecs.
5555
* [x] **fetch** - fetch the current remote or the given one, optionally just as dry-run.
5656
* **clone**
57-
* [ ] initialize a new **bare** repository and fetch all objects.
58-
* [ ] initialize a new repository, fetch all objects and checkout the main worktree.
57+
* [x] initialize a new **bare** repository and fetch all objects.
58+
* [x] initialize a new repository, fetch all objects and checkout the main worktree.
5959
* **credential**
6060
* [x] **fill/approve/reject** - The same as `git credential`, but implemented in Rust, calling helpers only when from trusted configuration.
6161
* **free** - no git repository necessary

git-repository/src/remote/connection/fetch/update_refs/mod.rs

+9-8
Original file line numberDiff line numberDiff line change
@@ -49,12 +49,13 @@ pub(crate) fn update(
4949
let mut edits = Vec::new();
5050
let mut updates = Vec::new();
5151

52-
for fetch::Mapping {
53-
remote,
54-
local,
55-
spec_index,
56-
} in mappings
57-
{
52+
for (remote, local, spec) in mappings.iter().filter_map(
53+
|fetch::Mapping {
54+
remote,
55+
local,
56+
spec_index,
57+
}| refspecs.get(*spec_index).map(|spec| (remote, local, spec)),
58+
) {
5859
let remote_id = remote.as_id();
5960
if dry_run == fetch::DryRun::No && !repo.objects.contains(remote_id) {
6061
updates.push(update::Mode::RejectedSourceObjectNotFound { id: remote_id.into() }.into());
@@ -83,14 +84,14 @@ pub(crate) fn update(
8384
let (mode, reflog_message) = if local_id == remote_id {
8485
(update::Mode::NoChangeNeeded, "no update will be performed")
8586
} else if let Some(git_ref::Category::Tag) = existing.name().category() {
86-
if refspecs[*spec_index].allow_non_fast_forward() {
87+
if spec.allow_non_fast_forward() {
8788
(update::Mode::Forced, "updating tag")
8889
} else {
8990
updates.push(update::Mode::RejectedTagUpdate.into());
9091
continue;
9192
}
9293
} else {
93-
let mut force = refspecs[*spec_index].allow_non_fast_forward();
94+
let mut force = spec.allow_non_fast_forward();
9495
let is_fast_forward = match dry_run {
9596
fetch::DryRun::No => {
9697
let ancestors = repo

git-repository/src/remote/connection/fetch/update_refs/update.rs

+5-2
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,9 @@ impl std::fmt::Display for Mode {
9494
impl Outcome {
9595
/// Produce an iterator over all information used to produce the this outcome, ref-update by ref-update, using the `mappings`
9696
/// used when producing the ref update.
97+
///
98+
/// Note that mappings that don't have a corresponding entry in `refspecs` these will be `None` even though that should never be the case.
99+
/// This can happen if the `refspecs` passed in aren't the respecs used to create the `mapping`, and it's up to the caller to sort it out.
97100
pub fn iter_mapping_updates<'a, 'b>(
98101
&self,
99102
mappings: &'a [fetch::Mapping],
@@ -102,15 +105,15 @@ impl Outcome {
102105
Item = (
103106
&super::Update,
104107
&'a fetch::Mapping,
105-
&'b git_refspec::RefSpec,
108+
Option<&'b git_refspec::RefSpec>,
106109
Option<&git_ref::transaction::RefEdit>,
107110
),
108111
> {
109112
self.updates.iter().zip(mappings.iter()).map(move |(update, mapping)| {
110113
(
111114
update,
112115
mapping,
113-
&refspecs[mapping.spec_index],
116+
refspecs.get(mapping.spec_index),
114117
update.edit_index.and_then(|idx| self.edits.get(idx)),
115118
)
116119
})

gitoxide-core/src/repository/fetch.rs

+1
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,7 @@ pub(crate) mod function {
9191
let mut last_spec_index = usize::MAX;
9292
let mut updates = update_refs
9393
.iter_mapping_updates(&map.mappings, refspecs)
94+
.filter_map(|(update, mapping, spec, edit)| spec.map(|spec| (update, mapping, spec, edit)))
9495
.collect::<Vec<_>>();
9596
updates.sort_by_key(|t| t.2);
9697
for (update, mapping, spec, edit) in updates {

0 commit comments

Comments
 (0)