From 7b153c3e82d2fce15cbd4d948b71aef7e56a2f04 Mon Sep 17 00:00:00 2001 From: Joey Hess Date: Mon, 21 Jul 2014 18:42:58 -0400 Subject: improve repair of bad branches The repair code assumed that if fsck found no broken objects, after removing bad objects and possibly pulling replacements from remote, all was well.. but this is not really true. Removing bad objects could leave some branches broken. fsck doesn't report any missing objects in this case, and its messages about broken branches are ignored by the fsck output parser. To deal with this, added a separate scan of all refs to find broken ones and remove them when --forced. This will also let anyone who ran into this bug run repair again to fix up the incomplete repair done before. This commit was sponsored by Aaron Whitehouse. --- Git/Repair.hs | 37 ++++++++++++++++++++++++++++++------- 1 file changed, 30 insertions(+), 7 deletions(-) (limited to 'Git') diff --git a/Git/Repair.hs b/Git/Repair.hs index 67ded359f..43f0a56fa 100644 --- a/Git/Repair.hs +++ b/Git/Repair.hs @@ -8,6 +8,7 @@ module Git.Repair ( runRepair, runRepairOf, + removeBadBranches, successfulRepair, cleanCorruptObjects, retrieveMissingObjects, @@ -191,8 +192,11 @@ isTrackingBranch b = "refs/remotes/" `isPrefixOf` fromRef b - any branches (filtered by a predicate) that reference them - Returns a list of all removed branches. -} -removeBadBranches :: (Ref -> Bool) -> MissingObjects -> GoodCommits -> Repo -> IO ([Branch], GoodCommits) -removeBadBranches removablebranch missing goodcommits r = +removeBadBranches :: (Ref -> Bool) -> Repo -> IO [Branch] +removeBadBranches removablebranch r = fst <$> removeBadBranches' removablebranch S.empty emptyGoodCommits r + +removeBadBranches' :: (Ref -> Bool) -> MissingObjects -> GoodCommits -> Repo -> IO ([Branch], GoodCommits) +removeBadBranches' removablebranch missing goodcommits r = go [] goodcommits =<< filter removablebranch <$> getAllRefs r where go removed gcs [] = return (removed, gcs) @@ -204,6 +208,11 @@ removeBadBranches removablebranch missing goodcommits r = nukeBranchRef b r go (b:removed) gcs' bs +badBranches :: MissingObjects -> Repo -> IO [Branch] +badBranches missing r = filterM isbad =<< getAllRefs r + where + isbad b = not . fst <$> verifyCommit missing emptyGoodCommits b r + {- Gets all refs, including ones that are corrupt. - git show-ref does not output refs to commits that are directly - corrupted, so it is not used. @@ -439,8 +448,12 @@ runRepair removablebranch forced g = do if foundBroken fsckresult then runRepair' removablebranch fsckresult forced Nothing g else do - putStrLn "No problems found." - return (True, []) + bad <- badBranches S.empty g + if null bad + then do + putStrLn "No problems found." + return (True, []) + else runRepair' removablebranch fsckresult forced Nothing g runRepairOf :: FsckResults -> (Ref -> Bool) -> Bool -> Maybe FilePath -> Repo -> IO (Bool, [Branch]) runRepairOf fsckresult removablebranch forced referencerepo g = do @@ -455,9 +468,9 @@ runRepair' removablebranch fsckresult forced referencerepo g = do case stillmissing of FsckFoundMissing s t | S.null s -> if repoIsLocalBare g - then successfulfinish [] + then checkbadbranches s else ifM (checkIndex g) - ( successfulfinish [] + ( checkbadbranches s , do putStrLn "No missing objects found, but the index file is corrupt!" if forced @@ -488,7 +501,7 @@ runRepair' removablebranch fsckresult forced referencerepo g = do | otherwise -> unsuccessfulfinish where repairbranches missing = do - (removedbranches, goodcommits) <- removeBadBranches removablebranch missing emptyGoodCommits g + (removedbranches, goodcommits) <- removeBadBranches' removablebranch missing emptyGoodCommits g let remotebranches = filter isTrackingBranch removedbranches unless (null remotebranches) $ putStrLn $ unwords @@ -503,6 +516,16 @@ runRepair' removablebranch fsckresult forced referencerepo g = do "Deleted these local branches, which could not be recovered due to missing objects:" return (resetbranches ++ deletedbranches) + checkbadbranches missing = do + bad <- badBranches missing g + case (null bad, forced) of + (True, _) -> successfulfinish [] + (False, False) -> do + displayList (map fromRef bad) + "Some git branches refer to missing objects:" + unsuccessfulfinish + (False, True) -> successfulfinish =<< repairbranches missing + forcerepair missing fscktruncated = do modifiedbranches <- repairbranches missing deindexedfiles <- rewriteIndex g -- cgit v1.2.3