c0c595345c
Commands that want to use it have to run their seek action inside allowConcurrentOutput. Which seems reasonable; perhaps some future command will want to support the -J flag but not use regions. The region state moved from Annex to MessageState. This makes sense organizationally, and note that some uses of onLocal use a different Annex state, but pass the MessageState into it, which is what is needed.
144 lines
4.1 KiB
Haskell
144 lines
4.1 KiB
Haskell
{- git-annex progress output
|
|
-
|
|
- Copyright 2010-2015 Joey Hess <id@joeyh.name>
|
|
-
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
-}
|
|
|
|
{-# LANGUAGE CPP #-}
|
|
|
|
module Messages.Progress where
|
|
|
|
import Common
|
|
import Messages
|
|
import Messages.Internal
|
|
import Utility.Metered
|
|
import Types
|
|
import Types.Messages
|
|
import Types.Key
|
|
|
|
#ifdef WITH_CONCURRENTOUTPUT
|
|
import System.Console.Concurrent
|
|
import System.Console.Regions
|
|
import Control.Concurrent
|
|
#endif
|
|
import Data.Progress.Meter
|
|
import Data.Progress.Tracker
|
|
import Data.Quantity
|
|
|
|
{- Shows a progress meter while performing a transfer of a key.
|
|
- The action is passed a callback to use to update the meter. -}
|
|
metered :: Maybe MeterUpdate -> Key -> AssociatedFile -> (MeterUpdate -> Annex a) -> Annex a
|
|
metered combinemeterupdate key af a = case keySize key of
|
|
Nothing -> nometer
|
|
Just size -> withOutputType (go $ fromInteger size)
|
|
where
|
|
go _ QuietOutput = nometer
|
|
go _ JSONOutput = nometer
|
|
#if 0
|
|
go size _ = do
|
|
showOutput
|
|
liftIO $ putStrLn ""
|
|
|
|
cols <- liftIO $ maybe 79 Terminal.width <$> Terminal.size
|
|
let desc = truncatepretty cols $ fromMaybe (key2file key) af
|
|
|
|
result <- liftIO newEmptyMVar
|
|
pg <- liftIO $ newProgressBar def
|
|
{ pgWidth = cols
|
|
, pgFormat = desc ++ " :percent :bar ETA :eta"
|
|
, pgTotal = size
|
|
, pgOnCompletion = do
|
|
ok <- takeMVar result
|
|
putStrLn $ desc ++ " " ++ endResult ok
|
|
}
|
|
r <- a $ liftIO . pupdate pg
|
|
|
|
liftIO $ do
|
|
-- See if the progress bar is complete or not.
|
|
sofar <- stCompleted <$> getProgressStats pg
|
|
putMVar result (sofar >= size)
|
|
-- May not be actually complete if the action failed,
|
|
-- but this just clears the progress bar.
|
|
complete pg
|
|
|
|
return r
|
|
#else
|
|
-- Old progress bar code, not suitable for concurrent output.
|
|
go _ (ConcurrentOutput _) = nometer
|
|
go size NormalOutput = do
|
|
showOutput
|
|
progress <- liftIO $ newProgress "" size
|
|
meter <- liftIO $ newMeter progress "B" 25 (renderNums binaryOpts 1)
|
|
r <- a $ liftIO . pupdate meter progress
|
|
liftIO $ clearMeter stdout meter
|
|
return r
|
|
#endif
|
|
|
|
#if 0
|
|
pupdate pg n = do
|
|
let i = fromBytesProcessed n
|
|
sofar <- stCompleted <$> getProgressStats pg
|
|
when (i > sofar) $
|
|
tickN pg (i - sofar)
|
|
threadDelay 100
|
|
#else
|
|
pupdate meter progress n = do
|
|
setP progress $ fromBytesProcessed n
|
|
displayMeter stdout meter
|
|
#endif
|
|
maybe noop (\m -> m n) combinemeterupdate
|
|
|
|
nometer = a (const noop)
|
|
|
|
{- Use when the progress meter is only desired for concurrent
|
|
- output; as when a command's own progress output is preferred. -}
|
|
concurrentMetered :: Maybe MeterUpdate -> Key -> AssociatedFile -> (MeterUpdate -> Annex a) -> Annex a
|
|
concurrentMetered combinemeterupdate key af a = withOutputType go
|
|
where
|
|
go (ConcurrentOutput _) = metered combinemeterupdate key af a
|
|
go _ = a (fromMaybe (const noop) combinemeterupdate)
|
|
|
|
{- Progress dots. -}
|
|
showProgressDots :: Annex ()
|
|
showProgressDots = outputMessage q "."
|
|
|
|
{- Runs a command, that may output progress to either stdout or
|
|
- stderr, as well as other messages.
|
|
-
|
|
- In quiet mode, the output is suppressed, except for error messages.
|
|
-}
|
|
progressCommand :: FilePath -> [CommandParam] -> Annex Bool
|
|
progressCommand cmd params = progressCommandEnv cmd params Nothing
|
|
|
|
progressCommandEnv :: FilePath -> [CommandParam] -> Maybe [(String, String)] -> Annex Bool
|
|
progressCommandEnv cmd params environ = ifM commandProgressDisabled
|
|
( do
|
|
oh <- mkOutputHandler
|
|
liftIO $ demeterCommandEnv oh cmd params environ
|
|
, liftIO $ boolSystemEnv cmd params environ
|
|
)
|
|
|
|
mkOutputHandler :: Annex OutputHandler
|
|
mkOutputHandler = OutputHandler
|
|
<$> commandProgressDisabled
|
|
<*> mkStderrEmitter
|
|
|
|
mkStderrRelayer :: Annex (Handle -> IO ())
|
|
mkStderrRelayer = do
|
|
quiet <- commandProgressDisabled
|
|
emitter <- mkStderrEmitter
|
|
return $ \h -> avoidProgress quiet h emitter
|
|
|
|
{- Generates an IO action that can be used to emit stderr.
|
|
-
|
|
- When a progress meter is displayed, this takes care to avoid
|
|
- messing it up with interleaved stderr from a command.
|
|
-}
|
|
mkStderrEmitter :: Annex (String -> IO ())
|
|
mkStderrEmitter = withOutputType go
|
|
where
|
|
#ifdef WITH_CONCURRENTOUTPUT
|
|
go (ConcurrentOutput _) = return errorConcurrent
|
|
#endif
|
|
go _ = return (hPutStrLn stderr)
|