aboutsummaryrefslogtreecommitdiffstats
path: root/app/Store.hs
diff options
context:
space:
mode:
Diffstat (limited to 'app/Store.hs')
-rw-r--r--app/Store.hs104
1 files changed, 88 insertions, 16 deletions
diff --git a/app/Store.hs b/app/Store.hs
index 704a1cc..ace8ea9 100644
--- a/app/Store.hs
+++ b/app/Store.hs
@@ -2,43 +2,104 @@
{-# LANGUAGE BlockArguments #-}
{-# LANGUAGE OverloadedRecordDot #-}
{-# LANGUAGE OverloadedStrings #-}
+{-# OPTIONS_GHC -fno-warn-orphans #-}
module Store
- ( withStore,
+ ( StoreT (..),
+ withStore,
listDirectory,
readFile,
)
where
import Control.Arrow (first)
-import Control.Monad.Catch
-import Control.Monad.IO.Unlift (MonadUnliftIO)
+import Control.Exception (Exception)
+import Control.Monad.Catch (MonadCatch (catch), MonadMask (generalBracket, mask, uninterruptibleMask), MonadThrow, throwM)
+import Control.Monad.IO.Unlift (MonadUnliftIO (withRunInIO))
import Control.Monad.Reader (ReaderT)
+import Control.Monad.State (MonadState (state), StateT (StateT), evalStateT, get)
+import Control.Monad.Trans (MonadIO, lift, liftIO)
import Data.ByteString qualified as B
import Data.ByteString.Lazy qualified as LB
import Data.ByteString.UTF8 qualified as B
import Data.List (isPrefixOf, sort)
+import Data.Tagged (Tagged (Tagged))
import Data.Text qualified as T
import Git
-import Git.Libgit2 (LgRepo, lgFactory)
+import Git.Libgit2 (HasLgRepo (getRepository), LgRepo, OidPtr, lgFactory)
+import Git.Libgit2 qualified as GB
import System.FilePath
import Prelude hiding (readFile)
+data StoreT m a = StoreT
+ { unStoreT :: StateT OidPtr m a
+ }
+
+-- XXX `DeriveAnyClass` does not work for `StoreT`
+--
+-- The instances generated by `DeriveAnyClass` are empty..
+
+instance (Monad m, Applicative m) => Applicative (StoreT m) where
+ pure = StoreT . pure
+ f <*> x = StoreT $ unStoreT f <*> unStoreT x
+
+instance Functor m => Functor (StoreT m) where
+ fmap f = StoreT . fmap f . unStoreT
+
+instance Monad m => Monad (StoreT m) where
+ m >>= k = StoreT $ unStoreT m >>= unStoreT . k
+
+instance MonadThrow m => MonadThrow (StoreT m) where
+ throwM = StoreT . throwM
+
+instance MonadCatch m => MonadCatch (StoreT m) where
+ m `catch` h = StoreT $ unStoreT m `catch` (unStoreT . h)
+
+instance MonadIO m => MonadIO (StoreT m) where
+ liftIO = StoreT . liftIO
+
+instance Monad m => MonadState OidPtr (StoreT m) where
+ state = StoreT . state
+
+instance MonadMask m => MonadMask (StoreT m) where
+ mask f = StoreT $ mask (\g -> unStoreT (f (\m -> (StoreT (g (unStoreT m))))))
+ uninterruptibleMask f = StoreT $ uninterruptibleMask (\g -> unStoreT (f (\m -> (StoreT (g (unStoreT m))))))
+ generalBracket m n k = StoreT (generalBracket (unStoreT m) ((unStoreT .) . n) (unStoreT . k))
+
+instance MonadUnliftIO m => MonadUnliftIO (StateT s m) where
+ withRunInIO f =
+ StateT $ \s ->
+ (,s) <$> withRunInIO (\g -> f (\m -> g (evalStateT m s)))
+
+instance MonadUnliftIO m => MonadUnliftIO (StoreT m) where
+ withRunInIO f =
+ StoreT (withRunInIO (\g -> f (\m -> g (unStoreT m))))
+
+instance (HasLgRepo m, Monad m) => HasLgRepo (StoreT m) where
+ getRepository = StoreT $ lift GB.getRepository
+
withStore ::
- (MonadMask m, MonadUnliftIO m) =>
+ (MonadUnliftIO m, MonadMask m, MonadFail m) =>
FilePath ->
- ReaderT LgRepo m a ->
+ RefName ->
+ StoreT (ReaderT LgRepo m) a ->
m a
-withStore = withRepository lgFactory
+withStore fp ref act = withRepository lgFactory fp do
+ Just cid <- resolveReference ref
+ evalStateT (unStoreT act) cid
listDirectory ::
- MonadGit r m =>
- CommitOid r ->
+ ( MonadState OidPtr m,
+ MonadMask m,
+ MonadUnliftIO m,
+ HasLgRepo m
+ ) =>
FilePath ->
m [FilePath]
-listDirectory cid dir' = do
+listDirectory dir' = do
let dir = normalise $ if hasDrive dir' then dir' else joinDrive "/" dir'
n = length (splitPath dir)
+ cid <- Tagged <$> get
tid <- (.commitTree) <$> lookupCommit cid
tree <- lookupTree tid
sort
@@ -67,7 +128,14 @@ data InappropriateType = InappropriateType String FilePath
instance Exception InappropriateType
class Readable a where
- readFile :: MonadGit r m => CommitOid r -> FilePath -> m a
+ readFile ::
+ ( MonadState OidPtr m,
+ MonadMask m,
+ MonadUnliftIO m,
+ HasLgRepo m
+ ) =>
+ FilePath ->
+ m a
instance Readable T.Text where
readFile = readFile' catBlobUtf8
@@ -79,12 +147,16 @@ instance Readable LB.ByteString where
readFile = readFile' catBlobLazy
readFile' ::
- MonadGit r m =>
- (BlobOid r -> m a) ->
- CommitOid r ->
+ ( MonadState OidPtr m,
+ MonadMask m,
+ MonadUnliftIO m,
+ HasLgRepo m
+ ) =>
+ (Tagged LgRepo OidPtr -> m b) ->
FilePath ->
- m a
-readFile' cat cid fp = do
+ m b
+readFile' cat fp = do
+ cid <- Tagged <$> get
tid <- (.commitTree) <$> lookupCommit cid
tree <- lookupTree tid
maybe