- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我编写了一个小的 *nix 实用程序,每次检测到文件系统更改时都会“重新运行”给定的命令。所以我将命令作为引用参数传递给它,例如
rerun "my command"
Rerun 是用 Python 写的,最后调用:
subprocess.call("my command", shell=True, executable=USERS_DEFAULT_SHELL)
在我的例子中,我的默认 shell 是“/bin/bash”。但是,subprocess.call 调用的 shell 不是“交互式”shell,因此无法识别我的 .bashrc 中定义的 shell 函数和别名。
Man bash 告诉我要启动交互式 shell,我将“-i”传递给/bin/bash。但是,可以预见的是,
subprocess.call(..., executable='/bin/bash -i')
不起作用 - 它无法找到该名称的可执行文件。 (即使它确实有效,我也在尝试为用户默认的任何 shell 创建此功能,而不仅仅是 Bash。可能 '-i' 不会对所有其他 shell 执行相同的操作。)
我怎样才能从 Python 执行“我的命令”,就像用户将它输入终端时所解释的那样?
最佳答案
Posix 需要 -i
shell 的选项以“交互模式”启动 shell。交互模式的精确定义因 shell 而异——显然 zsh
和 csh
不会尝试解释 .<b>bash</b>rc
中的命令-- 但使用 -i
flag 应该用所有合理的 shell 做正确的事。
通常,您会通过调用 subprocess.call
来传递参数(或一些 Popen
变体)带有列表:
subprocess.call(['bash', '-i'])
当然,这不会尊重用户的 shell 偏好。你应该能够从 SHELL
得到它环境变量:
subprocess.call([os.getenv('SHELL'), '-i'])
为了让shell执行特定的命令行,你需要使用-c
命令行选项,这也是 Posix 标准,因此它应该适用于所有 shell:
subprocess.call([os.getenv('SHELL'), '-i', '-c', command_to_run])
这在很多情况下都可以正常工作,但如果 shell 决定 exec
,它可能会失败。 command_to_run
中的最后(或唯一)命令(有关详细信息,请参阅 this answer 上的 http://unix.stackexchange.com。)您随后尝试调用另一个 shell 来执行另一个命令。
例如,考虑简单的 python 程序:
import subprocess
subprocess.call(['/bin/bash', '-i', '-c', 'ls'])
subprocess.call(['/bin/bash', '-i', '-c', 'echo second']);
第一个bash
过程开始。由于它是一个交互式 shell,它会创建一个新的进程组并将终端附加到该进程组。然后它检查要运行的命令,确定它是一个运行外部实用程序的简单命令,因此它可以 exec
命令。所以它这样做了,用 ls
代替自己实用程序,现在是终端进程组的领导者。当 ls
终止,终端进程组变为空,但终端仍附加到它。因此,当第二个 bash 进程启动时,它会尝试创建一个新的进程组并将终端附加到该进程组,但这是不可能的,因为终端处于一种不确定状态。根据 Posix 标准(基本定义,§11.1.2):
When there is no longer any process whose process ID or process group ID matches the foreground process group ID, the terminal shall have no foreground process group. It is unspecified whether the terminal has a foreground process group when there is a process whose process ID matches the foreground process group ID, but whose process group ID does not. No actions defined in POSIX.1-2008, other than allocation of a controlling terminal or a successful call to
tcsetpgrp()
, shall cause a process group to become the foreground process group of the terminal.
对于 bash,只有当字符串作为 -c
的值传递时才会发生这种情况。 argument 是一个简单的命令,因此有一个简单的解决方法:确保该字符串不是一个简单的命令。例如,
subprocess.call([os.getenv('SHELL'), '-i', '-c', ':;' + command_to_run])
在命令前添加一个空操作,使其成为一个复合命令。但是,这不适用于其他在尾调用优化方面更具侵略性的 shell。所以一般的解决方案需要遵循 Posix 建议的路径,同时注意 tcsetpgrp
的描述。系统调用:
Attempts to use
tcsetpgrp()
from a process which is a member of a background process group on a fildes associated with its controlling terminal shall cause the process group to be sent aSIGTTOU
signal. If the calling thread is blockingSIGTTOU
signals or the process is ignoringSIGTTOU
signals, the process shall be allowed to perform the operation, and no signal is sent.
由于对 SIGTTOU
的默认操作signal是停止进程,我们需要忽略或者阻塞这个signal。所以我们最终得到以下结果:
!/usr/bin/python
import signal
import subprocess
import os
# Ignore SIGTTOU
signal.signal(signal.SIGTTOU, signal.SIG_IGN)
def run_command_in_shell(cmd):
# Run the command
subprocess.call([os.getenv('SHELL'), '-i', '-c', cmd])
# Retrieve the terminal
os.tcsetpgrp(0,os.getpgrp())
run_command_in_shell('ls')
run_command_in_shell('ls')
关于python - 从 Python 启动一个可以解释函数和别名的 shell,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25099895/
我相信我在子 shell 中调用 exit 会导致我的程序继续: #!/bin/bash grep str file | while read line do exit 0 done
我相信我在子 shell 中调用 exit 会导致我的程序继续: #!/bin/bash grep str file | while read line do exit 0 done
我有几个脚本,它们的第一部分看起来是一样的。这部分的功能是识别脚本在哪台机器上运行并相应地设置几个变量。它看起来像这样: ENV=`echo $LOGNAME | cut -c1-8` if
这是我正在尝试做的事情。我有 4 个 shell 脚本。脚本 1 需要先运行,然后是 2,然后是 3,然后是 4,并且它们必须按此顺序运行。脚本 1 需要运行(并在后台等待)2 才能正常运行,但是脚本
我有一个名为 a.sh 的脚本,其中的内容是: //a.sh: #!/bin/bash temp=0 while [ "$temp" -ne 500 ] do echo `date`
在snakemake中,使用shell()函数执行多个命令的推荐方式是什么? 最佳答案 您可以调用shell()多次内run规则块(规则可以指定 run: 而不是 shell: ): rule pro
我有一个 shell 脚本,我向其中传递了一些参数。Test1.sh -a 1 -b 2 -c“一二三” 在 Test1.sh 中,我按以下方式调用另一个 shell 脚本。Test2.sh $* 我
我有 2 个 shell 脚本。 第二个shell脚本包含以下函数第二个.sh func1 func2 first.sh 将使用一些参数调用第二个 shell 脚本, 将使用特定于该函数的一些其他参数
我有一个 Unix shell 脚本 test.sh。在脚本中,我想调用另一个 shell,然后从子 shell 执行 shell 脚本中的其余命令并退出 说清楚: test.sh #! /bin/b
我想在 shell 脚本中更改路径环境变量。路径变量需要在shell脚本执行后修改。 最佳答案 我知道有两种方法可以做到这一点。第一种是在当前 shell 的上下文中运行脚本: . myscript.
此 shell 脚本按预期运行。 trap 'echo exit' EXIT foo() { exit } echo begin foo echo end 这是输出。 $ sh foo.sh
我正在使用 vimshell在 vim 中执行命令 nnoremap vs :VimShellPop 使用此键映射,我可以打开 vim shell 并执行诸如“捆绑安装”之类的命令,然后 输入 exi
我想连接到不同的 shell(csh、ksh 等)并在每个切换的 shell 中执行命令。 下面是反射(reflect)我的意图的示例程序: #!/bin/bash echo $SHELL csh e
我目前正在尝试使用 BNF 和 LL 解析器在 C 中重新编写 shell。 否则,我需要知道 shell 运算符的优先级是什么| , > , > , & , ; ? 有没有人可以提供给我? 谢谢 最
不幸的是,我没有suspend 命令(busybox/ash)。但是我可以使用 kill -STOP $$ 从后台 shell (sh &) 返回到父 shell(以及 fg 之后)。 但是我不想输入
我需要知道,当用户切换到另一个 shell 时,通过单击它。 我试过 shellListener.shellDeactivated()但是当 shell 失去对它自己的控件的焦点时,会触发此事件,这意
file1.txt aaaa bbbb cccc dddd eeee file2.txt DDDD cccc aaaa 结果 bbbb eeee 如果能不区分大小写就更好了! 谢谢! 最佳答案 gre
我见过解压缩目录中所有 zip 文件的循环。但是,在运行此之前,我宁愿确保我将要运行的内容正常工作: for i in dir; do cd $i; unzip '*.zip'; rm -rf *.z
我对编程还很陌生,但我想知道 vim、emacs、nano 等 shell 文本编辑器如何能够控制命令行窗口。我主要是一名 Windows 程序员,所以可能在 *nix 上有所不同。据我所知,只能将文
我有一个包含第 7 列日期的文件,我的要求是将它与今天的日期进行比较,如果小于它,则删除该完整行。 此外,如果第 7 列中提到的任何日期超过 15 天,则将其修改为最多 15 天 下面的例子- now
我是一名优秀的程序员,十分优秀!