{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE TupleSections #-}
{-# OPTIONS -Wall #-}
{-# OPTIONS -Werror=incomplete-patterns #-}
{-# OPTIONS -fno-warn-unused-top-binds #-}
module Network.DFINITY.RadixTree (
RadixDatabase(..)
, RadixError(..)
, RadixRoot
, RadixTree
, createRadixTree
, insertRadixTree
, deleteRadixTree
, merkleizeRadixTree
, lookupRadixTree
, isEmptyRadixTree
, isValidRadixRoot
, contentsRadixTree
, contentsMerkleizedRadixTree
, contentsNonMerkleizedRadixTree
, printRadixTree
, printMerkleizedRadixTree
, printNonMerkleizedRadixTree
) where
import Control.Exception (throw)
import Control.Monad (foldM, forM_)
import Control.Monad.IO.Class (MonadIO(..))
import Control.Monad.Trans.Resource (ResourceT)
import Data.BloomFilter as Bloom (elem, insert, insertList)
import Data.Bool (bool)
import Data.ByteString (ByteString)
import Data.ByteString.Short (fromShort)
import Data.Default.Class (def)
import Data.List.NonEmpty (NonEmpty(..), fromList)
import Data.LruCache as LRU (empty)
import Data.Map.Strict as Map (empty)
import Data.Maybe (fromJust, isJust, isNothing, listToMaybe)
import Data.Tuple (swap)
import Database.LevelDB (DB)
import Network.DFINITY.RadixTree.Bits
import Network.DFINITY.RadixTree.Bloom
import Network.DFINITY.RadixTree.Lenses
import Network.DFINITY.RadixTree.Memory
import Network.DFINITY.RadixTree.Types
import Network.DFINITY.RadixTree.Utilities
createRadixTree
:: RadixDatabase m database
=> Int
-> Int
-> Maybe RadixRoot
-> database
-> m (RadixTree database)
{-# SPECIALISE createRadixTree
:: Int
-> Int
-> Maybe RadixRoot
-> DB
-> ResourceT IO (RadixTree DB) #-}
createRadixTree bloomSize cacheSize checkpoint database
| bloomSize <= 0 = throw $ InvalidArgument "invalid Bloom filter size"
| cacheSize <= 0 = throw $ InvalidArgument "invalid LRU cache size"
| otherwise = do
(root, cache') <-
case checkpoint of
Nothing -> storeCold def cache database
Just root -> do
result <- loadCold root cache database
case snd <$> result of
Nothing -> throw $ StateRootDoesNotExist root
Just cache' -> pure (root, cache')
pure $ RadixTree bloom bloomSize Map.empty cache' cacheSize root database 0 root
where
bloom = emptyRadixBloom bloomSize
cache = LRU.empty cacheSize
isEmptyRadixTree
:: RadixTree database
-> Bool
{-# INLINABLE isEmptyRadixTree #-}
isEmptyRadixTree = (==) defaultRoot . _radixRoot
isValidRadixRoot
:: RadixDatabase m database
=> RadixRoot
-> RadixTree database
-> m Bool
{-# SPECIALISE isValidRadixRoot
:: RadixRoot
-> RadixTree DB
-> ResourceT IO Bool #-}
isValidRadixRoot root RadixTree {..} =
isJust <$> load _radixDatabase key
where
key = fromShort root
searchRadixTree
:: RadixDatabase m database
=> Bool
-> (RadixTree database -> m (Maybe (RadixNode, RadixCache)))
-> ByteString
-> RadixTree database
-> m (Either RadixError RadixSearchResult)
{-# SPECIALISE searchRadixTree
:: Bool
-> (RadixTree DB -> ResourceT IO (Maybe (RadixNode, RadixCache)))
-> ByteString
-> RadixTree DB
-> ResourceT IO (Either RadixError RadixSearchResult) #-}
searchRadixTree flag strategy = \ key tree@RadixTree {..} -> do
let key' = toBits key
let tree' = tree `bool` setRoot _radixCheckpoint tree $ flag
loop Nothing [] [] [] key' tree' where
loop implicit roots nodes prefixes key tree@RadixTree {..} = do
result <- strategy tree
case result of
Nothing -> pure $ Left $ StateRootDoesNotExist _radixRoot
Just (node@RadixNode {..}, cache') -> do
let bits = maybe id (:) implicit $ maybe [] toBits _radixPrefix
let prefix = matchBits bits key
let n = length prefix
let overflow = drop n bits
let roots' = _radixRoot:roots
let nodes' = node:nodes
let prefixes' = prefix:prefixes
let key' = drop n key
let residue = not $ null overflow
let bit = head key'
let child = bool _radixLeft _radixRight bit
if null key' || residue || isNothing child
then pure $ Right (fromList roots', fromList nodes', fromList prefixes', overflow, key', cache')
else do
let root' = fromJust child
let tree' = setCache cache' $ setRoot root' tree
let implicit' = Just bit
loop implicit' roots' nodes' prefixes' key' tree'
searchMerkleizedRadixTree
:: RadixDatabase m database
=> ByteString
-> RadixTree database
-> m (Either RadixError RadixSearchResult)
{-# SPECIALISE searchMerkleizedRadixTree
:: ByteString
-> RadixTree DB
-> ResourceT IO (Either RadixError RadixSearchResult) #-}
searchMerkleizedRadixTree =
searchRadixTree True $ \ RadixTree {..} ->
loadCold _radixRoot _radixCache _radixDatabase
searchNonMerkleizedRadixTree
:: RadixDatabase m database
=> ByteString
-> RadixTree database
-> m (Either RadixError RadixSearchResult)
{-# SPECIALISE searchNonMerkleizedRadixTree
:: ByteString
-> RadixTree DB
-> ResourceT IO (Either RadixError RadixSearchResult) #-}
searchNonMerkleizedRadixTree =
searchRadixTree False $ \ RadixTree {..} ->
loadHot _radixRoot _radixBuffer _radixCache _radixDatabase
insertRadixTree
:: RadixDatabase m database
=> ByteString
-> ByteString
-> RadixTree database
-> m (RadixTree database)
{-# SPECIALISE insertRadixTree
:: ByteString
-> ByteString
-> RadixTree DB
-> ResourceT IO (RadixTree DB) #-}
insertRadixTree key value tree =
if isEmptyRadixTree tree
then pure $ initializeRadixTree key value tree
else searchNonMerkleizedRadixTree key tree >>= \ case
Left err -> throw err
Right result@(_, _, _, [], [], _) ->
pure $ insertRadixTreeAt result value tree
Right result@(_, _, _, [], _, _) ->
pure $ insertRadixTreeAfter result value tree
Right result@(_, _, _, _, [], _) ->
pure $ insertRadixTreeBefore result value tree
Right result ->
pure $ insertRadixTreeBetween result value tree
initializeRadixTree
:: ByteString
-> ByteString
-> RadixTree database
-> RadixTree database
{-# INLINABLE initializeRadixTree #-}
initializeRadixTree key value tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setNonce nonce $ setRoot root tree
where
prefix = createPrefix $ toBits key
node = setPrefix prefix $ Just value `setLeaf` def
root = createRootFromNonce _radixNonce
bloom = insert root _radixBloom
nonce = _radixNonce + 1
buffer = storeHot root node _radixBuffer
insertRadixTreeAt
:: RadixSearchResult
-> ByteString
-> RadixTree database
-> RadixTree database
{-# INLINABLE insertRadixTreeAt #-}
insertRadixTreeAt (_:|roots, node:|nodes, prefix:|_, _, _, cache) value tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setNonce nonce $ setRoot state tree
where
node' = Just value `setLeaf` node
root' = createRootFromNonce _radixNonce
parent = listToMaybe $ zip3 roots nodes prefix
bloom = flip insertList _radixBloom $ root':roots
buffer = merkleSpoof root' parent $ storeHot root' node' _radixBuffer
nonce = _radixNonce + 1
state = bool _radixRoot root' $ isNothing parent
insertRadixTreeAfter
:: RadixSearchResult
-> ByteString
-> RadixTree database
-> RadixTree database
{-# INLINABLE insertRadixTreeAfter #-}
insertRadixTreeAfter (_:|roots, node:|nodes, prefix:|_, _, keyOverflow, cache) value tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setNonce nonce $ setRoot state tree
where
prefix' = createPrefix $ drop 1 keyOverflow
node' = setPrefix prefix' $ Just value `setLeaf` def
root' = createRootFromNonce _radixNonce
node'' = test `setChild` Just root' $ node
root'' = createRootFromNonce $ _radixNonce + 1
test = head keyOverflow
parent = listToMaybe $ zip3 roots nodes prefix
bloom = flip insertList _radixBloom $ root'':root':roots
buffer = merkleSpoof root'' parent $ storeHot root'' node'' $ storeHot root' node' _radixBuffer
nonce = _radixNonce + 2
state = bool _radixRoot root'' $ isNothing parent
insertRadixTreeBefore
:: RadixSearchResult
-> ByteString
-> RadixTree database
-> RadixTree database
{-# INLINABLE insertRadixTreeBefore #-}
insertRadixTreeBefore (_:|roots, node:|nodes, prefix:|_, prefixOverflow, _, cache) value tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setNonce nonce $ setRoot state tree
where
prefix' = createPrefix $ drop 1 prefixOverflow
node' = setPrefix prefix' node
root' = createRootFromNonce _radixNonce
prefix'' = createPrefix $ drop 1 prefix `bool` prefix $ isNothing parent
node'' = setPrefix prefix'' $ test `setChild` Just root' $ Just value `setLeaf` def
root'' = createRootFromNonce $ _radixNonce + 1
test = head prefixOverflow
parent = listToMaybe $ zip3 roots nodes prefix
bloom = flip insertList _radixBloom $ root'':root':roots
buffer = merkleSpoof root'' parent $ storeHot root'' node'' $ storeHot root' node' _radixBuffer
nonce = _radixNonce + 2
state = bool _radixRoot root'' $ isNothing parent
insertRadixTreeBetween
:: RadixSearchResult
-> ByteString
-> RadixTree database
-> RadixTree database
{-# INLINABLE insertRadixTreeBetween #-}
insertRadixTreeBetween (_:|roots, node:|nodes, prefix:|_, prefixOverflow, keyOverflow, cache) value tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setNonce nonce $ setRoot state tree
where
prefix' = createPrefix $ drop 1 keyOverflow
node' = setPrefix prefix' $ Just value `setLeaf` def
root' = createRootFromNonce _radixNonce
prefix'' = createPrefix $ drop 1 prefixOverflow
node'' = setPrefix prefix'' node
root'' = createRootFromNonce $ _radixNonce + 1
prefix''' = createPrefix $ drop 1 prefix `bool` prefix $ isNothing parent
node''' = setPrefix prefix''' $ setChildren children def
root''' = createRootFromNonce $ _radixNonce + 2
test = head keyOverflow
children = bool id swap test (Just root', Just root'')
parent = listToMaybe $ zip3 roots nodes prefix
bloom = flip insertList _radixBloom $ root''':root'':root':roots
buffer = merkleSpoof root''' parent $ storeHot root''' node''' $ storeHot root'' node'' $ storeHot root' node' _radixBuffer
nonce = _radixNonce + 3
state = bool _radixRoot root''' $ isNothing parent
deleteRadixTree
:: RadixDatabase m database
=> ByteString
-> RadixTree database
-> m (RadixTree database)
{-# SPECIALISE deleteRadixTree
:: ByteString
-> RadixTree DB
-> ResourceT IO (RadixTree DB) #-}
deleteRadixTree key tree@RadixTree {..} =
if isEmptyRadixTree tree
then pure tree
else searchNonMerkleizedRadixTree key tree >>= \ case
Left err -> throw err
Right result@(_, nodes, prefix:|_, [], [], cache) ->
case nodes of
RadixNode _ Nothing Nothing _:|[] ->
pure $ deleteRadixTreeNoChildrenNoParent result tree
RadixNode _ Nothing Nothing _:|parent:_ | isJust $ getLeaf parent ->
pure $ deleteRadixTreeNoChildrenParentWithLeaf result tree
RadixNode _ Nothing Nothing _:|parent:_ -> do
let test = not $ head prefix
let root = fromJust $ getChild test parent
loadHot root _radixBuffer cache _radixDatabase >>= \ case
Nothing -> throw $ StateRootDoesNotExist root
Just (node, cache') ->
pure $ deleteRadixTreeNoChildrenParentWithoutLeaf result node cache' test tree
RadixNode _ child Nothing _:|_ | isJust child -> do
let test = False
let root = fromJust child
loadHot root _radixBuffer cache _radixDatabase >>= \ case
Nothing -> throw $ StateRootDoesNotExist root
Just (node, cache') ->
pure $ deleteRadixTreeOneChild result node cache' test tree
RadixNode _ Nothing child _:|_ | isJust child -> do
let test = True
let root = fromJust child
loadHot root _radixBuffer cache _radixDatabase >>= \ case
Nothing -> throw $ StateRootDoesNotExist root
Just (node, cache') ->
pure $ deleteRadixTreeOneChild result node cache' test tree
_ -> pure $ deleteRadixTreeTwoChildren result tree
Right _ -> pure tree
deleteRadixTreeNoChildrenNoParent
:: RadixSearchResult
-> RadixTree database
-> RadixTree database
{-# INLINABLE deleteRadixTreeNoChildrenNoParent #-}
deleteRadixTreeNoChildrenNoParent (_, _, _, _, _, cache) tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setRoot state tree
where
bloom = insert defaultRoot _radixBloom
buffer = storeHot defaultRoot def _radixBuffer
state = defaultRoot
deleteRadixTreeNoChildrenParentWithLeaf
:: RadixSearchResult
-> RadixTree database
-> RadixTree database
{-# INLINABLE deleteRadixTreeNoChildrenParentWithLeaf #-}
deleteRadixTreeNoChildrenParentWithLeaf (_:|_:roots, _:|node:nodes, prefix:|prefixes, _, _, cache) tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setNonce nonce $ setRoot state tree
where
node' = setChild test Nothing node
root' = createRootFromNonce _radixNonce
test = head prefix
parent = listToMaybe $ zip3 roots nodes $ map head prefixes
bloom = flip insertList _radixBloom $ root':roots
buffer = merkleSpoof root' parent $ storeHot root' node' _radixBuffer
nonce = _radixNonce + 1
state = bool _radixRoot root' $ isNothing parent
deleteRadixTreeNoChildrenParentWithLeaf _ _ =
throw $ InvalidArgument "unknown parent"
deleteRadixTreeNoChildrenParentWithoutLeaf
:: RadixSearchResult
-> RadixNode
-> RadixCache
-> Bool
-> RadixTree database
-> RadixTree database
{-# INLINABLE deleteRadixTreeNoChildrenParentWithoutLeaf #-}
deleteRadixTreeNoChildrenParentWithoutLeaf (_:|_:roots, _:|_:nodes, _:|prefixes, _, _, _) node@RadixNode {..} cache test tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setNonce nonce $ setRoot state tree
where
prefix' = createPrefix $ drop 1 bits `bool` bits $ isNothing parent
node' = setPrefix prefix' node
root' = createRootFromNonce _radixNonce
bits = head prefixes ++ test:maybe [] toBits _radixPrefix
parent = listToMaybe $ zip3 roots nodes $ map head prefixes
bloom = flip insertList _radixBloom $ root':roots
buffer = merkleSpoof root' parent $ storeHot root' node' _radixBuffer
nonce = _radixNonce + 1
state = bool _radixRoot root' $ isNothing parent
deleteRadixTreeNoChildrenParentWithoutLeaf _ _ _ _ _ =
throw $ InvalidArgument "unknown parent"
deleteRadixTreeOneChild
:: RadixSearchResult
-> RadixNode
-> RadixCache
-> Bool
-> RadixTree database
-> RadixTree database
{-# INLINABLE deleteRadixTreeOneChild #-}
deleteRadixTreeOneChild (_:|roots, _:|nodes, prefix:|_, _, _, _) node@RadixNode {..} cache test tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setNonce nonce $ setRoot state tree
where
prefix' = createPrefix $ drop 1 bits `bool` bits $ isNothing parent
node' = setPrefix prefix' node
root' = createRootFromNonce _radixNonce
bits = prefix ++ test:maybe [] toBits _radixPrefix
parent = listToMaybe $ zip3 roots nodes prefix
bloom = flip insertList _radixBloom $ root':roots
buffer = merkleSpoof root' parent $ storeHot root' node' _radixBuffer
nonce = _radixNonce + 1
state = bool _radixRoot root' $ isNothing parent
deleteRadixTreeTwoChildren
:: RadixSearchResult
-> RadixTree database
-> RadixTree database
{-# INLINABLE deleteRadixTreeTwoChildren #-}
deleteRadixTreeTwoChildren (_:|roots, node:|nodes, prefix:|_, _, _, cache) tree@RadixTree {..} =
seq bloom $ setBloom bloom $ setBuffer buffer $ setCache cache $ setNonce nonce $ setRoot state tree
where
node' = setLeaf Nothing node
root' = createRootFromNonce _radixNonce
parent = listToMaybe $ zip3 roots nodes prefix
bloom = flip insertList _radixBloom $ root':roots
buffer = merkleSpoof root' parent $ storeHot root' node' _radixBuffer
nonce = _radixNonce + 1
state = bool _radixRoot root' $ isNothing parent
lookupRadixTree
:: RadixDatabase m database
=> ByteString
-> RadixTree database
-> m (Maybe (ByteString, RadixTree database))
{-# SPECIALISE lookupRadixTree
:: ByteString
-> RadixTree DB
-> ResourceT IO (Maybe (ByteString, RadixTree DB)) #-}
lookupRadixTree key tree = do
found <- searchNonMerkleizedRadixTree key tree
case found of
Left err -> throw err
Right (_, RadixNode {..}:|_, _, prefixOverflow, keyOverflow, cache') ->
if not $ null prefixOverflow && null keyOverflow
then pure Nothing
else pure $ do
value <- _radixLeaf
let tree' = setCache cache' tree
pure (value, tree')
merkleSpoof
:: RadixRoot
-> Maybe (RadixRoot, RadixNode, Bool)
-> RadixBuffer
-> RadixBuffer
{-# INLINABLE merkleSpoof #-}
merkleSpoof mask = \ case
Nothing -> id
Just (root, node, test) ->
storeHot root $ test `setChild` Just mask $ node
merkleizeRadixTree
:: RadixDatabase m database
=> RadixTree database
-> m (RadixRoot, RadixTree database)
{-# SPECIALISE merkleizeRadixTree
:: RadixTree DB
-> ResourceT IO (RadixRoot, RadixTree DB) #-}
merkleizeRadixTree RadixTree {..} = do
(root, cache) <- loop _radixRoot _radixCache
let tree = RadixTree bloom _radixBloomSize Map.empty cache _radixCacheSize root _radixDatabase 0 root
pure (root, tree)
where
bloom = emptyRadixBloom _radixBloomSize
loop root cache =
if not $ Bloom.elem root _radixBloom
then pure (root, cache)
else do
result <- loadHot root _radixBuffer cache _radixDatabase
case result of
Nothing -> throw $ StateRootDoesNotExist root
Just (node@RadixNode {..}, cache') ->
case (_radixLeft, _radixRight) of
(Nothing, Nothing) ->
storeCold node cache' _radixDatabase
(Just child, Nothing) -> do
(root', cache'') <- loop child cache'
let node' = False `setChild` Just root' $ node
storeCold node' cache'' _radixDatabase
(Nothing, Just child) -> do
(root', cache'') <- loop child cache'
let node' = True `setChild` Just root' $ node
storeCold node' cache'' _radixDatabase
(Just left, Just right) -> do
(root', cache'') <- loop left cache'
(root'', cache''') <- loop right cache''
let node' = setChildren (Just root', Just root'') node
storeCold node' cache''' _radixDatabase
contentsRadixTree'
:: RadixDatabase m database
=> Bool
-> (RadixTree database -> m (Maybe (RadixNode, RadixCache)))
-> RadixTree database
-> m [(ByteString, ByteString)]
{-# SPECIALISE contentsRadixTree'
:: Bool
-> (RadixTree DB -> ResourceT IO (Maybe (RadixNode, RadixCache)))
-> RadixTree DB
-> ResourceT IO [(ByteString, ByteString)] #-}
contentsRadixTree' flag strategy = \ tree@RadixTree {..} -> do
let tree' = tree `bool` setRoot _radixCheckpoint tree $ flag
loop tree' [] [] where
loop tree@RadixTree {..} prefix accum = do
result <- strategy tree
case fst <$> result of
Nothing -> throw $ StateRootDoesNotExist _radixRoot
Just RadixNode {..} -> do
let prefix' = prefix ++ maybe [] toBits _radixPrefix
let key = fromBits prefix'
let accum' = maybe accum (\ value -> (key, value):accum) _radixLeaf
let children = [(,False) <$> _radixLeft, (,True) <$> _radixRight]
flip foldM accum' `flip` children $ \ accum'' -> \ case
Nothing -> pure accum''
Just (root, test) -> do
let tree' = setRoot root tree
let prefix'' = prefix' ++ [test]
loop tree' prefix'' accum''
contentsRadixTree
:: RadixDatabase m database
=> RadixTree database
-> m [(ByteString, ByteString)]
{-# SPECIALISE contentsRadixTree
:: RadixTree DB
-> ResourceT IO [(ByteString, ByteString)] #-}
contentsRadixTree = contentsNonMerkleizedRadixTree
contentsMerkleizedRadixTree
:: RadixDatabase m database
=> RadixTree database
-> m [(ByteString, ByteString)]
{-# SPECIALISE contentsMerkleizedRadixTree
:: RadixTree DB
-> ResourceT IO [(ByteString, ByteString)] #-}
contentsMerkleizedRadixTree =
contentsRadixTree' True $ \ RadixTree {..} ->
loadCold _radixRoot _radixCache _radixDatabase
contentsNonMerkleizedRadixTree
:: RadixDatabase m database
=> RadixTree database
-> m [(ByteString, ByteString)]
{-# SPECIALISE contentsNonMerkleizedRadixTree
:: RadixTree DB
-> ResourceT IO [(ByteString, ByteString)] #-}
contentsNonMerkleizedRadixTree =
contentsRadixTree' False $ \ RadixTree {..} ->
loadHot _radixRoot _radixBuffer _radixCache _radixDatabase
printRadixTree'
:: MonadIO m
=> RadixDatabase m database
=> Bool
-> (RadixTree database -> m (Maybe (RadixNode, RadixCache)))
-> RadixTree database
-> m ()
{-# SPECIALISE printRadixTree'
:: Bool
-> (RadixTree DB -> ResourceT IO (Maybe (RadixNode, RadixCache)))
-> RadixTree DB
-> ResourceT IO () #-}
printRadixTree' flag strategy = \ tree@RadixTree {..} -> do
let tree' = tree `bool` setRoot _radixCheckpoint tree $ flag
loop tree' 0 where
loop tree@RadixTree {..} i = do
result <- strategy tree
case fst <$> result of
Nothing -> throw $ StateRootDoesNotExist _radixRoot
Just node@RadixNode {..} -> do
let indent = (++) $ concat $ replicate i "|"
liftIO $ putStrLn $ indent $ show node
let j = i + 1
forM_ [_radixLeft, _radixRight] $ \ case
Nothing -> pure ()
Just root -> setRoot root tree `loop` j
printRadixTree
:: MonadIO m
=> RadixDatabase m database
=> RadixTree database
-> m ()
{-# SPECIALISE printRadixTree
:: RadixTree DB
-> ResourceT IO () #-}
printRadixTree = printNonMerkleizedRadixTree
printMerkleizedRadixTree
:: MonadIO m
=> RadixDatabase m database
=> RadixTree database
-> m ()
{-# SPECIALISE printMerkleizedRadixTree
:: RadixTree DB
-> ResourceT IO () #-}
printMerkleizedRadixTree =
printRadixTree' True $ \ RadixTree {..} ->
loadCold _radixRoot _radixCache _radixDatabase
printNonMerkleizedRadixTree
:: MonadIO m
=> RadixDatabase m database
=> RadixTree database
-> m ()
{-# SPECIALISE printNonMerkleizedRadixTree
:: RadixTree DB
-> ResourceT IO () #-}
printNonMerkleizedRadixTree =
printRadixTree' False $ \ RadixTree {..} ->
loadHot _radixRoot _radixBuffer _radixCache _radixDatabase