goprogram

package module
v0.0.9 Latest Latest
Warning

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

Go to latest
Published: Aug 26, 2022 License: MIT Imports: 9 Imported by: 4

README

goprogram

CI Status

A go 1.18 package to create programs, originally designed for ggman. Documentation is a work in progress.

Changelog

0.0.10 (Upcoming)

0.0.9 (Released Aug 26 2022)

  • add BeforeKeyword, BeforeAlias, BeforeCommand hooks
  • add a method StdinIsATerminal to check if stdin is a terminal

0.0.8 (Released Aug 17 2022)

  • add ReadLine, ReadPassword and ReadPasswordStrict methods
  • minor go1.19 formatting

0.0.7 (Released May 2 2022)

  • remove BeforeRegister method and pass program in context
  • copy commands before executing and make sure they become pointers

0.0.6 (Released Apr 28 2022)

  • extend doccheck package into docfmt package

0.0.5 (Released Apr 18 2022)

  • add doccheck package

0.0.4 (Released Apr 15 2022)

  • refactor argument parsing

0.0.3 (Released Mar 29 2022)

  • add EmptyRequirement struct

0.0.2 (Released Mar 16 2022)

  • use golang.org/x/exp/slices package
  • name type parameters consistently

0.0.1 (Released Mar 9 2022)

  • initial release

Documentation

Overview

Package goprogram provides a program abstraction that can be used to create programs.

Index

Examples

Constants

This section is empty.

Variables

This section is empty.

Functions

func ValidateAllowedFlags

func ValidateAllowedFlags[F any](r Requirement[F], args Arguments[F]) error

Validate validates that every flag f in args.flags either passes the AllowsOption method of the given requirement, or has the zero value. If this is not the case returns an error of type ValidateAllowedFlags.

This function is intended to be used to implement the validate method of a Requirement.

Types

type AfterParseCommand added in v0.0.4

type AfterParseCommand[E any, P any, F any, R Requirement[F]] interface {
	Command[E, P, F, R]

	// AfterParse is called after arguments have been parsed, but before the command is being run.
	// It may perform additional argument checking and should return an error if needed.
	//
	// It is called only once and must return either nil or an error of type Error.
	AfterParse() error
}

AfterParseCommand represents a command with an AfterParse function

type Alias

type Alias struct {
	// Name is the name of this alias
	Name string

	// Command to invoke along with arguments
	Command string
	Args    []string

	// Description for the usage page
	Description string
}

Alias represents an alias for a command.

Expansion of an alias takes place at runtime. Aliases must not contain global flags; execution of the them will fail at runtime.

Aliases are not expanded recursively, meaning one alias may not refer to itself or another. An alias always takes precedence over a command with the same name.

Example

Register an alias for a program. See the test suite for instaniated types.

// create a new program that only has an echo command
// this code is reused across the test suite, hence not shown here.
p := makeProgram()
p.Register(makeEchoCommand("echo"))

// register an alias "hello" which expands into hello world
p.RegisterAlias(Alias{Name: "hello", Command: "echo", Args: []string{"hello world"}})

p.Main(stream.FromEnv(), "", []string{"hello"})
p.Main(stream.FromEnv(), "", []string{"hello", "again"})
Output:

[hello world]
[hello world again]

func (Alias) Expansion

func (a Alias) Expansion() []string

Expansion returns a slice representing the expansion of this alias.

func (Alias) Invoke

func (a Alias) Invoke(args []string) (command string, arguments []string)

Invoke returns command arguments that are to be used when invoking this alias args are additional arguments to pass to the command

type Arguments

type Arguments[F any] struct {
	Universals Universals `group:"universals"`
	Flags      F          `group:"flags"`

	Command string // command to run
	// contains filtered or unexported fields
}

Arguments represent a set of command-independent arguments passed to a command. These should be further parsed into CommandArguments using the appropriate Parse() method.

Command line argument are annotated using syntax provided by "github.com/jessevdk/go-flags".

type Command

type Command[E any, P any, F any, R Requirement[F]] interface {
	// Run runs this command in the given context.
	//
	// It is called only once and must return either nil or an error of type Error.
	Run(context Context[E, P, F, R]) error

	// Description returns a description of this command.
	// It may be called multiple times.
	Description() Description[F, R]
}

Command represents a command associated with a program. It takes the same type parameters as a program.

