Python 极简插件系统 pluggy 源码剖析

前言

本长文不适合手机端阅读,请酌情退出

公司架构组基于 pytest 自研了一套测试框架 sstest,目的是为了让业务组(也就是我在的组)更好的写单元测试,从而提高代码质量,单元测试的目的是为了回归校验,避免新提交的代码影响到项目中旧的功能。

我是组里第一个接入 sstest 的同学,踩了很多坑... 从而对 pytest 的源码产生了兴趣,在阅读 pytest 源码的过程中,发现 pluggy 插件系统其实是 pytest 的核心,可以说 pytest 只是将多个插件利用 pluggy 构建出来的项目,所以先分析 pluggy。

老规矩,在开始分析前,希望自己搞清楚的几个问题:

  • 1. 如何使用 pluggy?

  • 2. 插件代码如何做到灵活可插拔的?

  • 3. 外部系统如何调用插件逻辑?

随着分析的进行会有新的问题抛出,问题可以帮助我们理清目的,避免迷失在源码中。

整体把控

pluggy 插件系统与我此前研究的 python 插件系统不同,pluggy 不可以动态插入,即无法在程序运行的过程中利用插件添加新的功能。

pluggy 主要有 3 个概念:

  • 1.PluginManager:用于管理插件规范与插件本身

  • 2.HookspecMarker:定义插件调用规范,每个规范可以对应 1~N 个插件,每个插件都满足该规范,否则无法成功被外部调用

  • 3.HookimplMarker:定义插件,插件逻辑具体的实现在该类装饰的方法中

简单使用一下,代码如下。

import pluggy
# 创建插件规范类装饰器
hookspac = pluggy.HookspecMarker('example')
# 创建插件类装饰器
hookimpl = pluggy.HookimplMarker('example')

class MySpec(object):
    # 创建插件规范
    @hookspac
    def myhook(self, a, b):
        pass

class Plugin_1(object):
    # 定义插件
    @hookimpl
    def myhook(self, a, b):
        return a + b

class Plugin_2(object):
    @hookimpl
    def myhook(self, a, b):
        return a - b

# 创建manger和添加hook规范
pm = pluggy.PluginManager('example')
pm.add_hookspecs(MySpec)

# 注册插件
pm.register(Plugin_1())
pm.register(Plugin_2())

# 调用插件中的myhook方法
results = pm.hook.myhook(a=10, b=20)
print(results)

整段代码简单而言就是创建相应的类装饰器装饰类中的方法,通过这些类装饰器构建出了插件规范与插件本身。

首先,实例化 PluginManager 类,实例化时需要传入全局唯一的 project name,HookspecMarker 类与 HookimplMarker 类的实例化也需要使用相同的 project name。

创建完插件管理器后,通过 add_hookspecs 方法添加插件规范、通过 register 方法添加插件本身则可。

添加完插件调用规范与插件本身后,就可以通过插件管理器的 hook 属性直接调用插件了。

阅读到这里,关于问题「1,2,3」便有了答案。

pluggy 使用的过程可以分为 4 步:

  • 1. 通过 HookspecMarker 类装饰器定义插件调用规范

  • 2. 通过 HookimplMarker 类装饰器定义插件逻辑

  • 3. 创建 Plugin

  • 1
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

懒编程-二两

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值