是否可以在Git中使用shell的内置命令进行别名扩展?

时间:2015-09-17 08:04:43

标签: git shell alias

我的.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
$

有没有办法让它发挥作用?

1 个答案:

答案 0 :(得分:2)

原因

在阅读完问题之后,我还认为问题是由于history是内置shell的事实。

调试git

可以通过设置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内置函数进行了进一步的实验,这些实验得以实现,所以我认为它必须是别的东西:

没有非交互式shell的历史记录

阅读Bash手册说明Bash History Facilities仅在interactive shells中使用,而exec调用启动的shell是非交互式的。这就是运行history内置函数不打印任何输出的原因。

解决方案

选项1:打开非交互式shell

中的历史记录功能

感谢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),这就是我在此处不使用它的原因。

选项2:从历史文件中读取

更简单的解决方案是直接从shell历史文件中读取:

freq = !grep ^git $HOME/.bash_history | sort | uniq -c | sort -n -r | head -n 5

这会产生与在新的交互式shell中运行history命令相同的结果,因为当新的shell启动时,除了从历史文件中读取的命令之外,它的历史记录中没有命令。

此外,直接从历史文件中读取时不需要cut命令。

选项3:Bash别名

另一种选择是忘记使用Git别名并简单地使用Bash别名:

alias git-freq='history | cut -c 8- | grep ^git | sort | uniq -c | sort -n -r | head -n

保持shell历史文件最新

我以为我会添加以下内容作为有用的辅助信息。请注意,它适用于交互式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"