basic json support

This includes a generic JSONStream library built on top of Text.JSON
(somewhat hackishly).

It would be possible to stream out a single json document describing
all actions, but it's probably better for consumers if they can expect
one json document per line, so I did it that way instead.

Output from external programs used for transferring files is not
currently hidden when outputting json, which probably makes it not very
useful there. This may be dealt with if there is demand for json
output for --get or --move to be parsable.

The version, status, and find subcommands have hand-crafted output and
don't do json. The whereis subcommand needs to be modified to produce
useful json.
This commit is contained in:
Joey Hess 2011-09-01 15:16:31 -04:00
parent 57dd34c6be
commit 2f4d4d1c45
9 changed files with 123 additions and 22 deletions

View file

@ -64,7 +64,7 @@ data AnnexState = AnnexState
, cipher :: Maybe Cipher , cipher :: Maybe Cipher
} }
data OutputType = NormalOutput | QuietOutput data OutputType = NormalOutput | QuietOutput | JSONOutput
newState :: Git.Repo -> AnnexState newState :: Git.Repo -> AnnexState
newState gitrepo = AnnexState newState gitrepo = AnnexState

View file

@ -5,7 +5,22 @@
- Licensed under the GNU GPL version 3 or higher. - Licensed under the GNU GPL version 3 or higher.
-} -}
module Messages where module Messages (
showStart,
showNote,
showAction,
showProgress,
showSideAction,
showOutput,
showLongNote,
showEndOk,
showEndFail,
showEndResult,
showErr,
warning,
indent,
setupConsole
) where
import Control.Monad.State (liftIO) import Control.Monad.State (liftIO)
import System.IO import System.IO
@ -13,21 +28,15 @@ import Data.String.Utils
import Types import Types
import qualified Annex import qualified Annex
import qualified Messages.JSON as JSON
verbose :: Annex () -> Annex ()
verbose a = do
output <- Annex.getState Annex.output
case output of
Annex.NormalOutput -> a
_ -> return ()
showStart :: String -> String -> Annex () showStart :: String -> String -> Annex ()
showStart command file = verbose $ liftIO $ do showStart command file = handle (JSON.start command file) $ do
putStr $ command ++ " " ++ file ++ " " putStr $ command ++ " " ++ file ++ " "
hFlush stdout hFlush stdout
showNote :: String -> Annex () showNote :: String -> Annex ()
showNote s = verbose $ liftIO $ do showNote s = handle (JSON.note s) $ do
putStr $ "(" ++ s ++ ") " putStr $ "(" ++ s ++ ") "
hFlush stdout hFlush stdout
@ -35,28 +44,31 @@ showAction :: String -> Annex ()
showAction s = showNote $ s ++ "..." showAction s = showNote $ s ++ "..."
showProgress :: Annex () showProgress :: Annex ()
showProgress = verbose $ liftIO $ do showProgress = handle q $ do
putStr "." putStr "."
hFlush stdout hFlush stdout
showSideAction :: String -> Annex () showSideAction :: String -> Annex ()
showSideAction s = verbose $ liftIO $ putStrLn $ "(" ++ s ++ "...)" showSideAction s = handle q $ putStrLn $ "(" ++ s ++ "...)"
showOutput :: Annex () showOutput :: Annex ()
showOutput = verbose $ liftIO $ putStr "\n" showOutput = handle q $ putStr "\n"
showLongNote :: String -> Annex () showLongNote :: String -> Annex ()
showLongNote s = verbose $ liftIO $ putStr $ '\n' : indent s showLongNote s = handle (JSON.note s) $ putStr $ '\n' : indent s
showEndOk :: Annex () showEndOk :: Annex ()
showEndOk = verbose $ liftIO $ putStrLn "ok" showEndOk = showEndResult True
showEndFail :: Annex () showEndFail :: Annex ()
showEndFail = verbose $ liftIO $ putStrLn "failed" showEndFail = showEndResult False
showEndResult :: Bool -> Annex () showEndResult :: Bool -> Annex ()
showEndResult True = showEndOk showEndResult b = handle (JSON.end b) $ putStrLn msg
showEndResult False = showEndFail where
msg
| b = "ok"
| otherwise = "failed"
showErr :: (Show a) => a -> Annex () showErr :: (Show a) => a -> Annex ()
showErr e = liftIO $ do showErr e = liftIO $ do
@ -65,7 +77,7 @@ showErr e = liftIO $ do
warning :: String -> Annex () warning :: String -> Annex ()
warning w = do warning w = do
verbose $ liftIO $ putStr "\n" handle q $ putStr "\n"
liftIO $ do liftIO $ do
hFlush stdout hFlush stdout
hPutStrLn stderr $ indent w hPutStrLn stderr $ indent w
@ -85,3 +97,14 @@ setupConsole :: IO ()
setupConsole = do setupConsole = do
hSetBinaryMode stdout True hSetBinaryMode stdout True
hSetBinaryMode stderr True hSetBinaryMode stderr True
handle :: IO () -> IO () -> Annex ()
handle json normal = do
output <- Annex.getState Annex.output
case output of
Annex.NormalOutput -> liftIO normal
Annex.QuietOutput -> q
Annex.JSONOutput -> liftIO json
q :: Monad m => m ()
q = return ()

