/rust/registry/src/index.crates.io-6f17d22bba15001f/itertools-0.12.1/src/diff.rs
Line | Count | Source (jump to first uncovered line) |
1 | | //! "Diff"ing iterators for caching elements to sequential collections without requiring the new |
2 | | //! elements' iterator to be `Clone`. |
3 | | //! |
4 | | //! - [`Diff`] (produced by the [`diff_with`] function) |
5 | | //! describes the difference between two non-`Clone` iterators `I` and `J` after breaking ASAP from |
6 | | //! a lock-step comparison. |
7 | | |
8 | | use std::fmt; |
9 | | |
10 | | use crate::free::put_back; |
11 | | use crate::structs::PutBack; |
12 | | |
13 | | /// A type returned by the [`diff_with`] function. |
14 | | /// |
15 | | /// `Diff` represents the way in which the elements yielded by the iterator `I` differ to some |
16 | | /// iterator `J`. |
17 | | pub enum Diff<I, J> |
18 | | where |
19 | | I: Iterator, |
20 | | J: Iterator, |
21 | | { |
22 | | /// The index of the first non-matching element along with both iterator's remaining elements |
23 | | /// starting with the first mis-match. |
24 | | FirstMismatch(usize, PutBack<I>, PutBack<J>), |
25 | | /// The total number of elements that were in `J` along with the remaining elements of `I`. |
26 | | Shorter(usize, PutBack<I>), |
27 | | /// The total number of elements that were in `I` along with the remaining elements of `J`. |
28 | | Longer(usize, PutBack<J>), |
29 | | } |
30 | | |
31 | | impl<I, J> fmt::Debug for Diff<I, J> |
32 | | where |
33 | | I: Iterator, |
34 | | J: Iterator, |
35 | | PutBack<I>: fmt::Debug, |
36 | | PutBack<J>: fmt::Debug, |
37 | | { |
38 | 0 | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
39 | 0 | match self { |
40 | 0 | Self::FirstMismatch(idx, i, j) => f |
41 | 0 | .debug_tuple("FirstMismatch") |
42 | 0 | .field(idx) |
43 | 0 | .field(i) |
44 | 0 | .field(j) |
45 | 0 | .finish(), |
46 | 0 | Self::Shorter(idx, i) => f.debug_tuple("Shorter").field(idx).field(i).finish(), |
47 | 0 | Self::Longer(idx, j) => f.debug_tuple("Longer").field(idx).field(j).finish(), |
48 | | } |
49 | 0 | } |
50 | | } |
51 | | |
52 | | impl<I, J> Clone for Diff<I, J> |
53 | | where |
54 | | I: Iterator, |
55 | | J: Iterator, |
56 | | PutBack<I>: Clone, |
57 | | PutBack<J>: Clone, |
58 | | { |
59 | 0 | fn clone(&self) -> Self { |
60 | 0 | match self { |
61 | 0 | Self::FirstMismatch(idx, i, j) => Self::FirstMismatch(*idx, i.clone(), j.clone()), |
62 | 0 | Self::Shorter(idx, i) => Self::Shorter(*idx, i.clone()), |
63 | 0 | Self::Longer(idx, j) => Self::Longer(*idx, j.clone()), |
64 | | } |
65 | 0 | } |
66 | | } |
67 | | |
68 | | /// Compares every element yielded by both `i` and `j` with the given function in lock-step and |
69 | | /// returns a [`Diff`] which describes how `j` differs from `i`. |
70 | | /// |
71 | | /// If the number of elements yielded by `j` is less than the number of elements yielded by `i`, |
72 | | /// the number of `j` elements yielded will be returned along with `i`'s remaining elements as |
73 | | /// `Diff::Shorter`. |
74 | | /// |
75 | | /// If the two elements of a step differ, the index of those elements along with the remaining |
76 | | /// elements of both `i` and `j` are returned as `Diff::FirstMismatch`. |
77 | | /// |
78 | | /// If `i` becomes exhausted before `j` becomes exhausted, the number of elements in `i` along with |
79 | | /// the remaining `j` elements will be returned as `Diff::Longer`. |
80 | 0 | pub fn diff_with<I, J, F>(i: I, j: J, is_equal: F) -> Option<Diff<I::IntoIter, J::IntoIter>> |
81 | 0 | where |
82 | 0 | I: IntoIterator, |
83 | 0 | J: IntoIterator, |
84 | 0 | F: Fn(&I::Item, &J::Item) -> bool, |
85 | 0 | { |
86 | 0 | let mut i = i.into_iter(); |
87 | 0 | let mut j = j.into_iter(); |
88 | 0 | let mut idx = 0; |
89 | 0 | while let Some(i_elem) = i.next() { |
90 | 0 | match j.next() { |
91 | 0 | None => return Some(Diff::Shorter(idx, put_back(i).with_value(i_elem))), |
92 | 0 | Some(j_elem) => { |
93 | 0 | if !is_equal(&i_elem, &j_elem) { |
94 | 0 | let remaining_i = put_back(i).with_value(i_elem); |
95 | 0 | let remaining_j = put_back(j).with_value(j_elem); |
96 | 0 | return Some(Diff::FirstMismatch(idx, remaining_i, remaining_j)); |
97 | 0 | } |
98 | 0 | } |
99 | 0 | } |
100 | 0 | idx += 1; |
101 | | } |
102 | 0 | j.next() |
103 | 0 | .map(|j_elem| Diff::Longer(idx, put_back(j).with_value(j_elem))) |
104 | 0 | } |