当前位置:首页 > 文章列表 > Golang > Go问答 > 将 cobra 子命令源代码放入独立文件夹的方法

将 cobra 子命令源代码放入独立文件夹的方法

来源:stackoverflow 2024-03-27 14:09:27 0浏览 收藏

最近发现不少小伙伴都对Golang很感兴趣,所以今天继续给大家介绍Golang相关的知识,本文《将 cobra 子命令源代码放入独立文件夹的方法》主要内容涉及到等等知识点,希望能帮到你!当然如果阅读本文时存在不同想法,可以在评论中表达,但是请勿使用过激的措辞~

问题内容

我是 go 初学者,我正在尝试使用 cobra 创建 cli。为了引导该项目,我使用了 cobra generator,生成了一个命令、一个子命令,一切正常。

我现在有这种类型的布局:

cli
├── cmd
│   ├── command.go
│   ├── root.go
│   ├── subcommand.go
├── go.mod
├── go.sum
└── main.go

这不适合我,假设我的项目打算有很多命令,或者很多命令命名空间,每个命令都由不同的团队拥有,它会变得非常混乱并且难以维护。我更喜欢这样的布局:

cli
├── cmd
│   ├── command
│   │   ├── command.go
│   │   └── subcommand.go
│   └── root.go
├── go.mod
├── go.sum
└── main.go

现在,我对 go 中如何完成包和导入的方式缺乏一些了解(即使在阅读了这里和那里的文档之后),但据我所知,只要满足以下条件,就可以跨多个 go 源文件访问资源它们属于同一个包。但正如文档中所述,“实现可能要求包的所有源文件都位于同一目录中。”,因此要实现我想要的布局,我需要有多个包,例如每个命令命名空间都有一个,我认为这很好(或者至少,我不明白这样做有什么问题)。这就是我尝试过的:

  • cmd 目录中创建一个 command 目录
  • command.go 文件移至 command 目录内
  • package 子句从 command.go 文件更改为 command
  • subcommand.go 执行相同的操作

这构建得很好,但是找不到命令(error:“cli” 的未知命令“command”)。我以为是因为我从来没有导入过那个新的command包,所以我把它导入到main.go文件中,其中导入了cmd包。

构建失败,告诉我 undefined: rootcmdcommand.go 文件中。我猜测command包无法看到cmd中的资源,所以我在command.go文件中导入了cmd包,并将rootcmd替换为cmd.rootcmdrootcmd 是在 cli/cmd/root.txt 文件中创建的变量) go 文件,cobra 生成器提供的文件的一部分)。这次我确实抱有希望,但结果还是一样(undefined: cmd.rootcmd),现在我迷失了。

我是否正在尝试做一些 cobra 做不到的事情?

我是否正在尝试做一些可以用 cobra 实现但不使用 cobra 生成器的事情?

我是否试图做一些根本不应该做的事情(比如 go 试图保护我免受不良设计的影响)?

如果没有,我该如何进行?


正确答案


您无法使用 cobra-cli 命令获得所需的布局,但您当然可以手动进行设置。让我们从 cobra-cli 布局开始:

$ go mod init example
$ cobra-cli init

并添加几个命令:

$ cobra-cli add foo
$ cobra-cli add bar

这让我们:

.
├── cmd
│   ├── bar.go
│   ├── foo.go
│   └── root.go
├── go.mod
├── go.sum
├── license
└── main.go

让我们首先将命令移动到子目录中,这样我们就有了所需的布局。这让我们:

.
├── cmd
│   ├── bar
│   │   └── bar.go
│   ├── foo
│   │   └── foo.go
│   └── root.go
├── go.mod
├── go.sum
├── license
└── main.go

现在我们需要对代码进行一些更改才能使其正常工作。

  1. 由于我们的子命令现在位于单独的包中,因此我们需要重命名 rootcmd 以便可以导出。

    find cmd -name '*.go' -print | xargs sed -i 's/rootcmd/rootcmd/g'
  2. 我们需要每个子命令都位于自己的包中,因此我们将 cmd/foo/foo.go 顶部的 package cmd 替换为 package foo,并替换为 cmd/bar/bar 中的 package bar。 go

  3. 我们需要子命令来导入根命令,以便它们可以调用 rootcmd.addcommand。这意味着在 cmd/foo/foo.gocmd/bar/bar.go 中,我们需要将 example/cmd 添加到 import 部分(回想一下,我们通过 go mod init 命名了顶级包 example命令)。这意味着每个文件将以:

    package foo
    
    import (
            "fmt"
    
            "example/cmd"
            "github.com/spf13/cobra"
    )

    作为此更改的一部分,我们还需要更新对 addcommand 的引用以使用显式包名称:

    func init() {
      cmd.rootcmd.addcommand(foocmd)
    }
  4. 最后,我们需要将子命令导入到某个地方。现在,我们从不导入 foobar 包,因此代码实际上是不可见的(尝试在这些文件之一中引入严重的语法错误 - 您将看到 go build 将成功,因为这些文件在任何地方都没有引用)。

    我们可以在顶级 main.go 文件中执行此操作:

    package main
    
    import (
      "example/cmd"
      _ "example/cmd/bar"
      _ "example/cmd/foo"
    )
    
    func main() {
      cmd.execute()
    }

现在,如果我们构建并运行代码,我们会看到 foobar 命令可用:

$ go build
$ ./example
A longer description that spans multiple lines and likely contains
examples and usage of using your application. For example:

Cobra is a CLI library for Go that empowers applications.
This application is a tool to generate the needed files
to quickly create a Cobra application.

Usage:
  example [command]

Available Commands:
  bar         A brief description of your command
  completion  Generate the autocompletion script for the specified shell
  foo         A brief description of your command
  help        Help about any command

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

Use "example [command] --help" for more information about a command.

今天关于《将 cobra 子命令源代码放入独立文件夹的方法》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
linux jdk目录怎么查看linux jdk目录怎么查看
上一篇
linux jdk目录怎么查看
提高Java开发中集合操作的性能
下一篇
提高Java开发中集合操作的性能
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    516次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    500次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    485次学习
查看更多
AI推荐
  • ljg-skills -
    ljg-skills
    ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
    2184次使用
  • MELO音乐 - AI 音乐生成平台,支持多模态创作能力
    MELO音乐
    MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
    2000次使用
  • UniScribe - AI 免费在线音视频转文字平台
    UniScribe
    UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
    1941次使用
  • 剧云 - 免费 AI 智能中文剧本创作平台
    剧云
    剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
    2159次使用
  • 万象有声 - AI 一站式有声内容创作平台
    万象有声
    万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
    2122次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码