Skip to content
/ fisk Public

A fluent-style, type-safe command-line parser for Go.

License

Notifications You must be signed in to change notification settings

choria-io/fisk

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Overview

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.

Versions

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

Notable Changes

  • Renamed master branch to main
  • Incorporate github.com/alecthomas/units and github.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 since v0.3.0

UnNegatableBool

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.

Cheats

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.