Python实时获取cmd的输出


Posted in Python onDecember 13, 2015

最近发现一个问题,一个小伙儿写的console程序不够健壮,监听SOCKET的时候容易崩,造成程序的整体奔溃,无奈他没有找到问题的解决办法,一直解决不了,可是这又是一个监控程序,还是比较要紧的,又必须想办法解决。

(这是要搞死我的节奏啊....)由于个人不太懂他用的语言,只能在程序外围想办法。

环境描述:

1. 目标程序执行时会监听8080端口,TCP,并在每一次client连接后通过console输出client的IP地址。

2. 监听不是一次性完成的,而是一直监听,程序并不会退出

3. 为了监控需要,最好能对连接的IP进行排序,整理。

P.S. 系统基于windows平台。

想起来做监控程序,简单点比较好,于是想到了Python。

我的预想逻辑是这样的,通过python检测目标程序是否崩了,如果中标就启动目标程序,并进行监控,每输出一次,python进行一次数据运算整理,然后循环。

第一步,先搞定输出的捕获问题。

# this method is used for monitoring
import time
import subprocess
import locale
import codecs
mylist = []
ps = subprocess.Popen('netstat -a', stdin=subprocess.PIPE, stdout=subprocess.PIPE, shell=True)
while True:
  data = ps.stdout.readline()
  if data == b'':
    if ps.poll() is not None:
      break
  else:
    mylist.append(data.decode(codecs.lookup(locale.getpreferredencoding()).name))
    newlist = []
    for i in mylist:
      if i.find('192.168') > 0:
        newlist.append(i)
    newlist.sort()
    print('Sum of requests from LAN:', len(newlist))

我用netstat -a替代那个需要持续输出的程序,执行程序,发现程序和想象的不太一样,确实是实时获得数据了,但是感觉总是有点不太和谐,不管了,继续。

第二步,解决监控程序的问题

程序或者还是死的,有一点非常关键,就是监听端口,那只要检测一下端口就行了。三个办法:

1. 找端口检测的API

2. 连接一次目标端口,通了就是活的

3. netstat

第一种方法需要去找找有没有相关的API,第二种方法容易对目标程序的正常运行造成问题,第三种我想都没想就用了吧。这里需要用到cmd的重定向功能

# this method is used for monitoring
import time
import subprocess
import locale
import codecs
def getstdout(p):
  mylist = []
  while True:
    data = p.stdout.readline()
    if data == b'':
      if p.poll() is not None:
        break
    else:
      mylist.append(data.decode(codecs.lookup(locale.getpreferredencoding()).name))
  return mylist
while True:
  ps = subprocess.Popen('netstat -an | findstr "8080"', stdin=subprocess.PIPE, stdout=subprocess.PIPE, shell=True)
  resultlist = getstdout(ps)
  if len(resultlist) >= 1:
    pass
  else:
    print(time.strftime("%Y-%m-%d %H:%M:%S"))
    subprocess.Popen('taskkill.exe /f /im node.exe', shell=False)
 # 防止动作过快,把新建的程序整死了
    time.sleep(3)
    subprocess.Popen('start node D:\\app.js', shell=True)
  time.sleep(10)

netstat -an获得当前的端口监听情况,“|”将netstat的输出重定向到findstr函数

netstat -an | findstr "8080" 查找有8080端口的地址行,有就说明活着,否则就是挂了。

最后一步,整合

# this method is used for monitoring
import time
import subprocess
import locale
import codecs
def getstdout(p):
  mylist = []
  while True:
    data = p.stdout.readline()
    if data == b'':
      if p.poll() is not None:
        break
    else:
      mylist.append(data.decode(codecs.lookup(locale.getpreferredencoding()).name))
  return mylist
while True:
  ps = subprocess.Popen('netstat -an | findstr "8080"', stdin=subprocess.PIPE, stdout=subprocess.PIPE, shell=True)
  resultlist = getstdout(ps)
  if len(resultlist) >= 1:
    pass
  else:
    print(time.strftime("%Y-%m-%d %H:%M:%S"))
    subprocess.Popen('taskkill.exe /f /im node.exe', shell=False)
    time.sleep(3)
    pss = subprocess.Popen('start cmd.exe /k node app.js', stdin=subprocess.PIPE,
                stdout=subprocess.PIPE, shell=True)
    alist = getstdout(pss)
    newlist = []
    for i in alist:
      if i.find('192.168') > 0:
        newlist.append(i)
    newlist.sort()
    print('Sum of requests from LAN:', len(newlist))
  time.sleep(10)

然后发现有问题,程序完全不会定时检测,只会卡在readline()上。

各种找问题,发现那个process.stdout.readline()是个同步方法,没结果就不返回。有没有的能异步的方法? 

有人用fnctl,windows不支持,pass

asyncio?看了半天没太明白...

折腾了半天,最后关头我还是用c#解决这个问题了....

参考代码见http://www.jiamaocode.com/Cts/1031.html,打不开的话http://www.cnblogs.com/sode/archive/2012/07/10/2583941.html有转载

总算解决了这个问题,但是我心中还是不爽,思考了很久如何解决异步readline()的问题。忽然想起来多线程这个利器,干脆开

一个线程,不返回就等着,不就问题解决了。

