-
Notifications
You must be signed in to change notification settings - Fork 379
/
CommandLine.idr
546 lines (497 loc) · 20.9 KB
/
CommandLine.idr
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
module Idris.CommandLine
import Idris.Env
import Idris.Version
import Idris.Doc.Display
import Idris.Doc.String
import Idris.Pretty
import Core.Options
import Data.List
import Data.Maybe
import Data.String
import Data.Either
import System
%default total
public export
data PkgCommand
= Build
| Install
| InstallWithSrc
| MkDoc
| Typecheck
| Clean
| REPL
| Init
| DumpJson
| DumpInstallDir
export
Show PkgCommand where
show Build = "--build"
show Install = "--install"
show InstallWithSrc = "--install-with-src"
show MkDoc = "--mkdoc"
show Typecheck = "--typecheck"
show Clean = "--clean"
show REPL = "--repl"
show Init = "--init"
show DumpJson = "--dump-ipkg-json"
show DumpInstallDir = "--dump-installdir"
public export
data DirCommand
= LibDir | -- show top level package directory
||| Show the installation prefix
Prefix |
BlodwenPaths
export
Show DirCommand where
show LibDir = "--libdir"
show Prefix = "--prefix"
show BlodwenPaths = "--paths"
||| Help topics
public export
data HelpTopic
= ||| Interactive debugging topics
HelpLogging
| ||| The various pragmas
HelpPragma
recogniseHelpTopic : String -> Maybe HelpTopic
recogniseHelpTopic "logging" = pure HelpLogging
recogniseHelpTopic "pragma" = pure HelpPragma
recogniseHelpTopic _ = Nothing
||| CLOpt - possible command line options
public export
data CLOpt
=
||| Only typecheck the given file
CheckOnly |
||| The output file from the code generator
OutputFile String |
||| Execute a given function after checking the source file
ExecFn String |
||| Use a specific code generator
SetCG String |
||| Pass a directive to the code generator
Directive String |
||| Don't implicitly import Prelude
NoPrelude |
||| Set source directory
SourceDir String |
||| Set build directory
BuildDir String |
||| Set output directory
OutputDir String |
||| Generate profile data when compiling (backend dependent)
Profile |
||| Display Idris version
Version |
||| Display the TTC version currently used
TTCVersion |
||| Display help text
Help (Maybe HelpTopic) |
||| Suppress the banner
NoBanner |
||| Run Idris 2 in quiet mode
Quiet |
||| Show machine names when pretty printing
ShowMachineNames |
||| Show namespaces when pretty printing
ShowNamespaces |
||| Run Idris 2 in verbose mode (cancels quiet if it's the default)
Verbose |
||| Set the console width for REPL output
ConsoleWidth (Maybe Nat) |
||| Whether to use color in the console output
Color Bool |
||| Set the log level globally
Logging LogLevel |
||| Add a package as a dependency
PkgPath String |
||| List installed packages
ListPackages |
||| Build or install a given package, depending on PkgCommand
Package PkgCommand (Maybe String) |
||| Show locations of data/library directories
Directory DirCommand |
||| The input Idris file
InputFile String |
||| Whether or not to run in IdeMode (easily parsable for other tools)
IdeMode |
||| Whether or not to run IdeMode (using a socket instead of stdin/stdout)
IdeModeSocket String |
||| Run as a checker for the core language TTImp
Yaffle String |
||| Dump metadata from a .ttm file
Metadata String |
||| Dump cases before compiling
DumpCases String |
||| Dump lambda lifted defs before compiling
DumpLifted String |
||| Dump ANF defs before compiling
DumpANF String |
||| Dump VM code defs before compiling
DumpVMCode String |
||| Run a REPL command then exit immediately
RunREPL String |
IgnoreMissingIPKG |
FindIPKG |
Timing (Maybe Nat) |
DebugElabCheck |
AltErrorCount Nat |
||| Treat warnings as errors
WarningsAsErrors |
||| Do not print shadowing warnings
IgnoreShadowingWarnings |
||| Use SHA256 hashes to determine if a source file needs rebuilding instead
||| of modification time.
HashesInsteadOfModTime |
||| Apply experimental heuristics to case tree generation that
||| sometimes improves performance and reduces compiled code
||| size.
CaseTreeHeuristics |
||| Use incremental code generation, if the backend supports it
IncrementalCG String |
||| Use whole program compilation - overrides IncrementalCG if set
WholeProgram |
||| Generate bash completion info
BashCompletion String String |
||| Generate bash completion script
BashCompletionScript String |
||| Generate zsh completion script
ZshCompletionScript String |
||| Turn on %default total globally
Total |
||| Disable common subexpression elimination
NoCSE
||| Extract the host and port to bind the IDE socket to
export
ideSocketModeAddress : List CLOpt -> (String, Int)
ideSocketModeAddress [] = ("localhost", 0)
ideSocketModeAddress (IdeModeSocket hp :: _) =
let (h, p) = String.break (== ':') hp
port = fromMaybe 0 (portPart p >>= parsePositive)
host = if h == "" then "localhost" else h
in (host, port)
where
portPart : String -> Maybe String
portPart p = if p == ""
then Nothing
else Just $ assert_total $ prim__strTail p
ideSocketModeAddress (_ :: rest) = ideSocketModeAddress rest
formatSocketAddress : (String, Int) -> String
formatSocketAddress (host, port) = host ++ ":" ++ show port
data OptType
= Required String
| Optional String
| RequiredNat String
| AutoNat String
| RequiredLogLevel String
Show OptType where
show (Required a) = "<" ++ a ++ ">"
show (RequiredNat a) = "<" ++ a ++ ">"
show (RequiredLogLevel a) = "<" ++ a ++ ">"
show (Optional a) = "[" ++ a ++ "]"
show (AutoNat a) = "<" ++ a ++ ">"
ActType : List OptType -> Type
ActType [] = List CLOpt
ActType (Required a :: as) = String -> ActType as
ActType (RequiredNat a :: as) = Nat -> ActType as
ActType (RequiredLogLevel a :: as) = LogLevel -> ActType as
ActType (Optional a :: as) = Maybe String -> ActType as
ActType (AutoNat a :: as) = Maybe Nat -> ActType as
record OptDesc where
constructor MkOpt
flags : List String
argdescs : List OptType
action : ActType argdescs
help : Maybe String
optSeparator : OptDesc
optSeparator = MkOpt [] [] [] Nothing
showDefault : Show a => a -> String
showDefault x = "(default " ++ show x ++ ")"
options : List OptDesc
options = [MkOpt ["--check", "-c"] [] [CheckOnly]
(Just "Exit after checking source file"),
MkOpt ["--output", "-o"] [Required "file"] (\f => [OutputFile f, Quiet])
(Just "Specify output file"),
MkOpt ["--exec", "-x"] [Required "name"] (\f => [ExecFn f, Quiet])
(Just "Execute expression"),
MkOpt ["--no-prelude"] [] [NoPrelude]
(Just "Don't implicitly import Prelude"),
MkOpt ["--codegen", "--cg"] [Required "backend"] (\f => [SetCG f])
(Just $ "Set code generator " ++ showDefault (codegen defaultSession)),
MkOpt ["--incremental-cg", "--inc"] [Required "backend"] (\f => [IncrementalCG f])
(Just "Incremental code generation on given backend"),
MkOpt ["--whole-program", "--wp"] [] [WholeProgram]
(Just "Use whole program compilation (overrides --inc)"),
MkOpt ["--directive"] [Required "directive"] (\d => [Directive d])
(Just $ "Pass a directive to the current code generator"),
MkOpt ["--package", "-p"] [Required "package"] (\f => [PkgPath f])
(Just "Add a package as a dependency"),
MkOpt ["--source-dir"] [Required "dir"] (\d => [SourceDir d])
(Just $ "Set source directory"),
MkOpt ["--build-dir"] [Required "dir"] (\d => [BuildDir d])
(Just $ "Set build directory"),
MkOpt ["--output-dir"] [Required "dir"] (\d => [OutputDir d])
(Just $ "Set output directory"),
MkOpt ["--profile"] [] [Profile]
(Just "Generate profile data when compiling, if supported"),
MkOpt ["--no-cse"] [] [NoCSE]
(Just "Disable common subexpression elimination"),
optSeparator,
MkOpt ["--total"] [] [Total]
(Just "Require functions to be total by default"),
MkOpt ["-Werror"] [] [WarningsAsErrors]
(Just "Treat warnings as errors"),
MkOpt ["-Wno-shadowing"] [] [IgnoreShadowingWarnings]
(Just "Do not print shadowing warnings"),
optSeparator,
MkOpt ["-Xcheck-hashes"] [] [HashesInsteadOfModTime]
(Just "Use SHA256 hashes instead of modification time to determine if a source file needs rebuilding"),
MkOpt ["-Xcase-tree-opt"] [] [CaseTreeHeuristics]
(Just "Apply experimental optimizations to case tree generation"),
optSeparator,
MkOpt ["--prefix"] [] [Directory Prefix]
(Just "Show installation prefix"),
MkOpt ["--paths"] [] [Directory BlodwenPaths]
(Just "Show paths"),
MkOpt ["--libdir"] [] [Directory LibDir]
(Just "Show library directory"),
MkOpt ["--list-packages"] [] [ListPackages]
(Just "List installed packages"),
optSeparator,
MkOpt ["--init"] [Optional "package file"]
(\ f => [Package Init f])
(Just "Interactively initialise a new project"),
MkOpt ["--dump-ipkg-json"] [Optional "package file"]
(\ f => [Package DumpJson f])
(Just "Dump an Idris2 package file in the JSON format"),
MkOpt ["--dump-installdir"] [Optional "package file"]
(\ f => [Package DumpInstallDir f])
(Just "Dump the location where the given package will be installed"),
MkOpt ["--build"] [Optional "package file"]
(\f => [Package Build f])
(Just "Build modules/executable for the given package"),
MkOpt ["--install"] [Optional "package file"]
(\f => [Package Install f])
(Just "Install the given package"),
MkOpt ["--install-with-src"] [Optional "package file"]
(\f => [Package InstallWithSrc f])
(Just "Install the given package"),
MkOpt ["--mkdoc"] [Optional "package file"]
(\f => [Package MkDoc f])
(Just "Build documentation for the given package"),
MkOpt ["--typecheck"] [Optional "package file"]
(\f => [Package Typecheck f])
(Just "Typechecks the given package without code generation"),
MkOpt ["--clean"] [Optional "package file"] (\f => [Package Clean f])
(Just "Clean intermediate files/executables for the given package"),
MkOpt ["--repl"] [Optional "package file"] (\f => [Package REPL f])
(Just "Build the given package and launch a REPL instance."),
MkOpt ["--find-ipkg"] [] [FindIPKG]
(Just "Find and use an .ipkg file in a parent directory."),
MkOpt ["--ignore-missing-ipkg"] [] [IgnoreMissingIPKG]
(Just "Fail silently if a dependency is missing."),
optSeparator,
MkOpt ["--ide-mode"] [] [IdeMode]
(Just "Run the REPL with machine-readable syntax"),
MkOpt ["--ide-mode-socket"] [Optional "host:port"]
(\hp => [IdeModeSocket $ fromMaybe (formatSocketAddress (ideSocketModeAddress [])) hp])
(Just $ "Run the ide socket mode on given host and port (random open socket by default)"),
optSeparator,
MkOpt ["--client"] [Required "REPL command"] (\f => [RunREPL f])
(Just "Run a REPL command then quit immediately"),
MkOpt ["--timing"] [AutoNat "level"] (\ n => [Timing n])
(Just "Display timing logs"),
optSeparator,
MkOpt ["--no-banner"] [] [NoBanner]
(Just "Suppress the banner"),
MkOpt ["--quiet", "-q"] [] [Quiet]
(Just "Quiet mode; display fewer messages"),
MkOpt ["--console-width"] [AutoNat "console width"] (\l => [ConsoleWidth l])
(Just "Width for console output (0 for unbounded) (auto by default)"),
MkOpt ["--show-machine-names"] [] [ShowMachineNames]
(Just "Show machine names when pretty printing"),
MkOpt ["--show-namespaces"] [] [ShowNamespaces]
(Just "Show namespaces when pretty printing"),
MkOpt ["--color", "--colour"] [] ([Color True])
(Just "Forces colored console output (enabled by default)"),
MkOpt ["--no-color", "--no-colour"] [] ([Color False])
(Just "Disables colored console output"),
MkOpt ["--verbose"] [] [Verbose]
(Just "Verbose mode (default)"),
MkOpt ["--log"] [RequiredLogLevel "log level"] (\l => [Logging l])
(Just "Global log level (0 by default)"),
optSeparator,
MkOpt ["--version", "-v"] [] [Version]
(Just "Display version string"),
MkOpt ["--ttc-version"] [] [TTCVersion]
(Just "Display TTC version string"),
MkOpt ["--help", "-h", "-?"] [Optional "topic"] (\ tp => [Help (tp >>= recogniseHelpTopic)])
(Just "Display help text"),
-- Internal debugging options
MkOpt ["--yaffle", "--ttimp"] [Required "ttimp file"] (\f => [Yaffle f])
Nothing, -- run ttimp REPL rather than full Idris
MkOpt ["--ttm" ] [Required "ttimp file"] (\f => [Metadata f])
Nothing, -- dump metadata information from the given ttm file
MkOpt ["--dumpcases"] [Required "output file"] (\f => [DumpCases f])
Nothing, -- dump case trees to the given file
MkOpt ["--dumplifted"] [Required "output file"] (\f => [DumpLifted f])
Nothing, -- dump lambda lifted trees to the given file
MkOpt ["--dumpanf"] [Required "output file"] (\f => [DumpANF f])
Nothing, -- dump ANF to the given file
MkOpt ["--dumpvmcode"] [Required "output file"] (\f => [DumpVMCode f])
Nothing, -- dump VM Code to the given file
MkOpt ["--debug-elab-check"] [] [DebugElabCheck]
Nothing, -- do more elaborator checks (currently conversion in LinearCheck)
MkOpt ["--alt-error-count"] [RequiredNat "alternative count"] (\c => [AltErrorCount c])
(Just "Outputs errors for the given number of alternative parsing attempts."),
optSeparator,
-- bash completion
MkOpt ["--bash-completion"]
[ Required "input"
, Required "previous input"]
(\w1,w2 => [BashCompletion w1 w2])
(Just "Print bash autocompletion information"),
MkOpt ["--bash-completion-script"]
[ Required "function name" ]
(\n => [BashCompletionScript n])
(Just "Generate a bash script to activate autocompletion for Idris2"),
-- zsh completion
MkOpt ["--zsh-completion-script"]
[ Required "function name" ]
(\n => [ZshCompletionScript n])
(Just "Generate a zsh script (via bashcompinit) to activate autocompletion for Idris2")
]
optShow : OptDesc -> (String, Maybe String)
optShow (MkOpt [] _ _ _) = ("", Just "")
optShow (MkOpt flags argdescs action help) = (showSep ", " flags ++ " " ++
showSep " " (map show argdescs),
help)
where
showSep : String -> List String -> String
showSep sep [] = ""
showSep sep [x] = x
showSep sep (x :: xs) = x ++ sep ++ showSep sep xs
firstColumnWidth : Nat
firstColumnWidth = let maxOpt = foldr max 0 $ map (length . fst . optShow) options
maxEnv = foldr max 0 $ map (length . .name) envs in
max maxOpt maxEnv
makeTextFromOptionsOrEnvs : List (String, Maybe String) -> String
makeTextFromOptionsOrEnvs rows = concatMap (optUsage firstColumnWidth) rows
where
optUsage : Nat -> (String, Maybe String) -> String
optUsage maxOpt (optshow, help) = maybe "" -- Don't show anything if there's no help string (that means
-- it's an internal option)
(\h => " " ++ optshow ++
pack (List.replicate (minus (maxOpt + 2) (length optshow)) ' ') ++
h ++ "\n")
help
optsUsage : String
optsUsage = makeTextFromOptionsOrEnvs $ map optShow options
envsUsage : String
envsUsage = makeTextFromOptionsOrEnvs $ map (\e => (e.name, Just e.help)) envs
export
versionMsg : String
versionMsg = "Idris 2, version " ++ show version
export
usage : String
usage = """
\{ versionMsg }
Usage: idris2 [options] [input file]
Available options:
\{ optsUsage }
Environment variables:
\{ envsUsage }
"""
checkNat : Integer -> Maybe Nat
checkNat n = toMaybe (n >= 0) (integerToNat n)
processArgs : String -> (args : List OptType) -> List String -> ActType args ->
Either String (List CLOpt, List String)
processArgs flag [] xs f = Right (f, xs)
-- Missing required arguments
processArgs flag (opt@(Required _) :: as) [] f =
Left $ "Missing required argument " ++ show opt ++ " for flag " ++ flag
processArgs flag (opt@(RequiredNat _) :: as) [] f =
Left $ "Missing required argument " ++ show opt ++ " for flag " ++ flag
processArgs flag (opt@(RequiredLogLevel _) :: as) [] f =
Left $ "Missing required argument " ++ show opt ++ " for flag " ++ flag
processArgs flag (Optional a :: as) [] f =
processArgs flag as [] (f Nothing)
processArgs flag (opt@(AutoNat _) :: as) [] f =
Left $ "Missing required argument " ++ show opt ++ " for flag " ++ flag
-- Happy cases
processArgs flag (RequiredNat a :: as) (x :: xs) f =
do arg <- maybeToEither ("Expected Nat argument " ++ show x ++ " for flag " ++ flag)
(parseInteger x >>= checkNat)
processArgs flag as xs (f arg)
processArgs flag (RequiredLogLevel a :: as) (x :: xs) f =
do arg <- maybeToEither ("Expected LogLevel argument " ++ show x ++ " for flag " ++ flag)
(parseLogLevel x)
processArgs flag as xs (f arg)
processArgs flag (AutoNat a :: as) ("auto" :: xs) f =
processArgs flag as xs (f Nothing)
processArgs flag (AutoNat a :: as) (x :: xs) f =
do arg <- maybeToEither ("Expected Nat or \"auto\" argument " ++ show x ++ " for flag " ++ flag)
(parseInteger x >>= checkNat)
processArgs flag as xs (f (Just arg))
processArgs flag (Required a :: as) (x :: xs) f =
processArgs flag as xs (f x)
processArgs flag (Optional a :: as) (x :: xs) f =
if isPrefixOf "-" x
then processArgs flag as (x :: xs) (f Nothing)
else processArgs flag as xs (f $ Just x)
matchFlag : (d : OptDesc) -> List String ->
Either String (Maybe (List CLOpt, List String))
matchFlag d [] = Right Nothing -- Nothing left to match
matchFlag d (x :: xs)
= if x `elem` flags d
then do args <- processArgs x (argdescs d) xs (action d)
Right (Just args)
else Right Nothing
findMatch : List OptDesc -> List String ->
Either String (List CLOpt, List String)
findMatch [] [] = Right ([], [])
findMatch [] (f :: args) = Right ([InputFile f], args)
findMatch (d :: ds) args
= case !(matchFlag d args) of
Nothing => findMatch ds args
Just res => Right res
parseOpts : List OptDesc -> List String -> Either String (List CLOpt)
parseOpts opts [] = Right []
parseOpts opts args
= do (cl, rest) <- findMatch opts args
cls <- assert_total (parseOpts opts rest) -- 'rest' smaller than 'args'
pure (cl ++ cls)
export
getOpts : List String -> Either String (List CLOpt)
getOpts opts = parseOpts options opts
export covering
getCmdOpts : IO (Either String (List CLOpt))
getCmdOpts = do (_ :: opts) <- getArgs
| _ => pure (Left "Invalid command line")
pure $ getOpts opts
||| Find an opt that would have matched if only the user had prefixed it with
||| double-dashes. It is common to see the user error of specifying a command
||| like "idris2 repl ..." when they mean to write "idris2 --repl ..."
export
findNearMatchOpt : String -> Maybe String
findNearMatchOpt arg =
let argWithDashes = "--\{arg}"
in
case (getOpts [argWithDashes]) of
Right [(InputFile _)] => Nothing
Right [_] => Just argWithDashes
_ => Nothing
||| Suggest an opt that would have matched if only the user had prefixed it with
||| double-dashes. It is common to see the user error of specifying a command
||| like "idris2 repl ..." when they mean to write "idris2 --repl ..."
export
nearMatchOptSuggestion : String -> Maybe (Doc IdrisAnn)
nearMatchOptSuggestion arg =
findNearMatchOpt arg <&>
\opt =>
(reflow "Did you mean to type" <++>
(dquotes . meta $ pretty0 opt) <+> "?")
||| List of all command line option flags.
export
optionFlags : List String
optionFlags = options >>= flags