23
Messages/JSON.hs Normal file
View file

@ -0,0 +1,23 @@
{- git-annex JSON output
-
- Copyright 2011 Joey Hess <joey@kitenet.net>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Messages.JSON (
start,
end,
note
) where
import qualified Utility.JSONStream as Stream
start :: String -> String -> IO ()
start command file = putStr $ Stream.start [("command", command), ("file", file)]
end :: Bool -> IO ()
end b = putStr $ Stream.add [("success", b)] ++ Stream.end
note :: String -> IO ()
note s = putStr $ Stream.add [("note", s)]

View file

@ -30,6 +30,8 @@ commonOptions =
"avoid verbose output" "avoid verbose output"
, Option ['v'] ["verbose"] (NoArg (setoutput Annex.NormalOutput)) , Option ['v'] ["verbose"] (NoArg (setoutput Annex.NormalOutput))
"allow verbose output (default)" "allow verbose output (default)"
, Option ['j'] ["json"] (NoArg (setoutput Annex.JSONOutput))
"enable JSON output"
, Option ['d'] ["debug"] (NoArg (setdebug)) , Option ['d'] ["debug"] (NoArg (setdebug))
"show debug messages" "show debug messages"
, Option ['b'] ["backend"] (ReqArg setforcebackend paramName) , Option ['b'] ["backend"] (ReqArg setforcebackend paramName)

44
Utility/JSONStream.hs Normal file
View file

@ -0,0 +1,44 @@
{- Streaming JSON output.
-
- Copyright 2011 Joey Hess <joey@kitenet.net>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Utility.JSONStream (
start,
add,
end
) where
import Text.JSON
{- Text.JSON does not support building up a larger JSON document piece by
piece as a stream. To support streaming, a hack. The JSObject is converted
to a string with its final "}" is left off, allowing it to be added to
later. -}
start :: JSON a => [(String, a)] -> String
start l
| last s == endchar = take (length s - 1) s
| otherwise = bad s
where
s = encodeStrict $ toJSObject l
add :: JSON a => [(String, a)] -> String
add l
| head s == startchar = ',' : drop 1 s
| otherwise = bad s
where
s = start l
end :: String
end = [endchar, '\n']
startchar :: Char
startchar = '{'
endchar :: Char
endchar = '}'
bad :: String -> a
bad s = error $ "Text.JSON returned unexpected string: " ++ s

1
debian/changelog vendored
View file

@ -6,6 +6,7 @@ git-annex (3.20110820) UNRELEASED; urgency=low
* init: Make description an optional parameter. * init: Make description an optional parameter.
* unused, status: Sped up by avoiding unnecessary stats of annexed files. * unused, status: Sped up by avoiding unnecessary stats of annexed files.
* unused --remote: Reduced memory use to 1/4th what was used before. * unused --remote: Reduced memory use to 1/4th what was used before.
* Add --json switch, to produce machine-consumable output.
-- Joey Hess <joeyh@debian.org> Tue, 23 Aug 2011 13:41:01 -0400 -- Joey Hess <joeyh@debian.org> Tue, 23 Aug 2011 13:41:01 -0400

1
debian/control vendored
View file

@ -14,6 +14,7 @@ Build-Depends:
libghc-hs3-dev (>= 0.5.6), libghc-hs3-dev (>= 0.5.6),
libghc-testpack-dev [any-i386 any-amd64], libghc-testpack-dev [any-i386 any-amd64],
libghc-monad-control-dev, libghc-monad-control-dev,
libghc-json-dev,
ikiwiki, ikiwiki,
perlmagick, perlmagick,
git | git-core, git | git-core,

View file

@ -337,12 +337,18 @@ Many git-annex commands will stage changes for later `git commit` by you.
* --quiet * --quiet
Avoid the default verbose logging of what is done; only show errors Avoid the default verbose display of what is done; only show errors
and progress displays. and progress displays.
* --verbose * --verbose
Enable verbose logging. Enable verbose display.
* --json
Rather than the normal output, generate JSON. This is intended to be
parsed by programs that use git-annex. Each line of output is a JSON
object.
* --debug * --debug

View file

@ -28,6 +28,7 @@ To build and use git-annex, you will need:
* [QuickCheck 2](http://hackage.haskell.org/package/QuickCheck) * [QuickCheck 2](http://hackage.haskell.org/package/QuickCheck)
* [HTTP](http://hackage.haskell.org/package/HTTP) * [HTTP](http://hackage.haskell.org/package/HTTP)
* [hS3](http://hackage.haskell.org/package/hS3) (optional, but recommended) * [hS3](http://hackage.haskell.org/package/hS3) (optional, but recommended)
* [json](http://hackage.haskell.org/package/json)
* Shell commands * Shell commands
* [git](http://git-scm.com/) * [git](http://git-scm.com/)
* [uuid](http://www.ossp.org/pkg/lib/uuid/) * [uuid](http://www.ossp.org/pkg/lib/uuid/)