# this method is used for monitoring
import time
import subprocess
import locale
import codecs
import threading
alist = []
def getstdout(p, asy):
  if asy:
    alist.clear()
  mylist = []
  while True:
    data = p.stdout.readline()
    if data == b'':
      if p.poll() is not None:
        break
    else:
      if asy:
        alist.append(data.decode(codecs.lookup(locale.getpreferredencoding()).name))
      else:
        mylist.append(data.decode(codecs.lookup(locale.getpreferredencoding()).name))
  return mylist
while True:
  ps = subprocess.Popen('netstat -an | findstr ""', stdin=subprocess.PIPE, stdout=subprocess.PIPE, shell=True)
  resultlist = getstdout(ps, False)
  if len(resultlist) >= :
    newlist = []
    for i in alist:
      if i.find('.') > :
        newlist.append(i)
    newlist.sort()
    print('Sum of requests from LAN:', len(newlist))
  else:
    print(time.strftime("%Y-%m-%d %H:%M:%S"))
    subprocess.Popen('taskkill.exe /f /im node.exe', shell=False)
    time.sleep()
    pss = subprocess.Popen('start cmd.exe /k node app.js', stdin=subprocess.PIPE,
                stdout=subprocess.PIPE, shell=True)
    th = threading.Thread(target=getstdout, args=[pss, True])
    th.start()
  time.sleep()

总结

有时候简单的解决方法也可以实现同样的功能,对比python的实现与C#的实现,C#更面向事件一点,python应该也有不错的解决方案,继续摸索...

以上内容是小编给大家分享的Python实时获取cmd的输出的相关知识,希望大家喜欢。

Python 相关文章推荐
实例讲解Python编程中@property装饰器的用法
Jun 20 Python
Tornado协程在python2.7如何返回值(实现方法)
Jun 22 Python
python中关于for循环的碎碎念
Jun 30 Python
在python3环境下的Django中使用MySQL数据库的实例
Aug 29 Python
Python编程二分法实现冒泡算法+快速排序代码示例
Jan 15 Python
Python cookbook(数据结构与算法)保存最后N个元素的方法
Feb 13 Python
Python numpy 提取矩阵的某一行或某一列的实例
Apr 03 Python
Django2 连接MySQL及model测试实例分析
Dec 10 Python
Python3 xml.etree.ElementTree支持的XPath语法详解
Mar 06 Python
pandas DataFrame运算的实现
Jun 14 Python
python打开音乐文件的实例方法
Jul 21 Python
Python制作简单的剪刀石头布游戏
Dec 10 Python
一篇文章入门Python生态系统(Python新手入门指导)
Dec 11 #Python
深入源码解析Python中的对象与类型
Dec 11 #Python
Python实现各种排序算法的代码示例总结
Dec 11 #Python
Python操作MySQL数据库9个实用实例
Dec 11 #Python
使用Python编写简单的画图板程序的示例教程
Dec 08 #Python
一波神奇的Python语句、函数与方法的使用技巧总结
Dec 08 #Python
Python使用pygame模块编写俄罗斯方块游戏的代码实例
Dec 08 #Python
You might like
php 获取可变函数参数的函数
2009/08/26 PHP
PHP使用Alexa API获取网站的Alexa排名例子
2014/06/12 PHP
php通过strpos查找字符串出现位置的方法
2015/03/17 PHP
php 数组元素快速去重
2017/05/05 PHP
Ucren Virtual Desktop V2.0
2006/11/07 Javascript
Jquery replace 字符替换实现代码
2010/12/02 Javascript
用js来定义浏览器中一个左右浮动元素相对于页面主体宽度的位置的函数
2012/01/21 Javascript
jquery(hide方法)隐藏指定元素实例
2013/11/11 Javascript
jQuery异步获取json数据方法汇总
2014/12/22 Javascript
微信小程序之ES6与事项助手的功能实现
2016/11/30 Javascript
Angularjs 动态添加指令并绑定事件的方法
2017/04/13 Javascript
Node.js安装配置图文教程
2017/05/10 Javascript
详解Axios统一错误处理与后置
2018/09/26 Javascript
windows实现npm和cnpm安装步骤
2019/10/24 Javascript
vue实现权限控制路由(vue-router 动态添加路由)
2019/11/04 Javascript
Linux系统(CentOS)下python2.7.10安装
2018/09/26 Python
Python 读取xml数据,cv2裁剪图片实例
2020/03/10 Python
python自动下载图片的方法示例
2020/03/25 Python
Python编写万花尺图案实例
2021/01/03 Python
英国可持续奢侈品包包品牌:Elvis & Kresse
2018/08/05 全球购物
马德里著名的运动鞋商店:NOIRFONCE
2019/04/12 全球购物
Chemist Warehouse中文网:澳洲连锁大药房
2021/02/05 全球购物
新媒传信软件测试面试题
2013/02/24 面试题
艺术系大学生毕业个人自我评价
2013/09/19 职场文书
电厂职工自我鉴定
2014/02/20 职场文书
初三新学期计划书
2014/05/03 职场文书
教师求职自荐书
2014/06/14 职场文书
前台文员岗位职责
2015/02/04 职场文书
皇城相府导游词
2015/02/06 职场文书
实习班主任自我评价
2015/03/11 职场文书
上下班时间调整通知
2015/04/23 职场文书
高中升旗仪式主持词
2015/07/03 职场文书
2019大学毕业晚会主持词
2019/06/21 职场文书
2020年元旦晚会策划书模板
2019/12/30 职场文书
提升Nginx性能的一些建议
2021/03/31 Servers
oracle重置序列从0开始递增1
2022/02/28 Oracle