Flask之多应用

使用Flask框架进行多应用开发。

    这里所说的应用调度,平时也称为:多应用。

    官方文档:https://dormousehole.readthedocs.io/en/latest/patterns/appdispatch.html

    这是根据文档写的一个简单示例:

    #!/usr/bin/env python
    # -*- coding: utf-8 -*-
    from flask import Flask
    from werkzeug import run_simple
    from werkzeug.middleware.dispatcher import DispatcherMiddleware
    
    # 声明应用
    app1 = Flask(__name__)
    app2 = Flask(__name__)
    
    # 组合应用
    app = DispatcherMiddleware(app1, {
        '/a1': app1,
        '/a2': app2
    })
    
    
    @app1.route('/')
    def default():
        return 'hello it is default app1'
    
    
    @app1.route('/index')
    def index():
        return 'hello app1'
    
    
    @app2.route('/index')
    def index():
        return 'hello app2'
    
    
    if __name__ == "__main__":
        run_simple('localhost', 7777, app)

    运行代码,发现就是在访问路径里增加前缀即可访问不同应用的程序 。

    什么时候用呢?

    说到前缀的话,似乎用Blueprint(蓝图)也能实现?确实是,我个人的理解蓝图更多是用在程序的模块化,因为蓝图不是一个真正的应用。但若是你不需要每个应用独立配置的话,那选择蓝图确实是可以的。

    看上面的代码,主要的是DispatcherMiddleware,这个实现其实很好理解:

    class DispatcherMiddleware:
        def __init__(
            self,
            app: "WSGIApplication",
            mounts: t.Optional[t.Dict[str, "WSGIApplication"]] = None,
        ) -> None:
            self.app = app
            self.mounts = mounts or {}
    
        # 主要是这个,根据PATH_INFO分派应用程序
        def __call__(
            self, environ: "WSGIEnvironment", start_response: "StartResponse"
        ) -> t.Iterable[bytes]:
            script = environ.get("PATH_INFO", "")
            path_info = ""
    
            while "/" in script:
                if script in self.mounts:
                    app = self.mounts[script]
                    break
    
                script, last_item = script.rsplit("/", 1)
                path_info = f"/{last_item}{path_info}"
            else:
                app = self.mounts.get(script, self.app)
    
            original_script_name = environ.get("SCRIPT_NAME", "")
            environ["SCRIPT_NAME"] = original_script_name + script
            environ["PATH_INFO"] = path_info
            return app(environ, start_response)