This library is on the unstable version v0.X.X, which means there is a chance that any minor update may introduce a breaking change. Where I will endeavor to avoid this, care should be taken updating your dependency on this library until the first stable release v1.0.0 at which point any future breaking changes will result in a new major release.
A simple command-line tool-set for Golang modeled after HTTP routing patterns.
go get -u github.com/evilmonkeyinc/golang-cli
import "github.com/evilmonkeyinc/golang-cli/shell"
...
newShell := new(shell.Shell)
...
Options adds the ability to customize the shell's properties for your project.
newShell.Options(shell.OptionOutputWriter(customOutput))
Options should be set before performing any other actions on the shell
Middleware adds the ability to wrap all shell handler functions in additional logic.
newShell.Use(middleware.Recoverer())
The most common use-case has been included in the library, panic recovery, which makes it possible for the shell to recover from a panic without exiting the interactive-shell
Middleware can also be used to manipulate the shell handler functions ResponseWriter and Request to add or manipulate the existing functionality
Groups allow you to define a new inline-router to the shell router stack.
newShell.Group(func(r shell.Router) {
...
})
An inline-routers handlers will be evaluated at the same level as the containing router, but makes it possible to define middleware that will be executed for the inline-router handlers only.
Routes allow you do define a sub-router which is executed as a shell handler, but include additional routing instructions to handle sub-commands for the shell.
newShell.Route("users", func(r shell.Router) {
r.HandleFunction("add", func(rw shell.ResponseWriter, r *shell.Request) error { return nil })
r.HandleFunction("list", func(rw shell.ResponseWriter, r *shell.Request) error { return nil })
})
The sub commands would then be executed by including spaces between the commands:
./yourcli users list
Routes will also support specific middleware for these sub-commands in the same way as the inline-routers created by Group.
The Handle and HandleFunction functions add shell handlers to the router stack.
newShell.Handle("help", &commands.HelpCommand{})
newShell.HandleFunction("ping", func(rw shell.ResponseWriter, r *shell.Request) error {
fmt.Fprintln(rw, "pong")
return nil
})
./yourcli ping
pong
It is possible to define global flags directly on the shell, or on each route using the Flags()
function
newShell := new(shell.Shell)
newShell.Flags(flags.FlagHandlerFunction(func(fd flags.FlagDefiner) {
fd.Bool("toUpper", false, "")
}))
It is also possible to allow individual commands define flags if the Handler also conforms to the FlagHandler interface, as the sample Command struct does.
pingCommand := &commands.Command{
...
Flags: func(fd flags.FlagDefiner) {
fd.String("suffix", "", "")
},
...
}
These flags can be set on the command-line at any point after they are defined, so if they are defined globally on the shell then it could be set any time after the package is executed and would give the same result
.my-cli -toUpper ping
> PONG
.my-cli ping -toUpper
> PONG
but those flags defined in routes or commands will be ignored if used before they are defined
.my-cli ping -suffix=go
> ponggo
.my-cli -suffix=go ping
> flag provided but not defined: -suffix
> pong
-
CLI Example
Using this library for a standard command-line interface tool. -
Shell Example
Using this library for an interactive-shell interface.
The following projects were used as references and inspiration for this project