gookit/gcli v3.8:新增支持共享选项、文档生成和参数重排

20 小时 22 分钟前
 jxia

GCli v3.8 这一轮补的不是花哨功能,而是写复杂 CLI 时经常会撞到的几个缺口:参数可以写在位置参数后面,结构体绑定支持更多 Go 类型,父命令可以定义会被子命令继承的共享选项,还能直接生成 markdown 和 man 文档。

如果你的 CLI 已经有多级命令、共享配置和文档生成需求,这版会少写不少胶水代码。

v3.8 补了哪些缺口

先把范围列出来:

还有一个包名调整:events 改名为 gevent,同时在 gcli 包里补了 Evt* 事件名别名。

参数可以写在位置参数后面

GCli 以前和标准库 flag 一样,遇到第一个非 flag 参数后就停止解析选项。下面这种输入里,--name tom 可能不会按预期进入选项解析:

myapp build src/ --name tom

从 v3.6 开始,GCli 会在解析前做一次参数重排。下面两种写法现在结果一致:

myapp build --name tom src/
myapp build src/ --name tom

这个功能默认开启。边界也做了处理:取值型选项会带上自己的值,bool 选项不会误吞后续参数,--opt=val、负数、单独的 --- 之后的内容都会保留原语义。

多级命令里也做了限制:只重排最终执行命令的参数。遇到子命令名后,父命令和子命令的选项不会混在一起。

如果你的程序依赖标准库那种严格顺序,可以关掉:

c.ParserCfg().DisableReorderArgs = true

// 或通过配置函数关闭
gflag.WithReorderArgs(false)

示例在 _examples/cmd/reorder_demo.go

结构体绑定支持更多 Go 类型

结构体绑定是 GCli 里很常用的一块。v3.7 之前,如果要绑定列表、map 或时长,经常要绕到 gflag.StringsKVString 这类辅助类型上。现在可以直接使用常见 Go 类型:

type deployOpts struct {
    Names []string          `flag:"name=names;shorts=n;desc=name list"`
    Ports []int             `flag:"name=ports;shorts=p;desc=port list"`
    TTL   time.Duration     `flag:"name=ttl;desc=time to live, eg: 1h30m"`
    Meta  map[string]string `flag:"name=meta;shorts=m;desc=key=value metadata"`
    Lang  string            `flag:"name=lang;shorts=l;desc=language;enum=go,php,java"`
}

c.MustFromStruct(&deployOpts{})

对应命令可以这样写:

myapp deploy -n a -n b -p 80 -p 443 --ttl 1h30m -m k1=v1 -m k2=v2 -l go

实际变化有几条:

实现上,结构体绑定器不再使用 unsafe,改为通过 reflect.Value.Addr().Interface() 获取字段指针。用户侧基本无感,但维护成本低一些。

另外,匿名嵌套结构体现在在 namedsimplefield 三种标签规则下都会展开。v3.8 还修复了内嵌未导出类型此前不会展开的问题。

泛型 API 少记几个方法名

传统 API 仍然保留:BoolVarIntVarStrVarFloat64Var 这些都能继续用。

v3.7 新增了一个泛型入口,会根据指针类型选择对应的绑定器:

var (
    name string
    age  int
    tags []string
    ttl  time.Duration
)

gflag.Opt(&c.Flags, &name, "name", "n", "tom", "the user name")
gflag.Opt(&c.Flags, &age,  "age",  "a", 18,    "the user age")
gflag.Opt(&c.Flags, &tags, "tag",  "t", nil,   "the tags, repeatable")
gflag.Opt(&c.Flags, &ttl,  "ttl",  "",  time.Duration(0), "time to live")

如果需要完整控制选项元数据,可以用 gflag.BindVar[T]

var langs []string
gflag.BindVar(&c.Flags, &langs, gflag.NewOpt("langs", "language list", nil))

这不是替代所有旧 API 的大改造,只是给新代码一个更顺手的入口。

SharedOpts 补上了父子命令之间的中间层

v3.8 里最值得单独拿出来说的是 Command.SharedOpts()

GCli 以前有应用级全局选项和命令级局部选项,但缺少一个“父命令定义,子命令继承”的中间层。SharedOpts() 补的就是这层语义,接近 cobra 的 PersistentFlags

var gitDir string

top := &gcli.Command{Name: "git", Desc: "git-like demo"}
top.SharedOpts().StrOpt(&gitDir, "git-dir", "", ".git", "the git data dir")

top.Add(&gcli.Command{
    Name: "status",
    Func: func(c *gcli.Command, _ []string) error {
        gcli.Printf("git dir: %s\n", gitDir)
        return nil
    },
})

下面几种写法都会让 gitDir 得到 /x

myapp git --git-dir /x status
myapp git status --git-dir /x
myapp git status arg --git-dir /x

几个规则需要注意:

底层新增了 Parser.InheritOptsFrom(src, category...),继承选项会按底层 flag.Value 重新注册,所以父子命令写的是同一个绑定变量。

docgen 可以直接生成 markdown 和 man page

v3.8 新增了 docgen 包,可以把单个命令或整个应用导出成 markdown 或 man page:

import "github.com/gookit/gcli/v3/docgen"

docgen.MarkdownTree(app, "./docs")
docgen.ManTree(app, "./man")

md  := docgen.CmdMarkdown(cmd)
man := docgen.CmdMan(cmd)

也可以从命令行触发。把内置命令注册进去:

import "github.com/gookit/gcli/v3/builtin"

app.Add(builtin.GenDoc())

然后执行:

./cliapp gendoc -f md  -o ./docs
./cliapp gendoc -f man -o ./man

生成器会顺手处理命令行文档里容易漏掉的小细节:清理 Examples 里的颜色标签,展开 {$fullCmd} 这类内置变量,保留多行示例,并在应用概览里写入版本信息。选项表也会通过 gflag.CliOpt.TypeName() 带上类型。

迁移注意点

从 v3.5 升上来,有两处可能需要改代码。

事件包改名:

之前 现在
import "github.com/gookit/gcli/v3/events" import "github.com/gookit/gcli/v3/gevent"
events.OnCmdRunBefore gevent.OnCmdRunBeforegcli.EvtCmdRunBefore

应用级解析状态移动:

之前 现在
GlobalOpts.ShowHelp / ShowVersion / inCompletion / genCompletion 新的应用级 AppOptions,通过 app.AppOpts() 访问

App.Opts() 仍然返回进程级的 *GlobalOpts,所以 app.Opts().Verboseapp.Opts() == gcli.GOpts() 这类用法不变。移动的是那些描述单个 App 解析状态的字段,避免同一进程内多个 App 实例互相影响。

参数自动重排也是行为变化,只是它通常会让输入更宽容。如果你确实依赖旧行为,可以设置 Config.DisableReorderArgs = true

升级

go get -u github.com/gookit/gcli/v3@latest

升级后建议先看三个示例:reorder-argsstruct-typesstruct-flag。它们覆盖了这次最容易影响实际代码的几块能力。

问题和建议可以继续在 GitHub Issues 里提。

329 次点击
所在节点    Go 编程语言
0 条回复

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/1224117

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX