首页 > 解决方案 > Go exec 不捕获 git log 输出

问题描述

我正在将我们用 Ruby(使用 Thor)编写的 Cli 工具翻译成 Go。

到目前为止,我在传递 git 命令和捕获我想要的输出方面没有任何问题。

我想要输出时使用的一般代码是:

func output(c string, args []string, printOut bool) (string, error) {
    cmd := exec.Command(c, args...)

    if verbose {
        text.Command(c, args)
    }

    if printOut {
        cmd.Stdout = os.Stdout
        cmd.Stderr = os.Stderr
    }

    output, err := cmd.Output()

    if err != nil {
        reportError()

        return "", err
    }

    reportOk()

    return string(output), nil
}

我对以下命令有一些问题:

git --no-pager log --merges --grep='^merge.*(hotfix|release)|(.*from.*(hotfix|release))' --regexp-ignore-case --perl-regexp --pretty=format:%H -n 1

我这样调用输出:

    args := []string{
        "--no-pager", "log", "--merges", "--grep='^merge.*(hotfix|release)|(.*from.*(hotfix|release))'", "--regexp-ignore-case", "--perl-regexp", "--pretty=format:%H", "-n", "1",
    }

    out, err := output(cmd, args, false)

对于某些上下文,此命令用于查找最后一个版本/修补程序合并的哈希,以便查找从该提交到“现在”的所有相关提交以添加到更改日志中。

如果我删除--grep='^merge.*(hotfix|release)|(.*from.*(hotfix|release))' --regexp-ignore-case --perl-regexp,我会得到一个提交哈希输出,但我需要正则表达式来确保它是我想要的合并。

事情是cmd.Output()当我传递完整命令时不会返回错误,它只是返回一个空字符串。

完整的命令在终端上执行时有效。

有谁知道我可能做错了什么?

非常感谢!

标签: gitgocommand-line-interface

解决方案


您在调用git grep 时没有/bin/sh/bin/bash任何 shell 1首先在命令上得到它肮脏的小爪子。

使用时--grep,我们(我的意思是人类在 shell 中输入命令)编写如下内容:

git log --grep='^merge|whatever' whatever-else

在这里用单引号保护模式免受 shell 的影响。单引号确保竖线|字符不被视为管道,即,这不是命令:

git log --grep=^merge | whatever ...

这会将输出从管道git log --grep=^merge传输到whatever.

我们在这里需要引号,因为 shell 正在解释我们的命令,并特别对待某些字符;单引号破坏了这一点,并在此过程中被外壳消耗。 例如,这意味着Git可以看到--grep=^merge|whatever.

但是你没有调用 shell。没有什么可以保护的。因此,您正在指示 Git 查找'^merge,即单引号后跟帽子/插入符号^,后跟 anm后跟 an e,依此类推。这在您的任何提交中都不存在,因此您没有得到您期望的输出。

解决方案很简单:停止针对不存在的 shell 引用事物,这样您就不会将意外的引号传递给 Git。如果您出于某种原因确实需要使用 shell,请保留引号,但要非常注意其他所有内容,因为不小心调用 shell 会导致 28.41% 的 CVE(这个统计数据,就像所有统计数据的 72.93% 一样,是虚构的)当场)。


1在某些系统上,/bin/sh /bin/bash; 在一些它的dash; 在某些情况下,它是其他一些变体。这有时会造成便携性烦恼。在这里它并不重要,因为所有 shell 都以相同的方式对待它们。


推荐阅读