From 03c781c1a60759622e772ac7fb6a167111ed0bea Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Roman=20Smr=C5=BE?= Date: Sat, 21 Dec 2024 13:10:41 +0100 Subject: Repo module to abstract git access --- src/Repo.hs | 95 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 95 insertions(+) create mode 100644 src/Repo.hs (limited to 'src/Repo.hs') diff --git a/src/Repo.hs b/src/Repo.hs new file mode 100644 index 0000000..9e05ccd --- /dev/null +++ b/src/Repo.hs @@ -0,0 +1,95 @@ +module Repo ( + Repo(..), Commit(..), + CommitId, showCommitId, + TreeId, showTreeId, + + openRepo, + listCommits, + checkoutAt, + readTreeId, +) where + +import Control.Concurrent +import Control.Monad +import Control.Monad.IO.Class + +import Data.ByteString (ByteString) +import Data.ByteString.Char8 qualified as BC +import Data.Text (Text) +import Data.Text qualified as T + +import System.Directory +import System.Exit +import System.FilePath +import System.Process + + +data Repo + = GitRepo + { gitDir :: FilePath + , gitLock :: MVar () + } + +data Commit = Commit + { commitRepo :: Repo + , commitId :: CommitId + , commitDescription :: Text + } + + +newtype CommitId = CommitId ByteString + +showCommitId :: CommitId -> String +showCommitId (CommitId cid) = BC.unpack cid + +newtype TreeId = TreeId ByteString + +showTreeId :: TreeId -> String +showTreeId (TreeId tid) = BC.unpack tid + + +openRepo :: FilePath -> IO (Maybe Repo) +openRepo path = do + findGitDir >>= \case + Just gitDir -> do + gitLock <- newMVar () + return $ Just GitRepo {..} + Nothing -> do + return Nothing + where + tryGitDir gpath = readProcessWithExitCode "git" [ "rev-parse", "--resolve-git-dir", gpath ] "" >>= \case + ( ExitSuccess, out, _ ) | dir : _ <- lines out -> return (Just dir) + _ -> return Nothing + findGitDir = do + tryGitDir path >>= \case + Just dir -> return (Just dir) + Nothing -> tryGitDir (path ".git") >>= \case + Just dir -> return (Just dir) + _ -> return Nothing + + +listCommits :: MonadIO m => Repo -> String -> m [ Commit ] +listCommits commitRepo range = liftIO $ do + out <- readProcess "git" [ "log", "--pretty=oneline", "--first-parent", "--reverse", range ] "" + forM (lines out) $ \line -> do + let ( cid, desc ) = fmap (drop 1) $ (span (/=' ')) line + commitId = CommitId (BC.pack cid) + commitDescription = T.pack desc + return Commit {..} + + +checkoutAt :: (MonadIO m, MonadFail m) => Commit -> FilePath -> m () +checkoutAt Commit {..} dest = do + let GitRepo {..} = commitRepo + liftIO $ withMVar gitLock $ \_ -> do + "" <- readProcess "git" [ "clone", "--quiet", "--shared", "--no-checkout", gitDir, dest ] "" + "" <- readProcess "git" [ "-C", dest, "restore", "--worktree", "--source=" <> showCommitId commitId, "--", "." ] "" + removeDirectoryRecursive $ dest ".git" + + +readTreeId :: (MonadIO m, MonadFail m) => Commit -> m TreeId +readTreeId Commit {..} = do + let GitRepo {..} = commitRepo + liftIO $ withMVar gitLock $ \_ -> do + [ "tree", tid ] : _ <- map words . lines <$> readProcess "git" [ "--git-dir=" <> gitDir, "cat-file", "commit", showCommitId commitId ] "" + return $ TreeId $ BC.pack tid -- cgit v1.2.3