Go通过cobra快速构建命令行应用
来自jetbrains Go 语言现状调查报告 显示:在go开发者中使用go开发实用小程序的比例为31%仅次于web,go得益于跨平台、无依赖的特性,用来编写命令行或系统管理这类小程序非常不错。

本文主要介绍Steve Francia(spf13)大神写的用于快速构建命令行程序的golang包cobra,基于cobra写命令行的著名项目一只手数不过来:Docker CLI、Helm、istio、etcd、Git、Github CLI ...
下面进入正题
cobra能帮我们做啥?
cobra包提供以下功能:
- 轻松创建基于子命令的 CLI:如
app server、app fetch等。 - 自动添加
-h,--help等帮助性Flag - 自动生成命令和Flag的帮助信息
- 创建完全符合 POSIX 的Flag(标志)(包括长、短版本)
- 支持嵌套子命令
- 支持全局、本地和级联Flag
- 智能建议(
app srver... did you meanapp server?) - 为应用程序自动生成 shell 自动完成功能(bash、zsh、fish、powershell)
- 为应用程序自动生成man page
- 命令别名,可以在不破坏原有名称的情况下进行更改
- 支持灵活自定义help、usege等。
- 无缝集成viper构建12-factor应用
cobra遵循commands, arguments & flags结构。
举例来说
#appname command arguments
docker pull alpine:latest
#appname command flag
docker ps -a
#appname command flag argument
git commit -m "msg"
开发者可根据情况进行自组织。
cobra基础使用
安装cobra包和二进制工具cobra-cli,cobra-cli可以帮助我们快速创建出一个cobra基础代码结构。
go get -u github.com/spf13/cobra@latest
go install github.com/spf13/cobra-cli@latest
启用GO111MODULE=on,我们初始化一个xpower
# go mod init xpower
go: creating new go.mod: module xpower
使用cobra-cli初始化基础代码结构
# cobra-cli init
Your Cobra application is ready at /root/demo/xpower
#查看目录结构
# tree xpower
xpower
├── cmd
│ └── root.go
├── go.mod
├── go.sum
├── LICENSE
└── main.go
1 directory, 5 files
运行demo可以看到cobra包本身的一些提示信息。

查看main.go,cobra-cli为我们创建了一个cmd的包并且调用了包里面的Execute()函数
/*
Copyright 2022 NAME HERE <EMAIL ADDRESS>
*/
package main
import "xpower/cmd"
func main() {
cmd.Execute()
}
从上面的目录结构中可以看到cmd包目前只有一个root.go,我们可以在这里操作根命令相关的内容。
大多数时候CLI可能会包含多个子命令比如git clone、git add,cobra-cli可通过add 添加子命令。
现在我们添加wget和ping子命令,即接下来我们将通过xpower来重写wget和ping的部分功能。
cobra-cli add wget
cobra-cli add ping
现在的目录结构如下:
# tree xpower
xpower
├── cmd
│ ├── ping.go
│ ├── root.go
│ └── wget.go
├── go.mod
├── go.sum
├── LICENSE
└── main.go
ping和wget已经被集成到root.go中

