iter-set 2.0.2

Set operations on sorted, deduplicated iterators.
Documentation
use std::prelude::v1::*;
use std::vec;

use super::*;

const BITS: u32 = 8;

fn unpack(i: u32) -> Vec<u32> {
    (0..BITS).filter(|bit| i & (1u32 << bit) != 0).collect()
}

#[test]
fn test_cmp() {
    use std::cmp::Ordering::*;

    assert_eq!(cmp(&[1, 2], &[1, 2]), Some(Equal));
    assert_eq!(cmp(&[1], &[1, 2]), Some(Less));
    assert_eq!(cmp(&[1, 2, 3], &[1, 2]), Some(Greater));
    assert_eq!(cmp(&[1, 2, 3], &[1, 2, 4]), None);

    for i in 0..(1u32 << BITS) {
        for j in 0..(1u32 << BITS) {
            let lhs = unpack(i);
            let rhs = unpack(j);

            let expected = if i == j {
                Some(Equal)
            } else if (i & j) == i {
                Some(Less)
            } else if (i & j) == j {
                Some(Greater)
            } else {
                None
            };

            assert_eq!(cmp(&lhs, &rhs), expected);
        }
    }
}

#[test]
fn test_union() {
    fn slice_union(a: &[u32], b: &[u32]) -> Vec<u32> {
        union(a, b).cloned().collect()
    }

    assert_eq!(slice_union(&[1, 2], &[1, 2]), vec![1, 2]);
    assert_eq!(slice_union(&[1], &[1, 2]), vec![1, 2]);
    assert_eq!(slice_union(&[1, 2, 3], &[1, 2]), vec![1, 2, 3]);
    assert_eq!(slice_union(&[1, 2, 3], &[1, 2, 4]), vec![1, 2, 3, 4]);

    for i in 0..(1u32 << BITS) {
        for j in 0..(1u32 << BITS) {
            let lhs = unpack(i);
            let rhs = unpack(j);

            let expected = unpack(i | j);
            assert_eq!(slice_union(&lhs, &rhs), expected);
        }
    }
}

#[test]
fn test_intersection() {
    fn slice_intersection(a: &[u32], b: &[u32]) -> Vec<u32> {
        intersection(a, b).cloned().collect()
    }
    fn intersection_size_max(a: &[u32], b: &[u32]) -> Option<usize> {
        intersection(a, b).size_hint().1
    }

    assert_eq!(slice_intersection(&[1, 2], &[1, 2]), vec![1, 2]);
    assert_eq!(slice_intersection(&[1], &[1, 2]), vec![1]);
    assert_eq!(slice_intersection(&[1, 2, 3], &[1, 2]), vec![1, 2]);
    assert_eq!(slice_intersection(&[1, 2, 3], &[1, 2, 4]), vec![1, 2]);

    assert_eq!(intersection_size_max(&[1, 2], &[1, 2]), Some(2));
    assert_eq!(intersection_size_max(&[1], &[1, 2]), Some(1));
    assert_eq!(intersection_size_max(&[1, 2, 3], &[1, 2]), Some(2));
    assert_eq!(intersection_size_max(&[1, 2, 3], &[1, 2, 4]), Some(3));

    for i in 0..(1u32 << BITS) {
        for j in 0..(1u32 << BITS) {
            let lhs = unpack(i);
            let rhs = unpack(j);

            let expected = unpack(i & j);
            assert_eq!(slice_intersection(&lhs, &rhs), expected);
        }
    }
}

