-- | A combinator library for building command-line parsers.

module System.Console.MultiArg (

  -- | To say this library is inspired by Parsec would probably insult the
  -- creators of Parsec, as this library could not possibly be as
  -- elegant or throughly considered as Parsec is. Nevertheless this
  -- library can be used in a similar style as Parsec, but is
  -- specialized for parsing command lines.
  --
  -- This parser was built because I could not find anything that would
  -- readily parse command lines where the options took more than one
  -- argument. For example, for the @tail@ command on GNU systems, the
  -- --lines option takes one argument to specify how many lines you
  -- want to see. Well, what if you want to build a program with an
  -- option that takes /two/ arguments, like @--foo bar baz@? I found no
  -- such library so I built this one. Nevertheless, using this library
  -- you can build parsers to parse a variety of command line
  -- vocabularies, from simple to complex.

  -- * Terminology

  -- | Some terms are used throughout multiarg:
  --
  -- [@word@] When you run your program from the Unix shell prompt,
  -- your shell is responsible for splitting the command line into
  -- words. Typically you separate words with spaces, although quoting
  -- can affect this. multiarg parses lists of words. Each word can
  -- consist of a single long option, a single long option and an
  -- accompanying option argument, a single short option, multiple
  -- short options, and even one or more multiple short options and an
  -- accompanying short option argument. Or, a word can be a
  -- positional argument or a stopper. All these are described below.
  --
  -- [@option@] Options allow a user to specify ways to tune the
  -- operation of a program. Typically options are indeed optional,
  -- although some programs do sport \"required options\" (a bit of an
  -- oxymoron). Options can be either short options or long
  -- options. Also, options can take arguments.
  --
  -- [@short option@] An option that is specified with a single hyphen
  -- and a single letter. For example, for the program @tail(1)@,
  -- possible short options include @n@ and @v@. With multiarg it is
  -- possible to easily parse short options that are specified in
  -- different words or in the same word. For example, if a user wants
  -- to run @tail@ with two options, he might type @tail -v -f@ or he
  -- might type @tail -vf@.
  --
  -- [@long option@] An option that is specified using two hyphens and
  -- what is usually a mnemonic word, though it could be as short as a
  -- single letter. For example, @tail(1)@ has long options including
  -- @follow@ and @verbose@. The user would specify these on the
  -- command line by typing @tail --follow --verbose@.
  --
  -- [@option argument@] Some options take additional arguments that
  -- are specific to the option and change what the option does. For
  -- instance, the @lines@ option to @tail(1)@ takes a single,
  -- optional argument, which is the number of lines to show. Option
  -- arguments can be optional or required, and a single option can
  -- take a mulitple, fixed number of arguments and it can take a
  -- variable number of arguments. Option arguments can be given in
  -- various ways. They can be specified in the same word as a long
  -- option by using an equals sign; they can also be specified in the
  -- same word as a short option simply by placing them in the same
  -- word, or they can be specified in the following word. For
  -- example, these different command lines all mean the same thing;
  -- @tail --verbose --lines=20@, @tail --verbose --lines 20@, @tail
  -- -vn 20@, @tail -v -n20@, @tail -vn20@, and @tail -v -n 20@, and
  -- numerous other combinations also have the same meaning.
  --
  -- [@GNU-style option argument@] A long option with an argument
  -- given with an equal sign, such as [@lines=20@].
  --
  -- [@positional argument@] A word on the command line that is not an
  -- option or an argument to an option. For instance, with @tail(1)@,
  -- you specify the files you want to see by using positional
  -- arguments. In the command @tail -n 10 myfile@, @myfile@ is a
  -- positional argument. For some programs, such as @git@ or @darcs@,
  -- a positional argument might be a \"command\" or a \"mode\", such
  -- as the @commit@ in @git commit@ or the @whatsnew@ in @darcs
  -- whatsnew@. multiarg has no primitive parsers that treat these
  -- positional arguments specially but it is trivial to build a
  -- parser for command lines such as this, too.
  --
  -- [@stopper@] A single word consisting solely of two hyphens,
  -- @--@. The user types this to indicate that all subsequent words
  -- on the command line are positional arguments, even if they begin
  -- with hyphens and therefore look like they might be options.
  --
  -- [@pending@] The user might specify more than one short option, or
  -- a short option and a short option argument, in a single word. For
  -- example, she might type @tail -vl20@. After parsing the @v@
  -- option, the Parser makes @l20@ into a \"pending\". The next
  -- parser can then treat @l20@ as an option argument to the @v@
  -- option (which is probably not what was wanted) or the next parser
  -- can parse @l@ as a short option. This would result in a
  -- \"pending\" of @20@. Then, the next parser can treat @20@ as an
  -- option argument. After that parse there will be no pendings.

  -- * Getting started

  -- |If your needs are simple to moderately complicated just look at the
  -- "System.Console.MultiArg.SimpleParser" module, which uses the
  -- underlying combinators to build a simple parser for you. That
  -- module is already exported from this module for easy usage.
  --
  -- "System.Console.MultiArg.SimpleParser" also has a parser that can
  -- handle multi-mode commands (examples include @git@, @darcs@, and
  -- @cvs@.)
  --
  -- For maximum flexibility you will want to start with the
  -- "System.Console.MultiArg.Prim" module. Using those parsers you
  -- can easily build parsers that are quite complicated. The parsers
  -- can check for errors along the way, simplifying the sometimes
  -- complex task of ensuring that data a user supplied on the command
  -- line is good. You can easily build parsers for programs that take
  -- no options, take dozens of options, require that options be given
  -- in a particular order, require that some options be given, or bar
  -- some combinations of options. You might also require particular
  -- positional arguments. Other helpful functions are in
  -- "System.Console.MultiArg.Combinator". You will also want to
  -- examine the source code for "System.Console.MultiArg.Combinator"
  -- and "System.Console.MultiArg.SimpleParser" as these show some
  -- ways to use the primitive parsers and combinators.

  -- * Non-features and shortcomings
  --
  -- | multiarg isn't perfect; no software is. multiarg does not
  -- automatically make online help for your command line
  -- parsers. Getting this right would be tricky given the nature of
  -- the code and I don't even want to bother trying, as I just write
  -- my own online help in a text editor.
  --
  -- multiarg partially embraces \"The Tao of Option Parsing\" that
  -- Python's Optik (<http://optik.sourceforge.net/>) follows. Read
  -- \"The Tao of Option Parsing\" here:
  --
  -- <http://optik.sourceforge.net/doc/1.5/tao.html>
  --
  -- multiarg's philosophy is similar to that of Optik, which
  -- means you won't be able to use multiarg to (easily) build a clone
  -- to the UNIX @find(1)@ command. (You could do it, but multiarg won't
  -- help you very much.)
  --
  -- multiarg can be complicated, although I'd like to believe this is
  -- because it addresses a complicated problem in a flexible way.

  -- * Projects usings multiarg

  -- | * Penny, an extensible double-entry accounting
  -- system. <http://hackage.haskell.org/package/penny-lib> The code
  -- using multiarg is woven throughout the system; for example, see
  -- the Penny.Liberty module.


    module System.Console.MultiArg.Combinator
  , module System.Console.MultiArg.GetArgs
  , module System.Console.MultiArg.Option
  , module System.Console.MultiArg.Prim
  , module System.Console.MultiArg.SimpleParser
  , module Control.Monad.Exception.Synchronous
  ) where

import System.Console.MultiArg.Combinator
import System.Console.MultiArg.GetArgs
import System.Console.MultiArg.Option
import System.Console.MultiArg.Prim
import System.Console.MultiArg.SimpleParser
import Control.Monad.Exception.Synchronous