将Flask应用部署为Windows服务
我在这里使用了一个模板:在Windows中可以将Python脚本作为服务运行吗?如果可以,怎么做?
这是我的run.py,我按照上面链接中的说明把它安装成了一个服务。
from app import app
import win32serviceutil
import win32service
import win32event
import servicemanager
import socket
class AppServerSvc (win32serviceutil.ServiceFramework):
_svc_name_ = "Flask App"
_svc_display_name_ = "Flask App"
def __init__(self,args):
win32serviceutil.ServiceFramework.__init__(self,args)
self.hWaitStop = win32event.CreateEvent(None,0,0,None)
socket.setdefaulttimeout(60)
def SvcStop(self):
self.ReportServiceStatus(win32service.SERVICE_STOP_PENDING)
win32event.SetEvent(self.hWaitStop)
def SvcDoRun(self):
servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE,
servicemanager.PYS_SERVICE_STARTED,
(self._svc_name_,''))
self.main()
def main(self):
app.run(host = '192.168.1.6')
if __name__ == '__main__':
win32serviceutil.HandleCommandLine(AppServerSvc)
但是,当我尝试启动这个服务时,出现了这样的提示:
“本地计算机上的Flask应用服务启动后又停止了。 一些服务如果没有被其他服务或程序使用,会自动停止。”
你们觉得我哪里出错了?我尝试了不同的用户账户——我觉得这不是权限问题。
谢谢!
4 个回答
-3
我在SvcStop()的最后一行加了我的代码。
"self.ReportServiceStatus(win32service.SERVICE_STOPPED)"
在我的情况下,这段代码可以成功停止服务。
from app import app
import win32serviceutil
import win32service
import win32event
import servicemanager
import socket
class AppServerSvc (win32serviceutil.ServiceFramework):
_svc_name_ = "Flask App"
_svc_display_name_ = "Flask App"
def __init__(self,args):
win32serviceutil.ServiceFramework.__init__(self,args)
self.hWaitStop = win32event.CreateEvent(None,0,0,None)
socket.setdefaulttimeout(60)
def SvcStop(self):
self.ReportServiceStatus(win32service.SERVICE_STOP_PENDING)
win32event.SetEvent(self.hWaitStop)
# !important! to report "SERVICE_STOPPED"
self.ReportServiceStatus(win32service.SERVICE_STOPPED)
def SvcDoRun(self):
servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE,
servicemanager.PYS_SERVICE_STARTED,
(self._svc_name_,''))
self.main()
def main(self):
app.run(host = '192.168.1.6')
if __name__ == '__main__':
win32serviceutil.HandleCommandLine(AppServerSvc)
1
Flask不太适合直接用来提供服务。你可以查看Flask的文档来了解更多。
Flask应用对象实际上就是一个WSGI应用。
所以,我们需要一个WSGI容器来提供服务。在代码层面上,用gevent
来实现Flask的Windows服务是很简单的。例如:
# app.py
from flask import Flask
app = Flask(__name__)
@app.route('/')
def hello():
return 'hello world'
# server.py
import win32serviceutil
from gevent.pywsgi import WSGIServer
from app import app
class Service(win32serviceutil.ServiceFramework):
_svc_name_ = "flask_gevent_service_test"
_svc_display_name_ = "flask gevent service test display name"
_svc_description_ = "flask gevent service test description"
def __init__(self, *args):
super().__init__(*args)
self.http_server = WSGIServer(('127.0.0.1', 9854), app)
self.SvcStop = self.http_server.stop
self.SvcDoRun = self.http_server.serve_forever
if __name__ == '__main__':
win32serviceutil.HandleCommandLine(Service)
2
我搞明白了——我在app.run()里把调试选项开着。把它去掉后,就没问题了!
虽然服务可以正常启动和运行(我可以从网络上的另一台电脑访问我的flask应用),但它无法停止。在我用的那个模板的讨论中,作者提到要设置一个标志来正确停止服务。
有没有人知道他这是什么意思,以及怎么写代码才能正确停止服务呢?
13
我在Flask中无法在request
之外访问WSGIRequestHandler
,所以我使用了Process
。
import win32serviceutil
import win32service
import win32event
import servicemanager
from multiprocessing import Process
from app import app
class Service(win32serviceutil.ServiceFramework):
_svc_name_ = "TestService"
_svc_display_name_ = "Test Service"
_svc_description_ = "Tests Python service framework by receiving and echoing messages over a named pipe"
def __init__(self, *args):
super().__init__(*args)
def SvcStop(self):
self.ReportServiceStatus(win32service.SERVICE_STOP_PENDING)
self.process.terminate()
self.ReportServiceStatus(win32service.SERVICE_STOPPED)
def SvcDoRun(self):
self.process = Process(target=self.main)
self.process.start()
self.process.run()
def main(self):
app.run()
if __name__ == '__main__':
win32serviceutil.HandleCommandLine(Service)