2015-07-08 21:59:06 +00:00
|
|
|
{- git-annex command-line option parsing
|
2013-03-27 17:51:24 +00:00
|
|
|
-
|
2018-01-09 19:36:56 +00:00
|
|
|
- Copyright 2010-2018 Joey Hess <id@joeyh.name>
|
2013-03-27 17:51:24 +00:00
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
2014-01-26 20:25:55 +00:00
|
|
|
module CmdLine.GitAnnex.Options where
|
2013-03-27 17:51:24 +00:00
|
|
|
|
2015-07-08 21:59:06 +00:00
|
|
|
import Options.Applicative
|
2015-09-14 17:19:04 +00:00
|
|
|
import Options.Applicative.Builder.Internal
|
2015-11-05 23:57:41 +00:00
|
|
|
import Control.Concurrent
|
2018-01-09 19:36:56 +00:00
|
|
|
import qualified Data.Map as M
|
2013-03-27 17:51:24 +00:00
|
|
|
|
2016-01-20 20:36:33 +00:00
|
|
|
import Annex.Common
|
2013-03-27 17:51:24 +00:00
|
|
|
import qualified Git.Config
|
2015-09-14 17:19:04 +00:00
|
|
|
import qualified Git.Construct
|
2018-01-09 19:36:56 +00:00
|
|
|
import Git.Remote
|
2013-11-05 17:38:37 +00:00
|
|
|
import Git.Types
|
2017-02-24 19:16:56 +00:00
|
|
|
import Types.Key
|
2013-03-27 17:51:24 +00:00
|
|
|
import Types.TrustLevel
|
2014-01-21 20:08:19 +00:00
|
|
|
import Types.NumCopies
|
2014-01-18 15:54:43 +00:00
|
|
|
import Types.Messages
|
2015-07-08 21:59:06 +00:00
|
|
|
import Types.Command
|
2015-07-09 20:20:30 +00:00
|
|
|
import Types.DeferredParse
|
2015-07-10 04:55:53 +00:00
|
|
|
import Types.DesktopNotify
|
2016-09-09 16:57:42 +00:00
|
|
|
import Types.Concurrency
|
2013-03-27 17:51:24 +00:00
|
|
|
import qualified Annex
|
|
|
|
import qualified Remote
|
|
|
|
import qualified Limit
|
2013-10-28 18:50:17 +00:00
|
|
|
import qualified Limit.Wanted
|
2014-01-26 20:25:55 +00:00
|
|
|
import CmdLine.Option
|
|
|
|
import CmdLine.Usage
|
2015-07-10 06:18:08 +00:00
|
|
|
import CmdLine.GlobalSetter
|
2015-09-14 17:19:04 +00:00
|
|
|
import qualified Backend
|
|
|
|
import qualified Types.Backend as Backend
|
2013-03-27 17:51:24 +00:00
|
|
|
|
2015-07-10 04:55:53 +00:00
|
|
|
-- Global options that are accepted by all git-annex sub-commands,
|
2015-02-06 21:08:14 +00:00
|
|
|
-- although not always used.
|
2015-07-10 17:18:46 +00:00
|
|
|
gitAnnexGlobalOptions :: [GlobalOption]
|
2015-07-10 06:03:03 +00:00
|
|
|
gitAnnexGlobalOptions = commonGlobalOptions ++
|
|
|
|
[ globalSetter setnumcopies $ option auto
|
2015-07-10 04:55:53 +00:00
|
|
|
( long "numcopies" <> short 'N' <> metavar paramNumber
|
|
|
|
<> help "override default number of copies"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
|
|
|
, globalSetter (Remote.forceTrust Trusted) $ strOption
|
|
|
|
( long "trust" <> metavar paramRemote
|
|
|
|
<> help "override trust setting"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-09-14 17:19:04 +00:00
|
|
|
<> completeRemotes
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
|
|
|
, globalSetter (Remote.forceTrust SemiTrusted) $ strOption
|
|
|
|
( long "semitrust" <> metavar paramRemote
|
|
|
|
<> help "override trust setting back to default"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-09-14 17:19:04 +00:00
|
|
|
<> completeRemotes
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
|
|
|
, globalSetter (Remote.forceTrust UnTrusted) $ strOption
|
|
|
|
( long "untrust" <> metavar paramRemote
|
|
|
|
<> help "override trust setting to untrusted"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-09-14 17:19:04 +00:00
|
|
|
<> completeRemotes
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
|
|
|
, globalSetter setgitconfig $ strOption
|
|
|
|
( long "config" <> short 'c' <> metavar "NAME=VALUE"
|
|
|
|
<> help "override git configuration setting"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
|
|
|
, globalSetter setuseragent $ strOption
|
|
|
|
( long "user-agent" <> metavar paramName
|
|
|
|
<> help "override default User-Agent"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
|
|
|
, globalFlag (Annex.setFlag "trustglacier")
|
|
|
|
( long "trust-glacier"
|
|
|
|
<> help "Trust Amazon Glacier inventory"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
|
|
|
, globalFlag (setdesktopnotify mkNotifyFinish)
|
|
|
|
( long "notify-finish"
|
|
|
|
<> help "show desktop notification after transfer finishes"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
|
|
|
, globalFlag (setdesktopnotify mkNotifyStart)
|
|
|
|
( long "notify-start"
|
2015-07-13 22:15:24 +00:00
|
|
|
<> help "show desktop notification after transfer starts"
|
2015-07-10 06:18:08 +00:00
|
|
|
<> hidden
|
2015-07-10 04:55:53 +00:00
|
|
|
)
|
2015-02-06 21:08:14 +00:00
|
|
|
]
|
2013-03-27 17:51:24 +00:00
|
|
|
where
|
2015-07-10 04:55:53 +00:00
|
|
|
setnumcopies n = Annex.changeState $ \s -> s { Annex.forcenumcopies = Just $ NumCopies n }
|
2013-09-28 18:35:21 +00:00
|
|
|
setuseragent v = Annex.changeState $ \s -> s { Annex.useragent = Just v }
|
2016-01-22 17:47:41 +00:00
|
|
|
setgitconfig v = Annex.adjustGitRepo $ \r -> Git.Config.store v $
|
|
|
|
r { gitGlobalOpts = gitGlobalOpts r ++ [Param "-c", Param v] }
|
2015-07-10 04:55:53 +00:00
|
|
|
setdesktopnotify v = Annex.changeState $ \s -> s { Annex.desktopnotify = Annex.desktopnotify s <> v }
|
2013-07-03 17:02:42 +00:00
|
|
|
|
2015-07-10 17:18:46 +00:00
|
|
|
{- Parser that accepts all non-option params. -}
|
|
|
|
cmdParams :: CmdParamsDesc -> Parser CmdParams
|
|
|
|
cmdParams paramdesc = many $ argument str
|
|
|
|
( metavar paramdesc
|
2017-06-09 15:38:20 +00:00
|
|
|
<> action "file"
|
2015-07-10 17:18:46 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
parseAutoOption :: Parser Bool
|
|
|
|
parseAutoOption = switch
|
|
|
|
( long "auto" <> short 'a'
|
|
|
|
<> help "automatic mode"
|
|
|
|
)
|
|
|
|
|
2017-05-31 20:20:55 +00:00
|
|
|
parseRemoteOption :: RemoteName -> DeferredParse Remote
|
|
|
|
parseRemoteOption = DeferredParse
|
2015-09-14 17:19:04 +00:00
|
|
|
. (fromJust <$$> Remote.byNameWithUUID)
|
2017-05-31 20:20:55 +00:00
|
|
|
. Just
|
2015-07-09 19:23:14 +00:00
|
|
|
|
|
|
|
data FromToOptions
|
|
|
|
= FromRemote (DeferredParse Remote)
|
|
|
|
| ToRemote (DeferredParse Remote)
|
|
|
|
|
|
|
|
instance DeferredParseClass FromToOptions where
|
|
|
|
finishParse (FromRemote v) = FromRemote <$> finishParse v
|
|
|
|
finishParse (ToRemote v) = ToRemote <$> finishParse v
|
|
|
|
|
|
|
|
parseFromToOptions :: Parser FromToOptions
|
|
|
|
parseFromToOptions =
|
2017-05-31 20:20:55 +00:00
|
|
|
(FromRemote . parseRemoteOption <$> parseFromOption)
|
|
|
|
<|> (ToRemote . parseRemoteOption <$> parseToOption)
|
2015-07-09 19:23:14 +00:00
|
|
|
|
2017-05-31 20:20:55 +00:00
|
|
|
parseFromOption :: Parser RemoteName
|
|
|
|
parseFromOption = strOption
|
2015-07-09 19:23:14 +00:00
|
|
|
( long "from" <> short 'f' <> metavar paramRemote
|
|
|
|
<> help "source remote"
|
2015-09-14 17:19:04 +00:00
|
|
|
<> completeRemotes
|
2015-07-09 19:23:14 +00:00
|
|
|
)
|
|
|
|
|
2017-05-31 20:20:55 +00:00
|
|
|
parseToOption :: Parser RemoteName
|
|
|
|
parseToOption = strOption
|
2015-07-09 19:23:14 +00:00
|
|
|
( long "to" <> short 't' <> metavar paramRemote
|
|
|
|
<> help "destination remote"
|
2015-09-14 17:19:04 +00:00
|
|
|
<> completeRemotes
|
2015-07-09 19:23:14 +00:00
|
|
|
)
|
|
|
|
|
2015-07-08 21:59:06 +00:00
|
|
|
-- Options for acting on keys, rather than work tree files.
|
2015-07-09 16:44:03 +00:00
|
|
|
data KeyOptions
|
|
|
|
= WantAllKeys
|
|
|
|
| WantUnusedKeys
|
2016-08-03 16:37:12 +00:00
|
|
|
| WantFailedTransfers
|
2015-07-09 16:44:03 +00:00
|
|
|
| WantSpecificKey Key
|
|
|
|
| WantIncompleteKeys
|
--branch, stage 1
Added --branch option to copy, drop, fsck, get, metadata, mirror, move, and
whereis commands. This option makes git-annex operate on files that are
included in a specified branch (or other treeish).
The names of the files from the branch that are being operated on are not
displayed yet; only the keys. Displaying the filenames will need changes
to every affected command.
Also, note that --branch can be specified repeatedly. This is not really
documented, but seemed worth supporting, especially since we may later want
the ability to operate on all branches matching a refspec. However, when
operating on two branches that contain the same key, that key will be
operated on twice.
2016-07-20 16:05:22 +00:00
|
|
|
| WantBranchKeys [Branch]
|
2015-07-08 21:59:06 +00:00
|
|
|
|
2016-08-03 16:37:12 +00:00
|
|
|
parseKeyOptions :: Parser KeyOptions
|
|
|
|
parseKeyOptions = parseAllOption
|
|
|
|
<|> WantBranchKeys <$> some (option (str >>= pure . Ref)
|
|
|
|
( long "branch" <> metavar paramRef
|
|
|
|
<> help "operate on files in the specified branch or treeish"
|
|
|
|
))
|
|
|
|
<|> flag' WantUnusedKeys
|
|
|
|
( long "unused" <> short 'U'
|
|
|
|
<> help "operate on files found by last run of git-annex unused"
|
|
|
|
)
|
|
|
|
<|> (WantSpecificKey <$> option (str >>= parseKey)
|
|
|
|
( long "key" <> metavar paramKey
|
|
|
|
<> help "operate on specified key"
|
|
|
|
))
|
|
|
|
|
|
|
|
parseFailedTransfersOption :: Parser KeyOptions
|
|
|
|
parseFailedTransfersOption = flag' WantFailedTransfers
|
|
|
|
( long "failed"
|
|
|
|
<> help "operate on files that recently failed to be transferred"
|
|
|
|
)
|
|
|
|
|
|
|
|
parseIncompleteOption :: Parser KeyOptions
|
|
|
|
parseIncompleteOption = flag' WantIncompleteKeys
|
|
|
|
( long "incomplete"
|
|
|
|
<> help "resume previous downloads"
|
|
|
|
)
|
2015-07-08 21:59:06 +00:00
|
|
|
|
2015-07-09 23:03:21 +00:00
|
|
|
parseAllOption :: Parser KeyOptions
|
|
|
|
parseAllOption = flag' WantAllKeys
|
|
|
|
( long "all" <> short 'A'
|
|
|
|
<> help "operate on all versions of all files"
|
|
|
|
)
|
|
|
|
|
2015-07-08 21:59:06 +00:00
|
|
|
parseKey :: Monad m => String -> m Key
|
|
|
|
parseKey = maybe (fail "invalid key") return . file2key
|
|
|
|
|
2015-02-06 21:08:14 +00:00
|
|
|
-- Options to match properties of annexed files.
|
2015-07-10 17:18:46 +00:00
|
|
|
annexedMatchingOptions :: [GlobalOption]
|
2015-02-06 21:08:14 +00:00
|
|
|
annexedMatchingOptions = concat
|
|
|
|
[ nonWorkTreeMatchingOptions'
|
|
|
|
, fileMatchingOptions'
|
2015-07-10 17:18:46 +00:00
|
|
|
, combiningOptions
|
|
|
|
, [timeLimitOption]
|
2015-02-06 21:08:14 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
-- Matching options that don't need to examine work tree files.
|
2015-07-10 17:18:46 +00:00
|
|
|
nonWorkTreeMatchingOptions :: [GlobalOption]
|
|
|
|
nonWorkTreeMatchingOptions = nonWorkTreeMatchingOptions' ++ combiningOptions
|
2015-02-06 21:08:14 +00:00
|
|
|
|
2015-07-10 17:18:46 +00:00
|
|
|
nonWorkTreeMatchingOptions' :: [GlobalOption]
|
2015-02-06 21:08:14 +00:00
|
|
|
nonWorkTreeMatchingOptions' =
|
2015-07-10 17:18:46 +00:00
|
|
|
[ globalSetter Limit.addIn $ strOption
|
|
|
|
( long "in" <> short 'i' <> metavar paramRemote
|
|
|
|
<> help "match files present in a remote"
|
|
|
|
<> hidden
|
2015-09-14 17:19:04 +00:00
|
|
|
<> completeRemotes
|
2015-07-10 17:18:46 +00:00
|
|
|
)
|
|
|
|
, globalSetter Limit.addCopies $ strOption
|
|
|
|
( long "copies" <> short 'C' <> metavar paramRemote
|
|
|
|
<> help "skip files with fewer copies"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalSetter (Limit.addLackingCopies False) $ strOption
|
|
|
|
( long "lackingcopies" <> metavar paramNumber
|
|
|
|
<> help "match files that need more copies"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalSetter (Limit.addLackingCopies True) $ strOption
|
|
|
|
( long "approxlackingcopies" <> metavar paramNumber
|
|
|
|
<> help "match files that need more copies (faster)"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalSetter Limit.addInBackend $ strOption
|
|
|
|
( long "inbackend" <> short 'B' <> metavar paramName
|
|
|
|
<> help "match files using a key-value backend"
|
|
|
|
<> hidden
|
2015-09-14 17:19:04 +00:00
|
|
|
<> completeBackends
|
2015-07-10 17:18:46 +00:00
|
|
|
)
|
2017-02-27 19:02:38 +00:00
|
|
|
, globalFlag Limit.addSecureHash
|
|
|
|
( long "securehash"
|
|
|
|
<> help "match files using a cryptographically secure hash"
|
|
|
|
<> hidden
|
|
|
|
)
|
2015-07-10 17:18:46 +00:00
|
|
|
, globalSetter Limit.addInAllGroup $ strOption
|
|
|
|
( long "inallgroup" <> metavar paramGroup
|
|
|
|
<> help "match files present in all remotes in a group"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalSetter Limit.addMetaData $ strOption
|
|
|
|
( long "metadata" <> metavar "FIELD=VALUE"
|
|
|
|
<> help "match files with attached metadata"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalFlag Limit.Wanted.addWantGet
|
|
|
|
( long "want-get"
|
|
|
|
<> help "match files the repository wants to get"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalFlag Limit.Wanted.addWantDrop
|
|
|
|
( long "want-drop"
|
|
|
|
<> help "match files the repository wants to drop"
|
|
|
|
<> hidden
|
|
|
|
)
|
2015-02-06 21:08:14 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
-- Options to match files which may not yet be annexed.
|
2015-07-10 17:18:46 +00:00
|
|
|
fileMatchingOptions :: [GlobalOption]
|
|
|
|
fileMatchingOptions = fileMatchingOptions' ++ combiningOptions
|
2015-02-06 21:08:14 +00:00
|
|
|
|
2015-07-10 17:18:46 +00:00
|
|
|
fileMatchingOptions' :: [GlobalOption]
|
2015-02-06 21:08:14 +00:00
|
|
|
fileMatchingOptions' =
|
2015-07-10 17:18:46 +00:00
|
|
|
[ globalSetter Limit.addExclude $ strOption
|
|
|
|
( long "exclude" <> short 'x' <> metavar paramGlob
|
|
|
|
<> help "skip files matching the glob pattern"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalSetter Limit.addInclude $ strOption
|
|
|
|
( long "include" <> short 'I' <> metavar paramGlob
|
|
|
|
<> help "limit to files matching the glob pattern"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalSetter Limit.addLargerThan $ strOption
|
|
|
|
( long "largerthan" <> metavar paramSize
|
|
|
|
<> help "match files larger than a size"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
, globalSetter Limit.addSmallerThan $ strOption
|
|
|
|
( long "smallerthan" <> metavar paramSize
|
|
|
|
<> help "match files smaller than a size"
|
|
|
|
<> hidden
|
|
|
|
)
|
2015-02-06 21:08:14 +00:00
|
|
|
]
|
|
|
|
|
2015-07-10 17:18:46 +00:00
|
|
|
combiningOptions :: [GlobalOption]
|
2015-07-10 16:47:35 +00:00
|
|
|
combiningOptions =
|
2015-07-10 17:18:46 +00:00
|
|
|
[ longopt "not" "negate next option"
|
|
|
|
, longopt "and" "both previous and next option must match"
|
|
|
|
, longopt "or" "either previous or next option must match"
|
|
|
|
, shortopt '(' "open group of options"
|
|
|
|
, shortopt ')' "close group of options"
|
|
|
|
]
|
2015-04-10 21:08:07 +00:00
|
|
|
where
|
2015-07-10 17:18:46 +00:00
|
|
|
longopt o h = globalFlag (Limit.addToken o) ( long o <> help h <> hidden )
|
|
|
|
shortopt o h = globalFlag (Limit.addToken [o]) ( short o <> help h <> hidden )
|
2015-07-09 23:03:21 +00:00
|
|
|
|
2015-07-10 17:18:46 +00:00
|
|
|
jsonOption :: GlobalOption
|
2016-09-09 19:06:54 +00:00
|
|
|
jsonOption = globalFlag (Annex.setOutput (JSONOutput False))
|
2015-07-09 23:03:21 +00:00
|
|
|
( long "json" <> short 'j'
|
|
|
|
<> help "enable JSON output"
|
2015-07-10 17:18:46 +00:00
|
|
|
<> hidden
|
2015-07-09 23:03:21 +00:00
|
|
|
)
|
2015-03-25 21:06:14 +00:00
|
|
|
|
2016-09-09 19:06:54 +00:00
|
|
|
jsonProgressOption :: GlobalOption
|
|
|
|
jsonProgressOption = globalFlag (Annex.setOutput (JSONOutput True))
|
2017-01-30 16:46:42 +00:00
|
|
|
( long "json-progress"
|
2016-09-09 19:06:54 +00:00
|
|
|
<> help "include progress in JSON output"
|
|
|
|
<> hidden
|
|
|
|
)
|
|
|
|
|
2015-11-04 20:19:00 +00:00
|
|
|
-- Note that a command that adds this option should wrap its seek
|
|
|
|
-- action in `allowConcurrentOutput`.
|
2015-07-10 17:18:46 +00:00
|
|
|
jobsOption :: GlobalOption
|
2015-11-04 20:19:00 +00:00
|
|
|
jobsOption = globalSetter set $
|
2015-07-09 23:03:21 +00:00
|
|
|
option auto
|
|
|
|
( long "jobs" <> short 'J' <> metavar paramNumber
|
|
|
|
<> help "enable concurrent jobs"
|
2015-07-10 17:18:46 +00:00
|
|
|
<> hidden
|
2015-07-09 23:03:21 +00:00
|
|
|
)
|
2015-11-04 20:19:00 +00:00
|
|
|
where
|
2015-11-05 23:57:41 +00:00
|
|
|
set n = do
|
2016-09-09 16:57:42 +00:00
|
|
|
Annex.changeState $ \s -> s { Annex.concurrency = Concurrent n }
|
2015-11-05 23:57:41 +00:00
|
|
|
c <- liftIO getNumCapabilities
|
|
|
|
when (n > c) $
|
|
|
|
liftIO $ setNumCapabilities n
|
2015-07-09 23:03:21 +00:00
|
|
|
|
2015-07-10 17:18:46 +00:00
|
|
|
timeLimitOption :: GlobalOption
|
|
|
|
timeLimitOption = globalSetter Limit.addTimeLimit $ strOption
|
2015-07-09 23:03:21 +00:00
|
|
|
( long "time-limit" <> short 'T' <> metavar paramTime
|
|
|
|
<> help "stop after the specified amount of time"
|
2015-07-10 17:18:46 +00:00
|
|
|
<> hidden
|
2015-07-09 06:01:27 +00:00
|
|
|
)
|
2015-07-13 15:42:42 +00:00
|
|
|
|
|
|
|
data DaemonOptions = DaemonOptions
|
|
|
|
{ foregroundDaemonOption :: Bool
|
|
|
|
, stopDaemonOption :: Bool
|
|
|
|
}
|
|
|
|
|
|
|
|
parseDaemonOptions :: Parser DaemonOptions
|
|
|
|
parseDaemonOptions = DaemonOptions
|
|
|
|
<$> switch
|
|
|
|
( long "foreground"
|
|
|
|
<> help "do not daemonize"
|
|
|
|
)
|
|
|
|
<*> switch
|
|
|
|
( long "stop"
|
|
|
|
<> help "stop daemon"
|
|
|
|
)
|
2015-09-14 17:19:04 +00:00
|
|
|
completeRemotes :: HasCompleter f => Mod f a
|
|
|
|
completeRemotes = completer $ mkCompleter $ \input -> do
|
|
|
|
r <- maybe (pure Nothing) (Just <$$> Git.Config.read)
|
|
|
|
=<< Git.Construct.fromCwd
|
2018-01-09 19:36:56 +00:00
|
|
|
return $ filter (input `isPrefixOf`) $
|
|
|
|
map remoteKeyToRemoteName $
|
|
|
|
filter isRemoteKey $
|
|
|
|
maybe [] (M.keys . config) r
|
2015-09-14 17:19:04 +00:00
|
|
|
|
|
|
|
completeBackends :: HasCompleter f => Mod f a
|
2017-02-24 19:16:56 +00:00
|
|
|
completeBackends = completeWith $
|
|
|
|
map (formatKeyVariety . Backend.backendVariety) Backend.list
|