From 9351e031d533cb089b670239a44f15f52ab9189f Mon Sep 17 00:00:00 2001 From: konstin Date: Thu, 21 Dec 2023 11:08:18 +0100 Subject: [PATCH] Avoid loss of data in Range.simplify Co-authored-by: Zanie Blue Upstream port of #12 --- src/range.rs | 37 ++++++++++++++++++++++++++++--------- 1 file changed, 28 insertions(+), 9 deletions(-) diff --git a/src/range.rs b/src/range.rs index e3ba82e2..6e3ad43f 100644 --- a/src/range.rs +++ b/src/range.rs @@ -415,21 +415,26 @@ impl Range { Self { segments: output }.check_invariants() } - /// Returns a simpler Range that contains the same versions + /// Returns a simpler Range that contains the same versions. /// - /// For every one of the Versions provided in versions the existing range and - /// the simplified range will agree on whether it is contained. + /// For every one of the Versions provided in versions the existing range and the simplified range will agree on whether it is contained. /// The simplified version may include or exclude versions that are not in versions as the implementation wishes. - /// For example: - /// - If all the versions are contained in the original than the range will be simplified to `full`. - /// - If none of the versions are contained in the original than the range will be simplified to `empty`. /// - /// If versions are not sorted the correctness of this function is not guaranteed. + /// If none of the versions are contained in the original than the range will be returned unmodified. + /// If the range includes a single version, it will be returned unmodified. + /// If all the versions are contained in the original than the range will be simplified to `full`. + /// + /// If the given versions are not sorted the correctness of this function is not guaranteed. pub fn simplify<'v, I>(&self, versions: I) -> Self where I: Iterator + 'v, V: 'v, { + // Do not simplify singletons + if self.is_singleton() { + return self.clone(); + } + // Return the segment index in the range for each version in the range, None otherwise let version_locations = versions.scan(0, move |i, v| { while let Some(segment) = self.segments.get(*i) { @@ -441,7 +446,13 @@ impl Range { } Some(None) }); - let kept_segments = group_adjacent_locations(version_locations); + let mut kept_segments = group_adjacent_locations(version_locations).peekable(); + + // Do not return null sets + if kept_segments.peek().is_none() { + return self.clone(); + } + self.keep_segments(kept_segments) } @@ -462,6 +473,14 @@ impl Range { } Self { segments }.check_invariants() } + + /// Returns whether the range is just a single version. + pub fn is_singleton(&self) -> bool { + match self.segments.as_slice() { + [(Included(v1), Included(v2))] => v1 == v2, + _ => false, + } + } } impl VersionSet for Range { @@ -629,7 +648,7 @@ pub mod tests { segments.push((start_bound, Unbounded)); } - return Range { segments }.check_invariants(); + Range { segments }.check_invariants() }) }