OptionParser.parse
parse, go back to OptionParser module for more information.
Specs
Parses argv into a keyword list.
It returns a three-element tuple with the form {parsed, args, invalid}, where:
parsedis a keyword list of parsed switches with{switch_name, value}tuples in it;switch_nameis the atom representing the switch name whilevalueis the value for that switch parsed according toopts(see the "Examples" section for more information)argsis a list of the remaining arguments inargvas stringsinvalidis a list of invalid options as{option_name, value}whereoption_nameis the raw option andvalueisnilif the option wasn't expected or the string value if the value didn't have the expected type for the corresponding option
Elixir converts switches to underscored atoms, so --source-path becomes
:source_path. This is done to better suit Elixir conventions. However, this
means that switches can't contain underscores and switches that do contain
underscores are always returned in the list of invalid switches.
When parsing, it is common to list switches and their expected types:
iex> OptionParser.parse(["--debug"], strict: [debug: :boolean])
{[debug: true], [], []}
iex> OptionParser.parse(["--source", "lib"], strict: [source: :string])
{[source: "lib"], [], []}
iex> OptionParser.parse(
...> ["--source-path", "lib", "test/enum_test.exs", "--verbose"],
...> strict: [source_path: :string, verbose: :boolean]
...> )
{[source_path: "lib", verbose: true], ["test/enum_test.exs"], []}We will explore the valid switches and operation modes of option parser below.
Options
The following options are supported:
:switchesor:strict- see the "Switch definitions" section below:allow_nonexistent_atoms- see the "Parsing unknown switches" section below:aliases- see the "Aliases" section below
Switch definitions
Switches can be specified via one of two options:
:strict- defines strict switches and their types. Any switch inargvthat is not specified in the list is returned in the invalid options list. This is the preferred way to parse options.:switches- defines switches and their types. This function still attempts to parse switches that are not in this list.
Both these options accept a keyword list where the key is an atom
defining the name of the switch and value is the type of the
switch (see the "Types" section below for more information).
Note that you should only supply the :switches or the :strict option.
If you supply both, an ArgumentError exception will be raised.
Types
Switches parsed by OptionParser may take zero or one arguments.
The following switches types take no arguments:
:boolean- sets the value totruewhen given (see also the "Negation switches" section below):count- counts the number of times the switch is given
The following switches take one argument:
:integer- parses the value as an integer:float- parses the value as a float:string- parses the value as a string
If a switch can't be parsed according to the given type, it is returned in the invalid options list.
Modifiers
Switches can be specified with modifiers, which change how they behave. The following modifiers are supported:
:keep- keeps duplicated elements instead of overriding them; works with all types except:count. Specifyingswitch_name: :keepassumes the type of:switch_namewill be:string.
To use :keep with a type other than :string, use a list as the type
for the switch. For example: [foo: [:integer, :keep]].
Negation switches
In case a switch SWITCH is specified to have type :boolean, it may be
passed as --no-SWITCH as well which will set the option to false:
iex> OptionParser.parse(["--no-op", "path/to/file"], switches: [op: :boolean])
{[op: false], ["path/to/file"], []}Parsing unknown switches
When the :switches option is given, OptionParser will attempt to parse
unknown switches:
iex> OptionParser.parse(["--debug"], switches: [key: :string])
{[debug: true], [], []}Even though we haven't specified --debug in the list of switches, it is part
of the returned options. This would also work:
iex> OptionParser.parse(["--debug", "value"], switches: [key: :string])
{[debug: "value"], [], []}Switches followed by a value will be assigned the value, as a string. Switches
without an argument will be set automatically to true. Since we cannot assert
the type of the switch value, it is preferred to use the :strict option that
accepts only known switches and always verify their types.
If you do want to parse unknown switches, remember that Elixir converts switches
to atoms. Since atoms are not garbage-collected, OptionParser will only parse
switches that translate to atoms used by the runtime to avoid leaking atoms.
For instance, the code below will discard the --option-parser-example switch
because the :option_parser_example atom is never used anywhere:
OptionParser.parse(["--option-parser-example"], switches: [debug: :boolean])
# The :option_parser_example atom is not used anywhere belowHowever, the code below would work as long as :option_parser_example atom is
used at some point later (or earlier) in the same module. For example:
{opts, _, _} = OptionParser.parse(["--option-parser-example"], switches: [debug: :boolean])
# ... then somewhere in the same module you access it ...
opts[:option_parser_example]In other words, Elixir will only parse options that are used by the runtime,
ignoring all others. If you would like to parse all switches, regardless if
they exist or not, you can force creation of atoms by passing
allow_nonexistent_atoms: true as option. Use this option with care. It is
only useful when you are building command-line applications that receive
dynamically-named arguments and must be avoided in long-running systems.
Aliases
A set of aliases can be specified in the :aliases option:
iex> OptionParser.parse(["-d"], aliases: [d: :debug], strict: [debug: :boolean])
{[debug: true], [], []}Examples
Here are some examples of working with different types and modifiers:
iex> OptionParser.parse(["--unlock", "path/to/file"], strict: [unlock: :boolean])
{[unlock: true], ["path/to/file"], []}
iex> OptionParser.parse(
...> ["--unlock", "--limit", "0", "path/to/file"],
...> strict: [unlock: :boolean, limit: :integer]
...> )
{[unlock: true, limit: 0], ["path/to/file"], []}
iex> OptionParser.parse(["--limit", "3"], strict: [limit: :integer])
{[limit: 3], [], []}
iex> OptionParser.parse(["--limit", "xyz"], strict: [limit: :integer])
{[], [], [{"--limit", "xyz"}]}
iex> OptionParser.parse(["--verbose"], switches: [verbose: :count])
{[verbose: 1], [], []}
iex> OptionParser.parse(["-v", "-v"], aliases: [v: :verbose], strict: [verbose: :count])
{[verbose: 2], [], []}
iex> OptionParser.parse(["--unknown", "xyz"], strict: [])
{[], ["xyz"], [{"--unknown", nil}]}
iex> OptionParser.parse(
...> ["--limit", "3", "--unknown", "xyz"],
...> switches: [limit: :integer]
...> )
{[limit: 3, unknown: "xyz"], [], []}
iex> OptionParser.parse(
...> ["--unlock", "path/to/file", "--unlock", "path/to/another/file"],
...> strict: [unlock: :keep]
...> )
{[unlock: "path/to/file", unlock: "path/to/another/file"], [], []}