add git fsck to cronner, and UI for repository repair (not yet wired up)
This commit is contained in:
parent
44bb9a808f
commit
d345e5b52f
12 changed files with 163 additions and 18 deletions
|
@ -79,6 +79,22 @@ warningAlert name msg = Alert
|
||||||
, alertButton = Nothing
|
, alertButton = Nothing
|
||||||
}
|
}
|
||||||
|
|
||||||
|
errorAlert :: String -> AlertButton -> Alert
|
||||||
|
errorAlert msg button = Alert
|
||||||
|
{ alertClass = Error
|
||||||
|
, alertHeader = Just $ tenseWords ["error"]
|
||||||
|
, alertMessageRender = renderData
|
||||||
|
, alertData = [UnTensed $ T.pack msg]
|
||||||
|
, alertCounter = 0
|
||||||
|
, alertBlockDisplay = True
|
||||||
|
, alertClosable = True
|
||||||
|
, alertPriority = Pinned
|
||||||
|
, alertIcon = Just ErrorIcon
|
||||||
|
, alertCombiner = Nothing
|
||||||
|
, alertName = Nothing
|
||||||
|
, alertButton = Just button
|
||||||
|
}
|
||||||
|
|
||||||
activityAlert :: Maybe TenseText -> [TenseChunk] -> Alert
|
activityAlert :: Maybe TenseText -> [TenseChunk] -> Alert
|
||||||
activityAlert header dat = baseActivityAlert
|
activityAlert header dat = baseActivityAlert
|
||||||
{ alertHeader = header
|
{ alertHeader = header
|
||||||
|
@ -158,6 +174,9 @@ fsckAlert button n = baseActivityAlert
|
||||||
, alertButton = Just button
|
, alertButton = Just button
|
||||||
}
|
}
|
||||||
|
|
||||||
|
brokenRepositoryAlert :: AlertButton -> Alert
|
||||||
|
brokenRepositoryAlert = errorAlert "Your repository needs repairs."
|
||||||
|
|
||||||
pairingAlert :: AlertButton -> Alert
|
pairingAlert :: AlertButton -> Alert
|
||||||
pairingAlert button = baseActivityAlert
|
pairingAlert button = baseActivityAlert
|
||||||
{ alertData = [ UnTensed "Pairing in progress" ]
|
{ alertData = [ UnTensed "Pairing in progress" ]
|
||||||
|
|
|
@ -32,6 +32,8 @@ import Remote
|
||||||
import Assistant.WebApp.Types
|
import Assistant.WebApp.Types
|
||||||
#endif
|
#endif
|
||||||
import Git.Remote (RemoteName)
|
import Git.Remote (RemoteName)
|
||||||
|
import qualified Git.Fsck
|
||||||
|
import Logs.FsckResults
|
||||||
|
|
||||||
import Control.Concurrent.Async
|
import Control.Concurrent.Async
|
||||||
import Control.Concurrent.MVar
|
import Control.Concurrent.MVar
|
||||||
|
@ -182,15 +184,24 @@ runActivity urlrenderer activity nowt = do
|
||||||
runActivity' :: UrlRenderer -> ScheduledActivity -> Assistant ()
|
runActivity' :: UrlRenderer -> ScheduledActivity -> Assistant ()
|
||||||
runActivity' urlrenderer (ScheduledSelfFsck _ d) = do
|
runActivity' urlrenderer (ScheduledSelfFsck _ d) = do
|
||||||
program <- liftIO $ readProgramFile
|
program <- liftIO $ readProgramFile
|
||||||
void $ runFsck urlrenderer Nothing $
|
g <- liftAnnex gitRepo
|
||||||
batchCommand program (Param "fsck" : fsckParams d)
|
fsckresults <- showFscking urlrenderer Nothing $ tryNonAsync $ do
|
||||||
|
r <- Git.Fsck.findBroken True g
|
||||||
|
void $ batchCommand program (Param "fsck" : annexFsckParams d)
|
||||||
|
return r
|
||||||
|
when (Git.Fsck.foundBroken fsckresults) $ do
|
||||||
|
u <- liftAnnex getUUID
|
||||||
|
liftAnnex $ writeFsckResults u fsckresults
|
||||||
|
button <- mkAlertButton True (T.pack "Repair") urlrenderer $
|
||||||
|
RepairRepositoryR u
|
||||||
|
void $ addAlert $ brokenRepositoryAlert button
|
||||||
mapM_ reget =<< liftAnnex (dirKeys gitAnnexBadDir)
|
mapM_ reget =<< liftAnnex (dirKeys gitAnnexBadDir)
|
||||||
where
|
where
|
||||||
reget k = queueTransfers "fsck found bad file; redownloading" Next k Nothing Download
|
reget k = queueTransfers "fsck found bad file; redownloading" Next k Nothing Download
|
||||||
runActivity' urlrenderer (ScheduledRemoteFsck u s d) = go =<< liftAnnex (remoteFromUUID u)
|
runActivity' urlrenderer (ScheduledRemoteFsck u s d) = go =<< liftAnnex (remoteFromUUID u)
|
||||||
where
|
where
|
||||||
go (Just r) = void $ case Remote.remoteFsck r of
|
go (Just r) = void $ case Remote.remoteFsck r of
|
||||||
Nothing -> void $ runFsck urlrenderer (Just $ Remote.name r) $ do
|
Nothing -> void $ showFscking urlrenderer (Just $ Remote.name r) $ tryNonAsync $ do
|
||||||
program <- readProgramFile
|
program <- readProgramFile
|
||||||
batchCommand program $
|
batchCommand program $
|
||||||
[ Param "fsck"
|
[ Param "fsck"
|
||||||
|
@ -198,28 +209,28 @@ runActivity' urlrenderer (ScheduledRemoteFsck u s d) = go =<< liftAnnex (remoteF
|
||||||
, Param "--fast"
|
, Param "--fast"
|
||||||
, Param "--from"
|
, Param "--from"
|
||||||
, Param $ Remote.name r
|
, Param $ Remote.name r
|
||||||
] ++ fsckParams d
|
] ++ annexFsckParams d
|
||||||
Just mkfscker ->
|
Just mkfscker ->
|
||||||
{- Note that having mkfsker return an IO action
|
{- Note that having mkfsker return an IO action
|
||||||
- avoids running a long duration fsck in the
|
- avoids running a long duration fsck in the
|
||||||
- Annex monad. -}
|
- Annex monad. -}
|
||||||
void . runFsck urlrenderer (Just $ Remote.name r)
|
void . showFscking urlrenderer (Just $ Remote.name r) . tryNonAsync
|
||||||
=<< liftAnnex (mkfscker (fsckParams d))
|
=<< liftAnnex (mkfscker (annexFsckParams d))
|
||||||
go Nothing = debug ["skipping remote fsck of uuid without a configured remote", fromUUID u, fromSchedule s]
|
go Nothing = debug ["skipping remote fsck of uuid without a configured remote", fromUUID u, fromSchedule s]
|
||||||
|
|
||||||
runFsck :: UrlRenderer -> Maybe RemoteName -> IO Bool -> Assistant Bool
|
showFscking :: UrlRenderer -> Maybe RemoteName -> IO (Either E.SomeException a) -> Assistant a
|
||||||
runFsck urlrenderer remotename a = do
|
showFscking urlrenderer remotename a = do
|
||||||
#ifdef WITH_WEBAPP
|
#ifdef WITH_WEBAPP
|
||||||
button <- mkAlertButton False (T.pack "Configure") urlrenderer ConfigFsckR
|
button <- mkAlertButton False (T.pack "Configure") urlrenderer ConfigFsckR
|
||||||
r <- alertDuring (fsckAlert button remotename) $ liftIO $ do
|
r <- alertDuring (fsckAlert button remotename) $
|
||||||
E.try a :: IO (Either E.SomeException Bool)
|
liftIO a
|
||||||
either (liftIO . E.throwIO) return r
|
either (liftIO . E.throwIO) return r
|
||||||
#else
|
#else
|
||||||
a
|
a
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
fsckParams :: Duration -> [CommandParam]
|
annexFsckParams :: Duration -> [CommandParam]
|
||||||
fsckParams d =
|
annexFsckParams d =
|
||||||
[ Param "--incremental-schedule=1d"
|
[ Param "--incremental-schedule=1d"
|
||||||
, Param $ "--time-limit=" ++ fromDuration d
|
, Param $ "--time-limit=" ++ fromDuration d
|
||||||
]
|
]
|
||||||
|
|
|
@ -33,6 +33,7 @@ import Assistant.WebApp.Configurators.Fsck
|
||||||
import Assistant.WebApp.Documentation
|
import Assistant.WebApp.Documentation
|
||||||
import Assistant.WebApp.Control
|
import Assistant.WebApp.Control
|
||||||
import Assistant.WebApp.OtherRepos
|
import Assistant.WebApp.OtherRepos
|
||||||
|
import Assistant.WebApp.Repair
|
||||||
import Assistant.Types.ThreadedMonad
|
import Assistant.Types.ThreadedMonad
|
||||||
import Utility.WebApp
|
import Utility.WebApp
|
||||||
import Utility.Tmp
|
import Utility.Tmp
|
||||||
|
|
26
Assistant/WebApp/Repair.hs
Normal file
26
Assistant/WebApp/Repair.hs
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
{- git-annex assistant repository repair
|
||||||
|
-
|
||||||
|
- Copyright 2013 Joey Hess <joey@kitenet.net>
|
||||||
|
-
|
||||||
|
- Licensed under the GNU AGPL version 3 or higher.
|
||||||
|
-}
|
||||||
|
|
||||||
|
{-# LANGUAGE QuasiQuotes, TemplateHaskell, OverloadedStrings #-}
|
||||||
|
|
||||||
|
module Assistant.WebApp.Repair where
|
||||||
|
|
||||||
|
import Assistant.WebApp.Common
|
||||||
|
import Remote (prettyUUID)
|
||||||
|
|
||||||
|
getRepairRepositoryR :: UUID -> Handler Html
|
||||||
|
getRepairRepositoryR = postRepairRepositoryR
|
||||||
|
postRepairRepositoryR :: UUID -> Handler Html
|
||||||
|
postRepairRepositoryR u = page "Repair repository" Nothing $ do
|
||||||
|
repodesc <- liftAnnex $ prettyUUID u
|
||||||
|
$(widgetFile "control/repairrepository")
|
||||||
|
|
||||||
|
getRepairRepositoryRunR :: UUID -> Handler Html
|
||||||
|
getRepairRepositoryRunR = postRepairRepositoryR
|
||||||
|
postRepairRepositoryRunR :: UUID -> Handler Html
|
||||||
|
postRepairRepositoryRunR u = page "Repair repository" Nothing $ do
|
||||||
|
$(widgetFile "control/repairrepository/run")
|
|
@ -216,4 +216,3 @@ instance PathPiece ThreadName where
|
||||||
instance PathPiece ScheduledActivity where
|
instance PathPiece ScheduledActivity where
|
||||||
toPathPiece = pack . show
|
toPathPiece = pack . show
|
||||||
fromPathPiece = readish . unpack
|
fromPathPiece = readish . unpack
|
||||||
|
|
||||||
|
|
|
@ -108,4 +108,7 @@
|
||||||
/transfer/start/#Transfer StartTransferR GET POST
|
/transfer/start/#Transfer StartTransferR GET POST
|
||||||
/transfer/cancel/#Transfer CancelTransferR GET POST
|
/transfer/cancel/#Transfer CancelTransferR GET POST
|
||||||
|
|
||||||
|
/repair/#UUID RepairRepositoryR GET POST
|
||||||
|
/repair/run/#UUID RepairRepositoryRunR GET POST
|
||||||
|
|
||||||
/static StaticR Static getStatic
|
/static StaticR Static getStatic
|
||||||
|
|
17
Git/Fsck.hs
17
Git/Fsck.hs
|
@ -6,9 +6,11 @@
|
||||||
-}
|
-}
|
||||||
|
|
||||||
module Git.Fsck (
|
module Git.Fsck (
|
||||||
|
FsckResults,
|
||||||
|
MissingObjects,
|
||||||
findBroken,
|
findBroken,
|
||||||
|
foundBroken,
|
||||||
findMissing,
|
findMissing,
|
||||||
MissingObjects
|
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Common
|
import Common
|
||||||
|
@ -22,17 +24,20 @@ import qualified Data.Set as S
|
||||||
|
|
||||||
type MissingObjects = S.Set Sha
|
type MissingObjects = S.Set Sha
|
||||||
|
|
||||||
|
{- If fsck succeeded, Just a set of missing objects it found.
|
||||||
|
- If it failed, Nothing. -}
|
||||||
|
type FsckResults = Maybe MissingObjects
|
||||||
|
|
||||||
{- Runs fsck to find some of the broken objects in the repository.
|
{- 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
|
- May not find all broken objects, if fsck fails on bad data in some of
|
||||||
- the broken objects it does find. If the fsck fails generally without
|
- the broken objects it does find.
|
||||||
- finding any broken objects, returns Nothing.
|
|
||||||
-
|
-
|
||||||
- Strategy: Rather than parsing fsck's current specific output,
|
- Strategy: Rather than parsing fsck's current specific output,
|
||||||
- look for anything in its output (both stdout and stderr) that appears
|
- 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 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.
|
- to try to cat the object, and see if it fails.
|
||||||
-}
|
-}
|
||||||
findBroken :: Bool -> Repo -> IO (Maybe MissingObjects)
|
findBroken :: Bool -> Repo -> IO FsckResults
|
||||||
findBroken batchmode r = do
|
findBroken batchmode r = do
|
||||||
(output, fsckok) <- processTranscript command' (toCommand params') Nothing
|
(output, fsckok) <- processTranscript command' (toCommand params') Nothing
|
||||||
let objs = parseFsckOutput output
|
let objs = parseFsckOutput output
|
||||||
|
@ -46,6 +51,10 @@ findBroken batchmode r = do
|
||||||
| batchmode = toBatchCommand (command, params)
|
| batchmode = toBatchCommand (command, params)
|
||||||
| otherwise = (command, params)
|
| otherwise = (command, params)
|
||||||
|
|
||||||
|
foundBroken :: FsckResults -> Bool
|
||||||
|
foundBroken Nothing = True
|
||||||
|
foundBroken (Just s) = not (S.null s)
|
||||||
|
|
||||||
{- Finds objects that are missing from the git repsitory, or are corrupt.
|
{- Finds objects that are missing from the git repsitory, or are corrupt.
|
||||||
-
|
-
|
||||||
- Note that catting a corrupt object will cause cat-file to crash;
|
- Note that catting a corrupt object will cause cat-file to crash;
|
||||||
|
|
|
@ -48,7 +48,7 @@ import Data.Tuple.Utils
|
||||||
- To remove corrupt objects, unpack all packs, and remove the packs
|
- To remove corrupt objects, unpack all packs, and remove the packs
|
||||||
- (to handle corrupt packs), and remove loose object files.
|
- (to handle corrupt packs), and remove loose object files.
|
||||||
-}
|
-}
|
||||||
cleanCorruptObjects :: Maybe MissingObjects -> Repo -> IO MissingObjects
|
cleanCorruptObjects :: FsckResults -> Repo -> IO MissingObjects
|
||||||
cleanCorruptObjects mmissing r = check mmissing
|
cleanCorruptObjects mmissing r = check mmissing
|
||||||
where
|
where
|
||||||
check Nothing = do
|
check Nothing = do
|
||||||
|
|
|
@ -28,6 +28,7 @@ module Locations (
|
||||||
gitAnnexBadLocation,
|
gitAnnexBadLocation,
|
||||||
gitAnnexUnusedLog,
|
gitAnnexUnusedLog,
|
||||||
gitAnnexFsckState,
|
gitAnnexFsckState,
|
||||||
|
gitAnnexFsckResultsLog,
|
||||||
gitAnnexScheduleState,
|
gitAnnexScheduleState,
|
||||||
gitAnnexTransferDir,
|
gitAnnexTransferDir,
|
||||||
gitAnnexCredsDir,
|
gitAnnexCredsDir,
|
||||||
|
@ -66,6 +67,7 @@ import Data.Char
|
||||||
import Common
|
import Common
|
||||||
import Types
|
import Types
|
||||||
import Types.Key
|
import Types.Key
|
||||||
|
import Types.UUID
|
||||||
import qualified Git
|
import qualified Git
|
||||||
|
|
||||||
{- Conventions:
|
{- Conventions:
|
||||||
|
@ -193,6 +195,10 @@ gitAnnexUnusedLog prefix r = gitAnnexDir r </> (prefix ++ "unused")
|
||||||
gitAnnexFsckState :: Git.Repo -> FilePath
|
gitAnnexFsckState :: Git.Repo -> FilePath
|
||||||
gitAnnexFsckState r = gitAnnexDir r </> "fsckstate"
|
gitAnnexFsckState r = gitAnnexDir r </> "fsckstate"
|
||||||
|
|
||||||
|
{- .git/annex/fsckresults/uuid is used to store results of git fscks -}
|
||||||
|
gitAnnexFsckResultsLog :: UUID -> Git.Repo -> FilePath
|
||||||
|
gitAnnexFsckResultsLog u r = gitAnnexDir r </> "fsckresults" </> fromUUID u
|
||||||
|
|
||||||
{- .git/annex/schedulestate is used to store information about when
|
{- .git/annex/schedulestate is used to store information about when
|
||||||
- scheduled jobs were last run. -}
|
- scheduled jobs were last run. -}
|
||||||
gitAnnexScheduleState :: Git.Repo -> FilePath
|
gitAnnexScheduleState :: Git.Repo -> FilePath
|
||||||
|
|
43
Logs/FsckResults.hs
Normal file
43
Logs/FsckResults.hs
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
{- git-annex fsck results log files
|
||||||
|
-
|
||||||
|
- Copyright 2013 Joey Hess <joey@kitenet.net>
|
||||||
|
-
|
||||||
|
- Licensed under the GNU GPL version 3 or higher.
|
||||||
|
-}
|
||||||
|
|
||||||
|
module Logs.FsckResults (
|
||||||
|
writeFsckResults,
|
||||||
|
readFsckResults
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Common.Annex
|
||||||
|
import Utility.Tmp
|
||||||
|
import Git.Fsck
|
||||||
|
import Git.Types
|
||||||
|
|
||||||
|
import qualified Data.Set as S
|
||||||
|
|
||||||
|
writeFsckResults :: UUID -> FsckResults -> Annex ()
|
||||||
|
writeFsckResults u fsckresults = do
|
||||||
|
logfile <- fromRepo $ gitAnnexFsckResultsLog u
|
||||||
|
liftIO $
|
||||||
|
case fsckresults of
|
||||||
|
Nothing -> store S.empty logfile
|
||||||
|
Just s
|
||||||
|
| S.null s -> nukeFile logfile
|
||||||
|
| otherwise -> store s logfile
|
||||||
|
where
|
||||||
|
store s logfile = do
|
||||||
|
createDirectoryIfMissing True (parentDir logfile)
|
||||||
|
liftIO $ viaTmp writeFile logfile $ serialize s
|
||||||
|
serialize = unlines . map show . S.toList
|
||||||
|
|
||||||
|
readFsckResults :: UUID -> Annex FsckResults
|
||||||
|
readFsckResults u = do
|
||||||
|
logfile <- fromRepo $ gitAnnexFsckResultsLog u
|
||||||
|
liftIO $ catchDefaultIO (Just S.empty) $
|
||||||
|
deserialize <$> readFile logfile
|
||||||
|
where
|
||||||
|
deserialize l =
|
||||||
|
let s = S.fromList $ map Ref $ lines l
|
||||||
|
in if S.null s then Nothing else Just s
|
25
templates/control/repairrepository.hamlet
Normal file
25
templates/control/repairrepository.hamlet
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
<div .span9 .hero-unit>
|
||||||
|
<p>
|
||||||
|
A periodic #
|
||||||
|
<a href="@{ConfigFsckR}">
|
||||||
|
consistency check #
|
||||||
|
found corrupt data in the git repository #{repodesc}.
|
||||||
|
<p>
|
||||||
|
While this is not good, this problem can be automatically repaired,
|
||||||
|
often without data loss.
|
||||||
|
<p>
|
||||||
|
When possible, the corrupt data will be recovered from other #
|
||||||
|
repositories. You should make sure any other repositories #
|
||||||
|
are available before continuing. Ie, plug in any removable drive #
|
||||||
|
that contains a repository, or make sure your network connection #
|
||||||
|
to other repositories is active.
|
||||||
|
<p>
|
||||||
|
<a .btn .btn-primary href="@{RepairRepositoryRunR u}" onclick="$('#workingmodal').modal('show');">
|
||||||
|
Start Repair Process
|
||||||
|
<div .modal .fade #workingmodal>
|
||||||
|
<div .modal-header>
|
||||||
|
<h3>
|
||||||
|
Repairing #{repodesc} ...
|
||||||
|
<div .modal-body>
|
||||||
|
<p>
|
||||||
|
This may take some time.
|
3
templates/control/repairrepository/run.hamlet
Normal file
3
templates/control/repairrepository/run.hamlet
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
<div .span9 .hero-unit>
|
||||||
|
<p>
|
||||||
|
TODO
|
Loading…
Reference in a new issue