如何在子进程中启动崩溃(很少)应用程序

时间:2012-12-19 08:54:24

标签: windows python-3.x crash subprocess

我正在使用python应用程序,它需要每天执行大约20 000次专有应用程序(不时崩溃)。

问题是当应用程序崩溃时,Windows会自动触发WerFault,这将使程序保持挂起状态,因此python's subprocess.call()将永远等待用户输入(该应用程序必须在周末,节假日,24 / 7 ......所以这是不可接受的。)

如果关于使用sleep; poll; kill; terminate但是这意味着失去使用communicate()的能力,应用程序可以从几毫秒运行到2小时,因此设置固定超时将无效

我也尝试了turning on automatic debugging(使用一个脚本来获取应用程序的崩溃转储并终止id),但不知怎的,这个 howto 在我的服务器上不起作用(WerFault仍然出现并等待用户输入。)

其他几个教程如this也没有起任何作用。

问题: 有没有办法防止WerFault显示(等待用户输入)? 这是更多系统然后编程问题

备选问题:在python中是否有 graceful 方式如何检测应用程序崩溃(是否显示 WerFault

2 个答案:

答案 0 :(得分:2)

简单(和丑陋)回答,不时监视WerFault.exe个实例,特别是与违规应用程序的PID相关联的实例。杀了它。处理WerFault.exe很复杂,但您不想禁用它 - 请参阅Windows Error Reporting服务。

  1. 按名称获取与WerFault.exe匹配的流程列表。我使用psutil包。请注意psutil,因为进程已缓存,请使用psutil.get_pid_list()
  2. 使用argparse解码其命令行。这可能是过度的,但它利用了现有的python库。
  3. 根据PID确定应用程序的进程。
  4. 这是一个简单的实现。

    def kill_proc_kidnapper(self, child_pid, kidnapper_name='WerFault.exe'):
        """
        Look among all instances of 'WerFault.exe' process for an specific one
        that took control of another faulting process.
        When 'WerFault.exe' is launched it is specified the PID using -p argument:
    
        'C:\\Windows\\SysWOW64\\WerFault.exe -u -p 5012 -s 68'
                                 |               |
                                 +-> kidnapper   +-> child_pid
    
        Function uses `argparse` to properly decode process command line and get
        PID. If PID matches `child_pid` then we have found the correct parent
        process and can kill it.
        """
        parser = argparse.ArgumentParser()
        parser.add_argument('-u', action='store_false', help='User name')
        parser.add_argument('-p', type=int, help='Process ID')
        parser.add_argument('-s', help='??')
    
        kidnapper_p = None
        child_p = None
    
        for proc in psutil.get_pid_list():
            if kidnapper_name in proc.name:
                args, unknown_args = parser.parse_known_args(proc.cmdline)
                print proc.name, proc.cmdline
    
                if args.p == child_pid:
                    # We found the kidnapper, aim.
                    print 'kidnapper found: {0}'.format(proc.pid)
                    kidnapper_p = proc
    
        if psutil.pid_exists(child_pid):
            child_p = psutil.Process(child_pid)
    
        if kidnapper_p and child_pid:
            print 'Killing "{0}" ({1}) that kidnapped "{2}" ({3})'.format(
                kidnapper_p.name, kidnapper_p.pid, child_p.name, child_p.pid)
            self.taskkill(kidnapper_p.pid)
            return 1
        else:
            if not kidnapper_p:
                print 'Kidnapper process "{0}" not found'.format(kidnapper_name)
            if not child_p:
                print 'Child process "({0})" not found'.format(child_pid)
    
        return 0
    

    现在,taskkill函数使用正确的taskkill调用PID命令。

    def taskkill(self, pid):
        """
        Kill task and entire process tree for this process
        """
        print('Task kill for PID {0}'.format(pid))
        cmd = 'taskkill /f /t /pid {0}'.format(pid)
        subprocess.call(cmd.split())
    

答案 1 :(得分:-2)

我认为没有理由为什么你的程序需要崩溃,找到有问题的代码片段,并将其放入try-statement中。

http://docs.python.org/3.2/tutorial/errors.html#handling-exceptions