草庐IT

go - 通过 Golang 执行 shell 脚本后返回错误

coder 2024-07-11 原文

我有一个简单的 shell 脚本(名为 copy.sh),如下所示:-

#! /bin/sh

cp $1 $2

我执行了 chmod 777 copy.sh

我有一个执行上述 shell 代码的 golang 代码:-

package main

import (
    "fmt"
    "os/exec"
)

func main() {
    _, err := exec.Command("/Users/debraj/copy.sh", "/Users/debraj/temp.txt", "/Users/debraj/gotest/").Output()
    if err != nil {
        fmt.Println("Failed to execute command " + err.Error())
        return
    }
    fmt.Printf("\nCopy Successful - %v")
}

上面的代码显示为以下输出:-

jabongs-MacBook-Pro-4:src debraj$ go run copyerr.go 
Failed to execute command exit status 1
jabongs-MacBook-Pro-4:src debraj$ 

但是我从 shell 脚本收到的错误如下所示:-

jabongs-MacBook-Pro-4:~ debraj$ ./copy.sh /Users/debraj/temp.txt /Users/debraj/gotest/
cp: /Users/debraj/gotest/temp.txt: Permission denied     

谁能告诉我如何才能得到 shell 脚本返回的相同错误消息?

如果我不执行 chmod 777 copy.sh 并且文件具有如下权限:-

jabongs-MacBook-Pro-4:~ debraj$ ls -la copy.sh 
-rw-r--r--  1 debraj  staff  21 Sep 29 13:28 copy.sh

然后 golang 代码给出 shell 脚本给出的输出。有些人还可以让我知道为什么会这样吗?

我在

  • Go 语言 1.7
  • Mac OS X 10.11.4

最佳答案

好的!所以问题的核心是你有两个不同的地方可能会出错:

  1. 当 exec.Command().Output() 根本无法运行您的 shell 脚本时(因为在这种情况下它没有权限),并且
  2. 当您的 shell 脚本本身运行时,但返回错误代码(因为脚本中运行的最后一行是错误)。

在第一种情况下,你应该在 err 中得到一些合理的东西,因为 Go 本身在尝试做某事时遇到了一个定义明确的系统错误(立即失败)。在第二种情况下,Go 能够运行你的脚本,但它有输出(你忽略了)和一个返回码(在这种情况下,是你的脚本最后执行的退出代码)。由于输出实际上可能是任何东西(包括数兆字节和数兆字节的无意义文本),Go 采取了懒惰的方式,只返回状态代码。恕我直言,这是正确的做法,因为基本上每个人都在猖獗地滥用 stderr。

另外,您的脚本(如果稍作修改)实际上可以返回一个错误代码,而根本不显示错误消息!

也就是说,如果您想查看脚本中的实际错误信息,请执行以下操作之一:

// Display everything we got if error.
output, err := exec.Command(...).CombinedOutput()
if err != nil {
    fmt.Println("Error when running command.  Output:")
    fmt.Println(string(output))
    fmt.Printf("Got command status: %s\n", err.Error())
    return
}

或:

// Display just the stderr if an error occurs
cmd := exec.Command(...)
stderr := &bytes.Buffer{}    // make sure to import bytes
cmd.Stderr = stderr
err := cmd.Run()
if err != nil {
    fmt.Println("Error when running command.  Error log:")
    fmt.Println(stderr.String())
    fmt.Printf("Got command status: %s\n", err.Error())
    return
}

...或者您可以更高级地使用 StderrPipe()并阅读最后一行或类似内容。真的,有很多方法可以给这只猫换皮(因为 ​​cmd.Stderr 可以是任何 io.Writer,你可以轻松地编写几乎任何处理程序——包括一个解析流以寻找根本原因等的处理程序)。

需要考虑的是,所使用的 Output() 和 CombinedOutput() 仅在脚本完全运行完毕后才会显示内容。对于长时间运行的脚本,这可能是一个严重的 UI 问题,因为这意味着任何监视正在发生的事情的用户都将在处理过程中看到大量的空载。使用 StdoutPipe() 和 StderrPipe() 可能更可取,因为您可以在输出发生时解析/解释(或仅显示)输出。当然,如果您可以保证您的脚本不会运行超过几秒钟,或者它不会被人监视,那谁在乎呢?

如果您想探索如何在脚本运行时显示输出,StdoutPipe() 的示例是一个很好的起点。

关于go - 通过 Golang 执行 shell 脚本后返回错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39764600/

有关go - 通过 Golang 执行 shell 脚本后返回错误的更多相关文章

  1. ruby-openid:执行发现时未设置@socket - 2

    我在使用omniauth/openid时遇到了一些麻烦。在尝试进行身份验证时,我在日志中发现了这一点:OpenID::FetchingError:Errorfetchinghttps://www.google.com/accounts/o8/.well-known/host-meta?hd=profiles.google.com%2Fmy_username:undefinedmethod`io'fornil:NilClass重要的是undefinedmethodio'fornil:NilClass来自openid/fetchers.rb,在下面的代码片段中:moduleNetclass

  2. ruby-on-rails - Rails 常用字符串(用于通知和错误信息等) - 2

    大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje

  3. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  4. ruby - 通过 rvm 升级 ruby​​gems 的问题 - 2

    尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub

  5. ruby - 为什么 4.1%2 使用 Ruby 返回 0.0999999999999996?但是 4.2%2==0.2 - 2

    为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返

  6. ruby - 通过 erb 模板输出 ruby​​ 数组 - 2

    我正在使用puppet为ruby​​程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby​​不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这

  7. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

  8. ruby-on-rails - 独立 ruby​​ 脚本的配置文件 - 2

    我有一个在Linux服务器上运行的ruby​​脚本。它不使用rails或任何东西。它基本上是一个命令行ruby​​脚本,可以像这样传递参数:./ruby_script.rbarg1arg2如何将参数抽象到配置文件(例如yaml文件或其他文件)中?您能否举例说明如何做到这一点?提前谢谢你。 最佳答案 首先,您可以运行一个写入YAML配置文件的独立脚本:require"yaml"File.write("path_to_yaml_file",[arg1,arg2].to_yaml)然后,在您的应用中阅读它:require"yaml"arg

  9. ruby - 通过 RVM (OSX Mountain Lion) 安装 Ruby 2.0.0-p247 时遇到问题 - 2

    我的最终目标是安装当前版本的RubyonRails。我在OSXMountainLion上运行。到目前为止,这是我的过程:已安装的RVM$\curl-Lhttps://get.rvm.io|bash-sstable检查已知(我假设已批准)安装$rvmlistknown我看到当前的稳定版本可用[ruby-]2.0.0[-p247]输入命令安装$rvminstall2.0.0-p247注意:我也试过这些安装命令$rvminstallruby-2.0.0-p247$rvminstallruby=2.0.0-p247我很快就无处可去了。结果:$rvminstall2.0.0-p247Search

  10. ruby - Chef 执行非顺序配方 - 2

    我遵循了教程http://gettingstartedwithchef.com/,第1章。我的运行list是"run_list":["recipe[apt]","recipe[phpap]"]我的phpapRecipe默认Recipeinclude_recipe"apache2"include_recipe"build-essential"include_recipe"openssl"include_recipe"mysql::client"include_recipe"mysql::server"include_recipe"php"include_recipe"php::modul

随机推荐