python实现web应用框架之增加动态路由

2023-05-19 08:05:53 框架 增加 路由

今天我们将继续对该框架进行路由添加正则表达式。

本篇文章所依赖的python环境为:

路由添加正则表达式有什么用?

在介绍之前,我们首先要介绍一下Get获取资源的时候,有如下几种传参方式:

  • 使用key...value 的形式,将参数放到url问号?后面,如: Http://127.0.0.1/userInfo?name=pdudo
  • url中以路径参数的方式传递,例如: http://127.0.0.1/userInfo/pdudo/info
  • 在请求头中传递参数,请求头是以key...value的形式存储的,当然也可以自定义存放请求参数。

上述的第二种以路径参数的方式传参,就是我们这边文章所描述的点。

为什么需要这样做呢?

假设我们现在正在写一个功能,他需要返回传上来的用户信息,接口为: /userInfo/userid/infos,其中userid是动态的,所以接口可以是/userInfo/c12345/infos,也可以是/userInfo/d33456/infos等等。

如果是这种情况,我们应该怎么样样定义路由呢? 是在项目中给每个人都定义一个路由信息么?

很显然不是的,如果这个时候,注册的路由恰恰好是正则的,可以获取客户端传上来的参数,例如:/userInfo/pdudo/info,函数将会获取pdudo,这样不是很好么?

这就需要用到正则了。

如何定义动态路由

由于我们此前已经写好了定义静态路由的相关方法,所以我们在定义动态路由的时候,最好加一个识别参数,告诉框架,这个路由是静态的 还是 动态的。

由于是动态路由,所以在路由url中,有些值肯定是动态的,如何在区别于静态的值和动态的值呢? 在该框架中,我们将动态值用大括号括起来,比如: /userInfo/{userID}/infos,其中userID是动态的,可以是任何值组成,但是/userInfo/infos必须是静态的,而且先后顺序也是固定的。

所以说,我们可以约束一下,在定义动态路由的时候,需要告诉框架,我这个是动态路由,以及动态url

所以我们准备将动态路由定义规划如下:

@myWEB.routes(path="/jobs/{jobID}/startd",methods="post",regular=True)
def startJobs(r,cData):
    print("cData: " ,cData)

上述代码,是我们即将完善功能后的代码案例,其中regularTrue代表该路由是动态的,path的值为/jobs/{jobID}/startd,其中jobID也是动态的,可以被任何值代替。

最后是函数将接收2个参数,r的值代表wsgi中的environ信息。 而cData则代表从客户端上报的值,即:jobID的替代值。

注册动态路由

如上我们已经约束好了一个动态路由的添加规则,那么我们如何注册动态路由呢?

首先肯定的是,动态路由肯定不能和静态路由放在一起,所以我们定义了新的字典用于存放动态路由的信息,如:

mapGetRegularRouting = {}
mapPostRegularRouting = {}

而后则是我们需要将注册的路由信息转换为正则表达式,这里举个例子:

我们注册的路由是这样的: /jobs/{jobID}/startd,我们需要将其转换为正则表达式,正则表达式的值是这样的: ^/jobs/(.*?)/startd$

关于这段正则表达式,它的含义是 匹配以/jobs/开头和/startd结尾的字符串,并且将匹配到的信息(.*的值)存储到元组中。

注意这里.*后面的问号?代表不启用贪婪匹配,也就是最小化匹配。

现在的问题是,如何将{jobsID}给转换为(.*?)呢? 在Python中,可以使用re.sub进行替换操作,我们可以使用如下语句,将{jobsID}给替换为(.*?),代码如下:

import re
print(re.sub("{.*?}","(.*?)","/jobs/{jobID}/startd"))

如上代码执行的结果为:

最后将该路由信息存储到上述定义好的字典中即可,代码片段如下:

reFindall = re.compile(r"{(.*?)}")
reSubAll = re.compile(r"{.*?}")

parameter = re.findall(reFindall,self.path)
reText = re.sub(reSubAll,"(.*?)",self.path)
reText = f"{reText}$"
regular = {
    "original": self.path,
    "reText": reText,
    "parameter": parameter,
    "func": func
}
# ...
if self.re:
    mapPostRegularRouting[regular["reText"]] = regular

至此,我们的动态路由就已经注册好了。

匹配动态路由

相比于注册动态路由,当客户端请求来了之后,匹配动态路由会很麻烦,因为没有请求报文不是告诉你,这是动态路由还是静态路由,所以说,匹配动态路由很麻烦。

我们匹配动态路由想到的时候笨办法,即: 先匹配静态路由,若静态路由匹配不到,则再匹配动态路由,若动态路由都匹配不到的话,则选用默认路由。

匹配静态路由和动态路由,前面篇章已经介绍过了,所以这里不再赘述,这里就介绍如何匹配动态路由。

所谓的动态路由规则匹配,实际上就是拿着客户端上传的url,挨个对我们已有的正则表达式做轮训,若匹配成功则退出循环,若匹配失败,最后就返回一个默认路由即可,匹配正则表达式代码如下:

for key in RegularRouting.keys():
    if re.match(key, path):
        isRegular = True
        collText = re.findall(key, path)
        func = RegularRouting[key]["func"]
        break
else:
    func = Route["/*"]

框架运行效果展示

关于myWeb.py由于很长,有100多行,不好截图,也不好纯复制代码了,所以放到了gitee上面:gitee.com/pdudo/Golea…

我们简单写几个demo测试一下web框架,代码如下:

import myWeb
import wsgiref.simple_server

@myWeb.routes(path="/ip",methods="all")
def indx(r):
    print(r["REMOTE_ADDR"])
    return (200,r["REMOTE_ADDR"])

@myWeb.routes(path="/hello/{name}",methods="get",regular=True)
def helloWold(r,cData):
    name = cData[0]
    return (200,"hello %s" % (name))

def main():
    s = wsgiref.simple_server.make_server('', 8888, myWeb.application)
    s.serve_forever()

if __name__ == '__main__':
    main()

上述代码,我们首先引入了myWebwsgiref模块,前置是我们自己写的,后则是一个满足wsgi服务器框架,在代码中,我们定义了2个路由信息,一个是静态路由,一个是动态路由,其中静态路由的函数是indx,路由信息是/ip,匹配的客户端请求方法为get或者post,该方法主要返回ip地址。

动态路由的函数则是helloWold,它将匹配到以/hello/为首的路由信息,匹配客户端请求方法为get,还函数主要获取用户发送的动态值,并且以hello 等返回给客户端。

在主函数中,我们启动一个简单的wsgi服务器,入口为myWebapplication方法 。

代码运行效果如下:

我们分别使用getpost方法,请求本地/ip路由,可见都回复回来了,而后我们访问/hello/pdudo/hello/juejin,他们也分别回复了hello pdudohello juejin

总结

本篇文件介绍了将此前的web框架,路由修改为正则表达式形式,具体添加方式为: 在进行路由注册的时候,将其路由信息转换为正则表达式,而在进行路由匹配的时候,先匹配静态服务器,若静态服务器没有,再进行正则匹配,若正则匹配依然不行,则返回最后的默认页面。

以上就是python实现web应用框架之增加动态路由的详细内容,更多关于python增加动态路由的资料请关注其它相关文章!

相关文章