arg

package module
v0.0.0-...-ba9514f Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Oct 2, 2017 License: BSD-2-Clause Imports: 9 Imported by: 0

README

GoDoc Build Status Coverage Status Report Card

Structured argument parsing for Go

go get github.com/alexflint/go-arg

Declare the command line arguments your program accepts by defining a struct.

var args struct {
	Foo string
	Bar bool
}
arg.MustParse(&args)
fmt.Println(args.Foo, args.Bar)
$ ./example --foo=hello --bar
hello true

Required arguments

var args struct {
	ID      int `arg:"required"`
	Timeout time.Duration
}
arg.MustParse(&args)
$ ./example
Usage: example --id ID [--timeout TIMEOUT]
error: --id is required

Positional arguments

var args struct {
	Input   string   `arg:"positional"`
	Output  []string `arg:"positional"`
}
arg.MustParse(&args)
fmt.Println("Input:", args.Input)
fmt.Println("Output:", args.Output)
$ ./example src.txt x.out y.out z.out
Input: src.txt
Output: [x.out y.out z.out]

Environment variables

var args struct {
	Workers int `arg:"env"`
}
arg.MustParse(&args)
fmt.Println("Workers:", args.Workers)
$ WORKERS=4 ./example
Workers: 4
$ WORKERS=4 ./example --workers=6
Workers: 6

You can also override the name of the environment variable:

var args struct {
	Workers int `arg:"env:NUM_WORKERS"`
}
arg.MustParse(&args)
fmt.Println("Workers:", args.Workers)
$ NUM_WORKERS=4 ./example
Workers: 4

Usage strings

var args struct {
	Input    string   `arg:"positional"`
	Output   []string `arg:"positional"`
	Verbose  bool     `arg:"-v" help:"verbosity level"`
	Dataset  string   `help:"dataset to use"`
	Optimize int      `arg:"-O" help:"optimization level"`
}
arg.MustParse(&args)
$ ./example -h
Usage: [--verbose] [--dataset DATASET] [--optimize OPTIMIZE] [--help] INPUT [OUTPUT [OUTPUT ...]] 

Positional arguments:
  INPUT 
  OUTPUT

Options:
  --verbose, -v            verbosity level
  --dataset DATASET        dataset to use
  --optimize OPTIMIZE, -O OPTIMIZE
                           optimization level
  --help, -h               print this help message

As the example above shows, the help tag can be used in conjunction with arg, or instead. When used together, they can appear in either order.

Default values

var args struct {
	Foo string
	Bar bool
}
args.Foo = "default value"
arg.MustParse(&args)

Arguments with multiple values

var args struct {
	Database string
	IDs      []int64
}
arg.MustParse(&args)
fmt.Printf("Fetching the following IDs from %s: %q", args.Database, args.IDs)
./example -database foo -ids 1 2 3
Fetching the following IDs from foo: [1 2 3]

Arguments that can be specified multiple times, mixed with positionals

var args struct {
    Commands  []string `arg:"-c,separate"`
    Files     []string `arg:"-f,separate"`
    Databases []string `arg:"positional"`
}
./example -c cmd1 db1 -f file1 db2 -c cmd2 -f file2 -f file3 db3 -c cmd3
Commands: [cmd1 cmd2 cmd3]
Files [file1 file2 file3]
Databases [db1 db2 db3]

Custom validation

var args struct {
	Foo string
	Bar string
}
p := arg.MustParse(&args)
if args.Foo == "" && args.Bar == "" {
	p.Fail("you must provide one of --foo and --bar")
}
./example
Usage: samples [--foo FOO] [--bar BAR]
error: you must provide one of --foo and --bar

Version strings

type args struct {
	...
}

func (args) Version() string {
	return "someprogram 4.3.0"
}

func main() {
	var args args
	arg.MustParse(&args)
}
$ ./example --version
someprogram 4.3.0

Embedded structs

The fields of embedded structs are treated just like regular fields:


type DatabaseOptions struct {
	Host     string
	Username string
	Password string
}

type LogOptions struct {
	LogFile string
	Verbose bool
}

