Python 插件式程序设计与开发实践总结
插件式程序设计与开发实践总结
By:授客 QQ:1033553122
开发环境
win 10
python 3.6.5
代码结构
需求描述
如上,以user.py为程序入口脚本,运行该脚本时,需要创建一个user类对象,执行一系列动作(包含一系列动作的列表)。程序执行动作前,要求先获取动作名称,根据该名称,执行不同的操作。这些操作,对应不同的类函数。
实现思路
大致实现思路就是,把user对象需要运行的类函数(使用@classmethod修饰的函数,可不用创建对象进行调用),当作插件函数,并设置为user的属性,这样程序运行时,可通过该属性来调用对应的类函数。这里的问题是,程序怎么知道执行哪个类函数呢?到目前为止,程序只能根据动作名称来判断待执行的操作,所以,需要建立动作名称和类函数的映射关系。
怎么建立动作名称和类函数的映射关系呢?这里用到了装饰器,新建一个装饰器类ActionDecorator,为该类设置一个字典类型的类属性ACTION_FUNC_CLASS_MODULE_MAP,用这个类来存放动作名称和类函数的映射关系。我们把需要当作插件函数的类函数都用该装饰器进行修饰。
这里,笔者发现一个特性,就是对应模块被导入时,对应模块,对应类函数如果使用了装饰器,该装饰器函数会被执行。同时,笔者还发现另外一个特性,
首次对某个包执行import操作时,该包下面的__init__.py文件将优先被执行。基于这两个特性,我们把装饰器放在用于管理插件类函数的外围软件包下(例中的components包),同时,在该外围软件包下的__init__.py中加入动态加载插件模块的代码:遍历外围软件包下的所有非__init__.py文件,并且动态加载改模块。这样,当在user.py入口文件中,执行from components.decoraters.action_decorater import ActionDecorator时,会自动执行components/__init__.py文件,动态加载所有插件模块,并且自动触发装饰器的执行,装饰器方法执行,会自动根据提供的方法参数建立动作名称和类函数的映射关系。
然后,在初始化user对象时,给该对象动态设置属性,属性名称设置为动作名称,属性值设置为类方法,这样,执行动作时,就可以根据动作名称调用对应的类方法了。
代码实现
action_decorate.py
#!/usr/bin/env python
# -*- coding:utf-8 -*- '''
@CreateTime: 2020/12/09 14:58
@Author : shouke
''' class ActionDecorator(object):
'''
action 装饰器
''' ACTION_FUNC_CLASS_MODULE_MAP = {} @classmethod
def action_register(cls, action, class_name, function_name, module_path):
def wrapper(func):
cls.ACTION_FUNC_CLASS_MODULE_MAP.update({action: {'class_name':class_name, 'function_name':function_name, 'module_path':module_path}})
return func
return wrapper
components/__init__.py
#!/usr/bin/env python
# -*- coding:utf-8 -*- '''
@Author : shouke
''' import os.path
import importlib def load_plugin_modules():
'''递归加载当前目录下的所有模块''' head, tail = os.path.split(__file__)
package_father_path, package = os.path.split(head) def load_modules(dir_path):
nonlocal package_father_path
if not os.path.isdir(dir_path):
return
for name in os.listdir(dir_path):
full_path = os.path.join(dir_path, name)
if os.path.isdir(full_path):
load_modules(full_path)
elif not name.startswith('_') and name.endswith('.py'):
temp_path = full_path.replace(package_father_path, '')
relative_path = temp_path.replace('\\', '/').lstrip('/').replace('/', '.')
importlib.import_module(relative_path.rstrip('.py'), package=package)
load_modules(head) # 加载模块,自动触发装饰器,获取相关插件函数相关信息 load_plugin_modules()
assertioner.py
#!/usr/bin/env python
# -*- coding:utf-8 -*- '''
@Author : shouke
''' from components.decoraters.action_decorater import ActionDecorator class Assertioner(object):
@classmethod
@ActionDecorator.action_register('assert_equal', 'Assertioner', 'assert_equal', __name__)
def assert_equal(self, config:dict, *args, **kwargs):
print('执行断言')
print('断言配置:\n', config)
send_request.py
#!/usr/bin/env python
# -*- coding:utf-8 -*- '''
@Author : shouke
''' from components.decoraters.action_decorater import ActionDecorator class Requester(object):
@ActionDecorator.action_register('send_request', 'Requester', 'send_request', __name__)
@classmethod
def send_request(self, config:dict, *args, **kwargs):
print('发送请求')
print('请求配置:')
print(config)
example.py
#!/usr/bin/env python
# -*- coding:utf-8 -*- '''
@CreateTime: 2020/12/10 15:51
@Author : shouke
''' from components.decoraters.action_decorater import ActionDecorator class CustomClassName(object):
@ActionDecorator.action_register('custom_action_name', 'CustomClassName', 'action_func_name', __name__)
@classmethod
def action_func_name(self, config:dict, *args, **kwargs):
'''
example
user_instance: kwargs['user'] # 压测用户实例
''' # do something you want # 说明 plugings目录下可自由创建python包,管理插件,当然,也可以位于components包下其它任意位置创建python包,管理插件(不推荐)
user.py
#!/usr/bin/env python
# -*- coding:utf-8 -*- '''
@Author : shouke
''' from components.decoraters.action_decorater import ActionDecorator class User(object): def __init__(self):
for action, action_map in ActionDecorator.ACTION_FUNC_CLASS_MODULE_MAP.items():
module = __import__(action_map.get('module_path'), fromlist=['True'])
class_cls = getattr(module, action_map.get('class_name'))
setattr(self, action, getattr(class_cls, action_map.get('function_name'))) def run_actions(self, actions):
''' 执行一系列动作 ''' for step in actions:
action = step.get('action') if hasattr(self, action):
getattr(self, action)(step, user=self) if __name__ == '__main__':
actions = [{
"action": "send_request",
"name": "请求登录", #可选配置,默认为None
"method": "POST",
"path": "/api/v1/login",
"body": {
"account": "shouke",
"password": "123456"
},
"headers": {
"Content-Type": "application/json"
}
},
{
"action": "assert_equal",
"name": "请求响应断言",
"target": "body",
"rule": "assert_contain",
"patterns": ["shouke","token"],
"logic":"or"
}]
User().run_actions(actions)
运行结果
发送请求
请求配置:
{'action': 'send_request', 'name': '请求登录', 'method': 'POST', 'path': '/api/v1/login', 'body': {'account': 'shouke', 'password': '123456'}, 'headers': {'Content-Type': 'application/json'}}
执行断言
断言配置:
{'action': 'assert_equal', 'name': '请求响应断言', 'target': 'body', 'rule': 'assert_contain', 'patterns': ['shouke', 'token'], 'logic': 'or'}
Python 插件式程序设计与开发实践总结的更多相关文章
- 响应式WEB页面开发实践
转自:https://github.com/markyun/My-blog/issues/27 最近得到一个新任务单,让我用一套页面适应所有主流终端(Android.iPhone.iPad.PC),而 ...
- Chrome插件(Extensions)开发实践
内容摘自:http://www.cnblogs.com/mfryf/p/3701801.html
- WebApi 插件式构建方案:发现并加载程序集
插件式的 WebApi 开发,首要面对的问题就是程序集的发现.因为开发的过程中,都是在各自的解决方案下进行开发,部署后是分模块放在一个整体的的运行时网站下. 约定 这里我根据上一节的设定,把插件打包完 ...
- 《Flask Web开发——基于Python的Web应用开发实践》一字一句上机实践(上)
目录 前言 第1章 安装 第2章 程序的基本结构 第3章 模板 第4章 Web表单 第5章 数据库 第6章 电子邮件 第7章 大型程序的结构 前言 学习Python也有一个半月时间了,学到现在感觉 ...
- 基于Flask的Web应用程序插件式结构开发
事实上,很多应用程序基于插件式结构开发,可以很方便了扩展软件的功能,并且这些功能完全可以依托于第三方开发者,只要提供好接口和完备文档,比如wordpress.谷歌火狐浏览器等. Python这样的动态 ...
- 也来学学插件式开发续-利用MEF
前面一个博客:也来学学插件式开发中很多朋友留言说可以用MEF来实现.于是我就试着用MEF实现了一下. 步骤和上一篇差不多,只是加载插件的方式有所不同.这只是一个自己的示例程序,肯定有很多不足之处,欢迎 ...
- MEF 插件式开发 - 小试牛刀
原文:MEF 插件式开发 - 小试牛刀 目录 MEF 简介 实践出真知 面向接口编程 控制反转(IOC) 构建入门级 MEF 相关参考 MEF 简介 Managed Extensibility Fra ...
- 从零开始实现ASP.NET Core MVC的插件式开发(七) - 近期问题汇总及部分解决方案
标题:从零开始实现ASP.NET Core MVC的插件式开发(七) - 问题汇总及部分解决方案 作者:Lamond Lu 地址:https://www.cnblogs.com/lwqlun/p/12 ...
- AppDomain实现【插件式】开发
前言: 近期项目中需要实现"热插拔"式的插件程序,例如:定义一个插件接口:由不同开发人员实现具体的插件功能类库:并最终在应用中调用具体插件功能. 此时需要考虑:插件执行的安全性(隔 ...
- Redis的Python实践,以及四中常用应用场景详解——学习董伟明老师的《Python Web开发实践》
首先,简单介绍:Redis是一个基于内存的键值对存储系统,常用作数据库.缓存和消息代理. 支持:字符串,字典,列表,集合,有序集合,位图(bitmaps),地理位置,HyperLogLog等多种数据结 ...
随机推荐
- 为什么我们要用Spring Boot
最近我面试了不少人,其中不乏说对 Spring Boot 非常熟悉的,然后当我问到一些 Spring Boot 核心功能和原理的时候,没人能说得上来,或者说不到点上,可以说一个问题就问趴下了! 这是我 ...
- QShop商城-快速开始-uni-app小程序
QShop商城-快速开始-uni-app小程序 工具准备 HBuilderX 此项目为UniApp开发,开发工具为HBuilderX,下载地址: https://hx.dcloud.net.cn/Tu ...
- SpringBoot自定义注解失效原因(2022-10-3)
长话短说,我负责的是一个多模块项目,接手的时候没有注意 @ComponentScan 注解的扫描范围,所以打包的时候,没有扫到我新加包. 所以,重点检查下 @ComponentScan 注解的范围
- Easysearch 压缩功能的显著提升:从 8.7GB 到 1.4GB
引言 在海量数据的存储和处理中,索引膨胀率是一个不可忽视的关键指标.它直接影响了存储成本和查询性能.近期,Easysearch 在这方面取得了显著的进展,其压缩功能的效果远超过了之前的版本.本文将详细 ...
- Javascript高级程序设计第七章 | ch7 | 阅读笔记
迭代器与生成器 在软件开发领域,"迭代"的意思是按照顺序反复多次执行一段程序 理解迭代 在JavaScript中,计数循环就是最简单的迭代 但是这种迭代有点问题: 1. 迭代之前需 ...
- linux解压缩,复制,重命名,删除,目录按更新时间排序,grep递归搜索文档
linux解压缩,复制,重命名,删除,目录按更新时间排序,grep递归搜索文档 1.解压缩压缩命令 zip -p -r mymail-1026.zip mymail/ 解压命令 unzip mymai ...
- debian使用桌面管理器管理多个桌面系统
环境:debian12.x 前言:我安装了debian12版本的操作系统在虚拟机中,在安装的时候选择的是KDE桌面,便于以后日常使用linux操作系统 在安装KDE桌面后,会自动安装一个sddm,sd ...
- python webdriver.remote远程创建火狐浏览器会话报错,Unable to create new service: GeckoDriverService
问题: 使用selenium.webdriver.remote,远程指定地址的浏览器,并创建会话对象:创建火狐浏览器会话时,报错,错误信息如下: Message: Unable to create n ...
- springboot支持http2
现在http/3都出来了,但是很多项目还是没有采用https,这个是说不过去的. http3在2022/06/06 正式发布,具体见https://www.163.com/dy/article/H9B ...
- IS-IS总结
IS-IS 管理距离115 ISIS是链路状态协议 封装在数据链路层,所以没有协议号 使用SPF算法计算最短路径 没有骨干区的概念 使用IIH(ISIS ...