我在内置的cherrypy服务器上运行了一个web.py应用程序。 我想在用户发布到url时执行外部脚本,脚本将在后台运行python subprocess.Popen调用,web.py将重定向到另一个页面,其中使用jquery监视脚本的进度ajax请求。 但是我无法在这里正确实现这一点。 代码片段如下,如果需要,我可以提供更多信息。
import web
from multiprocessing import Process
import subprocess, shlex
import time, json
from login import authorize, sessidGen
def __callProcess(processString,mod='w',shell=False):
if not shell: args = shlex.split(processString)
else: args = processString
out = open('./bteq/logs/output.log',mod)
err = open('./bteq/logs/error.log',mod)
p = subprocess.Popen(args,stdout=out,stderr=err,shell=shell)
return p
def setExec():
__callProcess("chmod +x ./bteq/*.sh",shell=True)
def bteqExec(filename,system):
if system not in ['prod','da','cdw','cdw2','cert','']: return False
processString = " ".join([filename,system])
p = __callProcess(processString)
return p.pid
render = web.template.render('templates/',base='layout')
render_plain = web.template.render('templates/')
class Executor:
def GET(self):
authorize()
session = web.ctx.session
inputs = web.input(sessid={},type={})
if not inputs.sessid or session.id != inputs.sessid: web.seeother('/')
if inputs.sessid and inputs.type:
return render.executor('BTEQ Executor',inputs.type,inputs.sessid)
else: raise web.seeother('/')
def POST(self):
authorize()
session = web.ctx.session
inputs = web.input(sessid={},type={},act={})
if not inputs.sessid or session.id != inputs.sessid: web.seeother('/')
if inputs and inputs.act == 'start':
pid = bteqExec('python ./bteq/timer.py','')
session.id = sessidGen()
session.exctrpid = pid
return web.seeother('/progress.htm')
else: raise web.seeother('/')
class progress:
def GET(self):
authorize()
session = web.ctx.session
inputs = web.input(ajax={})
if inputs.ajax == 'true':
web.header('Content-Type', 'application/json')
if session.count >= 100: session.count = 0
session.count += 10
pid = session.exctrpid
out = open('./bteq/logs/output.log','r')
err = open('./bteq/logs/error.log','r')
output = ('<strong>OUTPUT:</strong><br>'+out.read()).replace('\n','<br>')
err = err.read()
if err:error = ('<strong>ERRORS:</strong><br>'+err.read()).replace('\n','<br>')
else: error = None
d = {'count':session.count,'msg':output,'err':error,'rc':pid,'session_id':session.session_id}
return json.dumps(d)
r = web.template.Template('$def with (title)\n$var title: $title\n')
return render_plain.layout_pgbar(r('progress test'))
由于subprocess.Popen对象不是pickleable,因此无法将其作为会话变量放入,我想从进度类中获取p.poll()和p.stdout.read()。
此外,我希望代码在linux和windows中都可运行,我在Windows中设置了我的开发设备,并将其部署在Linux服务器上。
有人可以帮助我吗...
谢谢。
答案 0 :(得分:0)
我将其设置为多处理 进程,并为子进程执行 p.wait()生成的Process和衍生过程负责其余的步骤并将结果更新到数据库中。
web.py进度页面将在数据库中检查执行进度,从而解决了这个问题。
代码:
from multiprocessing import Process
class ProcHandler(Process):
def __init__(self, *args, **kwargs):
#Initialize
Process.__init__(self, *args, **kwargs)
def run(self):
p = bteqExec('python ./bteq/timer.py','')
rc = p.wait()
# do update the resulting to database. and make the web.py
# Progress class read from database entry made by this Process.
return