gookit/gcli 一个 Golang 下的简单易用的命令行应用

发布于 2021-06-27 21:06:03 字数 22922 浏览 1494 评论 0

一个Golang下的简单易用的命令行应用,工具库。包含运行命令,颜色风格,数据展示,进度显示,交互方法等。

功能特色

  • 使用简单方便,功能丰富
  • 支持添加多个命令,并且支持给命令添加别名
  • 支持从结构体绑定命令选项
    • 示例 flag:"name=int0;shorts=i;required=true;desc=int option message"
  • 支持添加多级命令,每级命令均支持绑定自己的选项
  • option/flag 快速方便的添加选项绑定(--long),支持添加多个短选项(eg: -s
    • 选项支持设置 Required,表明为必须的选项参数
    • 选项支持设置 Validator,可以自定义验证输入参数
  • argument 支持绑定 参数 到指定名称(参数 是指flag绑定后剩余的参数信息)
    • 支持参数设置 可选/必须 Required,数组 isArray
    • 运行命令时将会自动检测,并按对应关系收集参数
  • colorable 支持丰富的颜色渲染输出, 由 gookit/color 提供
    • 同时支持html标签式的颜色渲染,兼容Windows
    • 内置info,error,success,danger等多种风格,可直接使用
  • interact 内置提供用户交互方法: ReadLine, Confirm, Select, MultiSelect
  • progress 内置提供进度显示方法: Txt, Bar, Loading, RoundTrip, DynamicText
  • 输入的命令错误时,将会提示相似命令(包含别名提示)
  • 自动处理返回错误,error 会自动渲染为错误提示信息
  • 自动根据命令生成帮助信息,并且支持颜色显示
  • 支持为当前CLI应用生成 zsh,bash 下的命令补全脚本文件
  • 支持将单个命令当做独立应用运行

安装

go get github.com/gookit/gcli/v3

快速开始

如下,引入当前包就可以快速的编写cli应用了

package main

import (
    "runtime"
    "github.com/gookit/gcli/v3"
    "github.com/gookit/gcli/v3/_examples/cmd"
)

// 测试运行: go run ./_examples/cliapp.go && ./cliapp
func main() {
    app := gcli.NewApp()
    app.Version = "1.0.3"
    app.Desc = "this is my cli application"
    // app.SetVerbose(gcli.VerbDebug)

    app.Add(cmd.Example)
    app.Add(&gcli.Command{
        Name: "demo",
        // allow color tag and {$cmd} will be replace to 'demo'
        Desc: "this is a description <info>message</> for command", 
        Aliases: []string{"dm"},
        Func: func (cmd *gcli.Command, args []string) error {
            gcli.Println("hello, in the demo command")
            return nil
        },
    })

    // .... add more ...

    app.Run()
}

使用说明

先使用本项目下的 demo 示例代码构建一个小的cli demo应用

% go build ./_examples/cliapp.go

打印版本信息

打印我们在创建cli应用时设置的版本信息。如果你还设置了字符LOGO,也会显示出来。

% ./cliapp --version
# or use -V
% ./cliapp -V

显示应用帮助信息

使用 ./cliapp 或者 ./cliapp -h 来显示应用的帮助信息,包含所有的可用命令和一些全局选项

示例:

./cliapp
./cliapp -h # can also
./cliapp --help # can also

显示一个命令的帮助

显示一个指定命令的帮助信息

示例:

./cliapp {command} -h
./cliapp {command} --help
./cliapp help {command}

相似命令提示

输入了错误的命令,但是有名称相似的会提示出来。

运行一个命令

语法结构:

./cliapp COMMAND [--OPTION VALUE -S VALUE ...] [ARGUMENT0 ARGUMENT1 ...]

示例

./cliapp ex -c some.txt -d ./dir --id 34 -n tom -n john val0 val1 val2 arrVal0 arrVal1 arrVal2

可以观察到选项和参数的搜集结果:

生成命令补全脚本

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

    // ...
    // 添加内置提供的生成命令
    app.Add(builtin.GenAutoComplete())

构建并运行生成命令(生成成功后可以去掉此命令):

% go build ./_examples/cliapp.go && ./cliapp genac -h // 使用帮助
% go build ./_examples/cliapp.go && ./cliapp genac // 开始生成, 你将会看到类似的信息
INFO: 
  {shell:zsh binName:cliapp output:auto-completion.zsh}

Now, will write content to file auto-completion.zsh
Continue? [yes|no](default yes): y

OK, auto-complete file generate successful

运行后就会在当前目录下生成一个 auto-completion.{zsh|bash} 文件, shell 环境名是自动获取的。当然你可以在运行时手动指定

生成的shell script 文件请参看:

预览效果:

编写命令

简单使用

app.Add(&gcli.Command{
    Name: "demo",
    // allow color tag and {$cmd} will be replace to 'demo'
    Desc: "this is a description <info>message</> for command", 
    Aliases: []string{"dm"},
    Func: func (cmd *gcli.Command, args []string) error {
        gcli.Print("hello, in the demo command\n")
        return nil
    },
})

使用独立的文件

the source file at: example.go

package cmd

import (
  "fmt"

  "github.com/gookit/color"
  "github.com/gookit/gcli/v3"
)

// options for the command
var exampleOpts = struct {
  id  int
  c   string
  dir string
  opt string
  names gcli.Strings
}{}

// ExampleCommand command definition
var ExampleCommand = &gcli.Command{
  Name:        "example",
  Desc: "this is a description message",
  Aliases:     []string{"exp", "ex"}, // 命令别名
  // {$binName} {$cmd} is help vars. '{$cmd}' will replace to 'example'
  Examples: `{$binName} {$cmd} --id 12 -c val ag0 ag1
<cyan>{$fullCmd} --names tom --names john -n c</> test use special option`,
  Config: func(c *gcli.Command) {
    // 绑定命令选项信息
    c.IntOpt(&exampleOpts.id, "id", "", 2, "the id option")
    c.StrOpt(&exampleOpts.c, "config", "c", "value", "the config option")
    // notice `DIRECTORY` will replace to option value type
    c.StrOpt(&exampleOpts.dir, "dir", "d", "", "the `DIRECTORY` option")
    // 支持设置选项短名称
    c.StrOpt(&exampleOpts.opt, "opt", "o", "", "the option message")
    // 支持绑定自定义变量, 但必须实现 flag.Value 接口
    c.VarOpt(&exampleOpts.names, "names", "n", "the option message")

    // 绑定命令参数信息,按参数位置绑定
    c.AddArg("arg0", "the first argument, is required", true)
    c.AddArg("arg1", "the second argument, is required", true)
    c.AddArg("arg2", "the optional argument, is optional")
    c.AddArg("arrArg", "the array argument, is array", false, true)
  },
  Func:  exampleExecute,
}

// 命令执行主逻辑代码
// example run:
//   go run ./_examples/cliapp.go ex -c some.txt -d ./dir --id 34 -n tom -n john val0 val1 val2 arrVal0 arrVal1 arrVal2
func exampleExecute(c *gcli.Command, args []string) error {
  fmt.Print("hello, in example command\n")

  magentaln := color.Magenta.Println

  magentaln("All options:")
  fmt.Printf("%+v\n", exampleOpts)
  magentaln("Raw args:")
  fmt.Printf("%v\n", args)

  magentaln("Get arg by name:")
  arr := c.Arg("arrArg")
  fmt.Printf("named array arg '%s', value: %v\n", arr.Name, arr.Value)

  magentaln("All named args:")
  for _, arg := range c.Args() {
    fmt.Printf("named arg '%s': %+v\n", arg.Name, *arg)
  }

  return nil
}
  • 查看此命令的帮助信息:
go build ./_examples/cliapp.go && ./cliapp example -h

漂亮的帮助信息就已经自动生成并展示出来了

添加选项

添加选项可用的方法:

BoolOpt(p *bool, name, shorts string, defValue bool, desc string)
BoolVar(p *bool, meta FlagMeta)
Float64Opt(p *float64, name, shorts string, defValue float64, desc string)
Float64Var(p *float64, meta FlagMeta)
Int64Opt(p *int64, name, shorts string, defValue int64, desc string)
Int64Var(p *int64, meta FlagMeta)
IntOpt(p *int, name, shorts string, defValue int, desc string)
IntVar(p *int, meta FlagMeta)
StrOpt(p *string, name, shorts, defValue, desc string)
StrVar(p *string, meta FlagMeta)
Uint64Opt(p *uint64, name, shorts string, defValue uint64, desc string)
Uint64Var(p *uint64, meta FlagMeta)
UintOpt(p *uint, name, shorts string, defValue uint, desc string)
UintVar(p *uint, meta FlagMeta)
Var(p flag.Value, meta FlagMeta)
VarOpt(p flag.Value, name, shorts, desc string)

Usage examples:

var id int
var b bool
var opt, dir string
var f1 float64
var names gcli.Strings

// bind options
cmd.IntOpt(&id, "id", "", 2, "the id option")
cmd.BoolOpt(&b, "bl", "b", false, "the bool option")
// notice `DIRECTORY` will replace to option value type
cmd.StrOpt(&dir, "dir", "d", "", "the `DIRECTORY` option")
// setting option name and short-option name
cmd.StrOpt(&opt, "opt", "o", "", "the option message")
// setting a special option var, it must implement the flag.Value interface
cmd.VarOpt(&names, "names", "n", "the option message")

绑定参数

关于参数定义:

  • 必须的 参数不能定义在 可选参数 之后
  • 只允许有一个数组参数(多个值的)
  • 数组参数只能定义在最后

绑定参数可用的方法:

Add(arg Argument) *Argument
AddArg(name, desc string, requiredAndIsArray ...bool) *Argument
AddArgument(arg *Argument) *Argument
BindArg(arg Argument) *Argument

用法示例:

cmd.AddArg("arg0", "the first argument, is required", true)
cmd.AddArg("arg1", "the second argument, is required", true)
cmd.AddArg("arg2", "the optional argument, is optional")
cmd.AddArg("arrArg", "the array argument, is array", false, true)

也可以使用 Add()/BindArg():

cmd.Add("arg0", gcli.Argument{
  Name: "ag0",
  Desc: "the first argument, is required",
  Require: true,
})
cmd.BindArg("arg0", gcli.Argument{
  Name: "ag0",
  Desc: "the second argument, is required",
  Require: true,
})
cmd.Add("arg2", gcli.Argument{
  Name: "ag0",
  Desc: "the third argument, is is optional",
})

cmd.BindArg("arrArg", gcli.Argument{
  Name: "arrArg",
  Desc: "the third argument, is is array",
  IsArray: true,
})

获取参数

可以通过 c.Arg(name string) *gcli.Argument 获取参数,通过上面内置的方法可以将参数转换文常用的数据类型

var MyCommand = &gcli.Command{
    Name: "example",
    Desc: "this is an example command",
    Config: func(c *gcli.Command) {
        cmd.BindArg("arg0", gcli.Argument{
            Name: "ag0",
            Desc: "the first argument, is required",
            Require: true,
        })
        cmd.Add("arg1", gcli.Argument{
            Name: "ag1",
            Desc: "the second argument, is is optional",
        })
    },
    Func: func(c *gcli.Command, args []string) error {
        arg0 := c.Arg("arg0").String()
        arg1 := c.Arg("arg1").Int()
        
        fmt.Println(arg0, arg1)
        return nil
    },
}

进度显示

progress.Bar 通用的进度条

Demo: ./cliapp prog bar

progress.Txt 文本进度条

Demo: ./cliapp prog txt

progress.LoadBar 加载中

  • progress.Counter 计数
  • progress.RoundTrip 来回滚动的进度条
[===     ] -> [    === ] -> [ ===    ]

  • progress.DynamicText 动态消息,执行进度到不同的百分比显示不同的消息

示例:

package main

import "time"
import "github.com/gookit/gcli/v3/progress"

func main()  {
  speed := 100
  maxSteps := 110
  p := progress.Bar(maxSteps)
  p.Start()

  for i := 0; i < maxSteps; i++ {
    time.Sleep(time.Duration(speed) * time.Millisecond)
    p.Advance()
  }

  p.Finish()
}

更多示例和使用请看 progress_demo.go

运行示例:

go run ./_examples/cliapp.go prog txt
go run ./_examples/cliapp.go prog bar
go run ./_examples/cliapp.go prog roundTrip

交互方法

控制台交互方法,包含读取输入,进行确认,单选,多选,询问问题等等

  • interact.ReadInput
  • interact.ReadLine
  • interact.ReadFirst
  • interact.Confirm
  • interact.Select/Choice
  • interact.MultiSelect/Checkbox
  • interact.Question/Ask
  • interact.ReadPassword

示例:

package main

import (
  "fmt"

  "github.com/gookit/gcli/v3/interact"
)

func main() {
  username, _ := interact.ReadLine("Your name?")
  password := interact.ReadPassword("Your password?")
  
  ok := interact.Confirm("ensure continue?")
  if !ok {
    // do something...
  }
    
  fmt.Printf("username: %s, password: %s\n", username, password)
}

读取输入

ans, _ := interact.ReadLine("Your name? ")

if ans != "" {
    color.Println("Your input: ", ans)
} else {
    color.Cyan.Println("No input!")
}

单选

ans := interact.SelectOne(
    "Your city name(use array)?",
    []string{"chengdu", "beijing", "shanghai"},
    "",
)
color.Comment.Println("your select is: ", ans)

多选

ans := interact.MultiSelect(
    "Your city name(use array)?",
    []string{"chengdu", "beijing", "shanghai"},
    nil,
)
color.Comment.Println("your select is: ", ans)

确认消息

if interact.Confirm("Ensure continue") {
    fmt.Println(emoji.Render(":smile: Confirmed"))
} else {
    color.Warn.Println("Unconfirmed")
}

读取密码输入

pwd := interact.ReadPassword()
color.Comment.Println("your input password is: ", pwd)

更多示例和使用请看 interact_demo.go

使用颜色输出

颜色输出使用 gookit/color 支持在windows cmd.exe powerShell 环境

  • 颜色输出展示

如何使用

package main

import (
    "github.com/gookit/color"
)

func main() {
  // simple usage
  color.Cyan.Printf("Simple to use %s\n", "color")

  // internal theme/style:
  color.Info.Tips("message")
  color.Info.Prompt("message")
  color.Info.Println("message")
  color.Warn.Println("message")
  color.Error.Println("message")
  
  // custom color
  color.New(color.FgWhite, color.BgBlack).Println("custom color style")

  // can also:
  color.Style{color.FgCyan, color.OpBold}.Println("custom color style")
  
  // use defined color tag
  color.Print("use color tag: <suc>he</><comment>llo</>, <cyan>wel</><red>come</>\n")

  // use custom color tag
  color.Print("custom color tag: <fg=yellow;bg=black;op=underscore;>hello, welcome</>\n")

  // set a style tag
  color.Tag("info").Println("info style text")

  // prompt message
  color.Info.Prompt("prompt style message")
  color.Warn.Prompt("prompt style message")

  // tips message
  color.Info.Tips("tips style message")
  color.Warn.Tips("tips style message")
}
  • 构建风格
// 仅设置前景色
color.FgCyan.Printf("Simple to use %s\n", "color")
// 仅设置背景色
color.BgRed.Printf("Simple to use %s\n", "color")

// 完全自定义 前景色 背景色 选项
style := color.New(color.FgWhite, color.BgBlack, color.OpBold)
style.Println("custom color style")

// can also:
color.Style{color.FgCyan, color.OpBold}.Println("custom color style")
  • 使用内置风格
color.Bold.Println("bold message")
color.Yellow.Println("yellow message")

扩展风格主题:

color.Info.Println("Info message")
color.Success.Println("Success message")
  • 使用颜色html标签

同样 支持 在windows cmd.exe powerShell 使用颜色标签

使用颜色标签可以非常方便简单的构建自己需要的任何格式

// 使用内置的 color tag
color.Print("<suc>he</><comment>llo</>, <cyan>wel</><red>come</>")
color.Println("<suc>hello</>")
color.Println("<error>hello</>")
color.Println("<warning>hello</>")

// 自定义颜色属性
color.Print("<fg=yellow;bg=black;op=underscore;>hello, welcome</>\n")

更多关于颜色库的使用请访问 gookit/color

参考项目

github 地址:https://github.com/gookit/gcli

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

JSmiles

生命进入颠沛而奔忙的本质状态,并将以不断告别和相遇的陈旧方式继续下去。

文章
评论
84963 人气
更多

推荐作者

微信用户

文章 0 评论 0

小情绪

文章 0 评论 0

ゞ记忆︶ㄣ

文章 0 评论 0

笨死的猪

文章 0 评论 0

彭明超

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文