new method for merging changes into adjusted branch that avoids unncessary merge conflicts

Still needs work when there are actual merge conflicts.
This commit is contained in:
Joey Hess 2016-04-06 15:33:29 -04:00
parent eb9ac8d6d7
commit b9e4e2ba84
Failed to extract signature
8 changed files with 201 additions and 240 deletions

View file

@ -5,6 +5,8 @@
- Licensed under the GNU GPL version 3 or higher. - Licensed under the GNU GPL version 3 or higher.
-} -}
{-# LANGUAGE BangPatterns #-}
module Annex.AdjustedBranch ( module Annex.AdjustedBranch (
Adjustment(..), Adjustment(..),
OrigBranch, OrigBranch,
@ -40,6 +42,9 @@ import Annex.CatFile
import Annex.Link import Annex.Link
import Annex.AutoMerge import Annex.AutoMerge
import Annex.Content import Annex.Content
import Annex.Perms
import Annex.GitOverlay
import Utility.Tmp
import qualified Database.Keys import qualified Database.Keys
import qualified Data.Map as M import qualified Data.Map as M
@ -137,7 +142,7 @@ originalBranch = fmap fromAdjustedBranch <$> inRepo Git.Branch.current
{- Enter an adjusted version of current branch (or, if already in an {- Enter an adjusted version of current branch (or, if already in an
- adjusted version of a branch, changes the adjustment of the original - adjusted version of a branch, changes the adjustment of the original
- branch). t a- branch).
- -
- Can fail, if no branch is checked out, or perhaps if staged changes - Can fail, if no branch is checked out, or perhaps if staged changes
- conflict with the adjusted branch. - conflict with the adjusted branch.
@ -225,80 +230,91 @@ adjustedBranchCommitMessage :: String
adjustedBranchCommitMessage = "git-annex adjusted branch" adjustedBranchCommitMessage = "git-annex adjusted branch"
{- Update the currently checked out adjusted branch, merging the provided {- Update the currently checked out adjusted branch, merging the provided
- branch into it. -} - branch into it. Note that the provided branch should be a non-adjusted
- branch. -}
updateAdjustedBranch :: Branch -> (OrigBranch, Adjustment) -> Git.Branch.CommitMode -> Annex Bool updateAdjustedBranch :: Branch -> (OrigBranch, Adjustment) -> Git.Branch.CommitMode -> Annex Bool
updateAdjustedBranch tomerge (origbranch, adj) commitmode = catchBoolIO $ updateAdjustedBranch tomerge (origbranch, adj) commitmode = catchBoolIO $
join $ preventCommits $ \commitsprevented -> go commitsprevented =<< (,) join $ preventCommits $ \commitsprevented ->
<$> inRepo (Git.Ref.sha tomerge) go commitsprevented =<< inRepo Git.Branch.current
<*> inRepo Git.Branch.current
where where
go commitsprevented (Just mergesha, Just currbranch) = go commitsprevented (Just currbranch) =
ifM (inRepo $ Git.Branch.changed currbranch mergesha) ifM (inRepo $ Git.Branch.changed currbranch tomerge)
( do ( do
void $ propigateAdjustedCommits' origbranch (adj, currbranch) commitsprevented (updatedorig, _) <- propigateAdjustedCommits' origbranch (adj, currbranch) commitsprevented
adjustedtomerge <- adjust adj mergesha changestomerge updatedorig currbranch
ifM (inRepo $ Git.Branch.changed currbranch adjustedtomerge)
( return $ do
-- Run after commit lock is dropped.
liftIO $ print ("autoMergeFrom", adjustedtomerge, (Just currbranch))
ifM (autoMergeFrom adjustedtomerge (Just currbranch) commitmode)
( preventCommits $ \_ ->
recommit currbranch mergesha =<< catCommit currbranch
, return False
)
, nochangestomerge
)
, nochangestomerge , nochangestomerge
) )
go _ _ = return $ return False go _ _ = return $ return False
nochangestomerge = return $ return True nochangestomerge = return $ return True
{- A merge commit has been made on the adjusted branch. {- Since the adjusted branch changes files, merging tomerge
- Now, re-do it, removing the old version of the adjusted branch - directly into it would likely result in unncessary merge
- from its history. - conflicts. To avoid those conflicts, instead merge tomerge into
- updatedorig. The result of the merge can the be
- adjusted to yield the final adjusted branch.
- -
- There are two possible scenarios; either some commits - In order to do a merge into a branch that is not checked out,
- were made on top of the adjusted branch's adjusting commit, - set the work tree to a temp directory, and set GIT_DIR
- or not. Those commits have already been propigated to the - to another temp directory, in which HEAD contains the
- orig branch, so we can just check if there are commits in the - updatedorig sha. GIT_COMMON_DIR is set to point to the real
- orig branch that are not present in tomerge. - git directory, and so git can read and write objects from there,
- but will use GIT_DIR for HEAD and index.
-
- (Doing the merge this way also lets it run even though the main
- index file is currently locked.)
-} -}
recommit currbranch mergedsha (Just mergecommit) = changestomerge (Just updatedorig) currbranch = do
ifM (inRepo $ Git.Branch.changed tomerge origbranch) misctmpdir <- fromRepo gitAnnexTmpMiscDir
( remerge currbranch mergedsha mergecommit void $ createAnnexDirectory misctmpdir
=<< inRepo (Git.Ref.sha origbranch) tmpwt <- fromRepo gitAnnexMergeDir
, fastforward currbranch mergedsha mergecommit withTmpDirIn misctmpdir "git" $ \tmpgit -> withWorkTreeRelated tmpgit $
withemptydir tmpwt $ withWorkTree tmpwt $ do
liftIO $ writeFile (tmpgit </> "HEAD") (fromRef updatedorig)
showAction $ "Merging into " ++ fromRef (Git.Ref.base origbranch)
ifM (autoMergeFrom tomerge (Just updatedorig) commitmode)
( do
!mergecommit <- liftIO $ extractSha <$> readFile (tmpgit </> "HEAD")
-- This is run after the commit lock is dropped.
return $ postmerge currbranch mergecommit
, return $ return False
)
changestomerge Nothing _ = return $ return False
withemptydir d a = bracketIO setup cleanup (const a)
where
setup = do
whenM (doesDirectoryExist d) $
removeDirectoryRecursive d
createDirectoryIfMissing True d
cleanup _ = removeDirectoryRecursive d
{- A merge commit has been made between the origbranch and
- tomerge. Update origbranch to point to that commit, adjust
- it to get the new adjusted branch, and check it out.
-
- But, there may be unstaged work tree changes that conflict,
- so the check out is done by making a normal merge of
- the new adjusted branch.
-}
postmerge currbranch (Just mergecommit) = do
inRepo $ Git.Branch.update "updating original branch" origbranch mergecommit
adjtree <- adjustTree adj mergecommit
-- Make currbranch be a parent, so that merging
-- this commit will be a fast-forward.
adjmergecommit <- commitAdjustedTree' adjtree mergecommit
[mergecommit, currbranch]
showAction "Merging into adjusted branch"
ifM (autoMergeFrom adjmergecommit (Just currbranch) commitmode)
-- The adjusted branch has a merge commit on top;
-- clean that up and propigate any changes made
-- in that merge to the origbranch.
( do
propigateAdjustedCommits origbranch (adj, currbranch)
return True
, return False
) )
recommit _ _ Nothing = return False postmerge _ Nothing = return False
{- Fast-forward scenario. The mergecommit is changed to a non-merge
- commit, with its parent being the mergedsha.
- The orig branch can simply be pointed at the mergedsha.
-}
fastforward currbranch mergedsha mergecommit = do
commitsha <- commitAdjustedTree (commitTree mergecommit) mergedsha
inRepo $ Git.Branch.update "fast-forward update of adjusted branch" currbranch commitsha
inRepo $ Git.Branch.update "updating original branch" origbranch mergedsha
return True
{- True merge scenario. -}
remerge currbranch mergedsha mergecommit (Just origsha) = do
-- Update origbranch by reverse adjusting the mergecommit,
-- yielding a merge between orig and tomerge.
treesha <- reverseAdjustedTree origsha adj
-- get 1-parent commit because
-- reverseAdjustedTree does not support merges
=<< commitAdjustedTree (commitTree mergecommit) origsha
revadjcommit <- inRepo $
Git.Branch.commitTree Git.Branch.AutomaticCommit
("Merge branch " ++ fromRef tomerge) [origsha, mergedsha] treesha
inRepo $ Git.Branch.update "updating original branch" origbranch revadjcommit
-- Update currbranch, reusing mergedsha, but making its
-- parent be the updated origbranch.
adjcommit <- commitAdjustedTree' (commitTree mergecommit) revadjcommit [revadjcommit]
inRepo $ Git.Branch.update rebaseOnTopMsg currbranch adjcommit
return True
remerge _ _ _ Nothing = return False
{- Check for any commits present on the adjusted branch that have not yet {- Check for any commits present on the adjusted branch that have not yet
- been propigated to the orig branch, and propigate them. - been propigated to the orig branch, and propigate them.
@ -308,16 +324,16 @@ updateAdjustedBranch tomerge (origbranch, adj) commitmode = catchBoolIO $
-} -}
propigateAdjustedCommits :: OrigBranch -> (Adjustment, AdjBranch) -> Annex () propigateAdjustedCommits :: OrigBranch -> (Adjustment, AdjBranch) -> Annex ()
propigateAdjustedCommits origbranch (adj, currbranch) = propigateAdjustedCommits origbranch (adj, currbranch) =
preventCommits $ \commitsprevented -> do preventCommits $ \commitsprevented ->
join $ propigateAdjustedCommits' origbranch (adj, currbranch) commitsprevented join $ snd <$> propigateAdjustedCommits' origbranch (adj, currbranch) commitsprevented
{- Returns action which will rebase the adjusted branch on top of the {- Returns sha of updated orig branch, and action which will rebase
- updated orig branch. -} - the adjusted branch on top of the updated orig branch. -}
propigateAdjustedCommits' propigateAdjustedCommits'
:: OrigBranch :: OrigBranch
-> (Adjustment, AdjBranch) -> (Adjustment, AdjBranch)
-> CommitsPrevented -> CommitsPrevented
-> Annex (Annex ()) -> Annex (Maybe Sha, Annex ())
propigateAdjustedCommits' origbranch (adj, currbranch) _commitsprevented = do propigateAdjustedCommits' origbranch (adj, currbranch) _commitsprevented = do
ov <- inRepo $ Git.Ref.sha (Git.Ref.under "refs/heads" origbranch) ov <- inRepo $ Git.Ref.sha (Git.Ref.under "refs/heads" origbranch)
case ov of case ov of
@ -329,11 +345,13 @@ propigateAdjustedCommits' origbranch (adj, currbranch) _commitsprevented = do
case v of case v of
Left e -> do Left e -> do
warning e warning e
return $ return () return (Nothing, return ())
Right newparent -> return $ Right newparent -> return
rebase currcommit newparent ( Just newparent
Nothing -> return $ return () , rebase currcommit newparent
Nothing -> return $ return () )
Nothing -> return (Nothing, return ())
Nothing -> return (Nothing, return ())
where where
newcommits = inRepo $ Git.Branch.changedCommits origbranch currbranch newcommits = inRepo $ Git.Branch.changedCommits origbranch currbranch
-- Get commits oldest first, so they can be processed -- Get commits oldest first, so they can be processed

View file

@ -35,7 +35,7 @@ import Control.Concurrent (threadDelay)
import Annex.Common import Annex.Common
import Annex.BranchState import Annex.BranchState
import Annex.Journal import Annex.Journal
import Annex.Index import Annex.GitOverlay
import qualified Git import qualified Git
import qualified Git.Command import qualified Git.Command
import qualified Git.Ref import qualified Git.Ref

View file

@ -37,7 +37,7 @@ import Annex.Perms
import Annex.ReplaceFile import Annex.ReplaceFile
import Annex.VariantFile import Annex.VariantFile
import Git.Index import Git.Index
import Annex.Index import Annex.GitOverlay
import Annex.LockFile import Annex.LockFile
import Annex.InodeSentinal import Annex.InodeSentinal

63
Annex/GitOverlay.hs Normal file
View file

@ -0,0 +1,63 @@
{- Temporarily changing the files git uses.
-
- Copyright 2014-2016 Joey Hess <id@joeyh.name>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Annex.GitOverlay where
import qualified Control.Exception as E
import Annex.Common
import Git
import Git.Types
import Git.Env
import qualified Annex
{- Runs an action using a different git index file. -}
withIndexFile :: FilePath -> Annex a -> Annex a
withIndexFile f = withAltRepo
(\g -> addGitEnv g "GIT_INDEX_FILE" f)
(\g g' -> g' { gitEnv = gitEnv g })
{- Runs an action using a different git work tree. -}
withWorkTree :: FilePath -> Annex a -> Annex a
withWorkTree d = withAltRepo
(\g -> return $ g { location = modlocation (location g) })
(\g g' -> g' { location = location g })
where
modlocation l@(Local {}) = l { worktree = Just d }
modlocation _ = error "withWorkTree of non-local git repo"
{- Runs an action with the git index file and HEAD, and a few other
- files that are related to the work tree coming from an overlay
- directory other than the usual. This is done by pointing
- GIT_COMMON_DIR at the regular git directory, and GIT_DIR at the
- overlay directory. -}
withWorkTreeRelated :: FilePath -> Annex a -> Annex a
withWorkTreeRelated d = withAltRepo modrepo unmodrepo
where
modrepo g = do
let g' = g { location = modlocation (location g) }
addGitEnv g' "GIT_COMMON_DIR" =<< absPath (localGitDir g)
unmodrepo g g' = g' { gitEnv = gitEnv g, location = location g }
modlocation l@(Local {}) = l { gitdir = d }
modlocation _ = error "withWorkTreeRelated of non-local git repo"
withAltRepo
:: (Repo -> IO Repo)
-- ^ modify Repo
-> (Repo -> Repo -> Repo)
-- ^ undo modifications; first Repo is the original and second
-- is the one after running the action.
-> Annex a
-> Annex a
withAltRepo modrepo unmodrepo a = do
g <- gitRepo
g' <- liftIO $ modrepo g
r <- tryNonAsync $ do
Annex.changeState $ \s -> s { Annex.repo = g' }
a
Annex.changeState $ \s -> s { Annex.repo = unmodrepo g (Annex.repo s) }
either E.throw return r

View file

@ -1,27 +0,0 @@
{- Using other git index files
-
- Copyright 2014 Joey Hess <id@joeyh.name>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Annex.Index (withIndexFile) where
import qualified Control.Exception as E
import Annex.Common
import Git.Types
import Git.Env
import qualified Annex
{- Runs an action using a different git index file. -}
withIndexFile :: FilePath -> Annex a -> Annex a
withIndexFile f a = do
g <- gitRepo
g' <- liftIO $ addGitEnv g "GIT_INDEX_FILE" f
r <- tryNonAsync $ do
Annex.changeState $ \s -> s { Annex.repo = g' }
a
Annex.changeState $ \s -> s { Annex.repo = (Annex.repo s) { gitEnv = gitEnv g} }
either E.throw return r

View file

@ -291,7 +291,8 @@ gitAnnexFeedStateDir r = addTrailingPathSeparator $ gitAnnexDir r </> "feedstate
gitAnnexFeedState :: Key -> Git.Repo -> FilePath gitAnnexFeedState :: Key -> Git.Repo -> FilePath
gitAnnexFeedState k r = gitAnnexFeedStateDir r </> keyFile k gitAnnexFeedState k r = gitAnnexFeedStateDir r </> keyFile k
{- .git/annex/merge/ is used for direct mode merges. -} {- .git/annex/merge/ is used as a empty work tree for direct mode merges and
- merges in adjusted branches. -}
gitAnnexMergeDir :: Git.Repo -> FilePath gitAnnexMergeDir :: Git.Repo -> FilePath
gitAnnexMergeDir r = addTrailingPathSeparator $ gitAnnexDir r </> "merge" gitAnnexMergeDir r = addTrailingPathSeparator $ gitAnnexDir r </> "merge"

View file

@ -23,7 +23,7 @@ import Annex.HashObject
import Git.Types import Git.Types
import Git.FilePath import Git.FilePath
import Annex.WorkTree import Annex.WorkTree
import Annex.Index import Annex.GitOverlay
import Annex.Link import Annex.Link
import Annex.CatFile import Annex.CatFile
import Logs.MetaData import Logs.MetaData

View file

@ -104,67 +104,54 @@ that are merged in, for object add/remove to work as described below.
When merging, there should never be any commits present on the When merging, there should never be any commits present on the
adjusted/master branch that have not yet been propigated back to the master adjusted/master branch that have not yet been propigated back to the master
branch. If there are any such commits, just propigate them into master before branch. If there are any such commits, just propigate them into master
beginning the merge. There may be staged changes, or changes in the work tree. before beginning the merge. There may be staged changes, or changes in the
work tree.
First adjust the new commit: First, merge origin/master into master. This is done in a temp work
tree and with a temp index, so does not affect the checked out adjusted
branch.
origin/master adjusted/master master (Note that the reason this is done, rather than adjusting origin/master
A A and merging it into the work tree, is that merge conflicts would be very
|--------------->A' | common with the naive approach, because the adjusted branch often changes
| | | files, and origin/master may change the same files.)
| |
B
|
|---------->B'
Then, merge that into adjusted/master: origin/master master adjusted/master
A------------->A- - - ->A'
| |
B------------->C
origin/master adjusted/master master While a fast-forward merge is shown here, other merges work the same way.
A A There may be merge conflicts; if so they're auto-resolved.
|--------------->A' |
| | |
| |
B |
| |
|----------->B'->B''
That merge will take care of updating the work tree. Then, adjust merge commit C, and merge that into adjusted/master.
(What if there is a merge conflict between A' and B'? Normally such a merge origin/master master adjusted/master
conflict should only affect the work tree/index, so can be resolved without A------------->A- - - ->A'
making a commit, but B'' may end up being made to resolve a merge | | |
conflict.) B------------->C- - C'->D'
Once the merge is done, we have a merge commit B'' on adjusted/master. This merge is done in-worktree, so the work tree gets updated.
To finish, redo that commit so it does not have A' as its parent. There may be more merge conflicts here; they're also auto-resolved.
origin/master adjusted/master master Now, D' is a merge commit, between A' and C'.
A A To finish, change that commit so it does not have A' as its parent.
|--------------->A' |
| | |
| | This can be accomplished by propigating the reverse-adjusted D'
B back to master, and then adjusting master to yield the final
| adjusted/master.
|--------------->B''
| |
Finally, update master, by reverse adjusting B''. origin/master master adjusted/master
A------------->A
| |
B------------->C
|
D - - -> D'
origin/master adjusted/master master Notice how similar this is to the commit graph. Indeed, "fast-forward"
A A
|--------------->A' |
| | |
| | |
B |
| |
|--------------->B'' - - - - - - -> B
| |
Notice how similar this is to the commit graph. So, "fast-forward"
merging the same B commit from origin/master will lead to an identical merging the same B commit from origin/master will lead to an identical
sha for B' as the original committer got. sha for B' as the original committer got!
Since the adjusted/master branch is not present on the remote, if the user Since the adjusted/master branch is not present on the remote, if the user
does a `git pull`, it won't merge in changes from origin/master. Which is does a `git pull`, it won't merge in changes from origin/master. Which is
@ -180,91 +167,6 @@ between the adjusted work tree and pulled changes. A post-merge hook would
be needed to re-adjust the work tree, and there would be a window where eg, be needed to re-adjust the work tree, and there would be a window where eg,
not present files would appear in the work tree.] not present files would appear in the work tree.]
## another merge scenario
Another merge scenario is when there's a new commit C on adjusted/master,
and also a new commit B on origin/master.
Start by adjusting B':
origin/master adjusted/master master
A A
|--------------->A' |
| | |
| C'
B
|
|---------->B'
Then, merge B' into adjusted/master:
origin/master adjusted/master master
A A
|--------------->A' |
| | |
| C'
B |
| |
|----------->B'->M'
Here M' is the correct tree, but it has A' as its grandparent,
which is the adjusted branch commit, so needs to be dropped in order to
get a commit that can be put on master.
We don't want to lose commit C', but it's an adjusted
commit, so needs to be de-adjusted.
origin/master adjusted/master master
A A
|--------------->A' |
| | |
| C'- - - - - - - - > C
B |
| |
|----------->B'->M'
|
Now, we generate a merge commit, between B and C, with known result M'
(so no actual merging done here).
origin/master adjusted/master master
A A
|--------------->A' |
| | |
| C'- - - - - - - - > C
B |
| |
|--------------->M'<-----------------|
|
Finally, update master, by reverse adjusting M'. The resulting commit
on master will also be a merge between B and C.
### avoiding conflicted merge
When merging origin/master with adjusted/master, origin/master is
adjusted first, and then merged into the checked out adjusted/master
branch.
This can lead to merge conflicts, when files in origin/master have
been renamed or modified.
This is because adjusted/master and origin/master (and also its adjusted
form) will both modify a file; the former by eg, unlocking it and
the latter by eg, deleting it.
This may need an out of work-tree merge to resolve. In an empty temp work
tree, merge the de-adjusted form of adjusted/master and origin/master. If
that has (real) merge conflicts, auto-resolve them.
The resulting merge commit can then be adjusted to yield the adjusted
merge commit. The parents of the adjusted merge commit also need to be
adjusted, to be the same as if adjusted(origin/master) was merged into
adjusted/master.
Finally, check out the adjusted merge commit, to update the real working
tree.
## annex object add/remove ## annex object add/remove
When objects are added/removed from the annex, the associated file has to When objects are added/removed from the annex, the associated file has to
@ -377,13 +279,16 @@ into adjusted view worktrees.]
will make copies of the content of annexed files, so this would need will make copies of the content of annexed files, so this would need
to checkout the adjusted branch some other way. Maybe generalize so this to checkout the adjusted branch some other way. Maybe generalize so this
more efficient checkout is available as a git-annex command? more efficient checkout is available as a git-annex command?
* sync in adjusted branch runs merge in overlay worktree,
but the merge conflict resolution code does not know to use that
worktree.
* sync in adjusted branch can trigger merge conflict detection where * sync in adjusted branch can trigger merge conflict detection where
there should be no conflict. there should be no conflict.
git init a git init a
cd a cd a
git annex init --version=6 git annex init --version=6
touch f echo hi > f
git annex add f git annex add f
git annex sync git annex sync
cd .. cd ..
@ -391,6 +296,7 @@ into adjusted view worktrees.]
git clone a b git clone a b
cd b cd b
git annex init --version=6 git annex init --version=6
git annex get
git annex adjust --unlock git annex adjust --unlock
cd .. cd ..