2010-11-08 19:15:21 +00:00
|
|
|
{- git-annex output messages
|
|
|
|
-
|
2011-11-15 04:30:00 +00:00
|
|
|
- Copyright 2010-2011 Joey Hess <joey@kitenet.net>
|
2010-11-08 19:15:21 +00:00
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
2011-09-01 19:16:31 +00:00
|
|
|
module Messages (
|
|
|
|
showStart,
|
|
|
|
showNote,
|
|
|
|
showAction,
|
|
|
|
showProgress,
|
|
|
|
showSideAction,
|
|
|
|
showOutput,
|
|
|
|
showLongNote,
|
|
|
|
showEndOk,
|
|
|
|
showEndFail,
|
|
|
|
showEndResult,
|
|
|
|
showErr,
|
|
|
|
warning,
|
|
|
|
indent,
|
2011-09-02 20:44:04 +00:00
|
|
|
maybeShowJSON,
|
2011-12-23 02:03:18 +00:00
|
|
|
showFullJSON,
|
2011-11-15 04:30:00 +00:00
|
|
|
showCustom,
|
|
|
|
showHeader,
|
|
|
|
showRaw,
|
2011-11-14 23:27:00 +00:00
|
|
|
|
2011-09-01 19:16:31 +00:00
|
|
|
setupConsole
|
|
|
|
) where
|
2010-11-08 19:15:21 +00:00
|
|
|
|
2011-09-02 20:44:04 +00:00
|
|
|
import Text.JSON
|
2010-11-08 19:15:21 +00:00
|
|
|
|
2011-10-04 02:24:57 +00:00
|
|
|
import Common
|
2010-11-08 19:15:21 +00:00
|
|
|
import Types
|
|
|
|
import qualified Annex
|
2011-09-01 19:16:31 +00:00
|
|
|
import qualified Messages.JSON as JSON
|
2010-11-08 19:15:21 +00:00
|
|
|
|
|
|
|
showStart :: String -> String -> Annex ()
|
2011-11-14 23:27:00 +00:00
|
|
|
showStart command file = handle (JSON.start command $ Just file) $
|
2011-10-11 18:43:45 +00:00
|
|
|
flushed $ putStr $ command ++ " " ++ file ++ " "
|
2010-11-08 19:15:21 +00:00
|
|
|
|
|
|
|
showNote :: String -> Annex ()
|
2011-10-11 18:43:45 +00:00
|
|
|
showNote s = handle (JSON.note s) $
|
|
|
|
flushed $ putStr $ "(" ++ s ++ ") "
|
2011-07-19 18:07:23 +00:00
|
|
|
|
|
|
|
showAction :: String -> Annex ()
|
|
|
|
showAction s = showNote $ s ++ "..."
|
2010-11-08 19:15:21 +00:00
|
|
|
|
|
|
|
showProgress :: Annex ()
|
2011-10-11 18:43:45 +00:00
|
|
|
showProgress = handle q $
|
|
|
|
flushed $ putStr "."
|
2011-07-19 18:07:23 +00:00
|
|
|
|
|
|
|
showSideAction :: String -> Annex ()
|
2011-10-11 18:43:45 +00:00
|
|
|
showSideAction s = handle q $
|
|
|
|
putStrLn $ "(" ++ s ++ "...)"
|
2011-07-19 18:07:23 +00:00
|
|
|
|
|
|
|
showOutput :: Annex ()
|
2011-10-11 18:43:45 +00:00
|
|
|
showOutput = handle q $
|
|
|
|
putStr "\n"
|
2010-11-08 19:15:21 +00:00
|
|
|
|
|
|
|
showLongNote :: String -> Annex ()
|
2011-10-11 18:43:45 +00:00
|
|
|
showLongNote s = handle (JSON.note s) $
|
|
|
|
putStrLn $ '\n' : indent s
|
2011-01-27 00:30:07 +00:00
|
|
|
|
2010-11-08 19:15:21 +00:00
|
|
|
showEndOk :: Annex ()
|
2011-09-01 19:16:31 +00:00
|
|
|
showEndOk = showEndResult True
|
2010-11-08 19:15:21 +00:00
|
|
|
|
|
|
|
showEndFail :: Annex ()
|
2011-09-01 19:16:31 +00:00
|
|
|
showEndFail = showEndResult False
|
2010-11-08 19:15:21 +00:00
|
|
|
|
2011-05-15 16:25:58 +00:00
|
|
|
showEndResult :: Bool -> Annex ()
|
2011-11-20 18:12:48 +00:00
|
|
|
showEndResult ok = handle (JSON.end ok) $ putStrLn msg
|
2011-09-01 19:16:31 +00:00
|
|
|
where
|
|
|
|
msg
|
2011-11-20 18:12:48 +00:00
|
|
|
| ok = "ok"
|
2011-09-01 19:16:31 +00:00
|
|
|
| otherwise = "failed"
|
2011-05-15 16:25:58 +00:00
|
|
|
|
2010-11-08 19:15:21 +00:00
|
|
|
showErr :: (Show a) => a -> Annex ()
|
2011-09-06 17:46:08 +00:00
|
|
|
showErr e = warning' $ "git-annex: " ++ show e
|
2010-11-08 19:15:21 +00:00
|
|
|
|
|
|
|
warning :: String -> Annex ()
|
2011-11-20 18:12:48 +00:00
|
|
|
warning = warning' . indent
|
2011-09-06 17:46:08 +00:00
|
|
|
|
|
|
|
warning' :: String -> Annex ()
|
|
|
|
warning' w = do
|
2011-09-01 19:16:31 +00:00
|
|
|
handle q $ putStr "\n"
|
2011-07-19 18:07:23 +00:00
|
|
|
liftIO $ do
|
|
|
|
hFlush stdout
|
2011-09-06 17:46:08 +00:00
|
|
|
hPutStrLn stderr w
|
2011-01-27 00:30:07 +00:00
|
|
|
|
|
|
|
indent :: String -> String
|
2011-11-20 18:12:48 +00:00
|
|
|
indent = join "\n" . map (\l -> " " ++ l) . lines
|
2011-02-10 18:21:44 +00:00
|
|
|
|
2011-12-23 02:03:18 +00:00
|
|
|
{- Shows a JSON fragment only when in json mode. -}
|
2011-11-15 04:30:00 +00:00
|
|
|
maybeShowJSON :: JSON a => [(String, a)] -> Annex ()
|
|
|
|
maybeShowJSON v = handle (JSON.add v) q
|
|
|
|
|
2011-12-23 02:03:18 +00:00
|
|
|
{- Shows a complete JSON value, only when in json mode. -}
|
|
|
|
showFullJSON :: JSON a => [(String, a)] -> Annex Bool
|
|
|
|
showFullJSON v = Annex.getState Annex.output >>= liftIO . go
|
|
|
|
where
|
|
|
|
go Annex.JSONOutput = JSON.complete v >> return True
|
|
|
|
go _ = return False
|
|
|
|
|
2011-11-15 04:30:00 +00:00
|
|
|
{- Performs an action that outputs nonstandard/customized output, and
|
|
|
|
- in JSON mode wraps its output in JSON.start and JSON.end, so it's
|
|
|
|
- a complete JSON document.
|
|
|
|
- This is only needed when showStart and showEndOk is not used. -}
|
|
|
|
showCustom :: String -> Annex Bool -> Annex ()
|
|
|
|
showCustom command a = do
|
|
|
|
handle (JSON.start command Nothing) q
|
|
|
|
r <- a
|
|
|
|
handle (JSON.end r) q
|
|
|
|
|
|
|
|
showHeader :: String -> Annex ()
|
2011-11-20 18:12:48 +00:00
|
|
|
showHeader h = handle q $
|
|
|
|
flushed $ putStr $ h ++ ": "
|
2011-11-15 04:30:00 +00:00
|
|
|
|
|
|
|
showRaw :: String -> Annex ()
|
|
|
|
showRaw s = handle q $ putStrLn s
|
|
|
|
|
2011-03-12 19:30:17 +00:00
|
|
|
{- By default, haskell honors the user's locale in its output to stdout
|
|
|
|
- and stderr. While that's great for proper unicode support, for git-annex
|
|
|
|
- all that's really needed is the ability to display simple messages
|
|
|
|
- (currently untranslated), and importantly, to display filenames exactly
|
|
|
|
- as they are written on disk, no matter what their encoding. So, force
|
|
|
|
- raw mode.
|
|
|
|
-
|
|
|
|
- NB: Once git-annex gets localized, this will need a rethink. -}
|
|
|
|
setupConsole :: IO ()
|
|
|
|
setupConsole = do
|
|
|
|
hSetBinaryMode stdout True
|
|
|
|
hSetBinaryMode stderr True
|
2011-09-01 19:16:31 +00:00
|
|
|
|
|
|
|
handle :: IO () -> IO () -> Annex ()
|
2011-11-20 18:12:48 +00:00
|
|
|
handle json normal = Annex.getState Annex.output >>= go
|
|
|
|
where
|
|
|
|
go Annex.NormalOutput = liftIO normal
|
|
|
|
go Annex.QuietOutput = q
|
Flush json output, avoiding a buffering problem that could result in doubled output.
The bug was that with --json, output lines were sometimes doubled. For
example, git annex init --json would output two lines, despite only running
one thing. Adding to the weirdness, this only occurred when the output
was redirected to a pipe or a file.
Strace showed two processes outputting the same buffered output.
The second process was this writer process (only needed to work around
bug #624389):
_ <- forkProcess $ do
hPutStr toh $ unlines paths
hClose toh
exitSuccess
The doubled output occurs when this process exits, and ghc flushes the
inherited stdout buffer. Why only when piping? I don't know, but ghc may
be behaving differently when stdout is not a terminal.
While this is quite possibly a ghc bug, there is a nice fix in git-annex.
Explicitly flushing after each chunk of json is output works around the
problem, and as a side effect, json is streamed rather than being output
all at the end when performing an expensive operaition.
However, note that this means all uses of putStr in git-annex must be
explicitly flushed. The others were, already.
2011-11-25 15:46:07 +00:00
|
|
|
go Annex.JSONOutput = liftIO $ flushed $ json
|
2011-09-01 19:16:31 +00:00
|
|
|
|
|
|
|
q :: Monad m => m ()
|
|
|
|
q = return ()
|
2011-10-11 18:43:45 +00:00
|
|
|
|
|
|
|
flushed :: IO () -> IO ()
|
|
|
|
flushed a = a >> hFlush stdout
|