将Flask应用部署为Windows服务

13 投票
4 回答
21220 浏览
提问于 2025-04-18 05:53

我在这里使用了一个模板:在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)

撰写回答