func main() {
	var args struct {
		DatabaseOptions
		LogOptions
	}
	arg.MustParse(&args)
}

As usual, any field tagged with arg:"-" is ignored.

Custom parsing

You can implement your own argument parser by implementing encoding.TextUnmarshaler:

package main

import (
	"fmt"
	"strings"

	"github.com/alexflint/go-arg"
)

// Accepts command line arguments of the form "head.tail"
type NameDotName struct {
	Head, Tail string
}

func (n *NameDotName) UnmarshalText(b []byte) error {
	s := string(b)
	pos := strings.Index(s, ".")
	if pos == -1 {
		return fmt.Errorf("missing period in %s", s)
	}
	n.Head = s[:pos]
	n.Tail = s[pos+1:]
	return nil
}

func main() {
	var args struct {
		Name *NameDotName
	}
	arg.MustParse(&args)
	fmt.Printf("%#v\n", args.Name)
}
$ ./example --name=foo.bar
&main.NameDotName{Head:"foo", Tail:"bar"}

$ ./example --name=oops
Usage: example [--name NAME]
error: error processing --name: missing period in "oops"

Description strings

type args struct {
	Foo string
}

func (args) Description() string {
	return "this program does this and that"
}

func main() {
	var args args
	arg.MustParse(&args)
}
$ ./example -h
this program does this and that
Usage: example [--foo FOO]

Options:
  --foo FOO
  --help, -h             display this help and exit

API Documentation

https://godoc.org/github.com/alexflint/go-arg

Rationale

There are many command line argument parsing libraries for Go, including one in the standard library, so why build another?

The shortcomings of the flag library that ships in the standard library are well known. Positional arguments must preceed options, so ./prog x --foo=1 does what you expect but ./prog --foo=1 x does not. Arguments cannot have both long (--foo) and short (-f) forms.

Many third-party argument parsing libraries are geared for writing sophisticated command line interfaces. The excellent codegangsta/cli is perfect for working with multiple sub-commands and nested flags, but is probably overkill for a simple script with a handful of flags.

The main idea behind go-arg is that Go already has an excellent way to describe data structures using Go structs, so there is no need to develop more levels of abstraction on top of this. Instead of one API to specify which arguments your program accepts, and then another API to get the values of those arguments, why not replace both with a single struct?

Backward Compatibility Notes

The tags have changed recently. Earlier versions required the help text to be part of the arg tag. This is still supported but is now deprecated. Instead, you should use a separate help tag, described above, which removes most of the limits on the text you can write. In particular, you will need to use the new help tag if your help text includes any commas.

Documentation

Overview

Package arg parses command line arguments using the fields from a struct.

For example,

var args struct {
	Iter int
	Debug bool
}
arg.MustParse(&args)

defines two command line arguments, which can be set using any of

./example --iter=1 --debug  // debug is a boolean flag so its value is set to true
./example -iter 1           // debug defaults to its zero value (false)
./example --debug=true      // iter defaults to its zero value (zero)

The fastest way to see how to use go-arg is to read the examples below.

Fields can be bool, string, any float type, or any signed or unsigned integer type. They can also be slices of any of the above, or slices of pointers to any of the above.

Tags can be specified using the `arg` and `help` tag names:

