当前位置: 首页 > news >正文

cobra - 更容易地构建命令行应用

cobra 是什么

cobra 的主要功能是创建强大的现代 cli 应用程序。目前市面上许多的著名的 Go 语言开源项目都是使用 Cobra 来构建的,例如:K8s、Hugo、etcd、Docker 等,是非常可靠的一个开源项目。

没有 cobra 之前用什么

如果不用 cobra,我们也可以使用 go 自带的 flag 标准库

flag 的基本用法

下面代码中,我们调用标准库 flagStringVar 方法实现了对命令行参数 name 的解析和绑定,其各个形参的含义分别为命令行标识位的名称、默认值、帮助信息。

命令行参数支持如下三种命令行标志语法:

  • -flag 仅支持布尔类型
  • -flag x 仅支持非布尔类型
  • -flag=x 均支持
package mainimport ("flag""fmt"
)func main() {var name string// 名称为 name、默认值为 "Go go go!"flag.StringVar(&name, "name", "Go go go!", "帮助信息")flag.StringVar(&name, "n", "Go go go!", "帮助信息")flag.Parse()fmt.Printf("name: %s\n", name)
}

执行:

// 均输出 abc
go run main.go -name=abc
go run main.go -n=abc

子命令实现

在我们日常使用的 CLI 应用中,另一个最常见的功能就是子命令的使用,一个工具它可能包含大量相关联的功能命令以此形成工具集,可以说是刚需,那么这个功能在标准库 flag 中可以如何实现呢,如下述示例:

package mainimport ("flag""log"
)var name stringfunc main() {flag.Parse()args := flag.Args()if len(args) == 0 {return}switch args[0] {case "go":goCmd := flag.NewFlagSet("go", flag.ExitOnError)goCmd.StringVar(&name, "name", "Go 语言", "帮助信息")_ = goCmd.Parse(args[1:])case "php":phpCmd := flag.NewFlagSet("php", flag.ExitOnError)phpCmd.StringVar(&name, "n", "PHP 语言", "帮助信息")_ = phpCmd.Parse(args[1:])}log.Printf("name: %s", name)
}

在上述代码中,我们首先调用了 flag.Parse 方法,将命令行解析为定义的标志,便于我们后续的参数使用。

另外由于我们需要处理子命令的情况,因此我们调用了 flag.NewFlagSet 方法,该方法会返回带有指定名称和错误处理属性的空命令集给我们去使用,相当于就是创建了一个新的命令集去支持子命令了。

这里需要特别注意的是 flag.NewFlagSet 方法的第二个参数是 ErrorHandling,用于指定处理异常错误,其内置提供以下三种模式:

const (// 返回错误描述ContinueOnError ErrorHandling = iota// 调用 os.Exit(2) 退出程序ExitOnError// 调用 panic 语句抛出错误异常PanicOnError
)
➜ go run main.go go    
2022/08/04 08:59:21 name: Go 语言
➜ go run main.go php  
2022/08/04 09:00:54 name: PHP 语言
➜ go run main.go php -n abc   
2022/08/04 09:01:05 name: abc

使用 cobra

安装:

go get -u github.com/spf13/cobra

示例:

package mainimport ("fmt""github.com/spf13/cobra""log"
)func main() {var wordCmd = &cobra.Command{Use:   "test",         // 子命令的命令标识Short: "测试",           // 简短说明Long:  "测试 cobra 子命令", // 完整说明Run: func(cmd *cobra.Command, args []string) {fmt.Println(args)},}var rootCmd = &cobra.Command{}rootCmd.AddCommand(wordCmd)err := rootCmd.Execute()if err != nil {log.Fatalf("cmd.Execute err: %v", err)}
}

在上面的例子中,我们添加了一个 test 子命令,使用:

➜  go run main.go test abc
[abc]

这里把 wordCmdrootCmd 都写在一块不是一个好的实践,如果命令多的情况下,这个文件会非常大。

http://www.lryc.cn/news/291504.html

相关文章:

  • windows10设置多个jar后台开机自启
  • 数据库||数据库相关知识练习题目与答案
  • YOLOv8改进 | 损失函数篇 | 更加聚焦的边界框损失Focaler-IoU、InnerFocalerIoU(二次创新)
  • 利用nginx宝塔免费防火墙实现禁止国外IP访问网站
  • 消息中间件(MQ)对比:RabbitMQ、Kafka、ActiveMQ 和 RocketMQ
  • MySQL索引原理以及SQL优化
  • [Bug] [OpenAI] [TypeError: fetch failed] { cause: [Error: AggregateError] }
  • @ 代码随想录算法训练营第5周(C语言)|Day31(贪心算法)
  • 面试手写第二期 Promsie相关
  • Windows冷知识:最小化远程桌面与ffmpeg
  • 12nm工艺,2.5GHz频率,低功耗Cortex-A72处理器培训
  • 网络编程套接字(2)
  • Elasticsearch:入门(二)
  • Debezium日常分享系列之:Debezium 2.6.0.Alpha1发布
  • Phoncent博客,探索Rie Kudan的GPT创作之举
  • 力扣hot100 划分字母区间 贪心 思维 满注释版
  • linux下使用swap分区扩展内存
  • 实现sleep函数
  • 汽车销量可视化分析
  • 代码随想录算法训练营DAY8 | 字符串(1)
  • 如何更改Outlook阅读邮件时的默认字体?
  • 【C++基础入门】三、运算符(算术运算符、赋值运算符、比较运算符、逻辑运算符)
  • ES7.17由于IP变化导致的故障及恢复
  • uniapp H5 touchstart touchend 切换背景会失效,或者没用
  • 【word visio绘图】关闭visio两线交叉的跳线(跨线)
  • meson、ninja编译dpdk
  • diff命令详解
  • Backtrader 文档学习- Broker - Slippage
  • 三子棋游戏小课堂
  • golang开源的可嵌入应用程序高性能的MQTT服务