log: Added --all option.
This commit is contained in:
parent
154c939830
commit
c4d011bf3e
6 changed files with 171 additions and 63 deletions
|
@ -24,6 +24,7 @@ git-annex (6.20160614) UNRELEASED; urgency=medium
|
|||
* Fix a similar crash when the webapp is used to delete a repository.
|
||||
* Support checking presence of content at a http url that redirects to
|
||||
a ftp url.
|
||||
* log: Added --all option.
|
||||
|
||||
-- Joey Hess <id@joeyh.name> Mon, 13 Jun 2016 21:52:24 -0400
|
||||
|
||||
|
|
200
Command/Log.hs
200
Command/Log.hs
|
@ -1,6 +1,6 @@
|
|||
{- git-annex command
|
||||
-
|
||||
- Copyright 2012 Joey Hess <id@joeyh.name>
|
||||
- Copyright 2012, 2016 Joey Hess <id@joeyh.name>
|
||||
-
|
||||
- Licensed under the GNU GPL version 3 or higher.
|
||||
-}
|
||||
|
@ -11,7 +11,6 @@ module Command.Log where
|
|||
|
||||
import qualified Data.Set as S
|
||||
import qualified Data.Map as M
|
||||
import qualified Data.ByteString.Lazy.Char8 as L
|
||||
import Data.Char
|
||||
import Data.Time.Clock.POSIX
|
||||
import Data.Time
|
||||
|
@ -21,8 +20,7 @@ import System.Locale
|
|||
|
||||
import Command
|
||||
import Logs
|
||||
import qualified Logs.Presence
|
||||
import Annex.CatFile
|
||||
import Logs.Location
|
||||
import qualified Annex.Branch
|
||||
import qualified Git
|
||||
import Git.Command
|
||||
|
@ -33,9 +31,13 @@ data RefChange = RefChange
|
|||
{ changetime :: POSIXTime
|
||||
, oldref :: Git.Ref
|
||||
, newref :: Git.Ref
|
||||
, changekey :: Key
|
||||
}
|
||||
deriving (Show)
|
||||
|
||||
type Outputter = Bool -> POSIXTime -> [UUID] -> Annex ()
|
||||
data LogChange = Added | Removed
|
||||
|
||||
type Outputter = LogChange -> POSIXTime -> [UUID] -> Annex ()
|
||||
|
||||
cmd :: Command
|
||||
cmd = withGlobalOptions annexedMatchingOptions $
|
||||
|
@ -44,6 +46,7 @@ cmd = withGlobalOptions annexedMatchingOptions $
|
|||
|
||||
data LogOptions = LogOptions
|
||||
{ logFiles :: CmdParams
|
||||
, allOption :: Bool
|
||||
, rawDateOption :: Bool
|
||||
, gourceOption :: Bool
|
||||
, passthruOptions :: [CommandParam]
|
||||
|
@ -52,6 +55,11 @@ data LogOptions = LogOptions
|
|||
optParser :: CmdParamsDesc -> Parser LogOptions
|
||||
optParser desc = LogOptions
|
||||
<$> cmdParams desc
|
||||
<*> switch
|
||||
( long "all"
|
||||
<> short 'A'
|
||||
<> help "display location log changes to all files"
|
||||
)
|
||||
<*> switch
|
||||
( long "raw-date"
|
||||
<> help "display seconds from unix epoch"
|
||||
|
@ -81,71 +89,106 @@ seek :: LogOptions -> CommandSeek
|
|||
seek o = do
|
||||
m <- Remote.uuidDescriptions
|
||||
zone <- liftIO getCurrentTimeZone
|
||||
withFilesInGit (whenAnnexed $ start m zone o) (logFiles o)
|
||||
let outputter = mkOutputter m zone o
|
||||
case (logFiles o, allOption o) of
|
||||
(fs, False) -> withFilesInGit (whenAnnexed $ start o outputter) fs
|
||||
([], True) -> commandAction (startAll o outputter)
|
||||
(_, True) -> error "Cannot specify both files and --all"
|
||||
|
||||
start
|
||||
:: M.Map UUID String
|
||||
-> TimeZone
|
||||
-> LogOptions
|
||||
-> FilePath
|
||||
-> Key
|
||||
-> CommandStart
|
||||
start m zone o file key = do
|
||||
(ls, cleanup) <- getLog key (passthruOptions o)
|
||||
showLog output (readLog ls)
|
||||
start :: LogOptions -> (FilePath -> Outputter) -> FilePath -> Key -> CommandStart
|
||||
start o outputter file key = do
|
||||
(changes, cleanup) <- getKeyLog key (passthruOptions o)
|
||||
showLogIncremental (outputter file) changes
|
||||
void $ liftIO cleanup
|
||||
stop
|
||||
where
|
||||
output
|
||||
| rawDateOption o = normalOutput lookupdescription file show
|
||||
| gourceOption o = gourceOutput lookupdescription file
|
||||
| otherwise = normalOutput lookupdescription file (showTimeStamp zone)
|
||||
lookupdescription u = fromMaybe (fromUUID u) $ M.lookup u m
|
||||
|
||||
showLog :: Outputter -> [RefChange] -> Annex ()
|
||||
showLog outputter ps = do
|
||||
startAll :: LogOptions -> (String -> Outputter) -> CommandStart
|
||||
startAll o outputter = do
|
||||
(changes, cleanup) <- getAllLog (passthruOptions o)
|
||||
showLog outputter changes
|
||||
void $ liftIO cleanup
|
||||
stop
|
||||
|
||||
{- Displays changes made. Only works when all the RefChanges are for the
|
||||
- same key. The method is to compare each value with the value
|
||||
- after it in the list, which is the old version of the value.
|
||||
-
|
||||
- This ncessarily buffers the whole list, so does not stream.
|
||||
- But, the number of location log changes for a single key tends to be
|
||||
- fairly small.
|
||||
-
|
||||
- This minimizes the number of reads from git; each logged value is read
|
||||
- only once.
|
||||
-
|
||||
- This also generates subtly better output when the git-annex branch
|
||||
- got diverged.
|
||||
-}
|
||||
showLogIncremental :: Outputter -> [RefChange] -> Annex ()
|
||||
showLogIncremental outputter ps = do
|
||||
sets <- mapM (getset newref) ps
|
||||
previous <- maybe (return genesis) (getset oldref) (lastMaybe ps)
|
||||
sequence_ $ compareChanges outputter $ sets ++ [previous]
|
||||
let l = sets ++ [previous]
|
||||
let changes = map (\((t, new), (_, old)) -> (t, new, old))
|
||||
(zip l (drop 1 l))
|
||||
sequence_ $ compareChanges outputter changes
|
||||
where
|
||||
genesis = (0, S.empty)
|
||||
getset select change = do
|
||||
s <- S.fromList <$> get (select change)
|
||||
s <- S.fromList <$> loggedLocationsRef (select change)
|
||||
return (changetime change, s)
|
||||
get ref = map toUUID . Logs.Presence.getLog . L.unpack <$>
|
||||
catObject ref
|
||||
|
||||
{- Displays changes made. Streams, and can display changes affecting
|
||||
- different keys, but does twice as much reading of logged values
|
||||
- as showLogIncremental. -}
|
||||
showLog :: (String -> Outputter) -> [RefChange] -> Annex ()
|
||||
showLog outputter cs = forM_ cs $ \c -> do
|
||||
let keyname = key2file (changekey c)
|
||||
new <- S.fromList <$> loggedLocationsRef (newref c)
|
||||
old <- S.fromList <$> loggedLocationsRef (oldref c)
|
||||
sequence_ $ compareChanges (outputter keyname)
|
||||
[(changetime c, new, old)]
|
||||
|
||||
mkOutputter :: M.Map UUID String -> TimeZone -> LogOptions -> FilePath -> Outputter
|
||||
mkOutputter m zone o file
|
||||
| rawDateOption o = normalOutput lookupdescription file show
|
||||
| gourceOption o = gourceOutput lookupdescription file
|
||||
| otherwise = normalOutput lookupdescription file (showTimeStamp zone)
|
||||
where
|
||||
lookupdescription u = fromMaybe (fromUUID u) $ M.lookup u m
|
||||
|
||||
normalOutput :: (UUID -> String) -> FilePath -> (POSIXTime -> String) -> Outputter
|
||||
normalOutput lookupdescription file formattime present ts us =
|
||||
normalOutput lookupdescription file formattime logchange ts us =
|
||||
liftIO $ mapM_ (putStrLn . format) us
|
||||
where
|
||||
time = formattime ts
|
||||
addel = if present then "+" else "-"
|
||||
addel = case logchange of
|
||||
Added -> "+"
|
||||
Removed -> "-"
|
||||
format u = unwords [ addel, time, file, "|",
|
||||
fromUUID u ++ " -- " ++ lookupdescription u ]
|
||||
|
||||
gourceOutput :: (UUID -> String) -> FilePath -> Outputter
|
||||
gourceOutput lookupdescription file present ts us =
|
||||
gourceOutput lookupdescription file logchange ts us =
|
||||
liftIO $ mapM_ (putStrLn . intercalate "|" . format) us
|
||||
where
|
||||
time = takeWhile isDigit $ show ts
|
||||
addel = if present then "A" else "M"
|
||||
addel = case logchange of
|
||||
Added -> "A"
|
||||
Removed -> "M"
|
||||
format u = [ time, lookupdescription u, addel, file ]
|
||||
|
||||
{- Generates a display of the changes (which are ordered with newest first),
|
||||
- by comparing each change with the previous change.
|
||||
{- Generates a display of the changes.
|
||||
- Uses a formatter to generate a display of items that are added and
|
||||
- removed. -}
|
||||
compareChanges :: Ord a => (Bool -> POSIXTime -> [a] -> b) -> [(POSIXTime, S.Set a)] -> [b]
|
||||
compareChanges format changes = concatMap diff $ zip changes (drop 1 changes)
|
||||
compareChanges :: Ord a => (LogChange -> POSIXTime -> [a] -> b) -> [(POSIXTime, S.Set a, S.Set a)] -> [b]
|
||||
compareChanges format changes = concatMap diff changes
|
||||
where
|
||||
diff ((ts, new), (_, old)) =
|
||||
[format True ts added, format False ts removed]
|
||||
where
|
||||
added = S.toList $ S.difference new old
|
||||
removed = S.toList $ S.difference old new
|
||||
diff (ts, new, old) =
|
||||
[ format Added ts $ S.toList $ S.difference new old
|
||||
, format Removed ts $ S.toList $ S.difference old new
|
||||
]
|
||||
|
||||
{- Gets the git log for a given location log file.
|
||||
{- Streams the git log for a given key's location log file.
|
||||
-
|
||||
- This is complicated by git log using paths relative to the current
|
||||
- directory, even when looking at files in a different branch. A wacky
|
||||
|
@ -156,42 +199,75 @@ compareChanges format changes = concatMap diff $ zip changes (drop 1 changes)
|
|||
- once the location log file is gone avoids it checking all the way back
|
||||
- to commit 0 to see if it used to exist, so generally speeds things up a
|
||||
- *lot* for newish files. -}
|
||||
getLog :: Key -> [CommandParam] -> Annex ([String], IO Bool)
|
||||
getLog key os = do
|
||||
getKeyLog :: Key -> [CommandParam] -> Annex ([RefChange], IO Bool)
|
||||
getKeyLog key os = do
|
||||
top <- fromRepo Git.repoPath
|
||||
p <- liftIO $ relPathCwdToFile top
|
||||
config <- Annex.getGitConfig
|
||||
let logfile = p </> locationLogFile config key
|
||||
inRepo $ pipeNullSplit $
|
||||
getGitLog [logfile] (Param "--remove-empty" : os)
|
||||
|
||||
{- Streams the git log for all git-annex branch changes. -}
|
||||
getAllLog :: [CommandParam] -> Annex ([RefChange], IO Bool)
|
||||
getAllLog = getGitLog []
|
||||
|
||||
getGitLog :: [FilePath] -> [CommandParam] -> Annex ([RefChange], IO Bool)
|
||||
getGitLog fs os = do
|
||||
(ls, cleanup) <- inRepo $ pipeNullSplit $
|
||||
[ Param "log"
|
||||
, Param "-z"
|
||||
, Param "--pretty=format:%ct"
|
||||
, Param "--raw"
|
||||
, Param "--abbrev=40"
|
||||
, Param "--remove-empty"
|
||||
] ++ os ++
|
||||
[ Param $ Git.fromRef Annex.Branch.fullname
|
||||
, Param "--"
|
||||
, Param logfile
|
||||
]
|
||||
] ++ map Param fs
|
||||
return (parseGitRawLog ls, cleanup)
|
||||
|
||||
readLog :: [String] -> [RefChange]
|
||||
readLog = mapMaybe (parse . lines)
|
||||
-- Parses chunked git log --raw output, which looks something like:
|
||||
--
|
||||
-- [ "timestamp\n:changeline"
|
||||
-- , "logfile"
|
||||
-- , ""
|
||||
-- , "timestamp\n:changeline"
|
||||
-- , "logfile"
|
||||
-- , ":changeline"
|
||||
-- , "logfile"
|
||||
-- , ""
|
||||
-- ]
|
||||
--
|
||||
-- The timestamp is not included before all changelines, so
|
||||
-- keep track of the most recently seen timestamp.
|
||||
parseGitRawLog :: [String] -> [RefChange]
|
||||
parseGitRawLog = parse epoch
|
||||
where
|
||||
parse (ts:raw:[]) = let (old, new) = parseRaw raw in
|
||||
Just RefChange
|
||||
{ changetime = parseTimeStamp ts
|
||||
, oldref = old
|
||||
, newref = new
|
||||
}
|
||||
parse _ = Nothing
|
||||
epoch = toEnum 0 :: POSIXTime
|
||||
parse oldts ([]:rest) = parse oldts rest
|
||||
parse oldts (c1:c2:rest) = case mrc of
|
||||
Just rc -> rc : parse ts rest
|
||||
Nothing -> parse ts (c2:rest)
|
||||
where
|
||||
(ts, cl) = case separate (== '\n') c1 of
|
||||
(cl', []) -> (oldts, cl')
|
||||
(tss, cl') -> (parseTimeStamp tss, cl')
|
||||
mrc = do
|
||||
(old, new) <- parseRawChangeLine cl
|
||||
key <- locationLogFileKey c2
|
||||
return $ RefChange
|
||||
{ changetime = ts
|
||||
, oldref = old
|
||||
, newref = new
|
||||
, changekey = key
|
||||
}
|
||||
parse _ _ = []
|
||||
|
||||
-- Parses something like ":100644 100644 oldsha newsha M"
|
||||
parseRaw :: String -> (Git.Ref, Git.Ref)
|
||||
parseRaw l = go $ words l
|
||||
-- Parses something like "100644 100644 oldsha newsha M"
|
||||
parseRawChangeLine :: String -> Maybe (Git.Ref, Git.Ref)
|
||||
parseRawChangeLine = go . words
|
||||
where
|
||||
go (_:_:oldsha:newsha:_) = (Git.Ref oldsha, Git.Ref newsha)
|
||||
go _ = error $ "unable to parse git log output: " ++ l
|
||||
go (_:_:oldsha:newsha:_) = Just (Git.Ref oldsha, Git.Ref newsha)
|
||||
go _ = Nothing
|
||||
|
||||
parseTimeStamp :: String -> POSIXTime
|
||||
parseTimeStamp = utcTimeToPOSIXSeconds . fromMaybe (error "bad timestamp") .
|
||||
|
|
|
@ -19,6 +19,7 @@ module Logs.Location (
|
|||
logChange,
|
||||
loggedLocations,
|
||||
loggedLocationsHistorical,
|
||||
loggedLocationsRef,
|
||||
isKnownKey,
|
||||
checkDead,
|
||||
setDead,
|
||||
|
@ -31,10 +32,12 @@ import qualified Annex.Branch
|
|||
import Logs
|
||||
import Logs.Presence
|
||||
import Annex.UUID
|
||||
import Git.Types (RefDate)
|
||||
import Annex.CatFile
|
||||
import Git.Types (RefDate, Ref)
|
||||
import qualified Annex
|
||||
|
||||
import Data.Time.Clock
|
||||
import qualified Data.ByteString.Lazy.Char8 as L
|
||||
|
||||
{- Log a change in the presence of a key's value in current repository. -}
|
||||
logStatus :: Key -> LogStatus -> Annex ()
|
||||
|
@ -61,6 +64,10 @@ loggedLocations = getLoggedLocations currentLogInfo
|
|||
loggedLocationsHistorical :: RefDate -> Key -> Annex [UUID]
|
||||
loggedLocationsHistorical = getLoggedLocations . historicalLogInfo
|
||||
|
||||
{- Gets the locations contained in a git ref. -}
|
||||
loggedLocationsRef :: Ref -> Annex [UUID]
|
||||
loggedLocationsRef ref = map toUUID . getLog . L.unpack <$> catObject ref
|
||||
|
||||
getLoggedLocations :: (FilePath -> Annex [String]) -> Key -> Annex [UUID]
|
||||
getLoggedLocations getter key = do
|
||||
config <- Annex.getGitConfig
|
||||
|
|
|
@ -34,6 +34,12 @@ showing each repository they were added to ("+") and removed from ("-").
|
|||
The [[git-annex-matching-options]](1)
|
||||
can be used to specify files to act on.
|
||||
|
||||
* `--all`
|
||||
|
||||
Shows location log changes to all files, with the most recent changes first.
|
||||
In this mode, the names of files are not available and keys are displayed
|
||||
instead.
|
||||
|
||||
# SEE ALSO
|
||||
|
||||
[[git-annex]](1)
|
||||
|
|
|
@ -3,3 +3,5 @@ It would be great to have something to call in post-update-annex which would giv
|
|||
This could be `git annex log --all --max-count=1` or somesuch.
|
||||
|
||||
This capability could alternatively be provided with a new post-transfer hook, called for every file.
|
||||
|
||||
> [[done]] --[[Joey]]
|
||||
|
|
|
@ -0,0 +1,16 @@
|
|||
[[!comment format=mdwn
|
||||
username="joey"
|
||||
subject="""comment 2"""
|
||||
date="2016-07-17T19:09:41Z"
|
||||
content="""
|
||||
Implelemented `git annex log --all`. It turned out to fit really well
|
||||
to add the functionality there.
|
||||
|
||||
You can use --max-count, or even --since to limit the log
|
||||
that's displayed.
|
||||
|
||||
The output streams, so you could just remember the first line you saw when
|
||||
running it before, and close the pipe when the subsequent run outputs that
|
||||
line. (Although this method may skip over changes that got merged in from
|
||||
another repository.)
|
||||
"""]]
|
Loading…
Reference in a new issue