我的.gitconfig
中有以下别名:
freq = !history | cut -c 8- | grep ^git | sort | uniq -c | sort -n -r | head -n 5
它应该显示我最常使用的5个git命令。取自这里:http://alblue.bandlem.com/2011/04/git-tip-of-week-aliases.html(我知道这篇文章是旧的)
当我运行命令时,它不会输出任何内容:
$ git freq
$
但如果我运行history | cut -c 8- | grep ^git | sort | uniq -c | sort -n -r | head -n 5
,则会输出:
$ history | cut -c 8- | grep ^git | sort | uniq -c | sort -n -r | head -n
5 git log
5 git status
5 git current
5 git co master
4 git co other-branch
我想这是因为history
是一个shell内置命令,如果我更改freq
别名,例如到:
freq = !history
我明白了:
$ git freq
error: cannot run history: No such file or directory
fatal: While expanding alias 'freq': 'history': No such file or directory
$
有没有办法让它发挥作用?
答案 0 :(得分:2)
在阅读完问题之后,我还认为问题是由于history
是内置shell的事实。
可以通过设置GIT_TRACE
环境变量来调试Git命令
至true
(有关详细信息,请参阅Scott Chacon的Git Book中Git Internals - Environment Variables的相关部分)。
$ GIT_TRACE=true git freq
10:14:11.901296 git.c:554 trace: exec: 'git-freq'
10:14:11.901296 run-command.c:341 trace: run_command: 'git-freq'
10:14:11.932496 run-command.c:341 trace: run_command: 'history | tail'
10:14:11.963697 run-command.c:192 trace: exec: '/bin/sh' '-c' 'history | tail' 'history | tail'
Checking the source显示外部命令由execv_shell_cmd
函数处理,该函数调用sane_execvp
,这是execvp
标准库的一个用户友好的前端函数 - 如果命令名称不包含斜杠 - 在PATH中的每个目录中搜索与该命令同名的可执行文件。由于history
是内置的shell,因此无法找到它,因此返回错误。参见
不是一个放手的人,我后来使用其他shell内置函数进行了进一步的实验,这些实验得以实现,所以我认为它必须是别的东西:
阅读Bash手册说明Bash History Facilities仅在interactive shells中使用,而exec
调用启动的shell是非交互式的。这就是运行history
内置函数不打印任何输出的原因。
感谢Gilles on Unix and Linux,事实证明,可以通过设置HISTFILE
shell变量然后运行set -o history
来启用非交互式shell中的历史记录功能通过运行:
bash -c "HISTFILE=~/.bash_history; set -o history; history"
在.gitconfig
中设置以下别名将起作用:
freq = "!HISTFILE=$HOME/.bash_history; set -o history; history | cut -c 8- | grep ^git | sort | uniq -c | sort -n -r | head -n 5"
注意:如果您使用的是Bash以外的shell,则需要指定shell将其命令历史记录保存到的文件的名称。如果您使用的是Bash,则HISTFILE
shell变量可能设置为$HOME/.bash_history
以外的其他值。
此外,不应将HISTFILE
等shell变量导出为环境变量(可用于git),这就是我在此处不使用它的原因。
更简单的解决方案是直接从shell历史文件中读取:
freq = !grep ^git $HOME/.bash_history | sort | uniq -c | sort -n -r | head -n 5
这会产生与在新的交互式shell中运行history
命令相同的结果,因为当新的shell启动时,除了从历史文件中读取的命令之外,它的历史记录中没有命令。
此外,直接从历史文件中读取时不需要cut
命令。
另一种选择是忘记使用Git别名并简单地使用Bash别名:
alias git-freq='history | cut -c 8- | grep ^git | sort | uniq -c | sort -n -r | head -n
我以为我会添加以下内容作为有用的辅助信息。请注意,它适用于交互式Bash shell,它可能适用于其他现代shell,也可能不适用。
我经常在同一主机上同时运行多个shell,并且我不希望在退出shell时覆盖已保存的历史命令。以下是我的.bashrc
设置,可确保在每个命令之后写入shell历史文件:
# Append history entries on logout - instead of over-writing the history file.
shopt -s histappend
# Don't store duplicate lines in the history.
# Ignore any commands that begin with a space.
HISTCONTROL=ignoredups:ignorespace
# Use a very large history file to store lots of commands.
# See http://mywiki.wooledge.org/BashFAQ/088
HISTFILESIZE=5000
# Keeping a large history requires system resources
HISTSIZE=3000
# Append to the history file after every command.
PROMPT_COMMAND="history -a"