gpt4 book ai didi

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

转载 作者:可可西里 更新时间:2023-11-01 13:49:59 28 4
gpt4 key购买 nike

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

问题是当应用程序崩溃时,Windows 会自动触发 WerFault这将使程序挂起,因此 python's subprocess.call()将永远等待用户输入(该应用程序必须在周末、节假日、24/7 全天候运行……所以这是 Not Acceptable )。

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

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

其他几个教程,例如 this也没有任何效果。

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

替代问题:在 python 中是否有优雅方法来检测应用程序崩溃(是否显示WerFault)

最佳答案

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

  1. 按名称获取与 WerFault.exe 匹配的进程列表。我用 psutil包裹。请小心使用 psutil,因为进程已缓存,请使用 psutil.get_pid_list()
  2. 使用argparse解码其命令行。这可能有点矫枉过正,但它利用了现有的 Python 库。
  3. 根据 PID 识别持有您的应用程序的进程。

这是一个简单的实现。

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 函数调用带有正确PIDtaskkill 命令。

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())

关于windows - 如何在子进程中启动崩溃(很少)的应用程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13948611/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com