2011-12-13 01:12:51 +00:00
|
|
|
{- git branch stuff
|
|
|
|
-
|
|
|
|
- Copyright 2011 Joey Hess <joey@kitenet.net>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
|
|
|
module Git.Branch where
|
|
|
|
|
2012-02-01 20:05:02 +00:00
|
|
|
import qualified Data.Text.Lazy as L
|
2011-12-13 01:12:51 +00:00
|
|
|
|
|
|
|
import Common
|
|
|
|
import Git
|
2011-12-14 19:30:14 +00:00
|
|
|
import Git.Sha
|
2011-12-14 19:56:11 +00:00
|
|
|
import Git.Command
|
2011-12-13 01:12:51 +00:00
|
|
|
|
2011-12-31 07:38:58 +00:00
|
|
|
{- The currently checked out branch. -}
|
|
|
|
current :: Repo -> IO (Maybe Git.Ref)
|
|
|
|
current r = parse <$> pipeRead [Param "symbolic-ref", Param "HEAD"] r
|
|
|
|
where
|
|
|
|
parse v
|
|
|
|
| L.null v = Nothing
|
|
|
|
| otherwise = Just $ Git.Ref $ firstLine $ L.unpack v
|
|
|
|
|
2011-12-13 01:12:51 +00:00
|
|
|
{- Checks if the second branch has any commits not present on the first
|
|
|
|
- branch. -}
|
|
|
|
changed :: Branch -> Branch -> Repo -> IO Bool
|
|
|
|
changed origbranch newbranch repo
|
|
|
|
| origbranch == newbranch = return False
|
|
|
|
| otherwise = not . L.null <$> diffs
|
|
|
|
where
|
2011-12-14 19:30:14 +00:00
|
|
|
diffs = pipeRead
|
2011-12-13 01:12:51 +00:00
|
|
|
[ Param "log"
|
|
|
|
, Param (show origbranch ++ ".." ++ show newbranch)
|
|
|
|
, Params "--oneline -n1"
|
|
|
|
] repo
|
|
|
|
|
|
|
|
{- Given a set of refs that are all known to have commits not
|
|
|
|
- on the branch, tries to update the branch by a fast-forward.
|
|
|
|
-
|
|
|
|
- In order for that to be possible, one of the refs must contain
|
|
|
|
- every commit present in all the other refs.
|
|
|
|
-}
|
|
|
|
fastForward :: Branch -> [Ref] -> Repo -> IO Bool
|
|
|
|
fastForward _ [] _ = return True
|
|
|
|
fastForward branch (first:rest) repo = do
|
|
|
|
-- First, check that the branch does not contain any
|
|
|
|
-- new commits that are not in the first ref. If it does,
|
|
|
|
-- cannot fast-forward.
|
|
|
|
diverged <- changed first branch repo
|
|
|
|
if diverged
|
|
|
|
then no_ff
|
|
|
|
else maybe no_ff do_ff =<< findbest first rest
|
|
|
|
where
|
|
|
|
no_ff = return False
|
|
|
|
do_ff to = do
|
2011-12-14 19:30:14 +00:00
|
|
|
run "update-ref"
|
2011-12-13 01:12:51 +00:00
|
|
|
[Param $ show branch, Param $ show to] repo
|
|
|
|
return True
|
|
|
|
findbest c [] = return $ Just c
|
|
|
|
findbest c (r:rs)
|
|
|
|
| c == r = findbest c rs
|
|
|
|
| otherwise = do
|
|
|
|
better <- changed c r repo
|
|
|
|
worse <- changed r c repo
|
|
|
|
case (better, worse) of
|
|
|
|
(True, True) -> return Nothing -- divergent fail
|
|
|
|
(True, False) -> findbest r rs -- better
|
|
|
|
(False, True) -> findbest c rs -- worse
|
|
|
|
(False, False) -> findbest c rs -- same
|
2011-12-13 19:08:44 +00:00
|
|
|
|
|
|
|
{- Commits the index into the specified branch (or other ref),
|
|
|
|
- with the specified parent refs, and returns the committed sha -}
|
|
|
|
commit :: String -> Branch -> [Ref] -> Repo -> IO Sha
|
|
|
|
commit message branch parentrefs repo = do
|
|
|
|
tree <- getSha "write-tree" $ asString $
|
|
|
|
pipeRead [Param "write-tree"] repo
|
|
|
|
sha <- getSha "commit-tree" $ asString $
|
|
|
|
ignorehandle $ pipeWriteRead
|
|
|
|
(map Param $ ["commit-tree", show tree] ++ ps)
|
|
|
|
(L.pack message) repo
|
|
|
|
run "update-ref" [Param $ show branch, Param $ show sha] repo
|
|
|
|
return sha
|
|
|
|
where
|
|
|
|
ignorehandle a = snd <$> a
|
|
|
|
asString a = L.unpack <$> a
|
|
|
|
ps = concatMap (\r -> ["-p", show r]) parentrefs
|