wget.go
package cmd
import (
"fmt"
"github.com/spf13/cobra"
)
// wgetCmd represents the wget command
var wgetCmd = &cobra.Command{
Use: "wget",
Example: "xpower wget iqsing.github.io/download.tar -o /tmp",
Short: "wget is a download cli.",
Long: `use wget to download everything you want from net.`,
Run: func(cmd *cobra.Command, args []string) {
fmt.Println("wget called")
},
}
func init() {
rootCmd.AddCommand(wgetCmd)
// Here you will define your flags and configuration settings.
}
在wget.go 中定义了一个wgetCmd结构体指针,可通过查看Command结构体原型添加或移除成员变量。这里我们添加了一个Example用于指示示例,Short和Long为命令简介,Run为wget命令的真正实现。
我们知道在go中包的init()函数会在import时执行,通过AddCommand(wgetCmd)将wegetCmd添加到结构体Command 成员变量commands中,包括后面我们编写的Flag也是如此。
接下来我们在结构体中添加Args用于验证(限制)参数数量,在init()函数中添加Flag -o用于保存下载的文件地址,并通过MarkFlagRequired约束flag的参数必须输入,最后在Run中调用Download即可。
package cmd
import (
"fmt"
"io"
"log"
"net/http"
"os"
"github.com/spf13/cobra"
)
var (
output string
)
// wgetCmd represents the wget command
var wgetCmd = &cobra.Command{
Use: "wget",
Example: "xpower wget iqsing.github.io/download.tar.gz -o /tmp/download.tar.gz",
Args: cobra.ExactArgs(1),
Short: "wget is a download cli.",
Long: `use wget to download everything you want from net.`,
Run: func(cmd *cobra.Command, args []string) {
fmt.Println("---wget running---")
Download(args[0], output)
},
}
func init() {
rootCmd.AddCommand(wgetCmd)
// Here you will define your flags and configuration settings.
wgetCmd.Flags().StringVarP(&output, "output", "o", "", "output file")
wgetCmd.MarkFlagRequired("output")
}
func Download(url string, path string) {
out, err := os.Create(path)
check(err)
defer out.Close()
res, err := http.Get(url)
check(err)
defer res.Body.Close()
_, err = io.Copy(out, res.Body)
check(err)
fmt.Println("save as" + path)
}
func check(err error) {
if err != nil {
log.Fatal(err)
}
}
args
Args: cobra.ExactArgs(1)
cobra内置的参数验证也是比较多,NoArgs、OnlyValidArgs、MinimumNArgs、MaximumNArgs等等可翻阅源码args.go,可以满足基本使用,如果有自己的特殊要求可以通过解析arg来实现。
flags
wgetCmd.Flags().StringVarP(&output, "output", "o", "", "output file(required)")
flag包含局部和全局两种,全局flag在父命令定义后子命令也会生效,而局部flag则在哪定义就在哪生效。
如上面的局部flag,我们在wgetCmd中定义的flag只有wget这个子命令能用。
全局flag
rootCmd.PersistentFlags().BoolVarP(&Verbose, "verbose", "v", false, "verbose output")
StringVarp、BoolVarP 用于flag数据类型限制。
简单的应用从命令行直接写入参数是很常见的,但是如果比较复杂的命令行应用参数需要非常多,再这样操作不太合理,cobra作者还写了另一个在go中很流行的包viper用于解析配置文件,比如kubectl 的yml,以及各种json
前面也说过可以无缝衔接,只需Bind一下即可。
var author string
func init() {
rootCmd.PersistentFlags().StringVar(&author, "author", "YOUR NAME", "Author name for copyright attribution")
viper.BindPFlag("author", rootCmd.PersistentFlags().Lookup("author"))
}
flag还可以做依赖,比如下面username和password必须同时接收到参数。
rootCmd.Flags().StringVarP(&u, "username", "u", "", "Username (required if password is set)")
rootCmd.Flags().StringVarP(&pw, "password", "p", "", "Password (required if username is set)")
rootCmd.MarkFlagsRequiredTogether("username", "password")
添加子命令可参考包go-ping/ping,这里不再赘述。
我们来看编译后使用如何?
通过-h查看帮助:

参数个数错误:

需要flag-o:

正确使用:
xpower 子命令ping:

xpower 子命令wget:

以上我们通过go中cobra包实现xpower命令,包含重写了简单功能的ping和wget两子命令,甚至我们还可以以此来实现自己的跨平台、无依赖的工具集。本文涉及代码已提交至仓库code/xpower
cobra包含很多开箱即用的功能,经过大量项目验证和完善,已满足大部分命令行应用构建需求。本文只介绍了一部分内容,更多内容可查看仓库spf13/cobra
通过博客阅读:iqsing.github.io
Go通过cobra快速构建命令行应用的更多相关文章
- Apache Commons CLI官方文档翻译 —— 快速构建命令行启动模式
昨天通过几个小程序以及Hangout源码学习了CLI的基本使用,今天就来尝试翻译一下CLI的官方使用手册. 下面将会通过几个部分简单的介绍CLI在应用中的使用场景. 昨天已经联系过几个基本的命令行参数 ...
- Windows 右键快速运行命令行
原文见:右键命令行 - yacper - 博客园 方法一:配置文件夹选项 1 打开人任意文件夹,[工具] --> [文件夹选项] --> [文件类型] --> [(无)资料夹] -- ...
- java基础-构建命令行运行的java程序简要注意
今天编写了一个运行在服务端的java工具类,才发现自己以前很少关注运营方面的内容,导致在服务端部署一个java的工具变得异常困难,其实这也是自己对java的了解不够造成的. 首先,当代码编写完成之后, ...
- 教你用Cobra开发类似docker的命令行
目录 前言 一.安装 二.初始化应用 gomod初始化 创建入口文件cmd/root.go 创建主程序main.go 三.生成Command 创建hello子命令 创建version子命令 四.如何设 ...
- go Cobra命令行工具入门
简介 Github:https://github.com/spf13/cobra Star:26.5K Cobra是一个用Go语言实现的命令行工具.并且现在正在被很多项目使用,例如:Kuberne ...
- 25 个 Linux 下最炫酷又强大的命令行神器,你用过其中哪几个呢?
本文首发于:微信公众号「运维之美」,公众号 ID:Hi-Linux. 「运维之美」是一个有情怀.有态度,专注于 Linux 运维相关技术文章分享的公众号.公众号致力于为广大运维工作者分享各类技术文章和 ...
- Node.js 命令行程序开发教程
nodejs开发命令行程序非常方便,具体操作方式查看下面几篇文章 http://www.ruanyifeng.com/blog/2015/05/command-line-with-node.html ...
- python命令行神器Click
原文: http://www.lengirl.com/code/python-click.html Click 是用Python写的一个第三方模块,用于快速创建命令行.我们知道,Python内置了一个 ...
- 【python】命令行神器 Click 简明笔记
全文拷贝自 命令行神器 Click 简明笔记 Click Click 是用 Python 写的一个第三方模块,用于快速创建命令行.我们知道,Python 内置了一个 Argparse 的标准库用于创建 ...
随机推荐
- count(*)这么慢,我该怎么办?
1)计算一个表有多少行数用什么命令? select count(*) from t 2)count(*)底层是怎样实现的? 在MYISAM中,是把这个总行数存到磁盘中去的,要的时候直接去读就行,特别快 ...
- Java并发编程虚假唤醒问题(生产者和消费者关系)
何为虚假唤醒: 当一个条件满足时,很多线程都被唤醒了,但是只有其中部分是有用的唤醒,其它的唤醒都是无用功:比如买货:如果商品本来没有货物,突然进了一件商品,这是所有的线程都被唤醒了,但是只能一个人买, ...
- CF487E Tourists(圆方树+树链剖分+multiset/可删堆)
CF487E Tourists(圆方树+树链剖分+multiset/可删堆) Luogu 给出一个带点权的无向图,两种操作: 1.修改某点点权. 2.询问x到y之间简单路径能走过的点的最小点权. 题解 ...
- Idea集成CSSO插件压缩css文件
首先需要本地已安装node环境,并且csso-cli已通过npm安装到本地目录,只要能找到就行. 1. 打开Settings配置,确认图中的 File Watchers 插件是否已存在,如果不存在,去 ...
- Oracle入门基础(七)一一集合运算
SQL> /* SQL> 查询10和20号部门的员工 SQL> 1. select * from emp where deptno=10 or deptno=20; SQL> ...
- jenkins-learning
常规的打包方式: 提交代码 拉去代码并打包:war包和jar包 上传到服务器 关闭当前程序 启动新的jar包 查看新的jar包是否起作用 jenkins自动化流程: CI(Continuous int ...
- OGNL(Object-Graph Navigation Language)使用
OGNL表达式:https://www.jianshu.com/p/6bc6752d11f4 Apache OGNL:http://commons.apache.org/proper/commons- ...
- Python - Datetime库简介
- 『现学现忘』Docker基础 — 36、CMD指令和ENTRYPOINT指令的区别
目录 1.CMD指令和ENTRYPOINT指令说明 2.CMD指令只有最后一条生效的原因 3.CMD指令演示 4.ENTRYPOINT指令演示 5.总结 CMD指令和ENTRYPOINT指令作用都是指 ...
- freeswitch对接WEBRTC的一个candidate问题
概述 近几年,WEBRTC的完善与成熟,使得网页上使用webrtc的应用越来越多. Freeswitch是一个开源的软交换平台,可以直接支持webrtc的对接方式. 最近在测试fs和webrtc的对接 ...