3

我需要创建一个基本上设置一些环境变量的脚本,启动一个子shell,并在退出时运行一些额外的代码,作为一个钩子。它需要同时在 Bash 和 ZSH 上工作,而且我知道也许我应该维护两个不同的脚本。

我需要它的原因是创建某种引导脚本。例如,我创建了一个名为Bootstrap.sh的文件,其中包含两个函数startexit,当我运行脚本时,start应该调用它并激活一个新的子 shell。当子shell 退出时,exit, 最终被调用。例如:

# Bootstrap.sh
start() {
    export CUSTOM_VAR="hello world"
}

exit() {
    echo "byebye"
}

$ bootstrap
$ # 找到“Bootstrap.sh”,加载它,然后调用“start”函数
(myenv) $ echo $CUSTOM_VAR
hello world
(myenv) $ exit
byebye # 这是我的退出钩子
$ # 回到“正常”shell

根据改进 Python 的virtualenv的建议,我正在使用这个脚本测试钩子。在这个文件中,我尝试使用陷阱 (a) 和 TRAPEXIT(b),但这些都不起作用:

# (a), the "bash way", I think
my_hook() {
    echo "exit hook called"
}
trap my_hook EXIT
exec "${@:-$SHELL}"
# (b), as explained in ZSH documentation
# http://zsh.sourceforge.net/Doc/Release/Functions.html
TRAPEXIT() {
    echo "exit hook called"
}
exec "${@:-$SHELL}"

那么,我该怎么做(如果可能,在 ZSH 和 Bash 中)?

4

2 回答 2

3

根据 ZSH 手册页,ZSH 还支持trap内置;那是,

trap 'any code here' EXIT

也适用于 ZSH,并且您可以将函数名称作为代码字符串。

如果您需要包装 exec,以下内容应该适用于 bash 和 zsh

function exec {
    echo "Execcing"
    builtin exec "$@"
}
于 2013-08-14T02:12:25.850 回答
1

问题是最后的'exec'。这(显然)不算作退出。当然,被替换的进程不会继承 shell 的退出处理程序。

解决方案:更换

exec "${@:-$SHELL}"

经过

"${@:-$SHELL}"

PS:我说的很明显,因为我从没想过它是否会(我的意思是算作EXIT)。我不得不尝试一下,但按道理,那个 exec 与退出无关。该进程仅在“{@:-$SHELL}”终止时退出。

于 2013-08-14T02:01:20.070 回答