Skip to content

Latest commit

 

History

History
175 lines (156 loc) · 8.27 KB

README.md

File metadata and controls

175 lines (156 loc) · 8.27 KB

cligen: A Native API-Inferred Command-Line Interface Generator For Nim

This approach to CLIs was inspired by Andrey Mikhaylenko's nice Python module 'argh'. The basic idea is that proc signatures encode/declare almost everything needed to generate a CLI - names, types, and default values. A little reflection/introspection then suffices to generate a parser-dispatcher that translates a seq[string] command input into calls of a wrapped proc. In Nim, adding a CLI can be as easy as adding one line of code:

proc foobar(foo=1, bar=2.0, baz="hi", verb=false, paths: seq[string]): int =
  ##Some existing API call
  result = 1          # Of course, real code would have real logic here
import cligen; dispatch(foobar) #Whoa..Just 1 line??

Compile it to foobar (e.g., nim c foobar.nim) and then run ./foobar --help to get a minimal (but not so useless!) help message:

Usage:
  foobar [optional-params] [paths: string...]
Some existing API call
  Options(opt-arg sep :|=|spc):
  -h, --help                    print this cligen-erated help
  --help-syntax                 advanced: prepend, multi-val,..
  -f=, --foo=    int     1      set foo
  -b=, --bar=    float   2.0    set bar
  --baz=         string  "hi"   set baz
  -v, --verb     bool    false  set verb

Other invocations (foobar --foo=2 --bar=2.7 ...) all work as expected. Default help tables work with automated "help to X" tools such as complete -F _longopt in bash, compdef _gnu_generic in zsh, or the GNU help2man. cligen-erated parsers accept any unambiguous prefix for long options. So, hitting the TAB key is unnecessary, though might pop-up useful help.

When you want more specific help than set foo and set bar, just add some parameter-keyed metadata with Nim's association-list literals:

dispatch(foobar, help = { "foo" : "the beginning", "bar" : "the rate" })

That's it. No specification language or complex arg parsing APIs. If you aren't immediately sold, here is some more MOTIVATION.


Most commands have some trailing variable length sequence of arguments like the paths in the above example. cligen automatically treats the first non-defaulted seq[T] proc parameter as such an optional sequence. cligen applies the same basic string-to-native type converters/parsers used for option values to such parameters. If a proc parameter has no explicit default value, it becomes mandatory/required input, but the syntax for input is the same as optional values. So, in the below

proc foobar(myMandatory: float, mynums: seq[int], foo=1, verb=false): int =
  ##Some API call
  result = 1        # Of course, real code would have real logic here
when isMainModule:  # Preserve ability to `import api` & call from Nim
  import cligen; dispatch(foobar)

the command-line user must give --myMandatory=something somewhere. Non-option arguments must be parsable as int with whitespace stripped, e.g. ./foobar --myMandatory=2.0 1 2 " -3".


dispatchMulti lets you expose two or more procs with subcommands a la git or nimble, just use in, say, a cmd.nim file. Each [] list in dispatchMulti is the argument list for each sub-dispatch. Tune command syntax and help strings in the same way as dispatch as in:

proc foo(myMandatory: int, mynums: seq[int], foo=1, verb=false) =
  ##Some API call
proc bar(yippee: int, myfloats: seq[float], verb=false) =
  ##Some other API call
when isMainModule:
  import cligen; dispatchMulti([foo, help={"myMandatory": "Need it!"}], [bar])

With that, a CLI user can run ./cmd foo -m1 or ./cmd bar -y10 1.0 2.0. ./cmd or ./cmd --help will emit a brief help message while ./cmd help emits a more comprehensive message, and ./cmd SUBCMD --help or ./cmd help SUBCMD emits just the message for SUBCMD (foo or bar in this example). Like long option keys or enum values, any unambiguous prefix is accepted. So, in the above ./cmd f -m1 would also work. This is patterned after, e.g. Mercurial, gdb, or gnuplot. Additionally, long option keys can be spelled flexibly, e.g. --dry-run or --dryRun, much like Nim's style-insensitive identifiers, but with extra insensitivity to so-called "kebab case".


Many CLI authors who have understood things this far can use cligen already. Enter illegal commands or --help to get help messages to exhibit the mappings.

More Controls For More Subtle Cases/More Picky CLI authors

You can manually control the short option for any parameter via the short macro parameter:

dispatch(foobar, short = { "bar" : 'r' }))

With that (and our first foobar example), "bar" gets 'r' while "baz" gets 'b' as short options. To suppress some long option getting any short option, specify '\0' as the value for its short key. To suppress all short options, give short a key of "".


The default exit protocol is quit(int(result)) or (echo $result or discard; quit(0)). If echoResult==true, it's just echo $result; quit(0), while if noAutoEcho==true it's just quit(int(result)) or (discard; quit(0)). So,

import editdistance, cligen   # gen a CLI for Nim stdlib's editDistance
dispatch(editDistanceASCII, echoResult=true)

prints the edit distance between two mandatory parameters, a, and b.

If these exit protocols are inadequate then you may need to call dispatchGen() and later call try: dispatchFoo(someSeqString) except: discard yourself. This is all dispatch itself does. Return types and values of the generated dispatcher matches the wrapped proc. { Other parameters to generated dispatchers are for internal use in dispatchMulti and probably don't matter to you. } A dispatcher raises 3 exception types: HelpOnly, VersionOnly, ParseError. These are hopefully self-explanatory.


If you want cligen to merge parameters from other sources like a $CMD environment variable then you can redefine mergeParams() after import cligen but before dispatch/dispatchMulti:

import cligen, os, strutils
proc mergeParams(cmdNames: seq[string], cmdLine=commandLineParams()): seq[string]=
  let e = os.getEnv(toUpperAscii(join(cmdNames, "_")))   #Get $MULTI_(FOO|_BAR)
  if e.len > 0: parseCmdLine(e) & cmdLine else: cmdLine  #See os.parseCmdLine
dispatchMulti([foo, short={"verb": 'v'}], [bar])

You can also just include cligen/mergeCfgEnv between import cligen and dispatch to merge ${CMD_CONFIG:-${XDG_CONFIG_HOME:-$HOME/.config}}/CMD (with Nim stdlib's parsecfg module) and then $CMD with parseCmdLine as shown above, if that works for you.

Rather than dispatching to a proc and exiting, you can also initialize the fields of an object/tuple from the command-line with initFromCL which has the same keyword parameters as the most salient features of dispatch:

type App* = object
  srcFile*: string
  show*: bool
const dfl* = App(srcFile: "junk") #set any defaults != default for type

proc logic*(a: var App) = echo "app is: ", a

when isMainModule:
    import cligen
    var app = initFromCL(dfl, help = { "srcFile": "yadda yadda" })
    app.logic()

Top-level types in the object/tuple just need in-scope argParse / argHelp definitions.

Even More Controls and Details

After many feature requests cligen grew many knobs & levers. First there are more DETAILS on the restrictions on wrappable procs and extending the parser to new argument types. A good starting point for various advanced usages is the many examples in my automated test suite: test/.

Then there is the documentation for the three main modules: parseopt3 argcvt cligen

Finally, I try to keep track of possibly breaking changes and new features in RELEASE-NOTES.