1
0
Fork 0
mirror of https://code.sup39.dev/repos/Wqawg synced 2025-01-15 05:25:09 +09:00

Weird initial (but complete) display of Darcs patches

This commit is contained in:
fr33domlover 2019-10-31 10:11:13 +00:00
parent e9f17ff220
commit b95e9a8006
2 changed files with 64 additions and 11 deletions

View file

@ -16,6 +16,7 @@
module Data.List.NonEmpty.Local module Data.List.NonEmpty.Local
( groupWithExtract ( groupWithExtract
, groupWithExtractBy , groupWithExtractBy
, groupWithExtract1
, groupWithExtractBy1 , groupWithExtractBy1
, groupAllExtract , groupAllExtract
, unionGroupsOrdWith , unionGroupsOrdWith
@ -50,6 +51,14 @@ groupWithExtractBy
-> [(b, NonEmpty c)] -> [(b, NonEmpty c)]
groupWithExtractBy eq f g = map (extract f g) . NE.groupBy (eq `on` f) groupWithExtractBy eq f g = map (extract f g) . NE.groupBy (eq `on` f)
groupWithExtract1
:: Eq b
=> (a -> b)
-> (a -> c)
-> NonEmpty a
-> NonEmpty (b, NonEmpty c)
groupWithExtract1 f g = NE.map (extract f g) . NE.groupWith1 f
groupWithExtractBy1 groupWithExtractBy1
:: (b -> b -> Bool) :: (b -> b -> Bool)
-> (a -> b) -> (a -> b)

View file

@ -26,17 +26,17 @@ where
import Prelude hiding (lookup) import Prelude hiding (lookup)
import Control.Applicative ((<|>)) import Control.Applicative ((<|>))
import Control.Arrow (second)
import Control.Monad.IO.Class import Control.Monad.IO.Class
import Control.Monad.Trans.Class (lift) import Control.Monad.Trans.Class (lift)
import Control.Monad.Trans.Except (ExceptT (ExceptT), runExceptT) import Control.Monad.Trans.Except (ExceptT (ExceptT), runExceptT)
import Darcs.Util.Path import Darcs.Util.Path
import Darcs.Util.Tree import Darcs.Util.Tree
import Darcs.Util.Tree.Hashed import Darcs.Util.Tree.Hashed
import Data.Bifunctor
import Data.Bool (bool) import Data.Bool (bool)
import Data.ByteString (ByteString) import Data.ByteString (ByteString)
import Data.Foldable hiding (find) import Data.Foldable hiding (find)
import Data.List.NonEmpty (NonEmpty (..)) import Data.List.NonEmpty (NonEmpty (..), nonEmpty)
import Data.Maybe (listToMaybe, mapMaybe, fromMaybe) import Data.Maybe (listToMaybe, mapMaybe, fromMaybe)
import Data.Text (Text) import Data.Text (Text)
import Data.Text.Encoding (encodeUtf8, decodeUtf8With, decodeUtf8) import Data.Text.Encoding (encodeUtf8, decodeUtf8With, decodeUtf8)
@ -53,11 +53,12 @@ import System.FilePath ((</>))
import Text.Email.Validate (emailAddress) import Text.Email.Validate (emailAddress)
import qualified Data.Attoparsec.Text as A import qualified Data.Attoparsec.Text as A
import qualified Data.ByteString as B
import qualified Data.ByteString.Lazy as BL (ByteString) import qualified Data.ByteString.Lazy as BL (ByteString)
import qualified Data.ByteString.Base16 as B16 (encode, decode) import qualified Data.ByteString.Base16 as B16 (encode, decode)
import qualified Data.Foldable as F (find) import qualified Data.Foldable as F (find)
import qualified Data.List.NonEmpty as N import qualified Data.List.NonEmpty as NE
import qualified Data.Text as T (unpack, takeWhile, stripEnd) import qualified Data.Text as T
import qualified Data.Vector as V (empty) import qualified Data.Vector as V (empty)
import qualified Database.Esqueleto as E import qualified Database.Esqueleto as E
@ -71,6 +72,7 @@ import Darcs.Local.Repository
import Data.Either.Local (maybeRight) import Data.Either.Local (maybeRight)
import Data.EventTime.Local import Data.EventTime.Local
import Data.List.Local import Data.List.Local
import Data.List.NonEmpty.Local
import Data.Text.UTF8.Local (decodeStrict) import Data.Text.UTF8.Local (decodeStrict)
import Data.Time.Clock.Local () import Data.Time.Clock.Local ()
@ -246,11 +248,37 @@ lastChange path now = fmap maybeRight $ runExceptT $ do
FriendlyConvert $ FriendlyConvert $
now `diffUTCTime` piTime pi now `diffUTCTime` piTime pi
data Change
= AddFile FilePath
| AddDir FilePath
| Move FilePath FilePath
| RemoveFile FilePath
| RemoveDir FilePath
| Replace FilePath Text Text Text
| Binary FilePath ByteString ByteString
| Pref Text Text Text
splitChange :: P.Change -> Either P.Hunk Change
splitChange = f
where
text = decodeUtf8
path = T.unpack . text
f (P.EditFile h) = Left h
f (P.AddFile p) = Right $ AddFile (path p)
f (P.AddDir p) = Right $ AddDir (path p)
f (P.Move old new) = Right $ Move (path old) (path new)
f (P.RemoveFile p) = Right $ RemoveFile (path p)
f (P.RemoveDir p) = Right $ RemoveDir (path p)
f (P.Replace p r old new) = Right $ Replace (path p) (text r) (text old) (text new)
f (P.Binary p old new) = Right $ Binary (path p) old new
f (P.Pref pref old new) = Right $ Pref (text pref) (text old) (text new)
-- | Group hunks by filename, assuming all the hunks for a given file are -- | Group hunks by filename, assuming all the hunks for a given file are
-- placed together in the patch file, and in increasing line number order. -- placed together in the patch file, and in increasing line number order.
groupHunksByFile groupHunksByFile
:: [P.Hunk] -> [(ByteString, NonEmpty (Int, [ByteString], [ByteString]))] :: NonEmpty (P.Hunk)
groupHunksByFile = groupMap P.hunkFile rest -> NonEmpty (ByteString, NonEmpty (Int, [ByteString], [ByteString]))
groupHunksByFile = groupWithExtract1 P.hunkFile rest
where where
rest h = (P.hunkLine h, P.hunkRemove h, P.hunkAdd h) rest h = (P.hunkLine h, P.hunkRemove h, P.hunkAdd h)
@ -263,7 +291,7 @@ joinHunks
:: NonEmpty (Int, [ByteString], [ByteString]) :: NonEmpty (Int, [ByteString], [ByteString])
-> NonEmpty (Bool, Int, Hunk) -> NonEmpty (Bool, Int, Hunk)
joinHunks = joinHunks =
N.map (mkHunk . second groupPairs) . NE.map (mkHunk . second groupPairs) .
groupMapBy1 consecutive lineNumber lines groupMapBy1 consecutive lineNumber lines
where where
consecutive (n1, r1, _) (n2, _, _) = n1 + length r1 == n2 consecutive (n1, r1, _) (n2, _, _) = n1 + length r1 == n2
@ -287,7 +315,7 @@ readPatch path hash = handle $ runExceptT $ do
li <- ExceptT $ readLatestInventory path latestInventoryAllP li <- ExceptT $ readLatestInventory path latestInventoryAllP
mp <- loop pih (liPatches li) (fst <$> liPrevTag li) mp <- loop pih (liPatches li) (fst <$> liPrevTag li)
for mp $ \ (pi, pch) -> do for mp $ \ (pi, pch) -> do
(_pir, hunks) <- (_pir, changes) <-
ExceptT $ readCompressedPatch path pch (P.patch <* A.endOfInput) ExceptT $ readCompressedPatch path pch (P.patch <* A.endOfInput)
(an, ae) <- (an, ae) <-
ExceptT . pure $ A.parseOnly (author <* A.endOfInput) $ piAuthor pi ExceptT . pure $ A.parseOnly (author <* A.endOfInput) $ piAuthor pi
@ -303,7 +331,11 @@ readPatch path hash = handle $ runExceptT $ do
, patchTitle = piTitle pi , patchTitle = piTitle pi
, patchDescription = fromMaybe "" $ piDescription pi , patchDescription = fromMaybe "" $ piDescription pi
, patchDiff = , patchDiff =
map (mkedit . second joinHunks) $ groupHunksByFile hunks let (befores, pairs, afters) = groupEithers $ map splitChange changes
befores' = mkedit befores
pairs' = map (bimap arrangeHunks mkedit) pairs
afters' = arrangeHunks <$> nonEmpty afters
in befores' ++ concatMap (NE.toList . uncurry (<>)) pairs' ++ maybe [] NE.toList afters'
} }
where where
handle a = do handle a = do
@ -329,8 +361,20 @@ readPatch path hash = handle $ runExceptT $ do
<* A.skip (== '<') <* A.skip (== '<')
<*> (A.takeWhile1 (/= '>') >>= email) <*> (A.takeWhile1 (/= '>') >>= email)
<* A.skip (== '>') <* A.skip (== '>')
mkedit (file, hunks) = arrangeHunks = NE.map (mkhunk . second joinHunks) . groupHunksByFile
EditTextFile (T.unpack $ decodeUtf8 file) V.empty hunks 0 0 where
mkhunk (file, hunks) =
EditTextFile (T.unpack $ decodeUtf8 file) V.empty hunks 0 0
mkedit = fmap mkedit'
where
mkedit' (AddFile fp) = AddTextFile fp 0 []
mkedit' (AddDir fp) = AddTextFile fp 0 []
mkedit' (Move old new) = MoveFile old 0 new 0
mkedit' (RemoveFile fp) = RemoveTextFile fp 0 []
mkedit' (RemoveDir fp) = RemoveTextFile fp 0 []
mkedit' (Replace fp regex old new) = AddTextFile "Replace" 0 [T.concat ["replace ", T.pack fp, " ", regex, " ", old, " ", new]]
mkedit' (Binary fp old new) = EditBinaryFile fp (fromIntegral $ B.length old) 0 (fromIntegral $ B.length new) 0
mkedit' (Pref pref old new) = AddTextFile "Pref" 0 [T.concat ["changepref ", pref, " ", old, " ", new]]
writePostApplyHooks :: WorkerDB () writePostApplyHooks :: WorkerDB ()
writePostApplyHooks = do writePostApplyHooks = do