use core::assert_eq; use core::iter::*; use core::num::NonZero; use super::*; #[test] fn test_iterator_flatten() { let xs = [0, 3, 6]; let ys = [0, 1, 2, 3, 4, 5, 6, 7, 8]; let it = xs.iter().map(|&x| (x..).step_by(1).take(3)).flatten(); let mut i = 0; for x in it { assert_eq!(x, ys[i]); i += 1; } assert_eq!(i, ys.len()); } /// Tests `Flatten::fold` with items already picked off the front and back, /// to make sure all parts of the `Flatten` are folded correctly. #[test] fn test_iterator_flatten_fold() { let xs = [0, 3, 6]; let ys = [1, 2, 3, 4, 5, 6, 7]; let mut it = xs.iter().map(|&x| x..x + 3).flatten(); assert_eq!(it.next(), Some(0)); assert_eq!(it.next_back(), Some(8)); let i = it.fold(0, |i, x| { assert_eq!(x, ys[i]); i + 1 }); assert_eq!(i, ys.len()); let mut it = xs.iter().map(|&x| x..x + 3).flatten(); assert_eq!(it.next(), Some(0)); assert_eq!(it.next_back(), Some(8)); let i = it.rfold(ys.len(), |i, x| { assert_eq!(x, ys[i - 1]); i - 1 }); assert_eq!(i, 0); } #[test] fn test_flatten_try_folds() { let f = &|acc, x| i32::checked_add(acc * 2 / 3, x); let mr = &|x| (5 * x)..(5 * x + 5); assert_eq!((0..10).map(mr).flatten().try_fold(7, f), (0..50).try_fold(7, f)); assert_eq!((0..10).map(mr).flatten().try_rfold(7, f), (0..50).try_rfold(7, f)); let mut iter = (0..10).map(mr).flatten(); iter.next(); iter.next_back(); // have front and back iters in progress assert_eq!(iter.try_rfold(7, f), (1..49).try_rfold(7, f)); let mut iter = (0..10).map(|x| (4 * x)..(4 * x + 4)).flatten(); assert_eq!(iter.try_fold(0, i8::checked_add), None); assert_eq!(iter.next(), Some(17)); assert_eq!(iter.try_rfold(0, i8::checked_add), None); assert_eq!(iter.next_back(), Some(35)); } #[test] fn test_flatten_advance_by() { let mut it = once(0..10).chain(once(10..30)).chain(once(30..40)).flatten(); assert_eq!(it.advance_by(5), Ok(())); assert_eq!(it.next(), Some(5)); assert_eq!(it.advance_by(9), Ok(())); assert_eq!(it.next(), Some(15)); assert_eq!(it.advance_back_by(4), Ok(())); assert_eq!(it.next_back(), Some(35)); assert_eq!(it.advance_back_by(9), Ok(())); assert_eq!(it.next_back(), Some(25)); assert_eq!(it.advance_by(usize::MAX), Err(NonZero::new(usize::MAX - 9).unwrap())); assert_eq!(it.advance_back_by(usize::MAX), Err(NonZero::new(usize::MAX).unwrap())); assert_eq!(it.advance_by(0), Ok(())); assert_eq!(it.advance_back_by(0), Ok(())); assert_eq!(it.size_hint(), (0, Some(0))); } #[test] fn test_flatten_non_fused_outer() { let mut iter = NonFused::new(once(0..2)).flatten(); assert_eq!(iter.next_back(), Some(1)); assert_eq!(iter.next(), Some(0)); assert_eq!(iter.next(), None); assert_eq!(iter.next(), None); let mut iter = NonFused::new(once(0..2)).flatten(); assert_eq!(iter.next(), Some(0)); assert_eq!(iter.next_back(), Some(1)); assert_eq!(iter.next_back(), None); assert_eq!(iter.next_back(), None); } #[test] fn test_flatten_non_fused_inner() { let mut iter = once(0..1).chain(once(1..3)).flat_map(NonFused::new); assert_eq!(iter.next_back(), Some(2)); assert_eq!(iter.next(), Some(0)); assert_eq!(iter.next(), Some(1)); assert_eq!(iter.next(), None); assert_eq!(iter.next(), None); let mut iter = once(0..1).chain(once(1..3)).flat_map(NonFused::new); assert_eq!(iter.next(), Some(0)); assert_eq!(iter.next_back(), Some(2)); assert_eq!(iter.next_back(), Some(1)); assert_eq!(iter.next_back(), None); assert_eq!(iter.next_back(), None); } #[test] fn test_double_ended_flatten() { let u = [0, 1]; let v = [5, 6, 7, 8]; let mut it = u.iter().map(|x| &v[*x..v.len()]).flatten(); assert_eq!(it.next_back().unwrap(), &8); assert_eq!(it.next().unwrap(), &5); assert_eq!(it.next_back().unwrap(), &7); assert_eq!(it.next_back().unwrap(), &6); assert_eq!(it.next_back().unwrap(), &8); assert_eq!(it.next().unwrap(), &6); assert_eq!(it.next_back().unwrap(), &7); assert_eq!(it.next_back(), None); assert_eq!(it.next(), None); assert_eq!(it.next_back(), None); } #[test] fn test_trusted_len_flatten() { fn assert_trusted_len(_: &T) {} let mut iter = IntoIterator::into_iter([[0; 3]; 4]).flatten(); assert_trusted_len(&iter); assert_eq!(iter.size_hint(), (12, Some(12))); iter.next(); assert_eq!(iter.size_hint(), (11, Some(11))); iter.next_back(); assert_eq!(iter.size_hint(), (10, Some(10))); let iter = IntoIterator::into_iter([[(); usize::MAX]; 1]).flatten(); assert_eq!(iter.size_hint(), (usize::MAX, Some(usize::MAX))); let iter = IntoIterator::into_iter([[(); usize::MAX]; 2]).flatten(); assert_eq!(iter.size_hint(), (usize::MAX, None)); let mut a = [(); 10]; let mut b = [(); 10]; let iter = IntoIterator::into_iter([&mut a, &mut b]).flatten(); assert_trusted_len(&iter); assert_eq!(iter.size_hint(), (20, Some(20))); core::mem::drop(iter); let iter = IntoIterator::into_iter([&a, &b]).flatten(); assert_trusted_len(&iter); assert_eq!(iter.size_hint(), (20, Some(20))); let iter = [(), (), ()].iter().flat_map(|_| [(); 1000]); assert_trusted_len(&iter); assert_eq!(iter.size_hint(), (3000, Some(3000))); let iter = [(), ()].iter().flat_map(|_| &a); assert_trusted_len(&iter); assert_eq!(iter.size_hint(), (20, Some(20))); } #[test] fn test_flatten_count() { let mut it = once(0..10).chain(once(10..30)).chain(once(30..40)).flatten(); assert_eq!(it.clone().count(), 40); assert_eq!(it.advance_by(5), Ok(())); assert_eq!(it.clone().count(), 35); assert_eq!(it.advance_back_by(5), Ok(())); assert_eq!(it.clone().count(), 30); assert_eq!(it.advance_by(10), Ok(())); assert_eq!(it.clone().count(), 20); assert_eq!(it.advance_back_by(8), Ok(())); assert_eq!(it.clone().count(), 12); assert_eq!(it.advance_by(4), Ok(())); assert_eq!(it.clone().count(), 8); assert_eq!(it.advance_back_by(5), Ok(())); assert_eq!(it.clone().count(), 3); assert_eq!(it.advance_by(3), Ok(())); assert_eq!(it.clone().count(), 0); } #[test] fn test_flatten_last() { let mut it = once(0..10).chain(once(10..30)).chain(once(30..40)).flatten(); assert_eq!(it.clone().last(), Some(39)); assert_eq!(it.advance_by(5), Ok(())); // 5..40 assert_eq!(it.clone().last(), Some(39)); assert_eq!(it.advance_back_by(5), Ok(())); // 5..35 assert_eq!(it.clone().last(), Some(34)); assert_eq!(it.advance_by(10), Ok(())); // 15..35 assert_eq!(it.clone().last(), Some(34)); assert_eq!(it.advance_back_by(8), Ok(())); // 15..27 assert_eq!(it.clone().last(), Some(26)); assert_eq!(it.advance_by(4), Ok(())); // 19..27 assert_eq!(it.clone().last(), Some(26)); assert_eq!(it.advance_back_by(5), Ok(())); // 19..22 assert_eq!(it.clone().last(), Some(21)); assert_eq!(it.advance_by(3), Ok(())); // 22..22 assert_eq!(it.clone().last(), None); } #[test] fn test_flatten_one_shot() { // This could be `filter_map`, but people often do flatten options. let mut it = (0i8..10).flat_map(|i| NonZero::new(i % 7)); assert_eq!(it.size_hint(), (0, Some(10))); assert_eq!(it.clone().count(), 8); assert_eq!(it.clone().last(), NonZero::new(2)); // sum -> fold let sum: i8 = it.clone().map(|n| n.get()).sum(); assert_eq!(sum, 24); // the product overflows at 6, remaining are 7,8,9 -> 1,2 let one = NonZero::new(1i8).unwrap(); let product = it.try_fold(one, |acc, x| acc.checked_mul(x)); assert_eq!(product, None); assert_eq!(it.size_hint(), (0, Some(3))); assert_eq!(it.clone().count(), 2); assert_eq!(it.advance_by(0), Ok(())); assert_eq!(it.clone().next(), NonZero::new(1)); assert_eq!(it.advance_by(1), Ok(())); assert_eq!(it.clone().next(), NonZero::new(2)); assert_eq!(it.advance_by(100), Err(NonZero::new(99).unwrap())); assert_eq!(it.next(), None); } #[test] fn test_flatten_one_shot_rev() { let mut it = (0i8..10).flat_map(|i| NonZero::new(i % 7)).rev(); assert_eq!(it.size_hint(), (0, Some(10))); assert_eq!(it.clone().count(), 8); assert_eq!(it.clone().last(), NonZero::new(1)); // sum -> Rev fold -> rfold let sum: i8 = it.clone().map(|n| n.get()).sum(); assert_eq!(sum, 24); // Rev try_fold -> try_rfold // the product overflows at 4, remaining are 3,2,1,0 -> 3,2,1 let one = NonZero::new(1i8).unwrap(); let product = it.try_fold(one, |acc, x| acc.checked_mul(x)); assert_eq!(product, None); assert_eq!(it.size_hint(), (0, Some(4))); assert_eq!(it.clone().count(), 3); // Rev advance_by -> advance_back_by assert_eq!(it.advance_by(0), Ok(())); assert_eq!(it.clone().next(), NonZero::new(3)); assert_eq!(it.advance_by(1), Ok(())); assert_eq!(it.clone().next(), NonZero::new(2)); assert_eq!(it.advance_by(100), Err(NonZero::new(98).unwrap())); assert_eq!(it.next(), None); } #[test] fn test_flatten_one_shot_arrays() { let it = (0..10).flat_map(|i| [i]); assert_eq!(it.size_hint(), (10, Some(10))); assert_eq!(it.sum::(), 45); let mut it = (0..10).flat_map(|_| -> [i32; 0] { [] }); assert_eq!(it.size_hint(), (0, Some(0))); assert_eq!(it.next(), None); }