【golang】15、cobra cli 命令行库


Cobra 是 golang 最流行的命令行库,文档见

一、脚手架

mkdir pt && cd pt && go mod init
cobra-cli init # 在项目下运行即可生成脚手架

# tree
.
├── LICENSE
├── cmd # 生成了cmd目录
│   └── root.go # 生成了root.go, 其中定义了rootCmt变量
├── go.mod
├── go.sum
└── main.go

# go run main.go
Usage:
  pt [command]

Available Commands:
  completion  Generate the autocompletion script for the specified shell
  help        Help about any command
  show        Display current time

Flags:
  -h, --help     help for pt
  -t, --toggle   Help message for toggle

二、子命令

# cobra-cli add show # 添加名为show的子命令, 则会自动生成cmd/show.go代码

# go run main.go show # 执行名为show的子命令
2023-08-28 22:00:58.263991 +0800 CST m=+0.000923251

# go run main.go show -h # 获取名为show的子命令的帮助信息
will display like 2023-08-28 22:00:58.263991 +0800 CST m=+0.000923251

Usage:
  pt show [flags]

Flags:
  -h, --help   help for show

三、flag 参数

3.1 定义

// 添加参数
func init() {
    
    
	rootCmd.AddCommand(proto2jsonCmd)
	proto2jsonCmd.Flags().StringP("content", "c", "", "byte content of a.proto")
	proto2jsonCmd.Flags().StringP("filepath", "f", "", "filepath of proto file")
}

// 输出如下:
with a.proto protocol

Usage:
  pt p2j [flags]

Flags:
  -c, --content string    byte content of a.proto
  -f, --filepath string   filepath of proto file
  -h, --help              help for p2j

3.2 使用

如果需要使用 全局flag 或者 局部flag,需要在合适的作用域内定义变量存储 flag 值,以便 flag 可在特定作用域内生效。

3.2.1 使用全局flag

让一个 flag 对所有命令生效,需要在 root.go 文件中创建一个变量存储 flag 值。

如需要定义一个全局flag name:

// 在root.go 文件中添加一个变量name
var name string
 
// 在init函数中添加全局flag,将flag值存储到变量name中
rootCmd.PersistentFlags().StringVar(&name, "name", "", "set name")
 
// 在子命令version的Run方法中输出name  
Run: func(cmd *cobra.Command, args []string) {
    
      
    fmt.Println("name is: ", name)
}
 
// 执行命令  
./demo version --name a
输出:
name is:  a

3.2 使用局部 flag

让一个 flag 对某个命令生效,需要在该命令文件中创建一个变量存储 flag 值。

如需要给version命令定义一个局部flag name:

// 定义变量content  
var content string
 
// 在version.go的init函数中添加flag  
versionCmd.Flags().StringVarP(&content, "content", "s", "false", "you are my sunshine")
 
// 在子命令version.go的Run方法中输出  
Run: func(cmd *cobra.Command, args []string) {
    
      
    fmt.Println("name is: ", name)
    fmt.Println("content is: ", content)
}
 
// 执行命令  
./demo version --name a --content b
输出:
name is:  a
content is:  b  

3.3 cmd 只能访问到自己定义的 flag 值

注意:将 flags 存储到本地变量当中,那么其他命令「不可以」用某个命令的 局部flag。因为局部flag虽然是定义在某个命令文件中作为局部变量,cmd 文件夹下的其他文件可以访问这个变量,但是其他命令如果没有定义自己的 局部flag 获取相同 flag 值的话,获取到的值是该局部变量的零值。示例如下:

// 1. 添加一个新命令helloworld  
> cobra add helloworld
 
// 2.输出content值  
Run: func(cmd *cobra.Command, args []string) {
    
      
    fmt.Println("content is: ", sunshine)
}

// 3.执行命令  
> ./demo helloworld --content b
Error: unknown flag: --content  // 输出错误未知flag, 原因就是该命令并未定义名为content的局部flag

3.4 必填 flag

// init文件中增加flag定义  
versionCmd.Flags().StringVarP(&sunshine, "content", "c", "", "my content")  
versionCmd.MarkFlagRequired("content")
 
// 执行  
> ./demo version
 
// 输出  
Error: required flag(s) "content" not set // 说明必须要设置flag content
 
// 传入content flag  
> ./demo version --content b // 输出: content is:  b

3.5 全局 flag 配置

MinimumNArgs(int) 当参数数目低于配置的最小参数个数时报错  
MaximumNArgs(int) 当参数数目大于配置的最大参数个数时报错  
ExactArgs(int)    如果参数数目不是配置的参数个数时报错  
NoArgs            没有参数则报错  

示例如下:

// 添加一个命令path  
> cobra add path
 
// 设置该命令需要且仅需要一个参数,并在Run方法中取出参数  
var pathCmd = &cobra.Command{
    
      
    Use:   "path [path]",
    Short: "A brief description of your command",
    Long: "",  
    Args: cobra.ExactArgs(1),
    Run: func(cmd *cobra.Command, args []string) {
    
    
        fmt.Println("path called")
        fmt.Println("path:", args[0])
    },
}
 
// 执行命令并输出  
> ./demo path /home // 输出path: /home  

猜你喜欢

转载自blog.csdn.net/jiaoyangwm/article/details/132548793