使用 Python 实现 Web 服务器

使用 Python 实现 Web 服务器

使用 Python 来实现一个 Web 服务器,可能大家都感觉到奇怪,为什么会有这样的需求呢?原因很简单,当我们需要在 Windows 或者其他操作系统上实现一个应用,而这个应用已经使用 Web 技术开发好的情况下,此时我们就需要实现一个能够嵌入到应用程序中的 Web 服务器。本文介绍的使用 Python 实现的 Web 服务器可以实现这个功能,而 Python 本身提供了 Windows 环境下的嵌入式版本。也就是说,我们通过 Python embed 版本结合自己编写一些代码即可实现一个嵌入式的 Web 服务器,该服务器可以随着你的应用程序的启动而启动,随着你的应用程序的关闭而终止。

由于前后端分离的开发方式,我们现在开发的 Web 应用程序大部分都是页面部分独立部署,API 部分独立部署,这样的话,我们的 Web 服务器就只需要实现静态页面的访问和 API 的请求,正如 Nginx 一样,我们在这里只需要实现能够访问静态页面,同时实现类似于 Nginx 的反向代理功能即可。

1.静态页面的访问

在 Python 中,可以实现 Web 服务器的框架有很多,包括:

  • Django
  • Flask
  • Tornado

是三大著名的 Python Web 框架,这三大框架都可以实现我们的需求,今天我们以 Flask 为例来说明。

对于静态页面的访问,在 Flask 中,只需要在 Flask 类实例中指定静态文件夹,并指定其url ,如:

app = Flask(__name__, static_folder='static', static_url_path="/xxxxxx")

上面的代码需要我们在项目所在目录下创建一个 static 目录,并把所有的静态页面或者文件存放到该目录,在浏览器中访问时的 url 为 /xxxxxx。

一般情况下,我们会将开发中编译生成的 html 文件放置到 static 目录下,并且在访问时直接在根路径访问,所以上面的 static_url_path 设置为 "/",这种情况下,我们需要对根目录的访问重定向一下,如:

@app.route("/")
def index():
    return redirect('/index.html')

2.实现类似于 Nginx 的反向代理

实现该功能需要在 Flask 中拦截所有请求,然后判断请求的 url 中的 path,如果是需要代理到其他服务器的请求,则请求其他服务,一般是 api,然后将请求结果返回即可。如:

proxy_list = {
    '/xxx_api': 'http://127.0.0.1:18888',
    '/yyy_api': 'http://127.0.0.1:18888',
    '/zzz_api': 'http://127.0.0.1:18888'
}

@app.before_request
def proxy():
    path = request.path
    method = request.method
    headers = request.headers
    data = request.get_data()
    params = request.args

    for key in proxy_list.keys():
        if path.startswith(key):
            url = proxy_list.get(key) + path
            resp = requests.request(\
                method=method, url=url, params=params, data=data, headers=headers)
            return Response(response=resp.content, status=resp.status_code)

3.完整的代码

# coding: utf-8
from flask import Flask
from flask import Response
from flask import redirect
from flask import request
import requests


app = Flask(__name__, static_folder='static_backstage', static_url_path="/")


@app.route("/")
def index():
    return redirect('/index.html')


proxy_list = {
    '/oth_api/': 'http://127.0.0.1:18888',
    '/qycommon_api': 'http://127.0.0.1:18888',
    '/qybackstage_api': 'http://127.0.0.1:18888'
}


@app.before_request
def proxy():
    path = request.path
    method = request.method
    headers = request.headers
    data = request.get_data()
    params = request.args

    for key in proxy_list.keys():
        if path.startswith(key):
            url = proxy_list.get(key) + path
            resp = requests.request(method=method, url=url, params=params, data=data, headers=headers)
            return Response(response=resp.content, status=resp.status_code)


def main(port: int):
    app.run(host='127.0.0.1', port=port)


if __name__ == "__main__":
    main(7000)



发表评论
留言与评论(共有 0 条评论) “”
   
验证码:

相关文章

推荐文章