module HashAddressed.Directory
(
ContentAddressedDirectory, init,
writeStreaming, writeLazy,
WriteResult (..), WriteType (..),
)
where
import Essentials
import HashAddressed.HashFunction
import Control.Monad.IO.Class (MonadIO)
import Data.Function (flip)
import Prelude (FilePath, IO)
import System.FilePath ((</>))
import qualified Control.Monad.Trans.Class as Monad
import qualified Control.Monad.Trans.Resource as Resource
import qualified Control.Monad.Trans.State as Monad
import qualified Control.Monad.Trans.State as State
import qualified Crypto.Hash.SHA256 as Hash
import qualified Data.ByteString as Strict
import qualified Data.ByteString as Strict.ByteString
import qualified Data.ByteString.Base16 as Base16
import qualified Data.ByteString.Char8 as Strict.ByteString.Char8
import qualified Data.ByteString.Lazy as Lazy
import qualified Data.ByteString.Lazy as Lazy.ByteString
import qualified System.Directory as Directory
import qualified System.IO as IO
import qualified System.IO.Temp as Temporary
data ContentAddressedDirectory =
ContentAddressedDirectory
{ ContentAddressedDirectory -> FilePath
directory :: FilePath
}
data WriteResult =
WriteResult
{ WriteResult -> FilePath
contentAddressedFile :: FilePath
, WriteResult -> WriteType
writeType :: WriteType
}
data WriteType = AlreadyPresent | NewContent
init ::
HashFunction
-> FilePath
-> ContentAddressedDirectory
init :: HashFunction -> FilePath -> ContentAddressedDirectory
init HashFunction
SHA_256 = FilePath -> ContentAddressedDirectory
ContentAddressedDirectory
writeStreaming ::
ContentAddressedDirectory
-> (forall m. MonadIO m => (Strict.ByteString -> m ()) -> m ())
-> IO WriteResult
writeStreaming :: ContentAddressedDirectory
-> (forall (m :: * -> *).
MonadIO m =>
(ByteString -> m ()) -> m ())
-> IO WriteResult
writeStreaming ContentAddressedDirectory
cafs forall (m :: * -> *). MonadIO m => (ByteString -> m ()) -> m ()
continue = forall (m :: * -> *) a. MonadUnliftIO m => ResourceT m a -> m a
Resource.runResourceT @IO
do
FilePath
temporaryRoot <- forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
Monad.lift IO FilePath
Temporary.getCanonicalTemporaryDirectory
(ReleaseKey
_, FilePath
temporaryDirectory) <- forall (m :: * -> *) a.
MonadResource m =>
IO a -> (a -> IO ()) -> m (ReleaseKey, a)
Resource.allocate
(FilePath -> FilePath -> IO FilePath
Temporary.createTempDirectory FilePath
temporaryRoot FilePath
"cafs")
FilePath -> IO ()
Directory.removeDirectoryRecursive
let temporaryFile :: FilePath
temporaryFile = FilePath
temporaryDirectory FilePath -> FilePath -> FilePath
</> FilePath
"cafs-file"
(ReleaseKey
handleRelease, Handle
handle) <- forall (m :: * -> *) a.
MonadResource m =>
IO a -> (a -> IO ()) -> m (ReleaseKey, a)
Resource.allocate
(FilePath -> IOMode -> IO Handle
IO.openBinaryFile FilePath
temporaryFile IOMode
IO.WriteMode)
Handle -> IO ()
IO.hClose
Ctx
hashState :: Hash.Ctx <- forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
Monad.lift forall a b. (a -> b) -> a -> b
$ forall a b c. (a -> b -> c) -> b -> a -> c
flip forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m s
Monad.execStateT Ctx
Hash.init forall a b. (a -> b) -> a -> b
$
forall (m :: * -> *). MonadIO m => (ByteString -> m ()) -> m ()
continue \ByteString
chunk ->
do
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
Monad.lift forall a b. (a -> b) -> a -> b
$ Handle -> ByteString -> IO ()
Strict.ByteString.hPut Handle
handle ByteString
chunk
forall (m :: * -> *) s. Monad m => (s -> s) -> StateT s m ()
State.modify' \Ctx
hashState -> Ctx -> ByteString -> Ctx
Hash.update Ctx
hashState ByteString
chunk
forall (m :: * -> *). MonadIO m => ReleaseKey -> m ()
Resource.release ReleaseKey
handleRelease
let contentAddressedFile :: FilePath
contentAddressedFile = ContentAddressedDirectory -> FilePath
directory ContentAddressedDirectory
cafs FilePath -> FilePath -> FilePath
</>
ByteString -> FilePath
Strict.ByteString.Char8.unpack
(ByteString -> ByteString
Base16.encode (Ctx -> ByteString
Hash.finalize Ctx
hashState))
WriteType
writeType <- forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
Monad.lift (FilePath -> IO Bool
Directory.doesPathExist FilePath
contentAddressedFile)
forall (f :: * -> *) a b. Functor f => f a -> (a -> b) -> f b
<&> \case{ Bool
True -> WriteType
AlreadyPresent; Bool
False -> WriteType
NewContent }
case WriteType
writeType of
WriteType
NewContent -> forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
Monad.lift forall a b. (a -> b) -> a -> b
$
FilePath -> FilePath -> IO ()
Directory.renamePath FilePath
temporaryFile FilePath
contentAddressedFile
WriteType
AlreadyPresent -> forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
pure WriteResult{ FilePath
contentAddressedFile :: FilePath
contentAddressedFile :: FilePath
contentAddressedFile, WriteType
writeType :: WriteType
writeType :: WriteType
writeType }
writeLazy ::
ContentAddressedDirectory
-> Lazy.ByteString
-> IO WriteResult
writeLazy :: ContentAddressedDirectory -> ByteString -> IO WriteResult
writeLazy ContentAddressedDirectory
cafs ByteString
lbs = ContentAddressedDirectory
-> (forall (m :: * -> *).
MonadIO m =>
(ByteString -> m ()) -> m ())
-> IO WriteResult
writeStreaming ContentAddressedDirectory
cafs \ByteString -> m ()
writeChunk ->
forall (t :: * -> *) (f :: * -> *) a b.
(Foldable t, Applicative f) =>
(a -> f b) -> t a -> f ()
traverse_ ByteString -> m ()
writeChunk (ByteString -> [ByteString]
Lazy.ByteString.toChunks ByteString
lbs)