2013-10-26 20:54:49 +00:00
|
|
|
{- git-annex assistant repository repair
|
|
|
|
-
|
|
|
|
- Copyright 2013 Joey Hess <joey@kitenet.net>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU AGPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
2013-10-26 21:16:29 +00:00
|
|
|
{-# LANGUAGE CPP #-}
|
2013-10-26 20:54:49 +00:00
|
|
|
|
|
|
|
module Assistant.Repair where
|
|
|
|
|
|
|
|
import Assistant.Common
|
|
|
|
import Command.Repair (repairAnnexBranch)
|
2013-10-27 19:38:59 +00:00
|
|
|
import Git.Fsck (FsckResults, foundBroken)
|
2013-10-26 20:54:49 +00:00
|
|
|
import Git.Repair (runRepairOf)
|
2013-10-27 19:38:59 +00:00
|
|
|
import qualified Git
|
|
|
|
import qualified Remote
|
|
|
|
import qualified Types.Remote as Remote
|
2013-10-26 20:54:49 +00:00
|
|
|
import Logs.FsckResults
|
|
|
|
import Annex.UUID
|
|
|
|
import Utility.Batch
|
|
|
|
import Config.Files
|
|
|
|
import Assistant.Sync
|
2013-10-26 21:16:29 +00:00
|
|
|
import Assistant.Alert
|
|
|
|
import Assistant.DaemonStatus
|
|
|
|
import Assistant.Types.UrlRenderer
|
|
|
|
#ifdef WITH_WEBAPP
|
|
|
|
import Assistant.WebApp.Types
|
|
|
|
#endif
|
2013-10-26 20:54:49 +00:00
|
|
|
|
2013-10-26 21:16:29 +00:00
|
|
|
import qualified Data.Text as T
|
2013-10-26 20:54:49 +00:00
|
|
|
import Control.Concurrent.Async
|
|
|
|
|
2013-10-27 19:38:59 +00:00
|
|
|
{- When the FsckResults require a repair, tries to do a non-destructive
|
|
|
|
- repair. If that fails, pops up an alert. -}
|
2013-10-27 20:42:13 +00:00
|
|
|
repairWhenNecessary :: UrlRenderer -> UUID -> Maybe Remote -> FsckResults -> Assistant Bool
|
2013-10-27 19:38:59 +00:00
|
|
|
repairWhenNecessary urlrenderer u mrmt fsckresults
|
|
|
|
| foundBroken fsckresults = do
|
|
|
|
liftAnnex $ writeFsckResults u fsckresults
|
|
|
|
repodesc <- liftAnnex $ Remote.prettyUUID u
|
2013-10-27 20:42:13 +00:00
|
|
|
ok <- alertWhile (repairingAlert repodesc)
|
2013-10-27 19:38:59 +00:00
|
|
|
(runRepair u mrmt False)
|
2013-10-26 21:16:29 +00:00
|
|
|
#ifdef WITH_WEBAPP
|
2013-10-27 20:42:13 +00:00
|
|
|
unless ok $ do
|
|
|
|
button <- mkAlertButton True (T.pack "Click Here") urlrenderer $
|
|
|
|
RepairRepositoryR u
|
|
|
|
void $ addAlert $ brokenRepositoryAlert button
|
|
|
|
return ok
|
2013-10-26 21:16:29 +00:00
|
|
|
#endif
|
2013-10-27 20:42:13 +00:00
|
|
|
| otherwise = return False
|
2013-10-26 21:16:29 +00:00
|
|
|
|
2013-10-27 19:38:59 +00:00
|
|
|
runRepair :: UUID -> Maybe Remote -> Bool -> Assistant Bool
|
|
|
|
runRepair u mrmt destructiverepair = do
|
2013-10-26 20:54:49 +00:00
|
|
|
fsckresults <- liftAnnex $ readFsckResults u
|
|
|
|
myu <- liftAnnex getUUID
|
2013-10-26 21:16:29 +00:00
|
|
|
ok <- if u == myu
|
2013-10-26 20:54:49 +00:00
|
|
|
then localrepair fsckresults
|
|
|
|
else remoterepair fsckresults
|
|
|
|
liftAnnex $ writeFsckResults u Nothing
|
2013-10-27 19:38:59 +00:00
|
|
|
debug [ "Repaired", show u, show ok ]
|
2013-10-26 21:16:29 +00:00
|
|
|
|
|
|
|
return ok
|
2013-10-26 20:54:49 +00:00
|
|
|
where
|
|
|
|
localrepair fsckresults = do
|
2013-10-27 19:38:59 +00:00
|
|
|
-- Stop the watcher from running while running repairs.
|
|
|
|
changeSyncable Nothing False
|
|
|
|
|
2013-10-26 20:54:49 +00:00
|
|
|
-- This intentionally runs the repair inside the Annex
|
|
|
|
-- monad, which is not strictly necessary, but keeps
|
|
|
|
-- other threads that might be trying to use the Annex
|
|
|
|
-- from running until it completes.
|
2013-10-27 19:38:59 +00:00
|
|
|
ok <- liftAnnex $ repair fsckresults Nothing
|
|
|
|
|
2013-10-26 21:16:29 +00:00
|
|
|
-- Run a background fast fsck if a destructive repair had
|
|
|
|
-- to be done, to ensure that the git-annex branch
|
|
|
|
-- reflects the current state of the repo.
|
2013-10-27 19:38:59 +00:00
|
|
|
when destructiverepair $
|
2013-10-26 20:54:49 +00:00
|
|
|
backgroundfsck [ Param "--fast" ]
|
2013-10-27 19:38:59 +00:00
|
|
|
|
|
|
|
-- Start the watcher running again. This also triggers it to
|
|
|
|
-- do a startup scan, which is especially important if the
|
|
|
|
-- git repo repair removed files from the index file. Those
|
|
|
|
-- files will be seen as new, and re-added to the repository.
|
|
|
|
when (ok || destructiverepair) $
|
|
|
|
changeSyncable Nothing True
|
|
|
|
|
2013-10-26 21:16:29 +00:00
|
|
|
return ok
|
2013-10-26 20:54:49 +00:00
|
|
|
|
2013-10-27 19:38:59 +00:00
|
|
|
remoterepair fsckresults = case Remote.repairRepo =<< mrmt of
|
|
|
|
Nothing -> return False
|
|
|
|
Just mkrepair -> do
|
|
|
|
thisrepopath <- liftIO . absPath
|
|
|
|
=<< liftAnnex (fromRepo Git.repoPath)
|
|
|
|
a <- liftAnnex $ mkrepair $
|
|
|
|
repair fsckresults (Just thisrepopath)
|
|
|
|
liftIO $ catchBoolIO a
|
|
|
|
|
|
|
|
repair fsckresults referencerepo = do
|
|
|
|
(ok, stillmissing, modifiedbranches) <- inRepo $
|
|
|
|
runRepairOf fsckresults destructiverepair referencerepo
|
|
|
|
when destructiverepair $
|
|
|
|
repairAnnexBranch stillmissing modifiedbranches
|
|
|
|
return ok
|
2013-10-26 20:54:49 +00:00
|
|
|
|
|
|
|
backgroundfsck params = liftIO $ void $ async $ do
|
|
|
|
program <- readProgramFile
|
|
|
|
batchCommand program (Param "fsck" : params)
|