ChatGPT解决这个技术问题 Extra ChatGPT

复杂的 Git 分支名称破坏了所有 Git 命令

我试图使用以下命令从 master 创建一个分支,

git branch SSLOC-201_Implement___str__()_of_ProductSearchQuery

当 Git 突然停止响应时。我怀疑未转义的 () 应该受到责备,不知何故。现在,每当我尝试运行任何 Git 命令时,都会收到相同的错误:

git:176: command not found: _of_ProductSearchQuery

每次输入命令时,git 之后的数字都会增加。

谁能解释发生了什么?以及如何恢复正常?我想删除那个分支,但我该怎么做呢?

我猜这与您的 zsh 环境有关,因为我能够在我的 bash shell 中运行创建分支而没有不良副作用(lubuntu 13.10),但是当我切换到完全香草 zsh 时我看到了错误
将来,引用看起来可疑的东西。 git branch "SSLOC-201_Implement___str__()_of_ProductSearchQuery" 工作得很好。
@Qix 最好完全避免有问题的字符。
@Jubobs 绝对是,尽管我看到某些公司强制执行这样奇怪的分支名称。
@DwightSpencer 您的链接特定于 Bash,但这个问题是特定于 zsh 的。这个问题实际上并没有发生在 Bash 中。

C
Community

问题

谁能解释发生了什么? [...] 我希望能够删除该分支,但 Git 不适合我。

通过运行

git branch SSLOC-201_Implement___str__()_of_ProductSearchQuery

在 zsh 中,你没有创建任何分支。相反,您不小心定义了三个 shell 函数,称为 gitbranchSSLOC-201_Implement___str__,它们忽略了它们的参数(如果有的话)并且其主体是 _of_ProductSearchQuery。您可以通过调用名为 functions 的内置 zsh 命令来检查自己是否确实发生了这种情况,该命令列出了所有现有的 shell 函数:

$ functions                                                     
SSLOC-201_Implement___str__ () {
    _of_ProductSearchQuery
}
branch () {
    _of_ProductSearchQuery
}
git () {
    _of_ProductSearchQuery
}

不幸的是,虽然其他两个 shell 函数没有问题,但 名为“git”的 shell 函数现在会影响 真正的 git 命令!

$ which git
git () {
    _of_ProductSearchQuery
}
# but the real "git" is a binary file that lives in /usr/local/bin/git (or some similar path)

因此,您随后将收到错误消息

command not found: _of_ProductSearchQuery

每当您尝试运行 Git 命令时,例如 git loggit status 等(当然,假设不存在名为 _of_ProductSearchQuery 的命令)。

边注

[...]我得到同样的错误:git:176:找不到命令:_of_ProductSearchQuery(每次我输入命令时git之后的数字都会增加)

该数字仅对应于 HISTCMD 的值,该环境变量包含

[t]交互式 shell 中的当前历史事件编号,即导致读取 $HISTCMD 的命令的事件编号。

有关详细信息,请参阅 zsh manual

解决方案

以及如何恢复正常?

只需删除有问题的 shell 函数(以及您在使用它时意外创建的另外两个):

unset -f git
unset -f branch SSLOC-201_Implement___str__

那么一切都应该没问题。

如果 unset 也被遮蔽了怎么办?!

Good question!我请您参考下面的Wumpus W. Wumbley's excellent comment

分支命名技巧

避免任何特殊的外壳字符

是的,正如评论中所指出的,括号是 Git 分支名称中的有效字符;您只需要适当地引用名称,例如

$ git branch 'foo()bar'
$ git branch
  foo()bar
* master
$ git checkout 'foo()bar'
Switched to branch 'foo()bar'

但是,当用作命令行参数时,每次都需要引用此类名称应该说服您避免在引用名称中使用括号。更一般地说,您应该(尽可能)避免在 shell 中具有特殊含义的字符,以防止像这样的意外。

使用简单的分支名称

无论如何,你应该保持你的分支名称简短而甜美。长描述如

SSLOC-201_Implement___str__()_of_ProductSearchQuery

属于提交消息,而不是分支名称。


该线程中没有任何内容表明括号是非法的。 Git 似乎很喜欢它。 Switched to a new branch 'abcd-()-foo'
看起来不错;使用它们绝对不是一个好主意,但它们在技术上并非无效。
如果有人也通过创建所谓的 shell 函数来隐藏 unset 会发生什么? (这可能吗?)
@codroipo 哈!那是个很好的观点。是的,有可能,在这种情况下,您最好重新启动 zsh
您可以使用 builtin unset。如果 builtinunset 都被函数遮蔽,则 unfunction。如果那也消失了,unhash -f。如果这四个都消失了,然后重新启动 shell。