2010-11-06 21:06:19 +00:00
|
|
|
{- git-annex command
|
|
|
|
-
|
2011-10-29 21:49:37 +00:00
|
|
|
- Copyright 2010-2011 Joey Hess <joey@kitenet.net>
|
2010-11-06 21:06:19 +00:00
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
|
|
|
module Command.Fsck where
|
|
|
|
|
2011-10-05 20:02:51 +00:00
|
|
|
import Common.Annex
|
2010-11-06 21:06:19 +00:00
|
|
|
import Command
|
2011-07-05 22:31:46 +00:00
|
|
|
import qualified Remote
|
|
|
|
import qualified Types.Backend
|
|
|
|
import qualified Types.Key
|
2011-10-29 21:49:37 +00:00
|
|
|
import qualified Backend
|
2011-10-04 04:40:47 +00:00
|
|
|
import Annex.Content
|
2011-10-15 20:21:08 +00:00
|
|
|
import Logs.Location
|
|
|
|
import Logs.Trust
|
2011-10-15 21:47:03 +00:00
|
|
|
import Annex.UUID
|
2011-07-06 00:36:43 +00:00
|
|
|
import Utility.DataUnits
|
2011-09-23 22:13:24 +00:00
|
|
|
import Utility.FileMode
|
2011-07-05 22:31:46 +00:00
|
|
|
import Config
|
2010-11-06 21:06:19 +00:00
|
|
|
|
2011-10-29 19:19:05 +00:00
|
|
|
def :: [Command]
|
|
|
|
def = [command "fsck" paramPaths seek "check for problems"]
|
2010-12-30 19:06:26 +00:00
|
|
|
|
2010-12-30 18:19:16 +00:00
|
|
|
seek :: [CommandSeek]
|
2011-11-11 03:35:08 +00:00
|
|
|
seek =
|
|
|
|
[ withNumCopies $ \n -> whenAnnexed $ start n
|
|
|
|
, withBarePresentKeys startBare
|
|
|
|
]
|
2010-11-15 22:22:50 +00:00
|
|
|
|
2011-12-31 08:11:39 +00:00
|
|
|
start :: Maybe Int -> FilePath -> (Key, Backend) -> CommandStart
|
2011-11-11 03:35:08 +00:00
|
|
|
start numcopies file (key, backend) = do
|
2010-11-15 22:22:50 +00:00
|
|
|
showStart "fsck" file
|
2011-09-15 20:24:47 +00:00
|
|
|
next $ perform key file backend numcopies
|
2010-11-15 22:22:50 +00:00
|
|
|
|
2011-12-31 08:11:39 +00:00
|
|
|
perform :: Key -> FilePath -> Backend -> Maybe Int -> CommandPerform
|
2011-10-29 21:49:37 +00:00
|
|
|
perform key file backend numcopies = check
|
2011-10-29 20:45:06 +00:00
|
|
|
-- order matters
|
|
|
|
[ verifyLocationLog key file
|
|
|
|
, checkKeySize key
|
2011-10-31 16:33:41 +00:00
|
|
|
, checkBackend backend key
|
2012-01-03 22:39:39 +00:00
|
|
|
, checkKeyNumCopies key file numcopies
|
2011-10-29 20:45:06 +00:00
|
|
|
]
|
2011-10-29 21:49:37 +00:00
|
|
|
|
|
|
|
{- To fsck a bare repository, fsck each key in the location log. -}
|
|
|
|
withBarePresentKeys :: (Key -> CommandStart) -> CommandSeek
|
2011-10-29 22:47:53 +00:00
|
|
|
withBarePresentKeys a params = isBareRepo >>= go
|
|
|
|
where
|
|
|
|
go False = return []
|
|
|
|
go True = do
|
2011-11-11 05:52:58 +00:00
|
|
|
unless (null params) $
|
2011-10-29 21:49:37 +00:00
|
|
|
error "fsck should be run without parameters in a bare repository"
|
2011-10-30 03:48:46 +00:00
|
|
|
prepStart a loggedKeys
|
2011-10-29 21:49:37 +00:00
|
|
|
|
|
|
|
startBare :: Key -> CommandStart
|
|
|
|
startBare key = case Backend.maybeLookupBackendName (Types.Key.keyBackendName key) of
|
|
|
|
Nothing -> stop
|
|
|
|
Just backend -> do
|
|
|
|
showStart "fsck" (show key)
|
|
|
|
next $ performBare key backend
|
|
|
|
|
|
|
|
{- Note that numcopies cannot be checked in a bare repository, because
|
|
|
|
- getting the numcopies value requires a working copy with .gitattributes
|
|
|
|
- files. -}
|
2011-12-31 08:11:39 +00:00
|
|
|
performBare :: Key -> Backend -> CommandPerform
|
2011-10-29 21:49:37 +00:00
|
|
|
performBare key backend = check
|
|
|
|
[ verifyLocationLog key (show key)
|
|
|
|
, checkKeySize key
|
2011-10-31 16:33:41 +00:00
|
|
|
, checkBackend backend key
|
2011-10-29 21:49:37 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
check :: [Annex Bool] -> CommandPerform
|
2011-11-01 03:39:55 +00:00
|
|
|
check = sequence >=> dispatch
|
2011-10-29 20:45:06 +00:00
|
|
|
where
|
2011-10-29 21:49:37 +00:00
|
|
|
dispatch vs
|
2011-10-29 20:45:06 +00:00
|
|
|
| all (== True) vs = next $ return True
|
|
|
|
| otherwise = stop
|
2011-03-02 18:30:36 +00:00
|
|
|
|
|
|
|
{- Checks that the location log reflects the current status of the key,
|
|
|
|
in this repository only. -}
|
2011-10-29 21:49:37 +00:00
|
|
|
verifyLocationLog :: Key -> String -> Annex Bool
|
|
|
|
verifyLocationLog key desc = do
|
2011-03-02 18:30:36 +00:00
|
|
|
present <- inAnnex key
|
|
|
|
|
2011-03-28 20:19:20 +00:00
|
|
|
-- Since we're checking that a key's file is present, throw
|
|
|
|
-- in a permission fixup here too.
|
2011-11-08 19:34:10 +00:00
|
|
|
when present $ do
|
2011-11-29 02:43:51 +00:00
|
|
|
f <- inRepo $ gitAnnexLocation key
|
2011-11-08 19:34:10 +00:00
|
|
|
liftIO $ do
|
|
|
|
preventWrite f
|
|
|
|
preventWrite (parentDir f)
|
2011-03-28 20:19:20 +00:00
|
|
|
|
2011-10-11 18:43:45 +00:00
|
|
|
u <- getUUID
|
2011-06-22 20:13:43 +00:00
|
|
|
uuids <- keyLocations key
|
2011-03-02 18:30:36 +00:00
|
|
|
|
|
|
|
case (present, u `elem` uuids) of
|
|
|
|
(True, False) -> do
|
2011-11-08 19:34:10 +00:00
|
|
|
fix u InfoPresent
|
2011-03-02 18:30:36 +00:00
|
|
|
-- There is no data loss, so do not fail.
|
|
|
|
return True
|
|
|
|
(False, True) -> do
|
2011-11-08 19:34:10 +00:00
|
|
|
fix u InfoMissing
|
2011-03-02 18:30:36 +00:00
|
|
|
warning $
|
2011-10-29 21:49:37 +00:00
|
|
|
"** Based on the location log, " ++ desc
|
2011-03-02 18:30:36 +00:00
|
|
|
++ "\n** was expected to be present, " ++
|
|
|
|
"but its content is missing."
|
|
|
|
return False
|
|
|
|
_ -> return True
|
|
|
|
|
|
|
|
where
|
2011-11-08 19:34:10 +00:00
|
|
|
fix u s = do
|
2011-03-02 18:30:36 +00:00
|
|
|
showNote "fixing location log"
|
2011-11-09 05:15:51 +00:00
|
|
|
logChange key u s
|
2011-07-05 22:31:46 +00:00
|
|
|
|
|
|
|
{- The size of the data for a key is checked against the size encoded in
|
|
|
|
- the key's metadata, if available. -}
|
|
|
|
checkKeySize :: Key -> Annex Bool
|
|
|
|
checkKeySize key = do
|
2011-11-29 02:43:51 +00:00
|
|
|
file <- inRepo $ gitAnnexLocation key
|
2011-07-05 22:31:46 +00:00
|
|
|
present <- liftIO $ doesFileExist file
|
|
|
|
case (present, Types.Key.keySize key) of
|
|
|
|
(_, Nothing) -> return True
|
|
|
|
(False, _) -> return True
|
|
|
|
(True, Just size) -> do
|
|
|
|
stat <- liftIO $ getFileStatus file
|
|
|
|
let size' = fromIntegral (fileSize stat)
|
|
|
|
if size == size'
|
|
|
|
then return True
|
|
|
|
else do
|
|
|
|
dest <- moveBad key
|
|
|
|
warning $ "Bad file size (" ++
|
|
|
|
compareSizes storageUnits True size size' ++
|
|
|
|
"); moved to " ++ dest
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2011-12-31 08:11:39 +00:00
|
|
|
checkBackend :: Backend -> Key -> Annex Bool
|
2011-11-11 05:52:58 +00:00
|
|
|
checkBackend = Types.Backend.fsckKey
|
2011-10-31 16:33:41 +00:00
|
|
|
|
2011-10-29 20:45:06 +00:00
|
|
|
checkKeyNumCopies :: Key -> FilePath -> Maybe Int -> Annex Bool
|
2011-07-05 22:31:46 +00:00
|
|
|
checkKeyNumCopies key file numcopies = do
|
|
|
|
needed <- getNumCopies numcopies
|
2011-09-06 21:19:29 +00:00
|
|
|
(untrustedlocations, safelocations) <- trustPartition UnTrusted =<< keyLocations key
|
2011-07-05 22:31:46 +00:00
|
|
|
let present = length safelocations
|
|
|
|
if present < needed
|
|
|
|
then do
|
2011-09-01 20:02:01 +00:00
|
|
|
ppuuids <- Remote.prettyPrintUUIDs "untrusted" untrustedlocations
|
2011-10-29 20:45:06 +00:00
|
|
|
warning $ missingNote file present needed ppuuids
|
2011-07-05 22:31:46 +00:00
|
|
|
return False
|
|
|
|
else return True
|
|
|
|
|
|
|
|
missingNote :: String -> Int -> Int -> String -> String
|
|
|
|
missingNote file 0 _ [] =
|
|
|
|
"** No known copies exist of " ++ file
|
|
|
|
missingNote file 0 _ untrusted =
|
|
|
|
"Only these untrusted locations may have copies of " ++ file ++
|
|
|
|
"\n" ++ untrusted ++
|
|
|
|
"Back it up to trusted locations with git-annex copy."
|
|
|
|
missingNote file present needed [] =
|
|
|
|
"Only " ++ show present ++ " of " ++ show needed ++
|
|
|
|
" trustworthy copies exist of " ++ file ++
|
|
|
|
"\nBack it up with git-annex copy."
|
|
|
|
missingNote file present needed untrusted =
|
|
|
|
missingNote file present needed [] ++
|
|
|
|
"\nThe following untrusted locations may also have copies: " ++
|
|
|
|
"\n" ++ untrusted
|