我正在尝试捕获许多进程的stderr和stdout,并使用python logging模块将其输出写入日志文件。下面的代码似乎实现了这一点。现在我调查每个进程stdout并写入记录器,如果有任何数据。有没有更好的方法来做到这一点。
此外,我还希望拥有所有单个Processese活动的主日志,换句话说,我想自动(不轮询)将每个进程的所有stdout / stderr写入主记录器。这可能吗?
由于
class MyProcess:
def __init__(self, process_name , param):
self.param = param
self.logfile = logs_dir + "Display_" + str(param) + ".log"
self.args = [process_name, str(param)]
self.logger_name = process_name + str(param)
self.start()
self.logger = self.initLogger()
def start(self):
self.process = Popen(self.args, bufsize=1, stdout=PIPE, stderr=STDOUT) #line buffered
# make each processes stdout non-blocking
fd = self.process.stdout
fl = fcntl.fcntl(fd, fcntl.F_GETFL)
fcntl.fcntl(fd, fcntl.F_SETFL, fl | os.O_NONBLOCK)
def initLogger(self):
f = logging.Formatter("%(levelname)s -%(name)s - %(asctime)s - %(message)s")
fh = logging.handlers.RotatingFileHandler(self.logfile, maxBytes=max_log_file_size, backupCount = 10)
fh.setFormatter(f)
logger = logging.getLogger(self.logger_name)
logger.setLevel(logging.DEBUG)
logger.addHandler(fh) #file handler
return logger
def getOutput(self): #non blocking read of stdout
try:
return self.process.stdout.readline()
except:
pass
def writeLog(self):
line = self.getOutput()
if line:
self.logger.debug(line.strip())
#print line.strip()
process_name = 'my_prog'
num_processes = 10
processes=[]
for param in range(num_processes)
processes.append(MyProcess(process_name,param))
while(1):
for p in processes:
p.writeLog()
sleep(0.001)
答案 0 :(得分:1)
您的选择是
非阻塞I / O:这就是您所做的:)
select
module:您可以使用poll()
或select()
来分配不同输入的读数。
主题:为要监视的每个文件描述符创建一个线程并使用阻塞I / O.不建议使用大量文件描述符,但至少它适用于Windows。
有关详细信息,请观看此video on non-blocking I/O with Python
由于您的方法似乎有效,我只会坚持下去,如果施加的处理器负载不会打扰您。如果确实如此,我会在Unix上使用select.select()
。
至于关于主记录器的问题:因为您想要切换各个输出,所以不能将所有内容重定向到主记录器。您必须手动执行此操作。