2019-02-03 08:27:44 +01:00
|
|
|
use std::collections::btree_map::Entry::{Occupied, Vacant};
|
2019-12-22 17:42:04 -05:00
|
|
|
use std::collections::BTreeMap;
|
2019-12-20 19:07:30 +01:00
|
|
|
use std::convert::TryFrom;
|
|
|
|
use std::fmt::Debug;
|
2019-12-22 17:42:04 -05:00
|
|
|
use std::iter::FromIterator;
|
2019-02-03 08:27:44 +01:00
|
|
|
use std::ops::Bound::{self, Excluded, Included, Unbounded};
|
2019-12-31 14:09:06 +01:00
|
|
|
use std::ops::RangeBounds;
|
2019-12-13 18:36:35 +01:00
|
|
|
use std::panic::catch_unwind;
|
2019-02-03 08:27:44 +01:00
|
|
|
use std::rc::Rc;
|
2019-12-13 18:36:35 +01:00
|
|
|
use std::sync::atomic::{AtomicU32, Ordering};
|
2015-03-10 23:58:16 -05:00
|
|
|
|
2016-05-29 13:30:13 +03:00
|
|
|
use super::DeterministicRng;
|
|
|
|
|
2015-03-10 23:58:16 -05:00
|
|
|
#[test]
|
|
|
|
fn test_basic_large() {
|
|
|
|
let mut map = BTreeMap::new();
|
2019-02-13 18:11:57 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
2015-03-10 23:58:16 -05:00
|
|
|
let size = 10000;
|
2019-02-13 18:11:57 +01:00
|
|
|
#[cfg(miri)]
|
2020-02-16 20:01:41 +01:00
|
|
|
let size = 144; // to obtain height 3 tree (having edges to both kinds of nodes)
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.len(), 0);
|
|
|
|
|
|
|
|
for i in 0..size {
|
2016-05-22 23:57:13 +05:30
|
|
|
assert_eq!(map.insert(i, 10 * i), None);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.len(), i + 1);
|
|
|
|
}
|
|
|
|
|
2020-02-04 22:19:05 +01:00
|
|
|
assert_eq!(map.first_key_value(), Some((&0, &0)));
|
|
|
|
assert_eq!(map.last_key_value(), Some((&(size - 1), &(10 * (size - 1)))));
|
|
|
|
assert_eq!(map.first_entry().unwrap().key(), &0);
|
|
|
|
assert_eq!(map.last_entry().unwrap().key(), &(size - 1));
|
|
|
|
|
2015-03-10 23:58:16 -05:00
|
|
|
for i in 0..size {
|
2016-05-22 23:57:13 +05:30
|
|
|
assert_eq!(map.get(&i).unwrap(), &(i * 10));
|
2015-03-10 23:58:16 -05:00
|
|
|
}
|
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
for i in size..size * 2 {
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.get(&i), None);
|
|
|
|
}
|
|
|
|
|
|
|
|
for i in 0..size {
|
2016-05-22 23:57:13 +05:30
|
|
|
assert_eq!(map.insert(i, 100 * i), Some(10 * i));
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.len(), size);
|
|
|
|
}
|
|
|
|
|
|
|
|
for i in 0..size {
|
2016-05-22 23:57:13 +05:30
|
|
|
assert_eq!(map.get(&i).unwrap(), &(i * 100));
|
2015-03-10 23:58:16 -05:00
|
|
|
}
|
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
for i in 0..size / 2 {
|
|
|
|
assert_eq!(map.remove(&(i * 2)), Some(i * 200));
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.len(), size - i - 1);
|
|
|
|
}
|
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
for i in 0..size / 2 {
|
|
|
|
assert_eq!(map.get(&(2 * i)), None);
|
|
|
|
assert_eq!(map.get(&(2 * i + 1)).unwrap(), &(i * 200 + 100));
|
2015-03-10 23:58:16 -05:00
|
|
|
}
|
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
for i in 0..size / 2 {
|
|
|
|
assert_eq!(map.remove(&(2 * i)), None);
|
|
|
|
assert_eq!(map.remove(&(2 * i + 1)), Some(i * 200 + 100));
|
|
|
|
assert_eq!(map.len(), size / 2 - i - 1);
|
2015-03-10 23:58:16 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_basic_small() {
|
|
|
|
let mut map = BTreeMap::new();
|
2019-12-20 19:07:30 +01:00
|
|
|
// Empty, shared root:
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.remove(&1), None);
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.len(), 0);
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.get(&1), None);
|
|
|
|
assert_eq!(map.get_mut(&1), None);
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.first_key_value(), None);
|
|
|
|
assert_eq!(map.last_key_value(), None);
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.keys().count(), 0);
|
|
|
|
assert_eq!(map.values().count(), 0);
|
2019-12-31 14:09:06 +01:00
|
|
|
assert_eq!(map.range(..).next(), None);
|
|
|
|
assert_eq!(map.range(..1).next(), None);
|
|
|
|
assert_eq!(map.range(1..).next(), None);
|
|
|
|
assert_eq!(map.range(1..=1).next(), None);
|
|
|
|
assert_eq!(map.range(1..2).next(), None);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.insert(1, 1), None);
|
2019-12-20 19:07:30 +01:00
|
|
|
|
|
|
|
// 1 key-value pair:
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.len(), 1);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.get(&1), Some(&1));
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.get_mut(&1), Some(&mut 1));
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.first_key_value(), Some((&1, &1)));
|
|
|
|
assert_eq!(map.last_key_value(), Some((&1, &1)));
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.keys().collect::<Vec<_>>(), vec![&1]);
|
|
|
|
assert_eq!(map.values().collect::<Vec<_>>(), vec![&1]);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.insert(1, 2), Some(1));
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.len(), 1);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.get(&1), Some(&2));
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.get_mut(&1), Some(&mut 2));
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.first_key_value(), Some((&1, &2)));
|
|
|
|
assert_eq!(map.last_key_value(), Some((&1, &2)));
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.keys().collect::<Vec<_>>(), vec![&1]);
|
|
|
|
assert_eq!(map.values().collect::<Vec<_>>(), vec![&2]);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.insert(2, 4), None);
|
2019-12-20 19:07:30 +01:00
|
|
|
|
|
|
|
// 2 key-value pairs:
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.len(), 2);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.get(&2), Some(&4));
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.get_mut(&2), Some(&mut 4));
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.first_key_value(), Some((&1, &2)));
|
|
|
|
assert_eq!(map.last_key_value(), Some((&2, &4)));
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.keys().collect::<Vec<_>>(), vec![&1, &2]);
|
|
|
|
assert_eq!(map.values().collect::<Vec<_>>(), vec![&2, &4]);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.remove(&1), Some(2));
|
2019-12-20 19:07:30 +01:00
|
|
|
|
|
|
|
// 1 key-value pair:
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.len(), 1);
|
|
|
|
assert_eq!(map.get(&1), None);
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.get_mut(&1), None);
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.get(&2), Some(&4));
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.get_mut(&2), Some(&mut 4));
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.first_key_value(), Some((&2, &4)));
|
|
|
|
assert_eq!(map.last_key_value(), Some((&2, &4)));
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.keys().collect::<Vec<_>>(), vec![&2]);
|
|
|
|
assert_eq!(map.values().collect::<Vec<_>>(), vec![&4]);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.remove(&2), Some(4));
|
2019-12-20 19:07:30 +01:00
|
|
|
|
|
|
|
// Empty but private root:
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.len(), 0);
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.get(&1), None);
|
|
|
|
assert_eq!(map.get_mut(&1), None);
|
2019-10-20 20:06:20 +02:00
|
|
|
assert_eq!(map.first_key_value(), None);
|
|
|
|
assert_eq!(map.last_key_value(), None);
|
2019-12-20 19:07:30 +01:00
|
|
|
assert_eq!(map.keys().count(), 0);
|
|
|
|
assert_eq!(map.values().count(), 0);
|
2019-12-31 14:09:06 +01:00
|
|
|
assert_eq!(map.range(..).next(), None);
|
|
|
|
assert_eq!(map.range(..1).next(), None);
|
|
|
|
assert_eq!(map.range(1..).next(), None);
|
|
|
|
assert_eq!(map.range(1..=1).next(), None);
|
|
|
|
assert_eq!(map.range(1..2).next(), None);
|
2015-03-10 23:58:16 -05:00
|
|
|
assert_eq!(map.remove(&1), None);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iter() {
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
2015-03-10 23:58:16 -05:00
|
|
|
let size = 10000;
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(miri)]
|
2019-02-13 18:11:57 +01:00
|
|
|
let size = 200;
|
2015-03-10 23:58:16 -05:00
|
|
|
|
|
|
|
let mut map: BTreeMap<_, _> = (0..size).map(|i| (i, i)).collect();
|
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
fn test<T>(size: usize, mut iter: T)
|
2019-12-22 17:42:04 -05:00
|
|
|
where
|
|
|
|
T: Iterator<Item = (usize, usize)>,
|
2016-05-22 23:57:13 +05:30
|
|
|
{
|
2015-03-10 23:58:16 -05:00
|
|
|
for i in 0..size {
|
|
|
|
assert_eq!(iter.size_hint(), (size - i, Some(size - i)));
|
|
|
|
assert_eq!(iter.next().unwrap(), (i, i));
|
|
|
|
}
|
|
|
|
assert_eq!(iter.size_hint(), (0, Some(0)));
|
|
|
|
assert_eq!(iter.next(), None);
|
|
|
|
}
|
|
|
|
test(size, map.iter().map(|(&k, &v)| (k, v)));
|
|
|
|
test(size, map.iter_mut().map(|(&k, &mut v)| (k, v)));
|
|
|
|
test(size, map.into_iter());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iter_rev() {
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
2015-03-10 23:58:16 -05:00
|
|
|
let size = 10000;
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(miri)]
|
2019-02-13 18:11:57 +01:00
|
|
|
let size = 200;
|
2015-03-10 23:58:16 -05:00
|
|
|
|
|
|
|
let mut map: BTreeMap<_, _> = (0..size).map(|i| (i, i)).collect();
|
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
fn test<T>(size: usize, mut iter: T)
|
2019-12-22 17:42:04 -05:00
|
|
|
where
|
|
|
|
T: Iterator<Item = (usize, usize)>,
|
2016-05-22 23:57:13 +05:30
|
|
|
{
|
2015-03-10 23:58:16 -05:00
|
|
|
for i in 0..size {
|
|
|
|
assert_eq!(iter.size_hint(), (size - i, Some(size - i)));
|
|
|
|
assert_eq!(iter.next().unwrap(), (size - i - 1, size - i - 1));
|
|
|
|
}
|
|
|
|
assert_eq!(iter.size_hint(), (0, Some(0)));
|
|
|
|
assert_eq!(iter.next(), None);
|
|
|
|
}
|
|
|
|
test(size, map.iter().rev().map(|(&k, &v)| (k, v)));
|
|
|
|
test(size, map.iter_mut().rev().map(|(&k, &mut v)| (k, v)));
|
|
|
|
test(size, map.into_iter().rev());
|
|
|
|
}
|
|
|
|
|
2019-12-20 19:07:30 +01:00
|
|
|
/// Specifically tests iter_mut's ability to mutate the value of pairs in-line
|
|
|
|
fn do_test_iter_mut_mutation<T>(size: usize)
|
|
|
|
where
|
|
|
|
T: Copy + Debug + Ord + TryFrom<usize>,
|
|
|
|
<T as std::convert::TryFrom<usize>>::Error: std::fmt::Debug,
|
|
|
|
{
|
|
|
|
let zero = T::try_from(0).unwrap();
|
|
|
|
let mut map: BTreeMap<T, T> = (0..size).map(|i| (T::try_from(i).unwrap(), zero)).collect();
|
|
|
|
|
|
|
|
// Forward and backward iteration sees enough pairs (also tested elsewhere)
|
|
|
|
assert_eq!(map.iter_mut().count(), size);
|
|
|
|
assert_eq!(map.iter_mut().rev().count(), size);
|
|
|
|
|
|
|
|
// Iterate forwards, trying to mutate to unique values
|
|
|
|
for (i, (k, v)) in map.iter_mut().enumerate() {
|
|
|
|
assert_eq!(*k, T::try_from(i).unwrap());
|
|
|
|
assert_eq!(*v, zero);
|
|
|
|
*v = T::try_from(i + 1).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate backwards, checking that mutations succeeded and trying to mutate again
|
|
|
|
for (i, (k, v)) in map.iter_mut().rev().enumerate() {
|
|
|
|
assert_eq!(*k, T::try_from(size - i - 1).unwrap());
|
|
|
|
assert_eq!(*v, T::try_from(size - i).unwrap());
|
|
|
|
*v = T::try_from(2 * size - i).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that backward mutations succeeded
|
|
|
|
for (i, (k, v)) in map.iter_mut().enumerate() {
|
|
|
|
assert_eq!(*k, T::try_from(i).unwrap());
|
|
|
|
assert_eq!(*v, T::try_from(size + i + 1).unwrap());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq, PartialOrd, Ord)]
|
|
|
|
#[repr(align(32))]
|
|
|
|
struct Align32(usize);
|
|
|
|
|
|
|
|
impl TryFrom<usize> for Align32 {
|
|
|
|
type Error = ();
|
|
|
|
|
|
|
|
fn try_from(s: usize) -> Result<Align32, ()> {
|
|
|
|
Ok(Align32(s))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iter_mut_mutation() {
|
|
|
|
// Check many alignments because various fields precede array in NodeHeader.
|
|
|
|
// Check with size 0 which should not iterate at all.
|
|
|
|
// Check with size 1 for a tree with one kind of node (root = leaf).
|
|
|
|
// Check with size 12 for a tree with two kinds of nodes (root and leaves).
|
|
|
|
// Check with size 144 for a tree with all kinds of nodes (root, internals and leaves).
|
|
|
|
do_test_iter_mut_mutation::<u8>(0);
|
|
|
|
do_test_iter_mut_mutation::<u8>(1);
|
|
|
|
do_test_iter_mut_mutation::<u8>(12);
|
|
|
|
do_test_iter_mut_mutation::<u8>(127); // not enough unique values to test 144
|
|
|
|
do_test_iter_mut_mutation::<u16>(1);
|
|
|
|
do_test_iter_mut_mutation::<u16>(12);
|
|
|
|
do_test_iter_mut_mutation::<u16>(144);
|
|
|
|
do_test_iter_mut_mutation::<u32>(1);
|
|
|
|
do_test_iter_mut_mutation::<u32>(12);
|
|
|
|
do_test_iter_mut_mutation::<u32>(144);
|
|
|
|
do_test_iter_mut_mutation::<u64>(1);
|
|
|
|
do_test_iter_mut_mutation::<u64>(12);
|
|
|
|
do_test_iter_mut_mutation::<u64>(144);
|
|
|
|
do_test_iter_mut_mutation::<u128>(1);
|
|
|
|
do_test_iter_mut_mutation::<u128>(12);
|
|
|
|
do_test_iter_mut_mutation::<u128>(144);
|
|
|
|
do_test_iter_mut_mutation::<Align32>(1);
|
|
|
|
do_test_iter_mut_mutation::<Align32>(12);
|
|
|
|
do_test_iter_mut_mutation::<Align32>(144);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_into_key_slice_with_shared_root_past_bounds() {
|
|
|
|
let mut map: BTreeMap<Align32, ()> = BTreeMap::new();
|
|
|
|
assert_eq!(map.get(&Align32(1)), None);
|
|
|
|
assert_eq!(map.get_mut(&Align32(1)), None);
|
|
|
|
}
|
|
|
|
|
2016-03-30 19:12:29 -04:00
|
|
|
#[test]
|
|
|
|
fn test_values_mut() {
|
|
|
|
let mut a = BTreeMap::new();
|
|
|
|
a.insert(1, String::from("hello"));
|
|
|
|
a.insert(2, String::from("goodbye"));
|
|
|
|
|
|
|
|
for value in a.values_mut() {
|
|
|
|
value.push_str("!");
|
|
|
|
}
|
|
|
|
|
|
|
|
let values: Vec<String> = a.values().cloned().collect();
|
2016-05-22 23:57:13 +05:30
|
|
|
assert_eq!(values, [String::from("hello!"), String::from("goodbye!")]);
|
2016-03-30 19:12:29 -04:00
|
|
|
}
|
|
|
|
|
2015-03-10 23:58:16 -05:00
|
|
|
#[test]
|
|
|
|
fn test_iter_mixed() {
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
2015-03-10 23:58:16 -05:00
|
|
|
let size = 10000;
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(miri)]
|
2019-02-13 18:11:57 +01:00
|
|
|
let size = 200;
|
2015-03-10 23:58:16 -05:00
|
|
|
|
|
|
|
let mut map: BTreeMap<_, _> = (0..size).map(|i| (i, i)).collect();
|
|
|
|
|
|
|
|
fn test<T>(size: usize, mut iter: T)
|
2019-12-22 17:42:04 -05:00
|
|
|
where
|
|
|
|
T: Iterator<Item = (usize, usize)> + DoubleEndedIterator,
|
2016-05-22 23:57:13 +05:30
|
|
|
{
|
2015-03-10 23:58:16 -05:00
|
|
|
for i in 0..size / 4 {
|
|
|
|
assert_eq!(iter.size_hint(), (size - i * 2, Some(size - i * 2)));
|
|
|
|
assert_eq!(iter.next().unwrap(), (i, i));
|
|
|
|
assert_eq!(iter.next_back().unwrap(), (size - i - 1, size - i - 1));
|
|
|
|
}
|
|
|
|
for i in size / 4..size * 3 / 4 {
|
|
|
|
assert_eq!(iter.size_hint(), (size * 3 / 4 - i, Some(size * 3 / 4 - i)));
|
|
|
|
assert_eq!(iter.next().unwrap(), (i, i));
|
|
|
|
}
|
|
|
|
assert_eq!(iter.size_hint(), (0, Some(0)));
|
|
|
|
assert_eq!(iter.next(), None);
|
|
|
|
}
|
|
|
|
test(size, map.iter().map(|(&k, &v)| (k, v)));
|
|
|
|
test(size, map.iter_mut().map(|(&k, &mut v)| (k, v)));
|
|
|
|
test(size, map.into_iter());
|
|
|
|
}
|
|
|
|
|
2019-12-31 14:09:06 +01:00
|
|
|
fn range_keys(map: &BTreeMap<i32, i32>, range: impl RangeBounds<i32>) -> Vec<i32> {
|
|
|
|
map.range(range)
|
|
|
|
.map(|(&k, &v)| {
|
|
|
|
assert_eq!(k, v);
|
|
|
|
k
|
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
2015-03-10 23:58:16 -05:00
|
|
|
#[test]
|
|
|
|
fn test_range_small() {
|
2019-12-31 14:09:06 +01:00
|
|
|
let size = 4;
|
|
|
|
|
|
|
|
let map: BTreeMap<_, _> = (1..=size).map(|i| (i, i)).collect();
|
|
|
|
let all: Vec<_> = (1..=size).collect();
|
|
|
|
let (first, last) = (vec![all[0]], vec![all[size as usize - 1]]);
|
|
|
|
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Excluded(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Included(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Included(size))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Unbounded)), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Excluded(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Included(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Included(size))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Unbounded)), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Excluded(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Included(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Included(size))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Unbounded)), all);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Excluded(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Included(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Included(size))), all);
|
|
|
|
assert_eq!(range_keys(&map, ..), all);
|
|
|
|
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Excluded(1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Included(0))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Included(0))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Excluded(1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Excluded(1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Included(0))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Excluded(2))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Included(1))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Excluded(2))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Included(1))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Excluded(2))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Included(1))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Excluded(2))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Included(1))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size - 1), Excluded(size + 1))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size - 1), Included(size + 1))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size - 1), Included(size))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size - 1), Unbounded)), last);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size), Excluded(size + 1))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size), Included(size + 1))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size), Included(size))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size), Unbounded)), last);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size), Excluded(size + 1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size), Included(size))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size), Unbounded)), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size + 1), Excluded(size + 1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size + 1), Included(size + 1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size + 1), Unbounded)), vec![]);
|
|
|
|
|
|
|
|
assert_eq!(range_keys(&map, ..3), vec![1, 2]);
|
|
|
|
assert_eq!(range_keys(&map, 3..), vec![3, 4]);
|
|
|
|
assert_eq!(range_keys(&map, 2..=3), vec![2, 3]);
|
|
|
|
}
|
2015-03-10 23:58:16 -05:00
|
|
|
|
2019-12-31 14:09:06 +01:00
|
|
|
#[test]
|
2020-02-16 20:01:41 +01:00
|
|
|
fn test_range_height_2() {
|
|
|
|
// Assuming that node.CAPACITY is 11, having 12 pairs implies a height 2 tree
|
2019-12-31 14:09:06 +01:00
|
|
|
// with 2 leaves. Depending on details we don't want or need to rely upon,
|
|
|
|
// the single key at the root will be 6 or 7.
|
|
|
|
|
|
|
|
let map: BTreeMap<_, _> = (1..=12).map(|i| (i, i)).collect();
|
|
|
|
for &root in &[6, 7] {
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(root), Excluded(root + 1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(root), Included(root + 1))), vec![root + 1]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(root), Excluded(root + 1))), vec![root]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(root), Included(root + 1))), vec![root, root + 1]);
|
|
|
|
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(root - 1), Excluded(root))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(root - 1), Excluded(root))), vec![root - 1]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(root - 1), Included(root))), vec![root]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(root - 1), Included(root))), vec![root - 1, root]);
|
2015-03-10 23:58:16 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-18 18:57:18 +00:00
|
|
|
#[test]
|
2019-12-31 14:09:06 +01:00
|
|
|
fn test_range_large() {
|
|
|
|
let size = 200;
|
2017-02-18 18:57:18 +00:00
|
|
|
|
2019-12-31 14:09:06 +01:00
|
|
|
let map: BTreeMap<_, _> = (1..=size).map(|i| (i, i)).collect();
|
|
|
|
let all: Vec<_> = (1..=size).collect();
|
|
|
|
let (first, last) = (vec![all[0]], vec![all[size as usize - 1]]);
|
|
|
|
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Excluded(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Included(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Included(size))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Unbounded)), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Excluded(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Included(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Included(size))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Unbounded)), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Excluded(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Included(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Included(size))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Unbounded)), all);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Excluded(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Included(size + 1))), all);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Included(size))), all);
|
|
|
|
assert_eq!(range_keys(&map, ..), all);
|
|
|
|
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Excluded(1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Included(0))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Included(0))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Excluded(1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Excluded(1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Included(0))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Excluded(2))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(0), Included(1))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Excluded(2))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Included(0), Included(1))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Excluded(2))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Included(1), Included(1))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Excluded(2))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Unbounded, Included(1))), first);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size - 1), Excluded(size + 1))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size - 1), Included(size + 1))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size - 1), Included(size))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size - 1), Unbounded)), last);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size), Excluded(size + 1))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size), Included(size + 1))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size), Included(size))), last);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size), Unbounded)), last);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size), Excluded(size + 1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size), Included(size))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Excluded(size), Unbounded)), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size + 1), Excluded(size + 1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size + 1), Included(size + 1))), vec![]);
|
|
|
|
assert_eq!(range_keys(&map, (Included(size + 1), Unbounded)), vec![]);
|
2017-02-18 18:57:18 +00:00
|
|
|
|
|
|
|
fn check<'a, L, R>(lhs: L, rhs: R)
|
2019-12-22 17:42:04 -05:00
|
|
|
where
|
|
|
|
L: IntoIterator<Item = (&'a i32, &'a i32)>,
|
|
|
|
R: IntoIterator<Item = (&'a i32, &'a i32)>,
|
2017-02-18 18:57:18 +00:00
|
|
|
{
|
|
|
|
let lhs: Vec<_> = lhs.into_iter().collect();
|
|
|
|
let rhs: Vec<_> = rhs.into_iter().collect();
|
|
|
|
assert_eq!(lhs, rhs);
|
|
|
|
}
|
|
|
|
|
2019-12-31 14:09:06 +01:00
|
|
|
check(map.range(..=100), map.range(..101));
|
2017-09-19 05:40:04 +00:00
|
|
|
check(map.range(5..=8), vec![(&5, &5), (&6, &6), (&7, &7), (&8, &8)]);
|
2019-12-31 14:09:06 +01:00
|
|
|
check(map.range(-1..=2), vec![(&1, &1), (&2, &2)]);
|
2017-02-18 18:57:18 +00:00
|
|
|
}
|
|
|
|
|
2017-02-18 19:09:11 +00:00
|
|
|
#[test]
|
|
|
|
fn test_range_inclusive_max_value() {
|
2019-02-02 10:34:36 +01:00
|
|
|
let max = std::usize::MAX;
|
2017-02-18 19:09:11 +00:00
|
|
|
let map: BTreeMap<_, _> = vec![(max, 0)].into_iter().collect();
|
|
|
|
|
2017-09-19 05:40:04 +00:00
|
|
|
assert_eq!(map.range(max..=max).collect::<Vec<_>>(), &[(&max, &0)]);
|
2017-02-18 19:09:11 +00:00
|
|
|
}
|
|
|
|
|
2017-02-02 00:58:18 -06:00
|
|
|
#[test]
|
|
|
|
fn test_range_equal_empty_cases() {
|
|
|
|
let map: BTreeMap<_, _> = (0..5).map(|i| (i, i)).collect();
|
|
|
|
assert_eq!(map.range((Included(2), Excluded(2))).next(), None);
|
|
|
|
assert_eq!(map.range((Excluded(2), Included(2))).next(), None);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn test_range_equal_excluded() {
|
|
|
|
let map: BTreeMap<_, _> = (0..5).map(|i| (i, i)).collect();
|
|
|
|
map.range((Excluded(2), Excluded(2)));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn test_range_backwards_1() {
|
|
|
|
let map: BTreeMap<_, _> = (0..5).map(|i| (i, i)).collect();
|
|
|
|
map.range((Included(3), Included(2)));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn test_range_backwards_2() {
|
|
|
|
let map: BTreeMap<_, _> = (0..5).map(|i| (i, i)).collect();
|
|
|
|
map.range((Included(3), Excluded(2)));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn test_range_backwards_3() {
|
|
|
|
let map: BTreeMap<_, _> = (0..5).map(|i| (i, i)).collect();
|
|
|
|
map.range((Excluded(3), Included(2)));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn test_range_backwards_4() {
|
|
|
|
let map: BTreeMap<_, _> = (0..5).map(|i| (i, i)).collect();
|
|
|
|
map.range((Excluded(3), Excluded(2)));
|
|
|
|
}
|
|
|
|
|
2015-03-10 23:58:16 -05:00
|
|
|
#[test]
|
|
|
|
fn test_range_1000() {
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
2015-03-10 23:58:16 -05:00
|
|
|
let size = 1000;
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(miri)]
|
2020-02-16 20:01:41 +01:00
|
|
|
let size = 144; // to obtain height 3 tree (having edges to both kinds of nodes)
|
2015-03-10 23:58:16 -05:00
|
|
|
let map: BTreeMap<_, _> = (0..size).map(|i| (i, i)).collect();
|
|
|
|
|
|
|
|
fn test(map: &BTreeMap<u32, u32>, size: u32, min: Bound<&u32>, max: Bound<&u32>) {
|
2016-12-23 23:51:32 +00:00
|
|
|
let mut kvs = map.range((min, max)).map(|(&k, &v)| (k, v));
|
2015-03-10 23:58:16 -05:00
|
|
|
let mut pairs = (0..size).map(|i| (i, i));
|
|
|
|
|
|
|
|
for (kv, pair) in kvs.by_ref().zip(pairs.by_ref()) {
|
|
|
|
assert_eq!(kv, pair);
|
|
|
|
}
|
|
|
|
assert_eq!(kvs.next(), None);
|
|
|
|
assert_eq!(pairs.next(), None);
|
|
|
|
}
|
|
|
|
test(&map, size, Included(&0), Excluded(&size));
|
|
|
|
test(&map, size, Unbounded, Excluded(&size));
|
|
|
|
test(&map, size, Included(&0), Included(&(size - 1)));
|
|
|
|
test(&map, size, Unbounded, Included(&(size - 1)));
|
|
|
|
test(&map, size, Included(&0), Unbounded);
|
|
|
|
test(&map, size, Unbounded, Unbounded);
|
|
|
|
}
|
|
|
|
|
2016-12-23 23:51:32 +00:00
|
|
|
#[test]
|
|
|
|
fn test_range_borrowed_key() {
|
|
|
|
let mut map = BTreeMap::new();
|
|
|
|
map.insert("aardvark".to_string(), 1);
|
|
|
|
map.insert("baboon".to_string(), 2);
|
|
|
|
map.insert("coyote".to_string(), 3);
|
|
|
|
map.insert("dingo".to_string(), 4);
|
2016-12-24 23:45:30 +00:00
|
|
|
// NOTE: would like to use simply "b".."d" here...
|
2019-12-22 17:42:04 -05:00
|
|
|
let mut iter = map.range::<str, _>((Included("b"), Excluded("d")));
|
2016-12-23 23:51:32 +00:00
|
|
|
assert_eq!(iter.next(), Some((&"baboon".to_string(), &2)));
|
|
|
|
assert_eq!(iter.next(), Some((&"coyote".to_string(), &3)));
|
|
|
|
assert_eq!(iter.next(), None);
|
|
|
|
}
|
|
|
|
|
2015-03-10 23:58:16 -05:00
|
|
|
#[test]
|
|
|
|
fn test_range() {
|
|
|
|
let size = 200;
|
2020-02-16 20:01:41 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
|
|
|
let step = 1;
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(miri)]
|
2020-02-16 20:01:41 +01:00
|
|
|
let step = 66;
|
2015-03-10 23:58:16 -05:00
|
|
|
let map: BTreeMap<_, _> = (0..size).map(|i| (i, i)).collect();
|
|
|
|
|
2020-02-16 20:01:41 +01:00
|
|
|
for i in (0..size).step_by(step) {
|
|
|
|
for j in (i..size).step_by(step) {
|
2016-12-23 23:51:32 +00:00
|
|
|
let mut kvs = map.range((Included(&i), Included(&j))).map(|(&k, &v)| (k, v));
|
2018-12-04 11:17:58 -08:00
|
|
|
let mut pairs = (i..=j).map(|i| (i, i));
|
2015-03-10 23:58:16 -05:00
|
|
|
|
|
|
|
for (kv, pair) in kvs.by_ref().zip(pairs.by_ref()) {
|
|
|
|
assert_eq!(kv, pair);
|
|
|
|
}
|
|
|
|
assert_eq!(kvs.next(), None);
|
|
|
|
assert_eq!(pairs.next(), None);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-24 23:46:00 +00:00
|
|
|
#[test]
|
|
|
|
fn test_range_mut() {
|
|
|
|
let size = 200;
|
2020-02-16 20:01:41 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
|
|
|
let step = 1;
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(miri)]
|
2020-02-16 20:01:41 +01:00
|
|
|
let step = 66;
|
2016-12-24 23:46:00 +00:00
|
|
|
let mut map: BTreeMap<_, _> = (0..size).map(|i| (i, i)).collect();
|
|
|
|
|
2020-02-16 20:01:41 +01:00
|
|
|
for i in (0..size).step_by(step) {
|
|
|
|
for j in (i..size).step_by(step) {
|
2016-12-24 23:46:00 +00:00
|
|
|
let mut kvs = map.range_mut((Included(&i), Included(&j))).map(|(&k, &mut v)| (k, v));
|
2018-12-04 11:17:58 -08:00
|
|
|
let mut pairs = (i..=j).map(|i| (i, i));
|
2016-12-24 23:46:00 +00:00
|
|
|
|
|
|
|
for (kv, pair) in kvs.by_ref().zip(pairs.by_ref()) {
|
|
|
|
assert_eq!(kv, pair);
|
|
|
|
}
|
|
|
|
assert_eq!(kvs.next(), None);
|
|
|
|
assert_eq!(pairs.next(), None);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-09 16:14:48 +02:00
|
|
|
#[test]
|
|
|
|
fn test_borrow() {
|
|
|
|
// make sure these compile -- using the Borrow trait
|
|
|
|
{
|
|
|
|
let mut map = BTreeMap::new();
|
|
|
|
map.insert("0".to_string(), 1);
|
|
|
|
assert_eq!(map["0"], 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
let mut map = BTreeMap::new();
|
|
|
|
map.insert(Box::new(0), 1);
|
|
|
|
assert_eq!(map[&0], 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
let mut map = BTreeMap::new();
|
|
|
|
map.insert(Box::new([0, 1]) as Box<[i32]>, 1);
|
|
|
|
assert_eq!(map[&[0, 1][..]], 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
let mut map = BTreeMap::new();
|
|
|
|
map.insert(Rc::new(0), 1);
|
|
|
|
assert_eq!(map[&0], 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-10 23:58:16 -05:00
|
|
|
#[test]
|
2016-05-22 23:57:13 +05:30
|
|
|
fn test_entry() {
|
2015-03-10 23:58:16 -05:00
|
|
|
let xs = [(1, 10), (2, 20), (3, 30), (4, 40), (5, 50), (6, 60)];
|
|
|
|
|
|
|
|
let mut map: BTreeMap<_, _> = xs.iter().cloned().collect();
|
|
|
|
|
|
|
|
// Existing key (insert)
|
|
|
|
match map.entry(1) {
|
|
|
|
Vacant(_) => unreachable!(),
|
|
|
|
Occupied(mut view) => {
|
|
|
|
assert_eq!(view.get(), &10);
|
|
|
|
assert_eq!(view.insert(100), 10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert_eq!(map.get(&1).unwrap(), &100);
|
|
|
|
assert_eq!(map.len(), 6);
|
|
|
|
|
|
|
|
// Existing key (update)
|
|
|
|
match map.entry(2) {
|
|
|
|
Vacant(_) => unreachable!(),
|
|
|
|
Occupied(mut view) => {
|
|
|
|
let v = view.get_mut();
|
|
|
|
*v *= 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert_eq!(map.get(&2).unwrap(), &200);
|
|
|
|
assert_eq!(map.len(), 6);
|
|
|
|
|
|
|
|
// Existing key (take)
|
|
|
|
match map.entry(3) {
|
|
|
|
Vacant(_) => unreachable!(),
|
|
|
|
Occupied(view) => {
|
|
|
|
assert_eq!(view.remove(), 30);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert_eq!(map.get(&3), None);
|
|
|
|
assert_eq!(map.len(), 5);
|
|
|
|
|
|
|
|
// Inexistent key (insert)
|
|
|
|
match map.entry(10) {
|
|
|
|
Occupied(_) => unreachable!(),
|
|
|
|
Vacant(view) => {
|
|
|
|
assert_eq!(*view.insert(1000), 1000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert_eq!(map.get(&10).unwrap(), &1000);
|
|
|
|
assert_eq!(map.len(), 6);
|
|
|
|
}
|
|
|
|
|
2015-06-03 12:38:42 +02:00
|
|
|
#[test]
|
|
|
|
fn test_extend_ref() {
|
|
|
|
let mut a = BTreeMap::new();
|
|
|
|
a.insert(1, "one");
|
|
|
|
let mut b = BTreeMap::new();
|
|
|
|
b.insert(2, "two");
|
|
|
|
b.insert(3, "three");
|
|
|
|
|
|
|
|
a.extend(&b);
|
|
|
|
|
|
|
|
assert_eq!(a.len(), 3);
|
|
|
|
assert_eq!(a[&1], "one");
|
|
|
|
assert_eq!(a[&2], "two");
|
|
|
|
assert_eq!(a[&3], "three");
|
|
|
|
}
|
|
|
|
|
2015-09-18 14:15:02 -04:00
|
|
|
#[test]
|
|
|
|
fn test_zst() {
|
|
|
|
let mut m = BTreeMap::new();
|
|
|
|
assert_eq!(m.len(), 0);
|
|
|
|
|
|
|
|
assert_eq!(m.insert((), ()), None);
|
|
|
|
assert_eq!(m.len(), 1);
|
|
|
|
|
|
|
|
assert_eq!(m.insert((), ()), Some(()));
|
|
|
|
assert_eq!(m.len(), 1);
|
|
|
|
assert_eq!(m.iter().count(), 1);
|
|
|
|
|
|
|
|
m.clear();
|
|
|
|
assert_eq!(m.len(), 0);
|
|
|
|
|
|
|
|
for _ in 0..100 {
|
|
|
|
m.insert((), ());
|
|
|
|
}
|
|
|
|
|
|
|
|
assert_eq!(m.len(), 1);
|
|
|
|
assert_eq!(m.iter().count(), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// This test's only purpose is to ensure that zero-sized keys with nonsensical orderings
|
|
|
|
// do not cause segfaults when used with zero-sized values. All other map behavior is
|
|
|
|
// undefined.
|
|
|
|
#[test]
|
|
|
|
fn test_bad_zst() {
|
|
|
|
use std::cmp::Ordering;
|
|
|
|
|
|
|
|
struct Bad;
|
|
|
|
|
|
|
|
impl PartialEq for Bad {
|
2016-05-22 23:57:13 +05:30
|
|
|
fn eq(&self, _: &Self) -> bool {
|
|
|
|
false
|
|
|
|
}
|
2015-09-18 14:15:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for Bad {}
|
|
|
|
|
|
|
|
impl PartialOrd for Bad {
|
2016-05-22 23:57:13 +05:30
|
|
|
fn partial_cmp(&self, _: &Self) -> Option<Ordering> {
|
|
|
|
Some(Ordering::Less)
|
|
|
|
}
|
2015-09-18 14:15:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Ord for Bad {
|
2016-05-22 23:57:13 +05:30
|
|
|
fn cmp(&self, _: &Self) -> Ordering {
|
|
|
|
Ordering::Less
|
|
|
|
}
|
2015-09-18 14:15:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut m = BTreeMap::new();
|
|
|
|
|
|
|
|
for _ in 0..100 {
|
|
|
|
m.insert(Bad, Bad);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-16 22:55:56 -06:00
|
|
|
#[test]
|
|
|
|
fn test_clone() {
|
|
|
|
let mut map = BTreeMap::new();
|
2020-02-16 20:01:41 +01:00
|
|
|
let size = 12; // to obtain height 2 tree (having edges to leaf nodes)
|
2016-01-16 22:55:56 -06:00
|
|
|
assert_eq!(map.len(), 0);
|
|
|
|
|
|
|
|
for i in 0..size {
|
2016-05-22 23:57:13 +05:30
|
|
|
assert_eq!(map.insert(i, 10 * i), None);
|
2016-01-16 22:55:56 -06:00
|
|
|
assert_eq!(map.len(), i + 1);
|
|
|
|
assert_eq!(map, map.clone());
|
|
|
|
}
|
|
|
|
|
|
|
|
for i in 0..size {
|
2016-05-22 23:57:13 +05:30
|
|
|
assert_eq!(map.insert(i, 100 * i), Some(10 * i));
|
2016-01-16 22:55:56 -06:00
|
|
|
assert_eq!(map.len(), size);
|
|
|
|
assert_eq!(map, map.clone());
|
|
|
|
}
|
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
for i in 0..size / 2 {
|
|
|
|
assert_eq!(map.remove(&(i * 2)), Some(i * 200));
|
2016-01-16 22:55:56 -06:00
|
|
|
assert_eq!(map.len(), size - i - 1);
|
|
|
|
assert_eq!(map, map.clone());
|
|
|
|
}
|
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
for i in 0..size / 2 {
|
|
|
|
assert_eq!(map.remove(&(2 * i)), None);
|
|
|
|
assert_eq!(map.remove(&(2 * i + 1)), Some(i * 200 + 100));
|
|
|
|
assert_eq!(map.len(), size / 2 - i - 1);
|
2016-01-16 22:55:56 -06:00
|
|
|
assert_eq!(map, map.clone());
|
|
|
|
}
|
2020-02-16 20:01:41 +01:00
|
|
|
|
|
|
|
// Full 2-level and minimal 3-level tree (sizes 143, 144 -- the only ones we clone for).
|
|
|
|
for i in 1..=144 {
|
|
|
|
assert_eq!(map.insert(i, i), None);
|
|
|
|
assert_eq!(map.len(), i);
|
|
|
|
if i >= 143 {
|
|
|
|
assert_eq!(map, map.clone());
|
|
|
|
}
|
|
|
|
}
|
2016-01-16 22:55:56 -06:00
|
|
|
}
|
|
|
|
|
2019-11-22 14:22:45 -05:00
|
|
|
#[test]
|
|
|
|
fn test_clone_from() {
|
|
|
|
let mut map1 = BTreeMap::new();
|
2020-02-16 20:01:41 +01:00
|
|
|
let max_size = 12; // to obtain height 2 tree (having edges to leaf nodes)
|
2019-11-22 14:22:45 -05:00
|
|
|
|
2020-02-16 20:01:41 +01:00
|
|
|
// Range to max_size inclusive, because i is the size of map1 being tested.
|
|
|
|
for i in 0..=max_size {
|
2019-11-22 14:22:45 -05:00
|
|
|
let mut map2 = BTreeMap::new();
|
|
|
|
for j in 0..i {
|
|
|
|
let mut map1_copy = map2.clone();
|
2020-02-16 20:01:41 +01:00
|
|
|
map1_copy.clone_from(&map1); // small cloned from large
|
2019-11-22 14:22:45 -05:00
|
|
|
assert_eq!(map1_copy, map1);
|
|
|
|
let mut map2_copy = map1.clone();
|
2020-02-16 20:01:41 +01:00
|
|
|
map2_copy.clone_from(&map2); // large cloned from small
|
2019-11-22 14:22:45 -05:00
|
|
|
assert_eq!(map2_copy, map2);
|
2020-01-28 11:39:48 -05:00
|
|
|
map2.insert(100 * j + 1, 2 * j + 1);
|
2019-11-22 14:22:45 -05:00
|
|
|
}
|
2020-02-16 20:01:41 +01:00
|
|
|
map2.clone_from(&map1); // same length
|
|
|
|
assert_eq!(map2, map1);
|
2020-01-28 11:39:48 -05:00
|
|
|
map1.insert(i, 10 * i);
|
2019-11-22 14:22:45 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-17 09:47:21 -06:00
|
|
|
#[test]
|
2016-01-30 16:19:37 -08:00
|
|
|
#[allow(dead_code)]
|
2016-01-17 09:47:21 -06:00
|
|
|
fn test_variance() {
|
2019-12-22 17:42:04 -05:00
|
|
|
use std::collections::btree_map::{IntoIter, Iter, Keys, Range, Values};
|
2016-01-17 09:47:21 -06:00
|
|
|
|
2016-05-22 23:57:13 +05:30
|
|
|
fn map_key<'new>(v: BTreeMap<&'static str, ()>) -> BTreeMap<&'new str, ()> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn map_val<'new>(v: BTreeMap<(), &'static str>) -> BTreeMap<(), &'new str> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn iter_key<'a, 'new>(v: Iter<'a, &'static str, ()>) -> Iter<'a, &'new str, ()> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn iter_val<'a, 'new>(v: Iter<'a, (), &'static str>) -> Iter<'a, (), &'new str> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn into_iter_key<'new>(v: IntoIter<&'static str, ()>) -> IntoIter<&'new str, ()> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn into_iter_val<'new>(v: IntoIter<(), &'static str>) -> IntoIter<(), &'new str> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn range_key<'a, 'new>(v: Range<'a, &'static str, ()>) -> Range<'a, &'new str, ()> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn range_val<'a, 'new>(v: Range<'a, (), &'static str>) -> Range<'a, (), &'new str> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn keys<'a, 'new>(v: Keys<'a, &'static str, ()>) -> Keys<'a, &'new str, ()> {
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn vals<'a, 'new>(v: Values<'a, (), &'static str>) -> Values<'a, (), &'new str> {
|
|
|
|
v
|
|
|
|
}
|
2016-01-17 09:47:21 -06:00
|
|
|
}
|
|
|
|
|
2016-03-14 16:35:08 -04:00
|
|
|
#[test]
|
|
|
|
fn test_occupied_entry_key() {
|
|
|
|
let mut a = BTreeMap::new();
|
|
|
|
let key = "hello there";
|
|
|
|
let value = "value goes here";
|
|
|
|
assert!(a.is_empty());
|
|
|
|
a.insert(key.clone(), value.clone());
|
|
|
|
assert_eq!(a.len(), 1);
|
|
|
|
assert_eq!(a[key], value);
|
|
|
|
|
|
|
|
match a.entry(key.clone()) {
|
|
|
|
Vacant(_) => panic!(),
|
|
|
|
Occupied(e) => assert_eq!(key, *e.key()),
|
|
|
|
}
|
|
|
|
assert_eq!(a.len(), 1);
|
|
|
|
assert_eq!(a[key], value);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_vacant_entry_key() {
|
|
|
|
let mut a = BTreeMap::new();
|
|
|
|
let key = "hello there";
|
|
|
|
let value = "value goes here";
|
|
|
|
|
|
|
|
assert!(a.is_empty());
|
|
|
|
match a.entry(key.clone()) {
|
|
|
|
Occupied(_) => panic!(),
|
|
|
|
Vacant(e) => {
|
|
|
|
assert_eq!(key, *e.key());
|
|
|
|
e.insert(value.clone());
|
2016-05-22 23:57:13 +05:30
|
|
|
}
|
2016-03-14 16:35:08 -04:00
|
|
|
}
|
|
|
|
assert_eq!(a.len(), 1);
|
|
|
|
assert_eq!(a[key], value);
|
|
|
|
}
|
|
|
|
|
2019-10-20 20:06:20 +02:00
|
|
|
#[test]
|
|
|
|
fn test_first_last_entry() {
|
|
|
|
let mut a = BTreeMap::new();
|
|
|
|
assert!(a.first_entry().is_none());
|
|
|
|
assert!(a.last_entry().is_none());
|
|
|
|
a.insert(1, 42);
|
|
|
|
assert_eq!(a.first_entry().unwrap().key(), &1);
|
|
|
|
assert_eq!(a.last_entry().unwrap().key(), &1);
|
|
|
|
a.insert(2, 24);
|
|
|
|
assert_eq!(a.first_entry().unwrap().key(), &1);
|
|
|
|
assert_eq!(a.last_entry().unwrap().key(), &2);
|
|
|
|
a.insert(0, 6);
|
|
|
|
assert_eq!(a.first_entry().unwrap().key(), &0);
|
|
|
|
assert_eq!(a.last_entry().unwrap().key(), &2);
|
|
|
|
let (k1, v1) = a.first_entry().unwrap().remove_entry();
|
|
|
|
assert_eq!(k1, 0);
|
|
|
|
assert_eq!(v1, 6);
|
|
|
|
let (k2, v2) = a.last_entry().unwrap().remove_entry();
|
|
|
|
assert_eq!(k2, 2);
|
|
|
|
assert_eq!(v2, 24);
|
|
|
|
assert_eq!(a.first_entry().unwrap().key(), &1);
|
|
|
|
assert_eq!(a.last_entry().unwrap().key(), &1);
|
|
|
|
}
|
|
|
|
|
2016-03-24 15:39:46 +01:00
|
|
|
macro_rules! create_append_test {
|
|
|
|
($name:ident, $len:expr) => {
|
|
|
|
#[test]
|
|
|
|
fn $name() {
|
|
|
|
let mut a = BTreeMap::new();
|
|
|
|
for i in 0..8 {
|
|
|
|
a.insert(i, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut b = BTreeMap::new();
|
|
|
|
for i in 5..$len {
|
2019-12-22 17:42:04 -05:00
|
|
|
b.insert(i, 2 * i);
|
2016-03-24 15:39:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
a.append(&mut b);
|
|
|
|
|
|
|
|
assert_eq!(a.len(), $len);
|
|
|
|
assert_eq!(b.len(), 0);
|
|
|
|
|
|
|
|
for i in 0..$len {
|
|
|
|
if i < 5 {
|
|
|
|
assert_eq!(a[&i], i);
|
|
|
|
} else {
|
2019-12-22 17:42:04 -05:00
|
|
|
assert_eq!(a[&i], 2 * i);
|
2016-03-24 15:39:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-22 17:42:04 -05:00
|
|
|
assert_eq!(a.remove(&($len - 1)), Some(2 * ($len - 1)));
|
|
|
|
assert_eq!(a.insert($len - 1, 20), None);
|
2016-03-24 15:39:46 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// These are mostly for testing the algorithm that "fixes" the right edge after insertion.
|
|
|
|
// Single node.
|
|
|
|
create_append_test!(test_append_9, 9);
|
|
|
|
// Two leafs that don't need fixing.
|
|
|
|
create_append_test!(test_append_17, 17);
|
|
|
|
// Two leafs where the second one ends up underfull and needs stealing at the end.
|
|
|
|
create_append_test!(test_append_14, 14);
|
|
|
|
// Two leafs where the second one ends up empty because the insertion finished at the root.
|
|
|
|
create_append_test!(test_append_12, 12);
|
|
|
|
// Three levels; insertion finished at the root.
|
|
|
|
create_append_test!(test_append_144, 144);
|
|
|
|
// Three levels; insertion finished at leaf while there is an empty node on the second level.
|
|
|
|
create_append_test!(test_append_145, 145);
|
|
|
|
// Tests for several randomly chosen sizes.
|
|
|
|
create_append_test!(test_append_170, 170);
|
|
|
|
create_append_test!(test_append_181, 181);
|
2020-02-16 20:01:41 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
2016-03-24 15:39:46 +01:00
|
|
|
create_append_test!(test_append_239, 239);
|
2019-02-13 17:18:47 +01:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
2016-03-24 15:39:46 +01:00
|
|
|
create_append_test!(test_append_1700, 1700);
|
|
|
|
|
2016-05-29 13:30:13 +03:00
|
|
|
fn rand_data(len: usize) -> Vec<(u32, u32)> {
|
|
|
|
let mut rng = DeterministicRng::new();
|
2016-10-16 15:48:22 +05:30
|
|
|
Vec::from_iter((0..len).map(|_| (rng.next(), rng.next())))
|
2016-05-29 13:30:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_split_off_empty_right() {
|
|
|
|
let mut data = rand_data(173);
|
|
|
|
|
|
|
|
let mut map = BTreeMap::from_iter(data.clone());
|
|
|
|
let right = map.split_off(&(data.iter().max().unwrap().0 + 1));
|
|
|
|
|
|
|
|
data.sort();
|
|
|
|
assert!(map.into_iter().eq(data));
|
|
|
|
assert!(right.into_iter().eq(None));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_split_off_empty_left() {
|
|
|
|
let mut data = rand_data(314);
|
|
|
|
|
|
|
|
let mut map = BTreeMap::from_iter(data.clone());
|
|
|
|
let right = map.split_off(&data.iter().min().unwrap().0);
|
|
|
|
|
|
|
|
data.sort();
|
|
|
|
assert!(map.into_iter().eq(None));
|
|
|
|
assert!(right.into_iter().eq(data));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_split_off_large_random_sorted() {
|
2019-08-09 11:43:25 +02:00
|
|
|
#[cfg(not(miri))] // Miri is too slow
|
2016-05-29 13:30:13 +03:00
|
|
|
let mut data = rand_data(1529);
|
2019-08-09 11:43:25 +02:00
|
|
|
#[cfg(miri)]
|
|
|
|
let mut data = rand_data(529);
|
2016-05-29 13:30:13 +03:00
|
|
|
// special case with maximum height.
|
|
|
|
data.sort();
|
|
|
|
|
|
|
|
let mut map = BTreeMap::from_iter(data.clone());
|
|
|
|
let key = data[data.len() / 2].0;
|
|
|
|
let right = map.split_off(&key);
|
|
|
|
|
|
|
|
assert!(map.into_iter().eq(data.clone().into_iter().filter(|x| x.0 < key)));
|
|
|
|
assert!(right.into_iter().eq(data.into_iter().filter(|x| x.0 >= key)));
|
|
|
|
}
|
2019-12-13 18:36:35 +01:00
|
|
|
|
|
|
|
#[test]
|
2020-03-06 14:32:54 +01:00
|
|
|
fn test_into_iter_drop_leak_1() {
|
2019-12-13 18:36:35 +01:00
|
|
|
static DROPS: AtomicU32 = AtomicU32::new(0);
|
|
|
|
|
|
|
|
struct D;
|
|
|
|
|
|
|
|
impl Drop for D {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
if DROPS.fetch_add(1, Ordering::SeqCst) == 3 {
|
|
|
|
panic!("panic in `drop`");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut map = BTreeMap::new();
|
|
|
|
map.insert("a", D);
|
|
|
|
map.insert("b", D);
|
|
|
|
map.insert("c", D);
|
|
|
|
map.insert("d", D);
|
|
|
|
map.insert("e", D);
|
|
|
|
|
|
|
|
catch_unwind(move || drop(map.into_iter())).ok();
|
|
|
|
|
|
|
|
assert_eq!(DROPS.load(Ordering::SeqCst), 5);
|
|
|
|
}
|
2020-03-06 14:32:54 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_into_iter_drop_leak_2() {
|
|
|
|
let size = 12; // to obtain tree with 2 levels (having edges to leaf nodes)
|
|
|
|
static DROPS: AtomicU32 = AtomicU32::new(0);
|
|
|
|
static PANIC_POINT: AtomicU32 = AtomicU32::new(0);
|
|
|
|
|
|
|
|
struct D;
|
|
|
|
impl Drop for D {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
if DROPS.fetch_add(1, Ordering::SeqCst) == PANIC_POINT.load(Ordering::SeqCst) {
|
|
|
|
panic!("panic in `drop`");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for panic_point in vec![0, 1, size - 2, size - 1] {
|
|
|
|
DROPS.store(0, Ordering::SeqCst);
|
|
|
|
PANIC_POINT.store(panic_point, Ordering::SeqCst);
|
|
|
|
let map: BTreeMap<_, _> = (0..size).map(|i| (i, D)).collect();
|
|
|
|
catch_unwind(move || drop(map.into_iter())).ok();
|
|
|
|
assert_eq!(DROPS.load(Ordering::SeqCst), size);
|
|
|
|
}
|
|
|
|
}
|