Each command is first initialized using any means by the user. Next, it is registered with a program using the Program.Register Method. Once a program is called with this command, the arguments for it are parsed by making use of the Description. Eventually the Run method of this command is invoked, using an appropriate new Context. See Context for details on the latter.

A command should be implemented as a struct or pointer to a struct. Typically command contains state that represents the parsed options. Flag parsing is implemented using the "github.com/jessevdk/go-flags" package.

To ensure consistency between runs, a command is copied before being executed. When command is a pointer, the underlying data (not the pointer to it) will be copied.

Example

Register a command for a program. See the test suite for instaniated types.

// create a new program that only has an echo command
// this code is reused across the test suite, hence not shown here.
p := makeProgram()
p.Register(makeEchoCommand("echo"))

// Execute the command with some arguments
p.Main(stream.FromEnv(), "", []string{"echo", "hello", "world"})
Output:

[hello world]

type Context

type Context[E any, P any, F any, R Requirement[F]] struct {
	// IOStream describes the input and output the command reads from and writes to.
	stream.IOStream

	// Args holds arguments passed to this command.
	Args Arguments[F]

	// Description is the description of the command being invoked
	Description Description[F, R]
	// Environment holds the environment for this command.
	Environment E

	// Program is the current program this command is executing under
	Program Program[E, P, F, R]
	// contains filtered or unexported fields
}

Context represents an execution environment for a command. it takes the same type parameters as a command and program.

type Description

type Description[F any, R Requirement[F]] struct {
	// Command and Description the name and human-readable description of this command.
	// Command must not be taken by any other command registered with the corresponding program.
	Command     string
	Description string

	ParserConfig parser.Config // information about how to configure a parser for this command

	// Requirements on the environment to be able to run the command
	Requirements R
}

Description describes a command, and specifies any potential requirements.

type EmptyRequirement added in v0.0.3

type EmptyRequirement[F any] struct{}

EmptyRequirement represents a requirement that allows any flag and validates all arguments

func (EmptyRequirement[F]) AllowsFlag added in v0.0.3

func (EmptyRequirement[F]) AllowsFlag(meta.Flag) bool

func (EmptyRequirement[F]) Validate added in v0.0.3

func (EmptyRequirement[F]) Validate(Arguments[F]) error

type Keyword

type Keyword[F any] func(args *Arguments[F], pos *[]string) error

Keywords are special "commands" that manipulate arguments and positionals before execution.

Keywords can not be stopped by calls to universal flags; they are expanded once before aliases and command expansion takes place.

type Program

type Program[E any, P any, F any, R Requirement[F]] struct {
	// Meta-information about the current program
	// Used to generate help and version pages
	Info meta.Info

	// The NewEnvironment function is used to create a new environment.
	// The returned error must be nil or of type exit.Error.
	//
	// NewEnvironment may be nil, in which case a new environment is assumed to be
	// the zero value of type E.
	NewEnvironment func(params P, context Context[E, P, F, R]) (E, error)

	// BeforeKeyword, BeforeAlias and BeforeCommand (if non-nil) are invoked right before their respective datum is executed.
	// They are intended to act as a guard before executing a particular datum.
	//
	// The returned error must be nil or of type exit.Error.
	// When non-nil, the error is returned to the caller of Main().
	BeforeKeyword func(context Context[E, P, F, R], keyword Keyword[F]) error
	BeforeAlias   func(context Context[E, P, F, R], alias Alias) error
	BeforeCommand func(context Context[E, P, F, R], command Command[E, P, F, R]) error
	// contains filtered or unexported fields
}

Program represents an executable program. A program is intended to be invoked on the command line. Each invocation of a program executes a command.

Programs have 4 type parameters: An environment of type E, a type of parameters P, a type of flags F and a type requirements R.

The Environment type E defines a runtime environment for commands to execute in. An Environment is created using the NewEnvironment function, taking parameters P.

The type of (global) command line flags F is backed by a struct type. It is jointed by a type of Requirements R which impose restrictions on flags for commands.

Internally a program also contains a list of commands, keywords and aliases.

See the Main method for a description of how program execution takes place.

func (Program[E, P, F, R]) AliasUsage

func (p Program[E, P, F, R]) AliasUsage(context Context[E, P, F, R], alias Alias) meta.Meta

AliasPage returns a usage page for the provided alias

func (Program[E, P, F, R]) Aliases

