Fisk is a fluent-style, type-safe command-line parser. It supports flags, nested commands, and positional arguments.
This is a fork of kingpin, a very nice CLI framework that has been in limbo for a few years. As this project and others we work on are heavily invested in Kingpin we thought to revive it for our needs. We'll likely make some breaking changes, so kingpin is kind of serving as a starting point here rather than this being designed as a direct continuation of that project.
For full help and intro see kingpin, this README will focus on our local changes.
We are not continuing the versioning scheme of Kingpin, the Go community has introduced onerous SemVer restrictions, we will start from 0.0.1 and probably never pass 0.x.x.
Some historical points in time are kept:
Tag | Description |
---|---|
v0.0.1 | Corresponds to the v2.2.6 release of Kingpin |
v0.0.2 | Corresponds with the master of Kingpin at the time of this fork |
v0.1.0 | The first release under choria-io org |
- Renamed
master
branch tomain
- Incorporate
github.com/alecthomas/units
andgithub.com/alecthomas/template
as local packages - Changes to make
staticcheck
happy - A new default template that shortens the help on large apps, old default preserved as
KingpinDefaultUsageTemplate
- Integration with cheat (see below)
- Unnegatable booleans using a new
UnNegatableBool()
flag type, backwards compatibility kept - Extended parsing for durations that include weeks (
w
,W
), months (M
), years (y
,Y
) and days (d
,D
) units (v0.1.3
or newer) - More contextually useful help when using
app.MustParseWithUsage(os.Args[1:])
(v0.1.4
or newer) - Default usage template is
CompactMainUsageTemplate
sincev0.3.0
Fisk will add to all Bool()
kind flags a negated version, in other words --force
will also get --no-force
added
and the usage will show these negatable booleans.
Often though one does not want to have the negatable version of a boolean added, with fisk you can achieve this using
our UnNegatableBool()
which would just be the basic boolean flag with no negatable version.
I really like cheat, a great little tool that gives access to bite-sized hints on what's great about a CLI tool.
Since v0.1.1
Fisk supports cheats natively, you can get cheat formatted hints right from the app with no extra dependencies or export cheats into the cheat
app for use via its interface and integrations.
$ nats cheat pub
# To publish 100 messages with a random body between 100 and 1000 characters
nats pub destination.subject "{{ Random 100 1000 }}" -H Count:{{ Count }} --count 100
Cheats are stored in a map[string]string
, meaning it's flat, does not support subs and when saving cheats, due to the
nature of the fluent api, 2 cheats with the same name will overwrite each other.
I therefore suggest you place your cheat in the top command for an intro and then place them where you need them in the first sub command only not deeper, this makes it easy to avoid clashes and easy for your users to discover them.
Let's look how that is done:
// WithCheats() enables cheats without adding any to the top, you
// can also just call Cheat() at the top to both set a cheat and enable it
// once enabled at the top all cheats in all sub commands are accessible
//
// Cheats can have multiple tags, here we set the tags "middleware", and "nats"
// that will be used when saving the cheats. If no tags are supplied the
// application name is used as the only tag
nats := fisk.New("nats", "NATS Utility").WithCheats("middleware", "nats")
pub := nats.Command("pub", "Publish utility")
// The cheat will be available as "pub", the if the first argument
// is empty the name of the command will be used
pub.Cheat("pub", `# To publish 100 messages with a random.....`)
After that your app will have a new command cheat
that gives access to the cheats. It will show
a list of cheats when trying to access a command without cheats or when running nats cheat --list
.
$ nats cheat unknown
Available Cheats:
pub
You can save your cheats to a directory of your choice with nats cheat --save /some/dir
, the directory
must not already exist.