module Data.FixFile.Tree23 (Tree23
,size
,depth
,Set
,createSetFile
,openSetFile
,insertSet
,lookupSet
,deleteSet
,partitionSet
,minSet
,maxSet
,toListSet
,fromListSet
,insertSetT
,lookupSetT
,deleteSetT
,partitionSetT
,minSetT
,maxSetT
,Map
,createMapFile
,openMapFile
,insertMap
,lookupMap
,deleteMap
,partitionMap
,alterMap
,minMap
,maxMap
,toListMap
,fromListMap
,insertMapT
,lookupMapT
,deleteMapT
,partitionMapT
,alterMapT
,minMapT
,maxMapT
,keysMap
,valuesMap
) where
import Prelude hiding (null)
import Data.Dynamic
import Data.Binary
import Data.Maybe
import Data.Monoid
import GHC.Generics (Generic)
import Unsafe.Coerce
import Data.FixFile
data Tree23F k v a =
Empty
| Leaf k v
| Two a k a
| Three a k a k a
deriving (Read, Show, Eq, Ord, Generic, Functor, Foldable, Traversable,
Typeable)
instance Null1 (Tree23F k v) where
empty1 = Empty
null1 Empty = True
null1 _ = False
type Tree23 d = Tree23F (TreeKey d) (TreeValue d)
data family TreeKey d
data family TreeValue d
instance (Binary a, Binary (TreeKey d), Binary (TreeValue d)) =>
Binary (Tree23F (TreeKey d) (TreeValue d) a)
leaf :: Fixed g => TreeKey d -> TreeValue d -> g (Tree23 d)
leaf k v = inf $ Leaf k v
two :: Fixed g => g (Tree23 d) -> TreeKey d ->
g (Tree23 d) -> g (Tree23 d)
two l v r = inf $ Two l v r
three :: Fixed g => g (Tree23 d) -> TreeKey d -> g (Tree23 d) ->
TreeKey d -> g (Tree23 d) -> g (Tree23 d)
three l t1 m t2 r =
inf $ Three l t1 m t2 r
size :: Fixed g => g (Tree23 d) -> Int
size = cata phi where
phi Empty = 0
phi (Leaf _ _) = 1
phi (Two l _ r) = l + r
phi (Three l _ m _ r) = l + m + r
depth :: Fixed g => g (Tree23 d) -> Int
depth = cata phi where
phi Empty = 0
phi (Leaf _ _) = 1
phi (Two l _ _) = l + 1
phi (Three l _ _ _ _) = l + 1
data Set k
newtype instance TreeKey (Set k) = SK k
deriving (Read, Show, Eq, Ord, Generic, Typeable)
data instance TreeValue (Set k) = SV
deriving (Read, Show, Eq, Ord, Generic, Typeable)
instance Binary k => Binary (TreeKey (Set k))
instance Binary (TreeValue (Set k))
insertSet :: (Fixed g, Ord k, f ~ Tree23 (Set k)) => k -> g f -> g f
insertSet k = alterTree23 (SK k) (maybe (Just $ Just SV) (const Nothing))
lookupSet :: (Fixed g, Ord k, f ~ Tree23 (Set k)) => k -> g f -> Bool
lookupSet k = isJust . lookupTree23 (SK k)
deleteSet :: (Fixed g, Ord k, f ~ Tree23 (Set k)) => k -> g f -> g f
deleteSet k = alterTree23 (SK k) (const $ Just Nothing)
partitionSet :: (Fixed g, Ord k, f ~ Tree23 (Set k)) => k -> g f -> (g f, g f)
partitionSet k = partitionTree23 (SK k)
minSet :: (Fixed g, Ord k, f ~ Tree23 (Set k)) => g f -> Maybe k
minSet t = do
(SK v, _) <- minTree23 t
return v
maxSet :: (Fixed g, Ord k, f ~ Tree23 (Set k)) => g f -> Maybe k
maxSet t = do
(SK v, _) <- maxTree23 t
return v
toListSet :: (Fixed g, Ord k, f ~ Tree23 (Set k)) => g f -> [k]
toListSet = ($ []) . cata phi where
phi Empty xs = xs
phi (Leaf (SK k) _) xs = k:xs
phi (Two la _ ra) xs = la . ra $ xs
phi (Three la _ ma _ ra) xs = la . ma . ra $ xs
fromListSet :: (Fixed g, Ord k, f ~ Tree23 (Set k)) => [k] -> g f
fromListSet = Prelude.foldr insertSet empty
createSetFile :: (Binary k, Typeable k, f ~ Tree23 (Set k)) =>
FilePath -> IO (FixFile (Ref f))
createSetFile fp = createFixFile (Ref empty) fp
openSetFile :: (Binary k, Typeable k, f ~ Tree23 (Set k)) =>
FilePath ->IO (FixFile (Ref f))
openSetFile fp = openFixFile fp
insertSetT :: (Binary k, Ord k, f ~ Tree23 (Set k)) =>
k -> Transaction (Ref f) s ()
insertSetT k = alterT (insertSet k)
lookupSetT :: (Binary k, Ord k, f ~ Tree23 (Set k)) =>
k -> Transaction (Ref f) s Bool
lookupSetT k = lookupT (lookupSet k)
deleteSetT :: (Binary k, Ord k, f ~ Tree23 (Set k)) =>
k -> Transaction (Ref f) s ()
deleteSetT k = alterT (deleteSet k)
partitionSetT :: (Binary k, Ord k, f ~ Tree23 (Set k)) => k ->
Transaction (Ref f) s (Stored s f, Stored s f)
partitionSetT k = lookupT (partitionSet k)
minSetT :: (Binary k, Ord k, f ~ Tree23 (Set k)) =>
Transaction (Ref f) s (Maybe k)
minSetT = lookupT minSet
maxSetT :: (Binary k, Ord k, f ~ Tree23 (Set k)) =>
Transaction (Ref f) s (Maybe k)
maxSetT = lookupT maxSet
instance FixedAlg (Tree23 (Set k)) where
type Alg (Tree23 (Set k)) = k
instance FixedFoldable (Tree23 (Set k)) where
foldMapF f = cata phi where
phi Empty = mempty
phi (Leaf (SK k) _) = f k
phi (Two l _ r) = l <> r
phi (Three l _ m _ r) = l <> m <> r
data Map k v
newtype instance TreeKey (Map k v) = MK k
deriving (Read, Show, Eq, Ord, Generic, Typeable)
newtype instance TreeValue (Map k v) = MV { fromMV :: v }
deriving (Read, Show, Eq, Ord, Generic, Typeable)
instance Binary k => Binary (TreeKey (Map k v))
instance Binary v => Binary (TreeValue (Map k v))
insertMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => k -> v -> g f -> g f
insertMap k v = alterTree23 (MK k) (const . Just . Just $ MV v)
lookupMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => k -> g f -> Maybe v
lookupMap k = fmap toV . lookupTree23 (MK k) where
toV (MV v) = v
deleteMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => k -> g f -> g f
deleteMap k = alterTree23 (MK k) (const . Just $ Nothing)
alterMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) =>
k -> (Maybe v -> Maybe v) -> g f -> g f
alterMap k f = alterTree23 (MK k) (Just . fmap MV . f . fmap fromMV)
partitionMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) =>
k -> g f -> (g f, g f)
partitionMap k = partitionTree23 (MK k)
toListMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => g f -> [(k,v)]
toListMap = ($ []) . cata phi where
phi Empty xs = xs
phi (Leaf (MK k) (MV v)) xs = (k,v):xs
phi (Two la _ ra) xs = la . ra $ xs
phi (Three la _ ma _ ra) xs = la . ma . ra $ xs
fromListMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => [(k,v)] -> g f
fromListMap = Prelude.foldr (uncurry insertMap) empty
keysMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => g f -> [k]
keysMap = fmap fst . toListMap
valuesMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => g f -> [v]
valuesMap = fmap snd . toListMap
minMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => g f -> Maybe (k, v)
minMap t = do
(MK k, MV v) <- minTree23 t
return (k, v)
maxMap :: (Fixed g, Ord k, f ~ Tree23 (Map k v)) => g f -> Maybe (k, v)
maxMap t = do
(MK k, MV v) <- maxTree23 t
return (k, v)
createMapFile :: (Binary k, Typeable k, Binary v, Typeable v,
f ~ Tree23 (Map k v)) =>
FilePath -> IO (FixFile (Ref f))
createMapFile fp = createFixFile (Ref empty) fp
openMapFile :: (Binary k, Typeable k, Binary v, Typeable v,
f ~ Tree23 (Map k v)) =>
FilePath -> IO (FixFile (Ref f))
openMapFile fp = openFixFile fp
insertMapT :: (Binary k, Binary v, Ord k, f ~ Tree23 (Map k v)) =>
k -> v -> Transaction (Ref f) s ()
insertMapT k v = alterT (insertMap k v)
lookupMapT :: (Binary k, Binary v, Ord k, f ~ Tree23 (Map k v)) =>
k -> Transaction (Ref f) s (Maybe v)
lookupMapT k = lookupT (lookupMap k)
deleteMapT :: (Binary k, Binary v, Ord k, f ~ Tree23 (Map k v)) =>
k -> Transaction (Ref f) s ()
deleteMapT k = alterT (deleteMap k)
partitionMapT :: (Binary k, Ord k, Binary v, f ~ Tree23 (Map k v)) =>
k -> Transaction (Ref f) s (Stored s f, Stored s f)
partitionMapT k = lookupT (partitionMap k)
alterMapT :: (Binary k, Binary v, Ord k, f ~ Tree23 (Map k v)) =>
k -> (Maybe v -> Maybe v) -> Transaction (Ref f) s ()
alterMapT k f = alterT (alterMap k f)
minMapT :: (Binary k, Binary v, Ord k, f ~ Tree23 (Map k v)) =>
Transaction (Ref f) s (Maybe (k, v))
minMapT = lookupT minMap
maxMapT :: (Binary k, Binary v, Ord k, f ~ Tree23 (Map k v)) =>
Transaction (Ref f) s (Maybe (k, v))
maxMapT = lookupT maxMap
lookupTree23 :: (Fixed g, Ord (TreeKey d)) => TreeKey d ->
g (Tree23 d) -> Maybe (TreeValue d)
lookupTree23 k = cata phi where
phi Empty = Nothing
phi (Leaf k' v)
| k == k' = Just v
| otherwise = Nothing
phi (Two la k' ra) =
case compare k k' of
LT -> la
_ -> ra
phi (Three la k1 ma k2 ra) =
case (compare k k1, compare k k2) of
(LT, _) -> la
(_, LT) -> ma
(_, _) -> ra
data Change g d =
NoChange
| Changed (Maybe (TreeKey d)) (g (Tree23 d))
| Unbalanced (Maybe (TreeKey d)) (g (Tree23 d))
| Hole
| Split (g (Tree23 d)) (TreeKey d) (g (Tree23 d))
alterTree23 :: (Fixed g, Ord (TreeKey d)) => TreeKey d ->
(Maybe (TreeValue d) -> Maybe (Maybe (TreeValue d))) ->
g (Tree23 d) -> g (Tree23 d)
alterTree23 k f t = processHead $ para phi t t where
processHead NoChange = t
processHead (Changed _ t') = t'
processHead Hole = empty
processHead (Unbalanced _ t') = t'
processHead (Split lt d rt) = two lt d rt
phi Empty _ = case f Nothing of
Just (Just v) -> Changed Nothing $ leaf k v
_ -> NoChange
phi (Leaf k' v') n
| k == k' = case f (Just v') of
Nothing -> NoChange
Just Nothing -> Hole
Just (Just v) -> Changed Nothing $ leaf k' v
| otherwise = case f Nothing of
Nothing -> NoChange
Just Nothing -> NoChange
Just (Just v) -> if k < k'
then Split (leaf k v) k' n
else Split n k (leaf k v)
phi (Two (ln, la) k' (rn, ra)) _
| k < k' = case la ln of
NoChange -> NoChange
Changed nk la' ->
Changed nk $ two la' k' rn
Split la' k'' ma'->
Changed Nothing $ three la' k'' ma' k' rn
Hole -> Unbalanced (Just k') rn
Unbalanced uk un -> case outf rn of
Three ln' k1 mn' k2 rn' -> Changed uk $
two (two un k' ln') k1 (two mn' k2 rn')
Two ln' k1 rn' -> Unbalanced uk $
three un k' ln' k1 rn'
_ -> error "Invalid Tree23"
| otherwise = case ra rn of
NoChange -> NoChange
Hole -> Unbalanced Nothing ln
Changed dk dn -> Changed Nothing $
two ln (maybe k' id dk) dn
Split ma' k'' ra' -> Changed Nothing $
three ln k' ma' k'' ra'
Unbalanced uk un -> case outf ln of
Three ln' k1 mn' k2 rn' -> Changed Nothing $
two (two ln' k1 mn') k2 (two rn' (maybe k' id uk) un)
Two ln' k1 rn' -> Unbalanced Nothing $
three ln' k1 rn' (maybe k' id uk) un
_ -> error "Invalid Tree23"
phi (Three (ln, la) k1 (mn, ma) k2 (rn, ra)) _
| k < k1 = case la ln of
NoChange -> NoChange
Hole -> Changed (Just k1) $ two mn k2 rn
Changed dk dn -> Changed dk $
three dn k1 mn k2 rn
Split ln' k' rn' -> Split
(two ln' k' rn') k1 (two mn k2 rn)
Unbalanced uk un -> case outf mn of
Three ln' k1' mn' k2' rn' -> Changed uk $
three (two un k1 ln') k1' (two mn' k2' rn') k2 rn
Two ln' k1' rn' -> Changed uk $
two (three un k1 ln' k1' rn') k2 rn
_ -> error "Invalid Tree23"
| k < k2 = case ma mn of
NoChange -> NoChange
Hole -> Changed Nothing $ two ln k2 rn
Changed dk dn -> Changed Nothing $
three ln (maybe k1 id dk) dn k2 rn
Split mn' k' rn' -> Split
(two ln k1 mn') k' (two rn' k2 rn)
Unbalanced uk un -> case outf rn of
Three ln' k1' mn' k2' rn' -> Changed Nothing $
three ln (maybe k1 id uk) (two un k2 ln')
k1' (two mn' k2' rn')
Two ln' k1' rn' -> Changed Nothing $
two ln (maybe k1 id uk) (three un k2 ln' k1' rn')
_ -> error "Invalid Tree23"
| otherwise = case ra rn of
NoChange -> NoChange
Hole -> Changed Nothing $ two ln k1 mn
Changed dk dn -> Changed Nothing $
three ln k1 mn (maybe k2 id dk) dn
Split mn' k' rn' -> Split
(two ln k1 mn) k2 (two mn' k' rn')
Unbalanced uk un -> case outf mn of
Three ln' k1' mn' k2' rn' -> Changed Nothing $
three ln k1 (two ln' k1' mn') k2'
(two rn' (maybe k2 id uk) un)
Two ln' k1' rn' -> Changed Nothing $ two ln k1
(three ln' k1' rn' (maybe k2 id uk) un)
_ -> error "Invalid Tree23"
data SkewDir = L | R
data Partition g d =
NoPartition
| Skew SkewDir
| Split2 (Int, g (Tree23 d)) (Int, g (Tree23 d))
merge :: (Fixed g, Ord (TreeKey d)) => Int -> g (Tree23 d) -> TreeKey d ->
Int -> g (Tree23 d) -> (Int, g (Tree23 d))
merge ld ln k rd rn
| ld == rd = (ld + 1, two ln k rn)
| ld < rd = case (rd ld, outf rn) of
(1, Two rln rk rrn) -> (rd, three ln k rln rk rrn)
(1, Three rln rk1 rmn rk2 rrn) ->
(rd + 1, two (two ln k rln) rk1 (two rmn rk2 rrn))
(_, Two rln rk rrn) ->
let (ld', rln') = merge ld ln k (rd 1) rln
in merge ld' rln' rk (rd 1) rrn
(_, Three rln rk1 rmn rk2 rrn) ->
let (ld', rln') = merge ld ln k (rd 1) rln
in merge ld' rln' rk1 (rd 1) (two rmn rk2 rrn)
_ -> error "Malformed Tree23"
| otherwise = case (ld rd, outf ln) of
(1, Two lln lk lrn) -> (ld, three lln lk lrn k rn)
(1, Three lln lk1 lmn lk2 lrn) ->
(ld + 1, two (two lln lk1 lmn) lk2 (two lrn k rn))
(_, Two lln lk lrn) ->
let (rd', lrn') = merge (ld 1) lrn k rd rn
in merge (ld 1) lln lk rd' lrn'
(_, Three lln lk1 lmn lk2 lrn) ->
let (rd', lrn') = merge (ld 1) lrn k rd rn
in merge (ld 1) (two lln lk1 lmn) lk2 rd' lrn'
_ -> error "Malformed Tree23"
partitionTree23 :: (Fixed g, Ord (TreeKey d)) => TreeKey d ->
g (Tree23 d) -> (g (Tree23 d), g (Tree23 d))
partitionTree23 k t = resp $ para phi t where
resp NoPartition = (t, t)
resp (Skew L) = (t, empty)
resp (Skew R) = (empty, t)
resp (Split2 (_, l) (_, r)) = (l, r)
phi Empty = NoPartition
phi (Leaf k' _)
| k' < k = Skew L
| otherwise = Skew R
phi (Two (ln, la) k' (rn, ra))
| k' == k = Split2 (1, ln) (1, rn)
| k' < k = case ra of
Skew L -> Skew L
Skew R -> Split2 (1, ln) (1, rn)
Split2 (lbal, lv) (rbal, rv) ->
Split2 (merge (1) ln k' lbal lv) (rbal 1, rv)
_ -> error "Malformed Tree23"
| otherwise = case la of
Skew L -> Split2 (1, ln) (1, rn)
Skew R -> Skew R
Split2 (lbal, lv) (rbal, rv) ->
Split2 (lbal 1, lv) (merge rbal rv k' (1) rn)
_ -> error "Malformed Tree23"
phi (Three (ln, la) k1 (mn, ma) k2 (rn, ra))
| k1 == k = Split2 (1, ln) (0, two mn k2 rn)
| k2 == k = Split2 (0, two ln k1 mn) (1, rn)
| k2 < k = case ra of
Skew L -> Skew L
Skew R -> Split2 (0, two ln k1 mn) (1, rn)
Split2 (lbal, lv) (rbal, rv) ->
Split2 (merge 0 (two ln k1 mn) k2 lbal lv) (rbal 1, rv)
_ -> error "Malformed Tree23"
| k1 < k = case ma of
Skew L -> Split2 (0, two ln k1 mn) (1, rn)
Skew R -> Split2 (1, ln) (0, two mn k2 rn)
Split2 (lbal, lv) (rbal, rv) ->
Split2 (merge (1) ln k1 lbal lv) (merge rbal rv k2 (1) rn)
_ -> error "Malformed Tree23"
| otherwise = case la of
Skew R -> Skew R
Skew L -> Split2 (1, ln) (0, two mn k2 rn)
Split2 (lbal, lv) (rbal, rv) ->
Split2 (lbal 1, lv) (merge rbal rv k2 0 (two mn k2 rn))
_ -> error "Malformed Tree23"
minTree23 :: Fixed g => g (Tree23 d) -> Maybe (TreeKey d, TreeValue d)
minTree23 = cata phi where
phi Empty = Nothing
phi (Leaf k v) = Just (k, v)
phi (Two l _ _) = l
phi (Three l _ _ _ _) = l
maxTree23 :: Fixed g => g (Tree23 d) -> Maybe (TreeKey d, TreeValue d)
maxTree23 = cata phi where
phi Empty = Nothing
phi (Leaf k v) = Just (k, v)
phi (Two _ _ r) = r
phi (Three _ _ _ _ r) = r
instance FixedAlg (Tree23 (Map k v)) where
type Alg (Tree23 (Map k v)) = v
instance FixedSub (Tree23 (Map k v)) where
type Sub (Tree23 (Map k v)) v v' = Tree23 (Map k v')
instance FixedFunctor (Tree23 (Map k v)) where
fmapF f = cata phi where
phi Empty = empty
phi (Leaf k (MV v)) = leaf (unsafeCoerce k) (MV (f v))
phi (Two l k r) = two l (unsafeCoerce k) r
phi (Three l k1 m k2 r) =
three l (unsafeCoerce k1) m (unsafeCoerce k2) r
instance FixedFoldable (Tree23 (Map k v)) where
foldMapF f = cata phi where
phi Empty = mempty
phi (Leaf _ (MV v)) = f v
phi (Two l _ r) = l <> r
phi (Three l _ m _ r) = l <> m <> r
instance FixedTraversable (Tree23 (Map k v)) where
traverseF f = cata phi where
phi Empty = pure empty
phi (Leaf k (MV v)) = (leaf (unsafeCoerce k) . MV) <$> f v
phi (Two l k r) = two <$> l <*> pure (unsafeCoerce k) <*> r
phi (Three l k1 m k2 r) = three <$> l <*> pure (unsafeCoerce k1) <*>
m <*> pure (unsafeCoerce k2) <*> r