func (p Program[E, P, F, R]) Aliases() []string

Aliases returns the names of all registered aliases. Aliases are returned in sorted order.

func (Program[E, P, F, R]) Command added in v0.0.5

func (p Program[E, P, F, R]) Command(name string) (Command[E, P, F, R], bool)

Command returns the command with the provided name and if it exists

func (Program[E, P, F, R]) CommandUsage

func (p Program[E, P, F, R]) CommandUsage(context Context[E, P, F, R]) meta.Meta

CommandUsage generates the usage information about a specific command

func (Program[E, P, F, R]) Commands

func (p Program[E, P, F, R]) Commands() []string

Commands returns a list of known commands

func (Program[E, P, F, R]) FmtCommands

func (p Program[E, P, F, R]) FmtCommands() string

FmtCommands returns a human readable string describing the commands. See also Commands.

func (Program[E, P, F, R]) Main

func (p Program[E, P, F, R]) Main(stream stream.IOStream, params P, argv []string) (err error)

Main invokes this program and returns an error of type exit.Error or nil.

Main takes input / output streams, parameters for the environment and a set of command-line arguments.

It first parses these into arguments for a specific command to be executed. Next, it executes any keywords and expands any aliases. Finally, it executes the requested command or displays a help or version page.

For keyword actions, see Keyword. For alias expansion, see Alias. For command execution, see Command.

For help pages, see MainUsage, CommandUsage, AliasUsage. For version pages, see FmtVersion.

func (Program[E, P, F, R]) MainUsage

func (p Program[E, P, F, R]) MainUsage() meta.Meta

MainUsage returns a help page about ggman

func (*Program[E, P, F, R]) Register

func (p *Program[E, P, F, R]) Register(c Command[E, P, F, R])

Register registers a command c with this program.

It expects that the command does not have a name that is already taken.

func (*Program[E, P, F, R]) RegisterAlias

func (p *Program[E, P, F, R]) RegisterAlias(alias Alias)

RegisterAlias registers a new alias. See also Alias.

If an alias already exists, RegisterAlias calls panic().

func (*Program[E, P, F, R]) RegisterKeyword

func (p *Program[E, P, F, R]) RegisterKeyword(name string, keyword Keyword[F])

RegisterKeyword registers a new keyword. See also Keyword.

If an keyword already exists, RegisterKeyword calls panic().

type Requirement

type Requirement[F any] interface {
	// AllowsFlag checks if the provided flag may be passed to fullfill this requirement
	// By default it is used only for help page generation, and may be inaccurate.
	AllowsFlag(flag meta.Flag) bool

	// Validate validates if this requirement is fullfilled for the provided global flags.
	// It should return either nil, or an error of type exit.Error.
	//
	// Validate does not take into account AllowsOption, see ValidateAllowedOptions.
	Validate(arguments Arguments[F]) error
}

Requirement describes a requirement on a type of Flags F.

A caller may use EmptyRequirement if no such abstraction is desired.

type Universals

type Universals struct {
	Help    bool `short:"h" long:"help" description:"print a help message and exit"`
	Version bool `short:"v" long:"version" description:"print a version message and exit"`
}

Universals holds flags added to every executable.

Command line arguments are annotated using syntax provided by "github.com/jessevdk/go-flags".

Directories

Path Synopsis
Package exit defines exit and error behavior of programs and commands.
Package exit defines exit and error behavior of programs and commands.
lib
docfmt
Package docfmt implements formatting and checking of user format strings.
Package docfmt implements formatting and checking of user format strings.
reflectx
Package reflectx provides extensions to the reflect package
Package reflectx provides extensions to the reflect package
testlib
Package testlib provides utilities for testing
Package testlib provides utilities for testing
text
Package text provides functions similar to strings.Join, but based on writers as opposed to strings
Package text provides functions similar to strings.Join, but based on writers as opposed to strings
wrap
Package wrap provides facilities to wrap text
Package wrap provides facilities to wrap text
Package meta contains facilities to provide meta-information about programs and commands.
Package meta contains facilities to provide meta-information about programs and commands.
Package parser implements compatiblity with the "github.com/jessevdk/go-flags" package
Package parser implements compatiblity with the "github.com/jessevdk/go-flags" package
Package stream provides NullStream
Package stream provides NullStream

Jump to

Keyboard shortcuts

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