98 lines
2.9 KiB
Haskell
98 lines
2.9 KiB
Haskell
{- git-annex command
|
|
-
|
|
- Copyright 2010-2014 Joey Hess <id@joeyh.name>
|
|
-
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
-}
|
|
|
|
module Command.Whereis where
|
|
|
|
import Common.Annex
|
|
import Command
|
|
import Remote
|
|
import Logs.Trust
|
|
import Logs.Web
|
|
import Remote.Web (getWebUrls)
|
|
|
|
import qualified Data.Map as M
|
|
|
|
cmd :: Command
|
|
cmd = noCommit $ withGlobalOptions (jsonOption : annexedMatchingOptions) $
|
|
command "whereis" SectionQuery
|
|
"lists repositories that have file content"
|
|
paramPaths (seek <$$> optParser)
|
|
|
|
data WhereisOptions = WhereisOptions
|
|
{ whereisFiles :: CmdParams
|
|
, keyOptions :: Maybe KeyOptions
|
|
}
|
|
|
|
optParser :: CmdParamsDesc -> Parser WhereisOptions
|
|
optParser desc = WhereisOptions
|
|
<$> cmdParams desc
|
|
<*> optional (parseKeyOptions False)
|
|
|
|
seek :: WhereisOptions -> CommandSeek
|
|
seek o = do
|
|
m <- remoteMap id
|
|
withKeyOptions (keyOptions o) False
|
|
(startKeys m)
|
|
(withFilesInGit $ whenAnnexed $ start m)
|
|
(whereisFiles o)
|
|
|
|
start :: M.Map UUID Remote -> FilePath -> Key -> CommandStart
|
|
start remotemap file key = start' remotemap key (Just file)
|
|
|
|
startKeys :: M.Map UUID Remote -> Key -> CommandStart
|
|
startKeys remotemap key = start' remotemap key Nothing
|
|
|
|
start' :: M.Map UUID Remote -> Key -> AssociatedFile -> CommandStart
|
|
start' remotemap key afile = do
|
|
showStart' "whereis" key afile
|
|
next $ perform remotemap key
|
|
|
|
perform :: M.Map UUID Remote -> Key -> CommandPerform
|
|
perform remotemap key = do
|
|
locations <- keyLocations key
|
|
(untrustedlocations, safelocations) <- trustPartition UnTrusted locations
|
|
let num = length safelocations
|
|
showNote $ show num ++ " " ++ copiesplural num
|
|
pp <- prettyPrintUUIDs "whereis" safelocations
|
|
unless (null safelocations) $ showLongNote pp
|
|
pp' <- prettyPrintUUIDs "untrusted" untrustedlocations
|
|
unless (null untrustedlocations) $ showLongNote $ untrustedheader ++ pp'
|
|
|
|
-- Since other remotes than the web remote can set urls
|
|
-- where a key can be downloaded, get and show all such urls
|
|
-- as a special case.
|
|
showRemoteUrls "web" =<< getWebUrls key
|
|
forM_ (mapMaybe (`M.lookup` remotemap) locations) $
|
|
performRemoteUrls key
|
|
if null safelocations then stop else next $ return True
|
|
where
|
|
copiesplural 1 = "copy"
|
|
copiesplural _ = "copies"
|
|
untrustedheader = "The following untrusted locations may also have copies:\n"
|
|
|
|
performRemoteUrls :: Key -> Remote -> Annex ()
|
|
performRemoteUrls key remote = do
|
|
ls <- (++)
|
|
<$> askremote
|
|
<*> claimedurls
|
|
showRemoteUrls (name remote) ls
|
|
where
|
|
askremote = maybe (pure []) (flip id key) (whereisKey remote)
|
|
claimedurls = do
|
|
us <- map fst
|
|
. filter (\(_, d) -> d == OtherDownloader)
|
|
. map getDownloader
|
|
<$> getUrls key
|
|
filterM (\u -> (==) <$> pure remote <*> claimingUrl u) us
|
|
|
|
showRemoteUrls :: String -> [String] -> Annex ()
|
|
showRemoteUrls nm us
|
|
| null us = return ()
|
|
| otherwise = do
|
|
let ls = unlines $ map (\u -> nm ++ ": " ++ u) us
|
|
outputMessage noop ('\n' : indent ls ++ "\n")
|
|
maybeShowJSON [("urls", us)]
|