#[test]
fn test_difference() {
    fn slice_difference(a: &[u32], b: &[u32]) -> Vec<u32> {
        difference(a, b).cloned().collect()
    }
    fn difference_size_max(a: &[u32], b: &[u32]) -> Option<usize> {
        difference(a, b).size_hint().1
    }

    assert_eq!(slice_difference(&[1, 2], &[1, 2]), vec![]);
    assert_eq!(slice_difference(&[1, 2], &[1]), vec![2]);
    assert_eq!(slice_difference(&[1, 2, 3], &[1, 2]), vec![3]);
    assert_eq!(slice_difference(&[1, 2, 3], &[1, 2, 4]), vec![3]);

    assert_eq!(difference_size_max(&[1, 2], &[1, 2]), Some(2));
    assert_eq!(difference_size_max(&[1], &[1, 2]), Some(1));
    assert_eq!(difference_size_max(&[1, 2, 3], &[1, 2]), Some(3));
    assert_eq!(difference_size_max(&[1, 2, 3], &[1, 2, 4]), Some(3));

    for i in 0..(1u32 << BITS) {
        for j in 0..(1u32 << BITS) {
            let lhs = unpack(i);
            let rhs = unpack(j);

            let expected = unpack(i & !j);
            assert_eq!(slice_difference(&lhs, &rhs), expected);
        }
    }
}

#[test]
fn test_symmetric_difference() {
    fn slice_symmetric_difference(a: &[u32], b: &[u32]) -> Vec<u32> {
        symmetric_difference(a, b).cloned().collect()
    }

    assert_eq!(slice_symmetric_difference(&[1, 2], &[1, 2]), vec![]);
    assert_eq!(slice_symmetric_difference(&[1, 2], &[1]), vec![2]);
    assert_eq!(slice_symmetric_difference(&[1, 2, 3], &[1, 2]), vec![3]);
    assert_eq!(
        slice_symmetric_difference(&[1, 2, 3], &[1, 2, 4]),
        vec![3, 4]
    );

    for i in 0..(1u32 << BITS) {
        for j in 0..(1u32 << BITS) {
            let lhs = unpack(i);
            let rhs = unpack(j);

            let expected = unpack(i ^ j);
            assert_eq!(slice_symmetric_difference(&lhs, &rhs), expected);
        }
    }
}

#[test]
fn test_classify_size_hint_lower() {
    let lhs = [1, 2, 3];
    let rhs = [1, 2, 3, 4];

    let mut classify = classify(&lhs, &rhs);

    let (lo, hi) = classify.size_hint();
    assert_eq!(lo, 4);
    assert_eq!(hi, Some(7));

    assert_eq!(classify.next(), Some(Inclusion::Both(&1, &1)));

    let (lo, hi) = classify.size_hint();
    assert_eq!(lo, 3);
    assert_eq!(hi, Some(5));

    assert_eq!(classify.count(), 3);
}

#[test]
fn test_classify_size_hint_upper() {
    let lhs = [1, 3, 5];
    let rhs = [2, 4, 6, 7];

    let mut classify = classify(&lhs, &rhs);

    let (lo, hi) = classify.size_hint();
    assert_eq!(lo, 4);
    assert_eq!(hi, Some(7));

    assert_eq!(classify.next(), Some(Inclusion::Left(&1)));

    let (lo, hi) = classify.size_hint();
    assert_eq!(lo, 4);
    assert_eq!(hi, Some(6));

    assert_eq!(classify.count(), 6);
}

#[test]
fn test_combine() {
    use std::cmp::Ordering;

    #[derive(Debug, Eq, PartialEq)]
    enum Foo {
        Zero,
        Some(Vec<u32>),
    }

    fn combine(l: &mut Foo, r: &mut Foo) -> Ordering {
        match (l, r) {
            (Foo::Zero, Foo::Zero) => Ordering::Equal,
            (Foo::Zero, Foo::Some(_)) => Ordering::Less,
            (Foo::Some(_), Foo::Zero) => Ordering::Greater,
            (Foo::Some(l), Foo::Some(r)) => {
                l.append(r);
                Ordering::Equal
            }
        }
    }

    let lhs = vec![Foo::Zero, Foo::Some(vec![1, 2])];
    let rhs = vec![Foo::Some(vec![3])];

    let union: Vec<_> = union_by(lhs, rhs, combine).collect();
    assert_eq!(union, vec![Foo::Zero, Foo::Some(vec![1, 2, 3])]);
}