summaryrefslogtreecommitdiff
path: root/Git/Repair.hs
diff options
context:
space:
mode:
Diffstat (limited to 'Git/Repair.hs')
-rw-r--r--Git/Repair.hs37
1 files changed, 30 insertions, 7 deletions
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