1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
// Copyright 2020 Graydon Hoare <graydon@pobox.com>
// Licensed under the MIT and Apache-2.0 licenses.

/*!

This is a small crate that provides a generic unital-join-semilattice type
(hereafter: "lattice") along with a few common instances.

Lattices are defined in two separate pieces: a _definition_ trait [`LatticeDef`]
that provides the type-and-functions for a given lattice and a _user interface_
struct [`LatticeElt`] that's parameterized by a [`LatticeDef`] and provides
convenient methods to work with (including impls of standard Rust operator
traits).

This unusual split exists because many types have multiple equally viable
lattices you can build on them (eg. `u32`-with-`min` or `u32`-with-`max`) and we
want to avoid both coupling any given lattice definition to the type _or_
accidentally inheriting an impl for any of the type's "standard semantics" as
the lattice semantics, eg. we don't want to inherit `u32`'s standard partial
order as any lattice's partial order, unless explicitly building such a lattice.


# Examples

## Simple u32-with-max lattice

```
use pergola::{MaxDef,LatticeElt};

type Def = MaxDef<u32>;      // lattice def for "u32 with max for join"
type Elt = LatticeElt<Def>;  // element struct, implementing std traits
let v = Elt::new_from(1);
let u = Elt::new_from(2);
let w = v + u;               // calls join(), which calls max()
assert!(v < u);
assert!(v < w);
```

## Composite tuple lattice
```
use pergola::{MaxDef,Tuple3,LatticeElt};

type NumDef = MaxDef<u32>;
type NumElt = LatticeElt<NumDef>;
type TupDef = Tuple3<NumDef,NumDef,NumDef>;
type TupElt = LatticeElt<TupDef>;
let n1: NumElt = (1).into();
let n2: NumElt = (2).into();
let n3: NumElt = (3).into();
let tup: TupElt = (n1, n2, n3).into();
assert!(tup.value.0 < tup.value.1);
assert!((tup.value.0 + tup.value.1) == tup.value.1);
let n1ref: &NumElt = &tup.value.0;
let n2ref: &NumElt = &tup.value.1;
assert!(n1ref < n2ref);
```

## Trickier union-map-of-union-bitsets lattice

```
use pergola::{BTreeMapWithUnion,BitSetWithUnion,LatticeElt};
use bit_set::BitSet;
use std::collections::BTreeMap;

type Def = BTreeMapWithUnion<String,BitSetWithUnion>;
type Elt = LatticeElt<Def>;
let bs_a1 = BitSet::from_bytes(&[0b11110000]);
let bs_a2 = BitSet::from_bytes(&[0b00001111]);
let bs_b = BitSet::from_bytes(&[0b10101010]);
let v = Elt::new_from([(String::from("a"),bs_a1.into()),
                       (String::from("b"),bs_b.into())].iter().cloned().collect());
let u = Elt::new_from([(String::from("a"),bs_a2.into())].iter().cloned().collect());
let w = &v + &u;
assert!(!(v < u));  // bs_a1 is not a subset of bs_a2,
                    // so v["a"] is unordered wrt. u["a"].
assert!(v < w);     // However, w is a join and join unions
                    // the values at common keys, so v["a"] < w["a"].
assert!(u < w);     // And likewise the other input to the join.
assert_eq!(w.value["a"].value.0, BitSet::from_bytes(&[0b11111111]));
assert_eq!(w.value["b"].value.0, BitSet::from_bytes(&[0b10101010]));
```

# Name

Wikipedia:

A pergola is an outdoor garden feature forming a shaded walkway, passageway, or
sitting area of vertical posts or pillars that usually support cross-beams and a
sturdy open lattice, often upon which woody vines are trained.

*/

// TODO: Maybe add hash maps and sets
// TODO: Maybe split struct into struct + trait
// TODO: Maybe split such trait into lattices with/without unit, upper bound

mod latticedef;
mod latticeelt;

pub use latticedef::{DefTraits, LatticeDef, ValTraits};
pub use latticedef::{MaxDef, MaxNum, MaxUnitDefault, MaxUnitMinValue, MinNum, MinOpt};

#[cfg(feature = "bit-set")]
pub use latticedef::{BitSetWithIntersection, BitSetWithUnion, BitSetWrapper};

#[cfg(feature = "im")]
pub use latticedef::{ArcOrdMapWithIntersection, ArcOrdMapWithUnion};
#[cfg(feature = "im")]
pub use latticedef::{ArcOrdSetWithIntersection, ArcOrdSetWithUnion};

#[cfg(feature = "im-rc")]
pub use latticedef::{RcOrdMapWithIntersection, RcOrdMapWithUnion};
#[cfg(feature = "im-rc")]
pub use latticedef::{RcOrdSetWithIntersection, RcOrdSetWithUnion};

pub use latticedef::{BTreeMapWithIntersection, BTreeMapWithUnion};
pub use latticedef::{BTreeSetWithIntersection, BTreeSetWithUnion};
pub use latticedef::{Tuple2, Tuple3, Tuple4, Tuple5};

pub use latticeelt::LatticeElt;

#[cfg(test)]
mod proptests;

#[cfg(test)]
mod quickchecks;

#[cfg(test)]
mod manualtests;