40ecf58d4b
This does not change the overall license of the git-annex program, which was already AGPL due to a number of sources files being AGPL already. Legally speaking, I'm adding a new license under which these files are now available; I already released their current contents under the GPL license. Now they're dual licensed GPL and AGPL. However, I intend for all my future changes to these files to only be released under the AGPL license, and I won't be tracking the dual licensing status, so I'm simply changing the license statement to say it's AGPL. (In some cases, others wrote parts of the code of a file and released it under the GPL; but in all cases I have contributed a significant portion of the code in each file and it's that code that is getting the AGPL license; the GPL license of other contributors allows combining with AGPL code.)
161 lines
4.8 KiB
Haskell
161 lines
4.8 KiB
Haskell
{- git fsck interface
|
|
-
|
|
- Copyright 2013 Joey Hess <id@joeyh.name>
|
|
-
|
|
- Licensed under the GNU AGPL version 3 or higher.
|
|
-}
|
|
|
|
{-# LANGUAGE BangPatterns, CPP #-}
|
|
|
|
module Git.Fsck (
|
|
FsckResults(..),
|
|
MissingObjects,
|
|
findBroken,
|
|
foundBroken,
|
|
findMissing,
|
|
isMissing,
|
|
knownMissing,
|
|
) where
|
|
|
|
import Common
|
|
import Git
|
|
import Git.Command
|
|
import Git.Sha
|
|
import Utility.Batch
|
|
import qualified Git.Version
|
|
|
|
import qualified Data.Set as S
|
|
import Control.Concurrent.Async
|
|
import qualified Data.Semigroup as Sem
|
|
import Prelude
|
|
|
|
data FsckResults
|
|
= FsckFoundMissing
|
|
{ missingObjects :: MissingObjects
|
|
, missingObjectsTruncated :: Bool
|
|
}
|
|
| FsckFailed
|
|
deriving (Show)
|
|
|
|
data FsckOutput
|
|
= FsckOutput MissingObjects Truncated
|
|
| NoFsckOutput
|
|
| AllDuplicateEntriesWarning
|
|
|
|
type MissingObjects = S.Set Sha
|
|
|
|
type Truncated = Bool
|
|
|
|
appendFsckOutput :: FsckOutput -> FsckOutput -> FsckOutput
|
|
appendFsckOutput (FsckOutput s1 t1) (FsckOutput s2 t2) =
|
|
FsckOutput (S.union s1 s2) (t1 || t2)
|
|
appendFsckOutput (FsckOutput s t) _ = FsckOutput s t
|
|
appendFsckOutput _ (FsckOutput s t) = FsckOutput s t
|
|
appendFsckOutput NoFsckOutput NoFsckOutput = NoFsckOutput
|
|
appendFsckOutput AllDuplicateEntriesWarning AllDuplicateEntriesWarning = AllDuplicateEntriesWarning
|
|
appendFsckOutput AllDuplicateEntriesWarning NoFsckOutput = AllDuplicateEntriesWarning
|
|
appendFsckOutput NoFsckOutput AllDuplicateEntriesWarning = AllDuplicateEntriesWarning
|
|
|
|
instance Sem.Semigroup FsckOutput where
|
|
(<>) = appendFsckOutput
|
|
|
|
instance Monoid FsckOutput where
|
|
mempty = NoFsckOutput
|
|
#if ! MIN_VERSION_base(4,11,0)
|
|
mappend = (Sem.<>)
|
|
#endif
|
|
|
|
{- Runs fsck to find some of the broken objects in the repository.
|
|
- May not find all broken objects, if fsck fails on bad data in some of
|
|
- the broken objects it does find.
|
|
-
|
|
- Strategy: Rather than parsing fsck's current specific output,
|
|
- look for anything in its output (both stdout and stderr) that appears
|
|
- to be a git sha. Not all such shas are of broken objects, so ask git
|
|
- to try to cat the object, and see if it fails.
|
|
-}
|
|
findBroken :: Bool -> Repo -> IO FsckResults
|
|
findBroken batchmode r = do
|
|
supportsNoDangling <- (>= Git.Version.normalize "1.7.10")
|
|
<$> Git.Version.installed
|
|
let (command, params) = ("git", fsckParams supportsNoDangling r)
|
|
(command', params') <- if batchmode
|
|
then toBatchCommand (command, params)
|
|
else return (command, params)
|
|
|
|
p@(_, _, _, pid) <- createProcess $
|
|
(proc command' (toCommand params'))
|
|
{ std_out = CreatePipe
|
|
, std_err = CreatePipe
|
|
}
|
|
(o1, o2) <- concurrently
|
|
(parseFsckOutput maxobjs r supportsNoDangling (stdoutHandle p))
|
|
(parseFsckOutput maxobjs r supportsNoDangling (stderrHandle p))
|
|
fsckok <- checkSuccessProcess pid
|
|
case mappend o1 o2 of
|
|
FsckOutput badobjs truncated
|
|
| S.null badobjs && not fsckok -> return FsckFailed
|
|
| otherwise -> return $ FsckFoundMissing badobjs truncated
|
|
NoFsckOutput
|
|
| not fsckok -> return FsckFailed
|
|
| otherwise -> return noproblem
|
|
-- If all fsck output was duplicateEntries warnings,
|
|
-- the repository is not broken, it just has some unusual
|
|
-- tree objects in it. So ignore nonzero exit status.
|
|
AllDuplicateEntriesWarning -> return noproblem
|
|
where
|
|
maxobjs = 10000
|
|
noproblem = FsckFoundMissing S.empty False
|
|
|
|
foundBroken :: FsckResults -> Bool
|
|
foundBroken FsckFailed = True
|
|
foundBroken (FsckFoundMissing s _) = not (S.null s)
|
|
|
|
knownMissing :: FsckResults -> MissingObjects
|
|
knownMissing FsckFailed = S.empty
|
|
knownMissing (FsckFoundMissing s _) = s
|
|
|
|
{- Finds objects that are missing from the git repsitory, or are corrupt.
|
|
-
|
|
- This does not use git cat-file --batch, because catting a corrupt
|
|
- object can cause it to crash, or to report incorrect size information.
|
|
-}
|
|
findMissing :: [Sha] -> Repo -> IO MissingObjects
|
|
findMissing objs r = S.fromList <$> filterM (`isMissing` r) objs
|
|
|
|
parseFsckOutput :: Int -> Repo -> Bool -> Handle -> IO FsckOutput
|
|
parseFsckOutput maxobjs r supportsNoDangling h = do
|
|
ls <- lines <$> hGetContents h
|
|
if null ls
|
|
then return NoFsckOutput
|
|
else if all ("duplicateEntries" `isInfixOf`) ls
|
|
then return AllDuplicateEntriesWarning
|
|
else do
|
|
let shas = findShas supportsNoDangling ls
|
|
let !truncated = length shas > maxobjs
|
|
missingobjs <- findMissing (take maxobjs shas) r
|
|
return $ FsckOutput missingobjs truncated
|
|
|
|
isMissing :: Sha -> Repo -> IO Bool
|
|
isMissing s r = either (const True) (const False) <$> tryIO dump
|
|
where
|
|
dump = runQuiet
|
|
[ Param "show"
|
|
, Param (fromRef s)
|
|
] r
|
|
|
|
findShas :: Bool -> [String] -> [Sha]
|
|
findShas supportsNoDangling = catMaybes . map extractSha . concat . map words . filter wanted
|
|
where
|
|
wanted l
|
|
| supportsNoDangling = True
|
|
| otherwise = not ("dangling " `isPrefixOf` l)
|
|
|
|
fsckParams :: Bool -> Repo -> [CommandParam]
|
|
fsckParams supportsNoDangling = gitCommandLine $ map Param $ catMaybes
|
|
[ Just "fsck"
|
|
, if supportsNoDangling
|
|
then Just "--no-dangling"
|
|
else Nothing
|
|
, Just "--no-reflogs"
|
|
]
|