aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLibravatar Alexander Foremny <aforemny@posteo.de>2024-02-09 02:40:03 +0100
committerLibravatar Alexander Foremny <aforemny@posteo.de>2024-02-09 02:40:03 +0100
commit11969ba7aed38e6e35d51d52e8eab9b46a0c6203 (patch)
treeca8d5dba9e182e62dc48b5063c26deee6d7cbb12
parent06402a97eb3d112b7affc52b24f698d60d4c8ece (diff)
add (simpler) `StoreT` type
-rw-r--r--app/Main.hs12
-rw-r--r--app/Store.hs141
2 files changed, 89 insertions, 64 deletions
diff --git a/app/Main.hs b/app/Main.hs
index 47f0280..9f04849 100644
--- a/app/Main.hs
+++ b/app/Main.hs
@@ -13,10 +13,8 @@ import Data.List (foldl', isSuffixOf)
import Data.Maybe (fromMaybe, mapMaybe)
import Data.Set qualified as S
import Data.String (IsString (fromString))
-import Data.Tagged (Tagged (Tagged))
import Data.Text qualified as T
import Debug.Trace (trace)
-import Git
import Store qualified as S
import System.Directory (setCurrentDirectory)
import System.FilePath ((</>))
@@ -146,9 +144,8 @@ query (Select fs c js ws) = do
where
ls c =
filter (not . (isSuffixOf "/"))
- <$> S.withStore "." do
- Just cid <- fmap Tagged <$> resolveReference "HEAD"
- S.listDirectory cid c
+ <$> S.withStore "." "HEAD" do
+ S.listDirectory c
combine :: [Record J.Value] -> [Join [Record J.Value]] -> [[Record J.Value]]
combine vs js = combine' (map (: []) vs) js
@@ -197,9 +194,8 @@ data DecodeException = DecodeException
instance Exception DecodeException
decodeFile :: J.FromJSON a => FilePath -> IO a
-decodeFile fp = S.withStore "." do
- Just cid <- fmap Tagged <$> resolveReference "HEAD"
- fromMaybe (throw DecodeException) . J.decode <$> S.readFile cid fp
+decodeFile fp = S.withStore "." "HEAD" do
+ fromMaybe (throw DecodeException) . J.decode <$> S.readFile fp
select :: FieldSelector -> [Record J.Value] -> J.Value
select All vs =
diff --git a/app/Store.hs b/app/Store.hs
index 704a1cc..3e0f3d3 100644
--- a/app/Store.hs
+++ b/app/Store.hs
@@ -1,5 +1,6 @@
{-# LANGUAGE AllowAmbiguousTypes #-}
{-# LANGUAGE BlockArguments #-}
+{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE OverloadedRecordDot #-}
{-# LANGUAGE OverloadedStrings #-}
@@ -11,50 +12,78 @@ module Store
where
import Control.Arrow (first)
-import Control.Monad.Catch
-import Control.Monad.IO.Unlift (MonadUnliftIO)
-import Control.Monad.Reader (ReaderT)
+import Control.Exception (Exception, finally)
+import Control.Monad.Catch (MonadCatch, MonadThrow, throwM)
+import Control.Monad.Reader (MonadReader, ReaderT, ask, runReaderT)
+import Control.Monad.State (MonadState, StateT, evalStateT, get)
+import Control.Monad.Trans (MonadIO, MonadTrans, lift)
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 System.FilePath
+import Git qualified as G
+import Git.Libgit2 qualified as GB
+import System.FilePath (addTrailingPathSeparator, hasDrive, joinDrive, makeRelative, normalise, splitPath)
import Prelude hiding (readFile)
-withStore ::
- (MonadMask m, MonadUnliftIO m) =>
- FilePath ->
- ReaderT LgRepo m a ->
- m a
-withStore = withRepository lgFactory
+newtype StoreT m a = StoreT
+ { runStoreT :: StateT GB.OidPtr (ReaderT GB.LgRepo m) a
+ }
+ deriving
+ ( Applicative,
+ Functor,
+ Monad,
+ MonadReader GB.LgRepo,
+ MonadState GB.OidPtr,
+ MonadCatch,
+ MonadThrow,
+ MonadIO
+ )
-listDirectory ::
- MonadGit r m =>
- CommitOid r ->
- FilePath ->
- m [FilePath]
-listDirectory cid dir' = do
- let dir = normalise $ if hasDrive dir' then dir' else joinDrive "/" dir'
- n = length (splitPath dir)
- tid <- (.commitTree) <$> lookupCommit cid
- tree <- lookupTree tid
- sort
- . map (makeRelative dir)
- . filter ((== n + 1) . length . splitPath)
- . filter (isPrefixOf (addTrailingPathSeparator dir))
- . map fst
- . map
- ( \e ->
- case snd e of
- BlobEntry _ _ -> e
- CommitEntry _ -> error "XXX commit entry"
- TreeEntry _ -> first addTrailingPathSeparator e
- )
- . map (first (("/" <>) . B.toString))
- <$> listTreeEntries tree
+instance MonadTrans StoreT where
+ lift = StoreT . lift . lift
+
+class MonadStore m where
+ getCommitOid :: m (G.CommitOid GB.LgRepo)
+ getRepository :: m GB.LgRepo
+
+instance Monad m => MonadStore (StoreT m) where
+ getCommitOid = Tagged <$> get
+ getRepository = ask
+
+type StoreM = StoreT IO
+
+withStore :: FilePath -> G.RefName -> StoreM a -> IO a
+withStore repoPath ref action = do
+ repo <- G.openRepository GB.lgFactory G.defaultRepositoryOptions {G.repoPath}
+ Just cid <- G.runRepository GB.lgFactory repo (G.resolveReference ref)
+ runReaderT (evalStateT (runStoreT action) cid) repo
+ `finally` G.runRepository GB.lgFactory repo G.closeRepository
+
+listDirectory :: FilePath -> StoreM [FilePath]
+listDirectory dir' = do
+ cid <- getCommitOid
+ repo <- getRepository
+ lift $ G.runRepository GB.lgFactory repo $ do
+ let dir = normalise $ if hasDrive dir' then dir' else joinDrive "/" dir'
+ n = length (splitPath dir)
+ tree <- G.lookupTree =<< (.commitTree) <$> G.lookupCommit cid
+ sort
+ . map (makeRelative dir)
+ . filter ((== n + 1) . length . splitPath)
+ . filter (isPrefixOf (addTrailingPathSeparator dir))
+ . map fst
+ . map
+ ( \e ->
+ case snd e of
+ G.BlobEntry _ _ -> e
+ G.CommitEntry _ -> error "XXX commit entry"
+ G.TreeEntry _ -> first addTrailingPathSeparator e
+ )
+ . map (first (("/" <>) . B.toString))
+ <$> G.listTreeEntries tree
data DoesNotExist = DoesNotExist String FilePath
deriving (Show)
@@ -67,32 +96,32 @@ data InappropriateType = InappropriateType String FilePath
instance Exception InappropriateType
class Readable a where
- readFile :: MonadGit r m => CommitOid r -> FilePath -> m a
+ readFile :: FilePath -> StoreM a
instance Readable T.Text where
- readFile = readFile' catBlobUtf8
+ readFile = readFile' G.catBlobUtf8
instance Readable B.ByteString where
- readFile = readFile' catBlob
+ readFile = readFile' G.catBlob
instance Readable LB.ByteString where
- readFile = readFile' catBlobLazy
+ readFile = readFile' G.catBlobLazy
readFile' ::
- MonadGit r m =>
- (BlobOid r -> m a) ->
- CommitOid r ->
+ (G.BlobOid GB.LgRepo -> ReaderT GB.LgRepo IO a) ->
FilePath ->
- m a
-readFile' cat cid fp = do
- tid <- (.commitTree) <$> lookupCommit cid
- tree <- lookupTree tid
- maybe
- (throwM (DoesNotExist "readFile" fp))
- ( \e ->
- case e of
- BlobEntry bid _ -> cat bid
- CommitEntry _ -> error "XXX commit entry"
- TreeEntry _ -> throwM (InappropriateType "readFile" fp)
- )
- =<< treeEntry tree (B.fromString fp)
+ StoreM a
+readFile' cat fp = do
+ cid <- getCommitOid
+ repo <- getRepository
+ lift $ G.runRepository GB.lgFactory repo do
+ tree <- G.lookupTree =<< (.commitTree) <$> G.lookupCommit cid
+ maybe
+ (throwM (DoesNotExist "readFile" fp))
+ ( \e ->
+ case e of
+ G.BlobEntry bid _ -> cat bid
+ G.CommitEntry _ -> error "XXX commit entry"
+ G.TreeEntry _ -> throwM (InappropriateType "readFile" fp)
+ )
+ =<< G.treeEntry tree (B.fromString fp)