Fix bug in initialization of clone from a repo with an adjusted branch that had not been synced back to master.

This bug caused broken tree objects to get built by a later git annex sync.

This is a somewhat unlikely but not impossible situation, and the test
suite's union_merge_regression test tickled it when it was run on FAT.
This commit is contained in:
Joey Hess 2016-06-09 14:11:00 -04:00
parent 8e4cbefbc6
commit 0249f3aff5
Failed to extract signature
2 changed files with 37 additions and 9 deletions

View file

@ -300,9 +300,20 @@ commitAdjustedTree' treesha (BasisBranch basis) parents =
mkcommit = Git.Branch.commitTree Git.Branch.AutomaticCommit mkcommit = Git.Branch.commitTree Git.Branch.AutomaticCommit
adjustedBranchCommitMessage parents treesha adjustedBranchCommitMessage parents treesha
{- This message should never be changed. -}
adjustedBranchCommitMessage :: String adjustedBranchCommitMessage :: String
adjustedBranchCommitMessage = "git-annex adjusted branch" adjustedBranchCommitMessage = "git-annex adjusted branch"
findAdjustingCommit :: AdjBranch -> Annex (Maybe Commit)
findAdjustingCommit (AdjBranch b) = go =<< catCommit b
where
go Nothing = return Nothing
go (Just c)
| commitMessage c == adjustedBranchCommitMessage = return (Just c)
| otherwise = case commitParent c of
[p] -> go =<< catCommit p
_ -> return Nothing
{- Update the currently checked out adjusted branch, merging the provided {- Update the currently checked out adjusted branch, merging the provided
- branch into it. Note that the provided branch should be a non-adjusted - branch into it. Note that the provided branch should be a non-adjusted
- branch. -} - branch. -}
@ -545,12 +556,22 @@ data AdjustedClone = InAdjustedClone | NotInAdjustedClone | NeedUpgradeForAdjust
{- Cloning a repository that has an adjusted branch checked out will {- Cloning a repository that has an adjusted branch checked out will
- result in the clone having the same adjusted branch checked out -- but - result in the clone having the same adjusted branch checked out -- but
- the origbranch won't exist in the clone, nor will the basis. - the origbranch won't exist in the clone, nor will the basis. So
- Create them. - to properly set up the adjusted branch, the origbranch and basis need
- to be set.
-
- We can't trust that the origin's origbranch matches up with the currently
- checked out adjusted branch; the origin could have the two branches
- out of sync (eg, due to another branch having been pushed to the origin's
- origbranch), or due to a commit on its adjusted branch not having been
- propigated back to origbranch.
-
- So, find the adjusting commit on the currently checked out adjusted
- branch, and use the parent of that commit as the basis, and set the
- origbranch to it.
- -
- The repository may also need to be upgraded to a new version, if the - The repository may also need to be upgraded to a new version, if the
- current version is too old to support adjusted branches. Returns True - current version is too old to support adjusted branches. -}
- when this is the case. -}
checkAdjustedClone :: Annex AdjustedClone checkAdjustedClone :: Annex AdjustedClone
checkAdjustedClone = ifM isBareRepo checkAdjustedClone = ifM isBareRepo
( return NotInAdjustedClone ( return NotInAdjustedClone
@ -561,12 +582,15 @@ checkAdjustedClone = ifM isBareRepo
go (Just currbranch) = case adjustedToOriginal currbranch of go (Just currbranch) = case adjustedToOriginal currbranch of
Nothing -> return NotInAdjustedClone Nothing -> return NotInAdjustedClone
Just (adj, origbranch) -> do Just (adj, origbranch) -> do
let remotebranch = Git.Ref.underBase "refs/remotes/origin" origbranch
let basis@(BasisBranch bb) = basisBranch (originalToAdjusted origbranch adj) let basis@(BasisBranch bb) = basisBranch (originalToAdjusted origbranch adj)
unlessM (inRepo $ Git.Ref.exists bb) $ unlessM (inRepo $ Git.Ref.exists bb) $ do
setBasisBranch basis remotebranch unlessM (inRepo $ Git.Ref.exists origbranch) $ do
unlessM (inRepo $ Git.Ref.exists origbranch) $ let remotebranch = Git.Ref.underBase "refs/remotes/origin" origbranch
inRepo $ Git.Branch.update' origbranch remotebranch inRepo $ Git.Branch.update' origbranch remotebranch
aps <- fmap commitParent <$> findAdjustingCommit (AdjBranch currbranch)
case aps of
Just [p] -> setBasisBranch basis p
_ -> error $ "Unable to clean up from clone of adjusted branch; perhaps you should check out " ++ Git.Ref.describe origbranch
ifM versionSupportsUnlockedPointers ifM versionSupportsUnlockedPointers
( return InAdjustedClone ( return InAdjustedClone
, return NeedUpgradeForAdjustedClone , return NeedUpgradeForAdjustedClone

View file

@ -24,6 +24,10 @@ git-annex (6.20160528) UNRELEASED; urgency=medium
and a directory with the same name when in an adjusted branch. and a directory with the same name when in an adjusted branch.
* Avoid a crash if getpwuid does not work, when querying the user's full * Avoid a crash if getpwuid does not work, when querying the user's full
name. name.
* Fix bug in initialization of clone from a repo with an adjusted branch
that had not been synced back to master.
(This bug caused broken tree objects to get built by a later git annex
sync.)
-- Joey Hess <id@joeyh.name> Fri, 27 May 2016 13:12:48 -0400 -- Joey Hess <id@joeyh.name> Fri, 27 May 2016 13:12:48 -0400