var args struct {
	Input string   `arg:"positional"`
	Log string     `arg:"positional,required"`
	Debug bool     `arg:"-d" help:"turn on debug mode"`
	RealMode bool  `arg:"--real"
	Wr io.Writer   `arg:"-"`
}

Any tag string that starts with a single hyphen is the short form for an argument (e.g. `./example -d`), and any tag string that starts with two hyphens is the long form for the argument (instead of the field name).

Other valid tag strings are `positional` and `required`.

Fields can be excluded from processing with `arg:"-"`.

Example

This example demonstrates basic usage

// These are the args you would pass in on the command line
os.Args = []string{"./example", "--foo=hello", "--bar"}

var args struct {
	Foo string
	Bar bool
}
MustParse(&args)
fmt.Println(args.Foo, args.Bar)
Output:

Example (DefaultValues)

This example demonstrates arguments that have default values

// These are the args you would pass in on the command line
os.Args = []string{"--help"}

var args struct {
	Foo string
	Bar bool
}
args.Foo = "default value"
MustParse(&args)
fmt.Println(args.Foo, args.Bar)
Output:

Example (MultipleMixed)

This eample demonstrates multiple value arguments that can be mixed with other arguments.

os.Args = []string{"./example", "-c", "cmd1", "db1", "-f", "file1", "db2", "-c", "cmd2", "-f", "file2", "-f", "file3", "db3", "-c", "cmd3"}
var args struct {
	Commands  []string `arg:"-c,separate"`
	Files     []string `arg:"-f,separate"`
	Databases []string `arg:"positional"`
}
MustParse(&args)
fmt.Println("Commands:", args.Commands)
fmt.Println("Files", args.Files)
fmt.Println("Databases", args.Databases)
Output:

Example (MultipleValues)

This example demonstrates arguments that have multiple values

// The args you would pass in on the command line
os.Args = []string{"--help"}

var args struct {
	Database string
	IDs      []int64
}
MustParse(&args)
fmt.Printf("Fetching the following IDs from %s: %q", args.Database, args.IDs)
Output:

Example (PositionalArguments)

This example demonstrates positional arguments

// These are the args you would pass in on the command line
os.Args = []string{"./example", "in", "out1", "out2", "out3"}

var args struct {
	Input  string   `arg:"positional"`
	Output []string `arg:"positional"`
}
MustParse(&args)
fmt.Println("Input:", args.Input)
fmt.Println("Output:", args.Output)
Output:

Example (RequiredArguments)

This example demonstrates arguments that are required

// These are the args you would pass in on the command line
os.Args = []string{"--foo=1", "--bar"}

var args struct {
	Foo string `arg:"required"`
	Bar bool
}
MustParse(&args)
Output:

Example (UsageString)

This example shows the usage string generated by go-arg

// These are the args you would pass in on the command line
os.Args = []string{"--help"}

var args struct {
	Input    string   `arg:"positional"`
	Output   []string `arg:"positional"`
	Verbose  bool     `arg:"-v,help:verbosity level"`
	Dataset  string   `arg:"help:dataset to use"`
	Optimize int      `arg:"-O,help:optimization level"`
}
MustParse(&args)
Output:

Index

Examples

Constants

This section is empty.

Variables

View Source
var ErrHelp = errors.New("help requested by user")

ErrHelp indicates that -h or --help were provided

View Source
var ErrVersion = errors.New("version requested by user")

ErrVersion indicates that --version was provided

Functions

func Parse

func Parse(dest ...interface{}) error

Parse processes command line arguments and stores them in dest

Types

type Config

type Config struct {
	Program string // Program is the name of the program used in the help text
}

Config represents configuration options for an argument parser

type Described

type Described interface {
	// Description returns the string that will be printed on a line by itself
	// at the top of the help message.
	Description() string
}

Described is the interface that the destination struct should implement to make a description string appear at the top of the help message.

type Parser

type Parser struct {
	// contains filtered or unexported fields
}

Parser represents a set of command line options with destination values

func MustParse

func MustParse(dest ...interface{}) *Parser

MustParse processes command line arguments and exits upon failure

func NewParser

func NewParser(config Config, dests ...interface{}) (*Parser, error)

NewParser constructs a parser from a list of destination structs

func (*Parser) Fail

func (p *Parser) Fail(msg string)

Fail prints usage information to stderr and exits with non-zero status

func (*Parser) Parse

func (p *Parser) Parse(args []string) error

Parse processes the given command line option, storing the results in the field of the structs from which NewParser was constructed

func (*Parser) WriteHelp

func (p *Parser) WriteHelp(w io.Writer)

WriteHelp writes the usage string followed by the full help string for each option

func (*Parser) WriteUsage

func (p *Parser) WriteUsage(w io.Writer)

WriteUsage writes usage information to the given writer

type Versioned

type Versioned interface {
	// Version returns the version string that will be printed on a line by itself
	// at the top of the help message.
	Version() string
}

Versioned is the interface that the destination struct should implement to make a version string appear at the top of the help message.

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL