From 060d13a759384fdc103aa96bfbbd0a84b90de503 Mon Sep 17 00:00:00 2001 From: UUBulb <35923940+uubulb@users.noreply.github.com> Date: Sat, 18 May 2024 22:38:39 +0800 Subject: [PATCH] Replace pflag with cobra, add service command (#17) * Replace pflag with cobra, add service command moved '-c' option to 'edit' command * Restart after installation & add some friendly outputs * Attempt to fix ci * Update dependencies & revert gosec to 2.19.0 gosec 2.19.0 is the last version supports Go 1.20 * Remove gosec checks --- .github/workflows/test-on-pr.yml | 10 +- .github/workflows/test.yml | 8 +- .gitignore | 1 + .goreleaser.yml | 6 +- cmd/agent/edit.go | 117 ++++++++++++++++++ main.go => cmd/agent/main.go | 163 ++++++------------------- main_test.go => cmd/agent/main_test.go | 0 cmd/agent/service.go | 112 +++++++++++++++++ go.mod | 5 +- go.sum | 13 ++ 10 files changed, 297 insertions(+), 138 deletions(-) create mode 100644 cmd/agent/edit.go rename main.go => cmd/agent/main.go (80%) rename main_test.go => cmd/agent/main_test.go (100%) create mode 100644 cmd/agent/service.go diff --git a/.github/workflows/test-on-pr.yml b/.github/workflows/test-on-pr.yml index a17ce5b..3cfabf0 100644 --- a/.github/workflows/test-on-pr.yml +++ b/.github/workflows/test-on-pr.yml @@ -21,11 +21,11 @@ jobs: go test -v ./... - name: Build test run: | - go build - - name: Run Gosec Security Scanner - run: | - go install github.com/securego/gosec/v2/cmd/gosec@latest - gosec -exclude=G104 ./... + go build ./cmd/agent + #- name: Run Gosec Security Scanner + # run: | + # go install github.com/securego/gosec/v2/cmd/gosec@v2.19.0 + # gosec -exclude=G104 ./... - name: Run GoReleaser uses: goreleaser/goreleaser-action@v2 with: diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 14674c8..021cc5b 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -27,10 +27,10 @@ jobs: - name: Unit test run: | go test -v ./... - - name: Run Gosec Security Scanner - run: | - go install github.com/securego/gosec/v2/cmd/gosec@latest - gosec -exclude=G104 ./... + #- name: Run Gosec Security Scanner + # run: | + # go install github.com/securego/gosec/v2/cmd/gosec@v2.19.0 + # gosec -exclude=G104 ./... - name: Run GoReleaser uses: goreleaser/goreleaser-action@v2 with: diff --git a/.gitignore b/.gitignore index 95a7ed0..db687ba 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,4 @@ .DS_Store /agent +/cmd/agent/agent *.pprof diff --git a/.goreleaser.yml b/.goreleaser.yml index 4734175..612d29d 100644 --- a/.goreleaser.yml +++ b/.goreleaser.yml @@ -27,7 +27,7 @@ builds: goarch: arm - goos: windows goarch: arm64 - main: . + main: ./cmd/agent binary: nezha-agent - id: darwin-amd64 env: @@ -40,7 +40,7 @@ builds: - darwin goarch: - amd64 - main: . + main: ./cmd/agent binary: nezha-agent - id: darwin-arm64 env: @@ -53,7 +53,7 @@ builds: - darwin goarch: - arm64 - main: . + main: ./cmd/agent binary: nezha-agent universal_binaries: - name_template: "nezha-agent" diff --git a/cmd/agent/edit.go b/cmd/agent/edit.go new file mode 100644 index 0000000..c733b67 --- /dev/null +++ b/cmd/agent/edit.go @@ -0,0 +1,117 @@ +package main + +import ( + "fmt" + "strings" + "net" + "errors" + + "github.com/spf13/cobra" + "github.com/AlecAivazis/survey/v2" + "github.com/shirou/gopsutil/v3/disk" + psnet "github.com/shirou/gopsutil/v3/net" + +) + +var editCmd = &cobra.Command{ + Use: "edit", + Short: "修改要监控的网卡/分区名单,修改自定义 DNS", + Run: editAgentConfig, + Args: cobra.NoArgs, +} + +func init() { + agentCmd.AddCommand(editCmd) +} + +// 修改Agent要监控的网卡与硬盘分区 +func editAgentConfig(cmd *cobra.Command, args []string) { + nc, err := psnet.IOCounters(true) + if err != nil { + panic(err) + } + var nicAllowlistOptions []string + for _, v := range nc { + nicAllowlistOptions = append(nicAllowlistOptions, v.Name) + } + + var diskAllowlistOptions []string + diskList, err := disk.Partitions(false) + if err != nil { + panic(err) + } + for _, p := range diskList { + diskAllowlistOptions = append(diskAllowlistOptions, fmt.Sprintf("%s\t%s\t%s", p.Mountpoint, p.Fstype, p.Device)) + } + + var qs = []*survey.Question{ + { + Name: "nic", + Prompt: &survey.MultiSelect{ + Message: "选择要监控的网卡", + Options: nicAllowlistOptions, + }, + }, + { + Name: "disk", + Prompt: &survey.MultiSelect{ + Message: "选择要监控的硬盘分区", + Options: diskAllowlistOptions, + }, + }, + { + Name: "dns", + Prompt: &survey.Input{ + Message: "自定义 DNS,可输入空格跳过,如 1.1.1.1:53,1.0.0.1:53", + Default: strings.Join(agentConfig.DNS, ","), + }, + }, + } + + answers := struct { + Nic []string + Disk []string + DNS string + }{} + + err = survey.Ask(qs, &answers, survey.WithValidator(survey.Required)) + if err != nil { + fmt.Println("选择错误", err.Error()) + return + } + + agentConfig.HardDrivePartitionAllowlist = []string{} + for _, v := range answers.Disk { + agentConfig.HardDrivePartitionAllowlist = append(agentConfig.HardDrivePartitionAllowlist, strings.Split(v, "\t")[0]) + } + + agentConfig.NICAllowlist = make(map[string]bool) + for _, v := range answers.Nic { + agentConfig.NICAllowlist[v] = true + } + + dnsServers := strings.TrimSpace(answers.DNS) + + if dnsServers != "" { + agentConfig.DNS = strings.Split(dnsServers, ",") + for _, s := range agentConfig.DNS { + host, _, err := net.SplitHostPort(s) + if err == nil { + if net.ParseIP(host) == nil { + err = errors.New("格式错误") + } + } + if err != nil { + panic(fmt.Sprintf("自定义 DNS 格式错误:%s %v", s, err)) + } + } + } else { + agentConfig.DNS = []string{} + } + + if err = agentConfig.Save(); err != nil { + panic(err) + } + + fmt.Println("修改自定义配置成功,重启 Agent 后生效") +} \ No newline at end of file diff --git a/main.go b/cmd/agent/main.go similarity index 80% rename from main.go rename to cmd/agent/main.go index 427ab11..4ea3c77 100644 --- a/main.go +++ b/cmd/agent/main.go @@ -16,7 +16,6 @@ import ( "strings" "time" - "github.com/AlecAivazis/survey/v2" bpc "github.com/DaRealFreak/cloudflare-bp-go" "github.com/blang/semver" "github.com/ebi-yade/altsvc-go" @@ -24,10 +23,8 @@ import ( "github.com/gorilla/websocket" "github.com/nezhahq/go-github-selfupdate/selfupdate" "github.com/quic-go/quic-go/http3" - "github.com/shirou/gopsutil/v3/disk" "github.com/shirou/gopsutil/v3/host" - psnet "github.com/shirou/gopsutil/v3/net" - flag "github.com/spf13/pflag" + "github.com/spf13/cobra" "google.golang.org/grpc" "google.golang.org/grpc/credentials" "google.golang.org/grpc/credentials/insecure" @@ -62,6 +59,15 @@ var ( inited bool ) +var agentCmd = &cobra.Command{ + Use: "agent", + Run: func(cmd *cobra.Command, args []string) { + run() + }, + PreRun: preRun, + PersistentPreRun: persistPreRun, +} + var ( agentCliParam AgentCliParam agentConfig model.AgentConfig @@ -108,7 +114,6 @@ func init() { } return nil, err } - flag.CommandLine.ParseErrorsWhitelist.UnknownFlags = true http.DefaultClient.Timeout = time.Second * 5 httpClient.Transport = bpc.AddCloudFlareByPass(httpClient.Transport, bpc.Options{ @@ -124,10 +129,32 @@ func init() { if err != nil { panic(err) } + + // 初始化运行参数 + agentCmd.PersistentFlags().StringVarP(&agentCliParam.Server, "server", "s", "localhost:5555", "管理面板RPC端口") + agentCmd.PersistentFlags().StringVarP(&agentCliParam.ClientSecret, "password", "p", "", "Agent连接Secret") + agentCmd.PersistentFlags().BoolVar(&agentCliParam.TLS, "tls", false, "启用SSL/TLS加密") + agentCmd.Flags().BoolVarP(&agentCliParam.Debug, "debug", "d", false, "开启调试信息") + agentCmd.Flags().IntVar(&agentCliParam.ReportDelay, "report-delay", 1, "系统状态上报间隔") + agentCmd.Flags().BoolVar(&agentCliParam.SkipConnectionCount, "skip-conn", false, "不监控连接数") + agentCmd.Flags().BoolVar(&agentCliParam.SkipProcsCount, "skip-procs", false, "不监控进程数") + agentCmd.Flags().BoolVar(&agentCliParam.DisableCommandExecute, "disable-command-execute", false, "禁止在此机器上执行命令") + agentCmd.Flags().BoolVar(&agentCliParam.DisableAutoUpdate, "disable-auto-update", false, "禁用自动升级") + agentCmd.Flags().BoolVar(&agentCliParam.DisableForceUpdate, "disable-force-update", false, "禁用强制升级") + agentCmd.Flags().Uint32VarP(&agentCliParam.IPReportPeriod, "ip-report-period", "u", 30*60, "本地IP更新间隔, 上报频率依旧取决于report-delay的值") + agentCmd.Flags().BoolVarP(&agentCliParam.Version, "version", "v", false, "查看当前版本号") + agentConfig.Read(filepath.Dir(ex) + "/config.yml") } func main() { + if err := agentCmd.Execute(); err != nil { + fmt.Println(err) + os.Exit(1) + } +} + +func persistPreRun(cmd *cobra.Command, args []string) { // windows环境处理 if runtime.GOOS == "windows" { hostArch, err := host.KernelArch() @@ -147,48 +174,26 @@ func main() { panic(fmt.Sprintf("与当前系统不匹配,当前运行 %s_%s, 需要下载 %s_%s", runtime.GOOS, arch, runtime.GOOS, hostArch)) } } +} +func preRun(cmd *cobra.Command, args []string) { // 来自于 GoReleaser 的版本号 monitor.Version = version - // 初始化运行参数 - var isEditAgentConfig bool - flag.BoolVarP(&agentCliParam.Debug, "debug", "d", false, "开启调试信息") - flag.BoolVarP(&isEditAgentConfig, "edit-agent-config", "c", false, "修改要监控的网卡/分区名单,修改自定义 DNS") - flag.StringVarP(&agentCliParam.Server, "server", "s", "localhost:5555", "管理面板RPC端口") - flag.StringVarP(&agentCliParam.ClientSecret, "password", "p", "", "Agent连接Secret") - flag.IntVar(&agentCliParam.ReportDelay, "report-delay", 1, "系统状态上报间隔") - flag.BoolVar(&agentCliParam.SkipConnectionCount, "skip-conn", false, "不监控连接数") - flag.BoolVar(&agentCliParam.SkipProcsCount, "skip-procs", false, "不监控进程数") - flag.BoolVar(&agentCliParam.DisableCommandExecute, "disable-command-execute", false, "禁止在此机器上执行命令") - flag.BoolVar(&agentCliParam.DisableAutoUpdate, "disable-auto-update", false, "禁用自动升级") - flag.BoolVar(&agentCliParam.DisableForceUpdate, "disable-force-update", false, "禁用强制升级") - flag.BoolVar(&agentCliParam.TLS, "tls", false, "启用SSL/TLS加密") - flag.BoolVarP(&agentCliParam.Version, "version", "v", false, "查看当前版本号") - flag.Uint32VarP(&agentCliParam.IPReportPeriod, "ip-report-period", "u", 30*60, "本地IP更新间隔, 上报频率依旧取决于report-delay的值") - flag.Parse() - if agentCliParam.Version { fmt.Println(version) - return - } - - if isEditAgentConfig { - editAgentConfig() - return + os.Exit(0) } if agentCliParam.ClientSecret == "" { - flag.Usage() - return + cmd.Help() + os.Exit(1) } if agentCliParam.ReportDelay < 1 || agentCliParam.ReportDelay > 4 { println("report-delay 的区间为 1-4") - return + os.Exit(1) } - - run() } func run() { @@ -629,98 +634,6 @@ func handleTerminalTask(task *pb.Task) { } } -// 修改Agent要监控的网卡与硬盘分区 -func editAgentConfig() { - nc, err := psnet.IOCounters(true) - if err != nil { - panic(err) - } - var nicAllowlistOptions []string - for _, v := range nc { - nicAllowlistOptions = append(nicAllowlistOptions, v.Name) - } - - var diskAllowlistOptions []string - diskList, err := disk.Partitions(false) - if err != nil { - panic(err) - } - for _, p := range diskList { - diskAllowlistOptions = append(diskAllowlistOptions, fmt.Sprintf("%s\t%s\t%s", p.Mountpoint, p.Fstype, p.Device)) - } - - var qs = []*survey.Question{ - { - Name: "nic", - Prompt: &survey.MultiSelect{ - Message: "选择要监控的网卡", - Options: nicAllowlistOptions, - }, - }, - { - Name: "disk", - Prompt: &survey.MultiSelect{ - Message: "选择要监控的硬盘分区", - Options: diskAllowlistOptions, - }, - }, - { - Name: "dns", - Prompt: &survey.Input{ - Message: "自定义 DNS,可输入空格跳过,如 1.1.1.1:53,1.0.0.1:53", - Default: strings.Join(agentConfig.DNS, ","), - }, - }, - } - - answers := struct { - Nic []string - Disk []string - DNS string - }{} - - err = survey.Ask(qs, &answers, survey.WithValidator(survey.Required)) - if err != nil { - fmt.Println("选择错误", err.Error()) - return - } - - agentConfig.HardDrivePartitionAllowlist = []string{} - for _, v := range answers.Disk { - agentConfig.HardDrivePartitionAllowlist = append(agentConfig.HardDrivePartitionAllowlist, strings.Split(v, "\t")[0]) - } - - agentConfig.NICAllowlist = make(map[string]bool) - for _, v := range answers.Nic { - agentConfig.NICAllowlist[v] = true - } - - dnsServers := strings.TrimSpace(answers.DNS) - - if dnsServers != "" { - agentConfig.DNS = strings.Split(dnsServers, ",") - for _, s := range agentConfig.DNS { - host, _, err := net.SplitHostPort(s) - if err == nil { - if net.ParseIP(host) == nil { - err = errors.New("格式错误") - } - } - if err != nil { - panic(fmt.Sprintf("自定义 DNS 格式错误:%s %v", s, err)) - } - } - } else { - agentConfig.DNS = []string{} - } - - if err = agentConfig.Save(); err != nil { - panic(err) - } - - fmt.Println("修改自定义配置成功,重启 Agent 后生效") -} - func println(v ...interface{}) { if agentCliParam.Debug { fmt.Printf("NEZHA@%s>> ", time.Now().Format("2006-01-02 15:04:05")) diff --git a/main_test.go b/cmd/agent/main_test.go similarity index 100% rename from main_test.go rename to cmd/agent/main_test.go diff --git a/cmd/agent/service.go b/cmd/agent/service.go new file mode 100644 index 0000000..915595e --- /dev/null +++ b/cmd/agent/service.go @@ -0,0 +1,112 @@ +package main + +import ( + "os" + "log" + + "github.com/spf13/cobra" + "github.com/nezhahq/service" +) + +type program struct { + exit chan struct{} + service service.Service +} + +var serviceCmd = &cobra.Command{ + Use: "service ", + Short: "服务与自启动设置", + Args: cobra.ExactArgs(1), + Run: runService, + PreRun: servicePreRun, +} + +func init() { + agentCmd.AddCommand(serviceCmd) +} + +func servicePreRun(cmd *cobra.Command, args []string) { + if args[0] == "install" { + if agentCliParam.ClientSecret == "" { + cmd.Help() + os.Exit(1) + } + } + + if agentCliParam.ReportDelay < 1 || agentCliParam.ReportDelay > 4 { + println("report-delay 的区间为 1-4") + os.Exit(1) + } +} + +func (p *program) Start(s service.Service) error { + go p.run() + return nil +} + +func (p *program) run() { + defer func() { + if service.Interactive() { + p.Stop(p.service) + } else { + p.service.Stop() + } + }() + + run() + + return +} + +func (p *program) Stop(s service.Service) error { + close(p.exit) + if service.Interactive() { + os.Exit(0) + } + return nil +} + +func runService(cmd *cobra.Command, args []string) { + var tlsoption string + + mode := args[0] + dir, err := os.Getwd() + if err != nil { + println("获取当前工作目录时出错: ", err) + return + } + + if agentCliParam.TLS { + tlsoption = "--tls" + } + + svcConfig := &service.Config{ + Name: "nezha-agent", + DisplayName: "Nezha Agent", + Description: "哪吒探针监控端", + Arguments: []string{ + "-s", agentCliParam.Server, + "-p", agentCliParam.ClientSecret, + tlsoption, + }, + WorkingDirectory: dir, + } + + prg := &program{ + exit: make(chan struct{}), + } + s, err := service.New(prg, svcConfig) + if err != nil { + log.Fatal("创建服务时出错: ", err) + } + + if mode == "install" { + initName := s.Platform() + log.Println("Init system is:", initName) + } + + err = service.Control(s, mode) + if err != nil { + log.Fatal(err) + } +} diff --git a/go.mod b/go.mod index b2b5352..c0233e8 100644 --- a/go.mod +++ b/go.mod @@ -15,9 +15,10 @@ require ( github.com/iamacarpet/go-winpty v1.0.4 github.com/json-iterator/go v1.1.12 github.com/nezhahq/go-github-selfupdate v0.0.0-20240418134522-9d84a13bbf2d + github.com/nezhahq/service v0.0.0-20240518110122-594be3ba1962 github.com/quic-go/quic-go v0.40.1 github.com/shirou/gopsutil/v3 v3.24.4 - github.com/spf13/pflag v1.0.5 + github.com/spf13/cobra v1.8.0 github.com/spf13/viper v1.18.2 google.golang.org/grpc v1.63.2 google.golang.org/protobuf v1.34.0 @@ -38,6 +39,7 @@ require ( github.com/google/uuid v1.6.0 // indirect github.com/hashicorp/hcl v1.0.0 // indirect github.com/inconshreveable/go-update v0.0.0-20160112193335-8152e7eb6ccf // indirect + github.com/inconshreveable/mousetrap v1.1.0 // indirect github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 // indirect github.com/lufia/plan9stats v0.0.0-20230110061619-bbe2e5e100de // indirect github.com/magiconair/properties v1.8.7 // indirect @@ -59,6 +61,7 @@ require ( github.com/sourcegraph/conc v0.3.0 // indirect github.com/spf13/afero v1.11.0 // indirect github.com/spf13/cast v1.6.0 // indirect + github.com/spf13/pflag v1.0.5 // indirect github.com/subosito/gotenv v1.6.0 // indirect github.com/tcnksm/go-gitconfig v0.1.2 // indirect github.com/tklauser/go-sysconf v0.3.12 // indirect diff --git a/go.sum b/go.sum index 860ced8..4da736d 100644 --- a/go.sum +++ b/go.sum @@ -17,6 +17,7 @@ github.com/blang/semver v3.5.1+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnweb github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= +github.com/cpuguy83/go-md2man/v2 v2.0.3/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= github.com/creack/pty v1.1.17/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4= github.com/creack/pty v1.1.21 h1:1/QdRyBaHHJP61QkWMXlOIBfsgdDeeKfK8SYVUWJKf0= github.com/creack/pty v1.1.21/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4= @@ -68,6 +69,8 @@ github.com/iamacarpet/go-winpty v1.0.4/go.mod h1:50yLtqN2hFb5sYO5Qm2LegB166oAEw/ github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/inconshreveable/go-update v0.0.0-20160112193335-8152e7eb6ccf h1:WfD7VjIE6z8dIvMsI4/s+1qr5EL+zoIGev1BQj1eoJ8= github.com/inconshreveable/go-update v0.0.0-20160112193335-8152e7eb6ccf/go.mod h1:hyb9oH7vZsitZCiBt0ZvifOrB+qc8PS5IiilCIb87rg= +github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= +github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 h1:Z9n2FFNUXsshfwJMBgNA0RU6/i7WVaAegv3PtuIHPMs= @@ -98,6 +101,12 @@ github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9G github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= github.com/nezhahq/go-github-selfupdate v0.0.0-20240418134522-9d84a13bbf2d h1:gUt6JLTE/HH7qXS5F7SSYrmsj8NXx3i5CaF6mzYw6pA= github.com/nezhahq/go-github-selfupdate v0.0.0-20240418134522-9d84a13bbf2d/go.mod h1:fOsabb2tjwCe7/kGSsout6oL2cp0sKhOwYQp6fP/Xfg= +github.com/nezhahq/service v0.0.0-20240518043736-9ae0db11e8df h1:V9mIEc9CKm+hFuPmrEjByCScbWZGoWLOUcPjrAis9ew= +github.com/nezhahq/service v0.0.0-20240518043736-9ae0db11e8df/go.mod h1:1CemEvuMOM4B88ckxMZvLT0dehlP5+bqQOYRLMslSdE= +github.com/nezhahq/service v0.0.0-20240518100746-f22fc3c61e5d h1:OmxtpsZfT0wLBgwo0uw4vusU+nEVWljye0vWfRpbFig= +github.com/nezhahq/service v0.0.0-20240518100746-f22fc3c61e5d/go.mod h1:1CemEvuMOM4B88ckxMZvLT0dehlP5+bqQOYRLMslSdE= +github.com/nezhahq/service v0.0.0-20240518110122-594be3ba1962 h1:2zrICSRNedjDMlmDb58B03QHcLvpKIBe931FV7kHvas= +github.com/nezhahq/service v0.0.0-20240518110122-594be3ba1962/go.mod h1:1CemEvuMOM4B88ckxMZvLT0dehlP5+bqQOYRLMslSdE= github.com/onsi/ginkgo/v2 v2.9.5 h1:+6Hr4uxzP4XIUyAkg61dWBw8lb/gc4/X5luuxN/EC+Q= github.com/onsi/ginkgo/v2 v2.9.5/go.mod h1:tvAoo1QUJwNEU2ITftXTpR7R1RbCzoZUOs3RonqW57k= github.com/onsi/gomega v1.27.6 h1:ENqfyGeS5AX/rlXDd/ETokDz93u0YufY1Pgxuy/PvWE= @@ -117,6 +126,7 @@ github.com/quic-go/qtls-go1-20 v0.4.1/go.mod h1:X9Nh97ZL80Z+bX/gUXMbipO6OxdiDi58 github.com/quic-go/quic-go v0.40.1 h1:X3AGzUNFs0jVuO3esAGnTfvdgvL4fq655WaOi1snv1Q= github.com/quic-go/quic-go v0.40.1/go.mod h1:PeN7kuVJ4xZbxSv/4OX6S1USOX8MJvydwpTx31vx60c= github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= +github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= github.com/sagikazarmark/locafero v0.4.0 h1:HApY1R9zGo4DBgr7dqsTH/JJxLTTsOt7u6keLGt6kNQ= github.com/sagikazarmark/locafero v0.4.0/go.mod h1:Pe1W6UlPYUk/+wc/6KFhbORCfqzgYEpgQ3O5fPuL3H4= github.com/sagikazarmark/slog-shim v0.1.0 h1:diDBnUNK9N/354PgrxMywXnAwEr1QZcOr6gto+ugjYE= @@ -133,6 +143,8 @@ github.com/spf13/afero v1.11.0 h1:WJQKhtpdm3v2IzqG8VMqrr6Rf3UYpEF239Jy9wNepM8= github.com/spf13/afero v1.11.0/go.mod h1:GH9Y3pIexgf1MTIWtNGyogA5MwRIDXGUr+hbWNoBjkY= github.com/spf13/cast v1.6.0 h1:GEiTHELF+vaR5dhz3VqZfFSzZjYbgeKDpBxQVS4GYJ0= github.com/spf13/cast v1.6.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= +github.com/spf13/cobra v1.8.0 h1:7aJaZx1B85qltLMc546zn58BxxfZdR/W22ej9CFoEf0= +github.com/spf13/cobra v1.8.0/go.mod h1:WXLWApfZ71AjXPya3WOlMsY9yMs7YeiHhFVlvLyhcho= github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spf13/viper v1.18.2 h1:LUXCnvUvSM6FXAsj6nnfc8Q2tp1dIgUfY9Kc8GsSOiQ= @@ -195,6 +207,7 @@ golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5h golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20201015000850-e3ed0017c211/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=