module UHC.Util.RelMap
( Rel
, empty
, toDomList, toRngList
, toList, fromList
, singleton
, dom, rng
, restrictDom, restrictRng
, mapDom, mapRng
, union, unions
, insert
, deleteDom, delete
, deleteRng
, applyDomMbSet, applyRngMbSet
, applyDomSet, applyRngSet
, applyDom, applyRng
, apply, applyInverse
, lookupDom, lookupRng
, lookup, lookupInverse
, toDomMap, toRngMap
)
where
import Prelude hiding (lookup)
import Control.Monad
import qualified Data.Map as Map
import qualified Data.Set as Set
import UHC.Util.AssocL
import UHC.Util.Binary
import UHC.Util.Serialize
type RelMap a b = Map.Map a (Set.Set b)
relmapDeleteRng :: Ord b => b -> RelMap a b -> RelMap a b
relmapDeleteRng x r = snd $ Map.mapEither (eith x) r
where eith x ds = if Set.null ds' then Left ds else Right ds'
where (ds1,ds2) = Set.split x ds
ds' = Set.union ds1 ds2
data Rel a b
= Rel
{ relDomMp :: RelMap a b
, relRngMp :: RelMap b a
}
toDomList :: Rel a b -> [(a,[b])]
toDomList (Rel m _) = [ (d, Set.toList rs) | (d,rs) <- Map.toList m ]
toRngList :: Rel a b -> [([a],b)]
toRngList (Rel _ m) = [ (Set.toList ds, r) | (r,ds) <- Map.toList m ]
toList :: Rel a b -> [(a,b)]
toList rel = [ (d,r) | (d,rs) <- toDomList rel, r <- rs ]
fromList :: (Ord a, Ord b) => [(a,b)] -> Rel a b
fromList = unions . map (uncurry singleton)
singleton :: (Ord a, Ord b) => a -> b -> Rel a b
singleton a b = Rel (relMapSingleton a b) (relMapSingleton b a)
empty :: Rel a b
empty = Rel Map.empty Map.empty
dom :: (Ord a, Ord b) => Rel a b -> Set.Set a
dom = Map.keysSet . relDomMp
rng :: (Ord a, Ord b) => Rel a b -> Set.Set b
rng = Map.keysSet . relRngMp
restrictDom :: (Ord a, Ord b) => (a -> Bool) -> Rel a b -> Rel a b
restrictDom p (Rel d r) = Rel d' r'
where d' = Map.filterWithKey (\d r -> p d) d
r' = relMapInverse d'
restrictRng :: (Ord a, Ord b) => (b -> Bool) -> Rel a b -> Rel a b
restrictRng p (Rel d r) = Rel d' r'
where r' = Map.filterWithKey (\r d -> p r) r
d' = relMapInverse r'
mapDom :: (Ord b, Ord x) => (a -> x) -> Rel a b -> Rel x b
mapDom f = fromList . assocLMapKey f . toList
mapRng :: (Ord a, Ord x) => (b -> x) -> Rel a b -> Rel a x
mapRng f = fromList . assocLMapElt f . toList
union :: (Ord a, Ord b) => Rel a b -> Rel a b -> Rel a b
union (Rel d1 r1) (Rel d2 r2) = Rel (Map.unionWith Set.union d1 d2) (Map.unionWith Set.union r1 r2)
unions :: (Ord a, Ord b) => [Rel a b] -> Rel a b
unions [ ] = empty
unions [r] = r
unions rs = foldr union empty rs
insert :: (Ord a, Ord b) => a -> b -> Rel a b -> Rel a b
insert x y r = singleton x y `union` r
deleteDom, delete :: (Ord a, Ord b) => a -> Rel a b -> Rel a b
deleteDom x (Rel d r) = Rel (Map.delete x d) (relmapDeleteRng x r)
delete = deleteDom
deleteRng :: (Ord a, Ord b) => b -> Rel a b -> Rel a b
deleteRng x (Rel d r) = Rel (relmapDeleteRng x d) (Map.delete x r)
applyDomMbSet :: (Ord a) => Rel a b -> a -> Maybe (Set.Set b)
applyDomMbSet r a = Map.lookup a (relDomMp r)
applyRngMbSet :: (Ord b) => Rel a b -> b -> Maybe (Set.Set a)
applyRngMbSet r b = Map.lookup b (relRngMp r)
applyDomSet :: (Ord a) => Rel a b -> a -> Set.Set b
applyDomSet r a = maybe Set.empty id $ applyDomMbSet r a
applyRngSet :: (Ord b) => Rel a b -> b -> Set.Set a
applyRngSet r b = maybe Set.empty id $ applyRngMbSet r b
applyDom :: (Ord a) => Rel a b -> a -> Maybe b
applyDom r a = fmap Set.findMin $ applyDomMbSet r a
applyRng :: (Ord b) => Rel a b -> b -> Maybe a
applyRng r b = fmap Set.findMin $ applyRngMbSet r b
apply :: (Ord a) => Rel a b -> a -> [b]
apply r a = maybe [] Set.toList $ applyDomMbSet r a
applyInverse :: (Ord b) => Rel a b -> b -> [a]
applyInverse r b = maybe [] Set.toList $ applyRngMbSet r b
lookupDom, lookup :: (Ord a) => a -> Rel a b -> Maybe b
lookupDom = flip applyDom
lookup = lookupDom
lookupRng, lookupInverse :: (Ord b) => b -> Rel a b -> Maybe a
lookupRng = flip applyRng
lookupInverse = lookupRng
toDomMap :: Ord a => Rel a b -> Map.Map a [b]
toDomMap = Map.map Set.toList . relDomMp
toRngMap :: Ord b => Rel a b -> Map.Map b [a]
toRngMap = Map.map Set.toList . relRngMp
relMapSingleton :: (Ord a, Ord b) => a -> b -> RelMap a b
relMapSingleton d r = Map.singleton d (Set.singleton r)
relMapInverse :: (Ord a, Ord b) => RelMap a b -> RelMap b a
relMapInverse m = Map.unionsWith Set.union [ relMapSingleton r d | (d,rs) <- Map.toList m, r <- Set.toList rs ]
instance (Ord b, Ord a, Binary a, Binary b) => Binary (Rel a b) where
put = put . toList
get = liftM fromList get
instance (Ord b, Ord a, Serialize a, Serialize b) => Serialize (Rel a b) where
sput = sput . toList
sget = liftM fromList sget