- 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/
每当我运行命令以将 Virtualbox 驱动程序启动 Minishift 到操作系统主机时,它都需要一段疯狂的时间,而且它永远不会结束。有时我什至收到有关达到存储限制的错误消息。 不知道是不是描述h
您好,我正在使用 npm 运行一个基本的 React 项目,我正尝试在 docker 容器中启动它。但是我实际上无法让项目运行。我的 dockerfile 看起来像这样: FROM node:7.8.
所以我想从我的 SSH 终端开始游戏。 这真的很奇怪,当我直接从 Linux GUI 执行此操作时,它可以工作。但是当我使用 SSH 客户端进行远程连接时,它就崩溃了。似乎与我的显示驱动程序有关。 U
我有一个显示图像的动态壁纸。我在 Activity 中更改了该图像。然后我需要通知动态壁纸,以便它知道重新加载资源。 Intent 似乎是完美、简单的解决方案: Intent intent = new
我有一个似乎无法解决的问题。我在 Boot Dashboard 中使用 STS 3.9.2 从 Eclipse (Oxygen) 启动 Spring Boot 应用程序没有任何问题: 但是,当我尝试从
全新的 Python,在我开始摆弄东西之前先设置和安装东西。我的理解是 Python 2.7 和 Python 3.3 之间存在一些显着差异/不兼容,尽管这两个版本都得到了很好的使用,所以我认为最好安
在使用了很长时间的 jQuery 之后,我有一个问题,我正在使用 jQuery 模式(样式)编写一个简单的代码, (function(window, undefined) { var jQu
我正在尝试在 spring boot 应用程序下的非 spring 托管类中配置 Autowired。我在 tomcat 服务器下部署的 Web 应用程序下成功运行了这个。但是当我想在 spring
我对 xmonad 完全陌生,但我想开始使用它来提高我的工作效率。 这是我一直在使用的指南(我使用的是 Apple OS X Snow Leopard) http://xmonad.org/tour.
我试图将Spring Boot指南中的Managing Transactions示例扩展到两个数据源,但是@Transaction注释似乎仅对其中一个数据源有效。 在“Application.java
conEmu 有没有办法默认打开多个不同的选项卡? 我看到这个页面解释了如何使用 splits , 我意识到我可以按 Ctrl + T, 1, Enter,但我希望有一种方法可以自动执行此操作! "%
我正在寻找快速而肮脏的答案。我当时脑子一片空白,盯着屏幕看了 12 个小时以上,我想我中枪了。 我想做一个简单的 SignalR 应用程序作为教程。我找到了这个example ,但我不断收到票证未定义
我正在使用 Azure Powershell cmdlet 来启动/停止 VM。 Start-AzureVM [-ServiceName] [-Name] [ ] Stop-AzureVM [-S
我想使用Powershell脚本代码启动/停止iis和mssql 意味着当我运行ps脚本时,我想启动/停止iis和mssql 我在网上搜索了它,发现了一些代码,但按照我的要求无法正常工作 码: $ii
我在 liferay 工作。我们在我们的项目中使用一个模块来创建 liferay 主题。我使用命令 ant -Ddeploy.war=true 将它部署在服务器中。 war 文件在 liferay 部
我想在已安装 Python 2.7 的 Windows XP 计算机上运行 IPython(版本 0.12)。 我通过 Windows 二进制安装程序安装,但安装后 IPython 没有显示在菜单中,
我从创建了自己的简单图片。 FROM python:2.7.11 RUN mkdir /extra/later/ \ && mkdir /yyy 现在,我可以执行以下步骤: docker run
$(document).ready(function () { setTimeout(function() { window.location.reload(); }, 2000); // 2
我刚刚创建了一个帐户 OpenWeatherMap 我想通过城市 ID API 调用获取当前位置的天气: http://api.openweathermap.org/data/2.5/weather?
我注意到,如果我更改 xcasset 中的图像,启动 Storyboard不会更新。 例如,假设您的启动 Storyboard中有一个 UIImage View ,其中包含一个名为“logo”的蓝色图
我是一名优秀的程序员,十分优秀!