python - 如何在使用 python 在提示符下发出命令后截取 stdout 的每一行

标签 python linux bash shell

我已经编写了一个正在执行以下操作的 python 脚本;

  1. 连接到 LUN
  2. 安装驱动器
  3. 将我的主驱动器备份到该 LUN

以上所有内容都保存在日志文件中,一切都正常工作,但我没有从日志文件中得到我想要的内容。

import subprocess
from datetime import datetime
from sys import stderr

logfile="/var/tmp/home-backup.log"
TimeStamp = datetime.now().strftime("%a %d %b %Y %H:%M:%S - ")
BackStamp = datetime.now().strftime("%d-%b-%Y")
Portal = "192.168.1.59:3260"
Target = "iqn.2014-07.com.synology:fabdskst.linux-tgt1.888dddfd71"

File_obj = open(logfile, "a",0)

MyCommand = "sudo iscsiadm -m node --targetname " + Target + " --portal \"" + Portal + "\"" + " --login"
Discover = "sudo iscsiadm -d 0 -m discovery -t sendtargets -p 192.168.1.59:3260"
MyMount = "sudo mount -t ext4 -v /dev/sdc /media/linuxLUN01/"
MyBackup = "tar -cvzf /media/linuxLUN01/Mint_backup_" + BackStamp + ".tgz /home/fabrice/"

File_obj.write(TimeStamp)
subprocess.call(MyCommand, shell=True, stdout=File_obj, stderr=subprocess.PIPE)
File_obj.write(TimeStamp)
subprocess.call(MyMount, shell=True, stdout=File_obj, stderr=subprocess.PIPE)
File_obj.write(TimeStamp)
subprocess.call(MyBackup, shell=True, stdout=File_obj, stderr=subprocess.PIPE)

File_obj.close()

我希望当输出出现在屏幕上时,标准输出中的每一行都带有实时时间戳(在我的例子中,标准输出被重定向到日志文件)。相反,只有命令的第一个输出带有时间戳,其余部分则没有。

我正在学习Python,所以并不是我不好(你可能已经注意到我现在没有做任何测试,稍后会进行)。

我正在考虑如何解决这个问题,但我只是想出了一个不优雅的解决方案,在时间方面并不严格精确;

  1. 我会在开始时保存开始时间
  2. 将所有输出保存到临时文件
  3. 我会节省完成时间
  4. 我会计算临时文件中有多少条目,并获取开始时间和结束时间之间的差异,并将结果除以条目数((完成 - 开始)/计数)。这将为我提供构建真实日志文件的平均时间
  5. 逐行读取临时文件并像这样传输每个条目(开始时间 + ((n-1) * 平均时间) + “临时文件中的行”,并将 n 增加到每行的计数

任何人有其他想法可以使这个更准确吗?理想情况下,我想拦截每个标准输出条目并在其出现时为其添加时间戳。目前备份大约需要 5 分钟。

非常感谢任何建议

根据 Ram 的要求,从日志文件中摘录

Sat 04 Oct 2014 22:15:04 - Logging in to [iface: default, target: iqn.2014-07.com.synology:fabdskst.linux-tgt1.888dddfd71, portal: 192.168.1.59,3260] (multiple)
Login to [iface: default, target: iqn.2014-07.com.synology:fabdskst.linux-tgt1.888dddfd71, portal: 192.168.1.59,3260] successful.
Sat 04 Oct 2014 22:15:04 - Sat 04 Oct 2014 22:15:04 - /home/fabrice/
/home/fabrice/.gnome2_private/
/home/fabrice/.profile
/home/fabrice/.ICEauthority
/home/fabrice/.linuxmint/
/home/fabrice/.linuxmint/mintUpload/
/home/fabrice/.linuxmint/mintUpload/services/
/home/fabrice/Downloads/
/home/fabrice/Downloads/mshell.zip
/home/fabrice/Downloads/bash-4.3.tar.gz
/home/fabrice/Downloads/VMwareTools-9.2.4-1398046.tar.gz
/home/fabrice/Downloads/vmware-tools-distrib/
/home/fabrice/Downloads/vmware-tools-distrib/vmware-install.pl
/home/fabrice/Downloads/vmware-tools-distrib/lib/
/home/fabrice/Downloads/vmware-tools-distrib/lib/lib32/
/home/fabrice/Downloads/vmware-tools-distrib/lib/lib32/libgdkmm-2.4.so.1/
/home/fabrice/Downloads/vmware-tools-distrib/lib/lib32/libgdkmm-2.4.so.1/libgdkmm-2.4.so.1

您可以看到,只有每个命令的输出的第一行带有时间戳

至于stderr=subprocess.PIPE,我使用它没有什么特别的原因,我只是把它放在那里,就像我在网上找到的示例一样,我仍然需要了解它的作用。

最佳答案

不是经过测试的解决方案,但我相信它工作正常:

import subprocess as sub

[...]

with open(logfile, "a", 0) as File_obj:
    for cmd in (MyCommand, MyMount, MyBackup):
        with sub.Popen(cmd, shell=True, stdout=sub.PIPE, universal_newlines=True) as proc:
            while proc.poll() is None:
                log = [TimeStamp + line for line in proc.stdout.readlines()]
                File_obj.writelines(log)

观察:您希望 stderr 输出到终端,对吗?

关于python - 如何在使用 python 在提示符下发出命令后截取 stdout 的每一行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26195502/

相关文章:

Bash 怪异 : Exit Codes of Arithmetic Expressions in Relation to Conditionals. 真/假与成功/失败

python - 如何使用 twill 查看当前 url?

Python - 按日期透视日志数据

python - 如何替换 pandas 上过滤的多列中的 nan 值?

python - 我收到 urllib2.HTTPError : HTTP Error 502: Bad Gateway

linux - 如何检测子目录中的某些文件是否已更改?

linux - 从批处理 Linux 到 Windows 的 awk 和 sed 命令。如何转换?

linux - ONELINER 打印系统组及其标识符,适用于标识符以 '1' 开头的所有组

linux - Unix/巴什 : Redirect results of find command so files are used as input for other command

linux - 当我启动 bash 脚本时,如何找到 Ubuntu 中的所有用户并通过参数显示用户信息?