219 lines
6.5 KiB
Haskell
219 lines
6.5 KiB
Haskell
{-# LANGUAGE TemplateHaskell #-}
|
|
|
|
module Nix.Options.Parser where
|
|
|
|
import Control.Arrow ( second )
|
|
import Data.Char ( isDigit )
|
|
import Data.Maybe ( fromMaybe )
|
|
import Data.Text ( Text )
|
|
import qualified Data.Text as Text
|
|
import Data.Time
|
|
import Nix.Options
|
|
import Options.Applicative hiding ( ParserResult(..) )
|
|
import Data.Version ( showVersion )
|
|
import Development.GitRev ( gitCommitDate
|
|
, gitBranch
|
|
, gitHash )
|
|
import Paths_hnix ( version )
|
|
|
|
decodeVerbosity :: Int -> Verbosity
|
|
decodeVerbosity 0 = ErrorsOnly
|
|
decodeVerbosity 1 = Informational
|
|
decodeVerbosity 2 = Talkative
|
|
decodeVerbosity 3 = Chatty
|
|
decodeVerbosity 4 = DebugInfo
|
|
decodeVerbosity _ = Vomit
|
|
|
|
argPair :: Mod OptionFields (Text, Text) -> Parser (Text, Text)
|
|
argPair = option $ str >>= \s -> case Text.findIndex (== '=') s of
|
|
Nothing ->
|
|
errorWithoutStackTrace "Format of --arg/--argstr in hnix is: name=expr"
|
|
Just i -> pure $ second Text.tail $ Text.splitAt i s
|
|
|
|
nixOptions :: UTCTime -> Parser Options
|
|
nixOptions current =
|
|
Options
|
|
<$> (fromMaybe Informational <$>
|
|
optional
|
|
(option
|
|
|
|
(do
|
|
a <- str
|
|
if all isDigit a
|
|
then pure $ decodeVerbosity (read a)
|
|
else fail "Argument to -v/--verbose must be a number"
|
|
)
|
|
|
|
( short 'v'
|
|
<> long "verbose"
|
|
<> help "Verbose output"
|
|
)
|
|
|
|
)
|
|
)
|
|
<*> switch
|
|
( long "trace"
|
|
<> help "Enable tracing code (even more can be seen if built with --flags=tracing)"
|
|
)
|
|
<*> switch
|
|
( long "thunks"
|
|
<> help "Enable reporting of thunk tracing as well as regular evaluation"
|
|
)
|
|
<*> switch
|
|
( long "values"
|
|
<> help "Enable reporting of value provenance in error messages"
|
|
)
|
|
<*> switch
|
|
( long "scopes"
|
|
<> help "Enable reporting of scopes in evaluation traces"
|
|
)
|
|
<*> optional
|
|
(strOption
|
|
( long "reduce"
|
|
<> help "When done evaluating, output the evaluated part of the expression to FILE"
|
|
)
|
|
)
|
|
<*> switch
|
|
( long "reduce-sets"
|
|
<> help "Reduce set members that aren't used; breaks if hasAttr is used"
|
|
)
|
|
<*> switch
|
|
( long "reduce-lists"
|
|
<> help "Reduce list members that aren't used; breaks if elemAt is used"
|
|
)
|
|
<*> switch
|
|
( long "parse"
|
|
<> help "Whether to parse the file (also the default right now)"
|
|
)
|
|
<*> switch
|
|
( long "parse-only"
|
|
<> help "Whether to parse only, no pretty printing or checking"
|
|
)
|
|
<*> switch
|
|
( long "find"
|
|
<> help "If selected, find paths within attr trees"
|
|
)
|
|
<*> optional
|
|
(strOption
|
|
( long "find-file"
|
|
<> help "Look up the given files in Nix's search path"
|
|
)
|
|
)
|
|
<*> switch
|
|
( long "strict"
|
|
<> help "When used with --eval, recursively evaluate list elements and attributes"
|
|
)
|
|
<*> switch
|
|
( long "eval"
|
|
<> help "Whether to evaluate, or just pretty-print"
|
|
)
|
|
<*> switch
|
|
( long "json"
|
|
<> help "Print the resulting value as an JSON representation"
|
|
)
|
|
<*> switch
|
|
( long "xml"
|
|
<> help "Print the resulting value as an XML representation"
|
|
)
|
|
<*> optional
|
|
(strOption
|
|
( short 'A'
|
|
<> long "attr"
|
|
<> help "Select an attribute from the top-level Nix expression being evaluated"
|
|
)
|
|
)
|
|
<*> many
|
|
(strOption
|
|
( short 'I'
|
|
<> long "include"
|
|
<> help "Add a path to the Nix expression search path"
|
|
)
|
|
)
|
|
<*> switch
|
|
( long "check"
|
|
<> help "Whether to check for syntax errors after parsing"
|
|
)
|
|
<*> optional
|
|
(strOption
|
|
( long "read"
|
|
<> help "Read in an expression tree from a binary cache"
|
|
)
|
|
)
|
|
<*> switch
|
|
( long "cache"
|
|
<> help "Write out the parsed expression tree to a binary cache"
|
|
)
|
|
<*> switch
|
|
( long "repl"
|
|
<> help "After performing any indicated actions, enter the REPL"
|
|
)
|
|
<*> switch
|
|
( long "ignore-errors"
|
|
<> help "Continue parsing files, even if there are errors"
|
|
)
|
|
<*> optional
|
|
(strOption
|
|
( short 'E'
|
|
<> long "expr"
|
|
<> help "Expression to parse or evaluate")
|
|
)
|
|
<*> many
|
|
(argPair
|
|
( long "arg"
|
|
<> help "Argument to pass to an evaluated lambda")
|
|
)
|
|
<*> many
|
|
(argPair
|
|
( long "argstr"
|
|
<> help "Argument string to pass to an evaluated lambda"
|
|
)
|
|
)
|
|
<*> optional
|
|
(strOption
|
|
( short 'f'
|
|
<> long "file"
|
|
<> help "Parse all of the files given in FILE; - means stdin"
|
|
)
|
|
)
|
|
<*> option
|
|
(parseTimeOrError True defaultTimeLocale "%Y/%m/%d %H:%M:%S" <$> str)
|
|
( long "now"
|
|
<> value current
|
|
<> help "Set current time for testing purposes"
|
|
)
|
|
<*> many
|
|
(strArgument
|
|
( metavar "FILE"
|
|
<> help "Path of file to parse"
|
|
)
|
|
)
|
|
|
|
-- 2020-09-12: CLI --version option mechanism is tied to meta modules specificly generated by Cabal. It is possible to use Template Haskell to resolve the version, as also a g
|
|
versionOpt :: Parser (a -> a)
|
|
versionOpt = shortVersionOpt <*> debugVersionOpt
|
|
where
|
|
shortVersionOpt :: Parser (a -> a)
|
|
shortVersionOpt = infoOption
|
|
(showVersion version)
|
|
( long "version"
|
|
<> help "Show release version"
|
|
)
|
|
|
|
-- 2020-09-13: NOTE: Does not work for direct `nix-build`s, works for `nix-shell` `cabal` builds.
|
|
debugVersionOpt :: Parser (a -> a)
|
|
debugVersionOpt = infoOption
|
|
( concat
|
|
[ "Version: ", showVersion version
|
|
, "\nCommit: ", $(gitHash)
|
|
, "\n date: ", $(gitCommitDate)
|
|
, "\n branch: ", $(gitBranch)
|
|
]
|
|
)
|
|
( long "long-version"
|
|
<> help "Show long debug version form"
|
|
)
|
|
|
|
nixOptionsInfo :: UTCTime -> ParserInfo Options
|
|
nixOptionsInfo current = info (helper <*> versionOpt <*> nixOptions current)
|
|
(fullDesc <> progDesc "" <> header "hnix")
|