python|web应用框架|增加动态路由

简介: python|web应用框架|增加动态路由

昨天我们已经已经修改了web路由注册,将其从函数方式注册,修改为类装饰器注册,如果你还没有看过此前那篇文章,建议你先看下,以便做到承上启下:

python|web应用框架|使用类装饰器注册路由:juejin.cn/post/722879…



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

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


image.png



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


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


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


为什么需要这样做呢?


假设我们现在正在写一个功能,他需要返回传上来的用户信息,接口为:

/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"))

如上代码执行的结果为:

image.png


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

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方法 。


代码运行效果如下:

image.png

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



总结


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



相关文章
|
9天前
|
前端开发 JavaScript API
探索现代Web开发中的动态数据交互——前端与后端整合实战
本文探讨了现代Web开发中前端与后端整合的关键技术点,通过实际案例演示了如何利用JavaScript和Node.js实现动态数据交互,全面解析从数据请求到响应的全过程。
|
2天前
|
分布式计算 并行计算 安全
在Python Web开发中,Python的全局解释器锁(Global Interpreter Lock,简称GIL)是一个核心概念,它直接影响了Python程序在多线程环境下的执行效率和性能表现
【6月更文挑战第30天】Python的GIL是CPython中的全局锁,限制了多线程并行执行,尤其是在多核CPU上。GIL确保同一时间仅有一个线程执行Python字节码,导致CPU密集型任务时多线程无法充分利用多核,反而可能因上下文切换降低性能。然而,I/O密集型任务仍能受益于线程交替执行。为利用多核,开发者常选择多进程、异步IO或使用不受GIL限制的Python实现。在Web开发中,理解GIL对于优化并发性能至关重要。
17 0
|
4天前
|
前端开发 JavaScript Java
使用Spring Boot和Thymeleaf构建动态Web页面
使用Spring Boot和Thymeleaf构建动态Web页面
|
4天前
|
测试技术
Appium+python自动化(三十九)-Appium自动化测试框架综合实践 - 代码实现(超详解)
Appium+python自动化(三十九)-Appium自动化测试框架综合实践 - 代码实现(超详解)
|
4天前
|
测试技术 Python
python接口自动化测试 - unittest框架suite、runner详细使用
python接口自动化测试 - unittest框架suite、runner详细使用
|
6天前
|
前端开发 JavaScript Python
Python之Tornado web 框架详解
Python之Tornado web 框架详解
|
29天前
|
存储 数据管理 测试技术
构建Python构建自动化测试框架(原理与实践)
当谈到软件质量保证时,自动化测试是一个不可或缺的步骤。Python作为一种简单易学的编程语言,具有丰富的测试框架和库,使得构建自动化测试框架变得相对简单。本文将介绍如何使用Python构建自动化测试框架,包括选择合适的测试框架、编写测试用例、执行测试和生成报告等方面。
构建Python构建自动化测试框架(原理与实践)
|
22天前
|
IDE 测试技术 持续交付
Python作为一种简洁、易读且功能强大的编程语言,其自动化测试和单元测试框架的丰富性和易用性为开发者提供了极大的便利
【6月更文挑战第10天】本文探讨了Python自动化测试与单元测试框架在提升代码质量和效率中的作用。Selenium、Appium和pytest是常用的自动化测试框架,分别支持Web和移动应用的测试。unittest是Python的标准单元测试框架,提供断言方法和测试组织结构。通过制定测试计划、编写高质量测试用例、持续集成与测试、以及有效利用测试报告,开发者能提高代码质量和开发效率。
30 1
|
18天前
|
机器人 测试技术 持续交付
Python进行自动化测试测试框架的选择与应用
【6月更文挑战第9天】本文介绍了Python自动化测试的重要性及选择测试框架的考量因素,如功能丰富性、易用性、灵活性和集成性。文中列举了常用的Python测试框架,包括unittest、pytest、nose2和Robot Framework,并提供了使用pytest进行单元测试的示例代码。此外,还展示了如何使用Robot Framework进行验收测试和Web UI测试。选择合适的测试框架对提升测试效率和软件质量至关重要,团队应根据项目需求、社区支持、集成性和学习曲线等因素进行选择。通过不断学习和实践,可以优化自动化测试流程,确保软件的稳定性和可靠性。
23 0
|
2月前
|
监控 数据可视化 IDE
python自动化测试实战 —— 单元测试框架
python自动化测试实战 —— 单元测试框架
38 2