Loading [MathJax]/jax/input/TeX/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >从 0 到 1,如何徒手撸一个 Python 插件系统?

从 0 到 1,如何徒手撸一个 Python 插件系统?

作者头像
程序员荒生
发布于 2022-04-02 09:06:10
发布于 2022-04-02 09:06:10
74100
代码可运行
举报
运行总次数:0
代码可运行

号主从事算法服务开发多年,临近三月的尾巴,输出一个插件化部署算法服务的解决方案。篇幅内容都经过生产实践

插件机制背景

插件化机制使框架业务模块的实现相解耦,框架服务抽象出统一的交互接口,业务模块只要符合交互标准即可做到插件替换。

微内核架构是插件架构模式的一种典型实现,常常把微内核架构也叫做插件式架构。 当前微内核架构也被应用在许多我们熟知的产品,比如:操作系统、Chrome浏览器、Eclipse编辑器等

微内核架构包含两个组件:核心系统(core system)和插件模块(plugin component)

  • 核心系统只包含让系统可以运作的最小功能
  • 插件模块则包含一些特殊处理逻辑、额外的功能,用于提供更多的业务能力

具体如下图:

Python 插件核心

__import__()函数是我们本篇的核心,具体实现都围绕这个函数定制展开。

作用:用于动态加载类和函数。如果一个模块经常变化就可以使用 import() 来动态载入。 语法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
__import__(name, globals=None, locals=None, fromlist=(), level=0)
 name[必须]:模块名称
 globals - 全局变量集合,默认为None
 locals - 局部变量集合,默认为None
 fromlist - 是否导入子模块,看上去是导入模块的列表。但实际是一个判断条件,只要设置为非空的值,且模块名称是带有子模块的,将导入子模块。例如:sys.path。当不设置时,返回sys,如果设置为非空值,则返回ntpath(path模块)
 level - 绝对或者相对导入

此处我们做个简单的验证,通过__import__实现和import一样的导入能力。分别验证导入os.path和导入其对应的子模块,具体如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
>>> __import__('os.path')
<module 'os' from '/opt/anaconda3/envs/python38/lib/python3.8/os.py'>
>>> __import__('os.path', fromlist=['None'])
<module 'posixpath' from '/opt/anaconda3/envs/python38/lib/python3.8/posixpath.py'>
>>> __import__('os.path', fromlist=['path'])
<module 'posixpath' from '/opt/anaconda3/envs/python38/lib/python3.8/posixpath.py'>
>>>

以上只是在 shell 里面进行验证,那 Python 代码里面该如何写呢?有两种方式: 直接引用加载

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
p = __import__('widget.demo.infer', fromlist='infer')  # 导入
print(p.Demo)      # 取出类变量
# 打印结果
<class 'widget.demo.infer.Demo'>

Importlib包方式

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import importlib    # 导入工具包
demo = importlib.import_module('.demo.infer', package='widget')
print(demo)
# 打印结果
<module 'widget.demo.infer' from '/Users/**/language-bootcamp/python/plugin_system/widget/demo/infer.py'>
print(demo.Demo)
# 打印结果
<class 'widget.demo.infer.Demo'>

插件服务基本组成

PluginCore:通过Plugin Manager调用算法,负责业务逻辑的实现 PluginManger: 通过读取配置文件,负责各种插件的加载、管理、甚至热更新

此处,Python 插件相关的知识点已经讲完了,下面进行算法插件化部署相关内容,Go Go Go!

算法服务基本组成

如上,一个完整的算法服务包含三部分:API逻辑算法逻辑模型文件

API逻辑:服务相关逻辑,比如:HTTP 相关的请求/响应设置或 gRPC 远程交互约定等。 算法逻辑:算法相关逻辑,比如:数据的前置预处理、后置标签映射等。 模型文件:算法运行依赖项。

可能某些号友会发问:一定是三个部分,难道不能两个,甚至一个? 此处只是从纯业务功能划分,便于理解。具体代码实现,都可以。

插件管理器定义

定义了插件管理的方法,插件功能加载、插件获取方法实现

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class BaseManager(object):
    """插件管理基类"""
    
    def __init__(self) -> None:
        """加载插件配置文件"""
        pass

    def get_plugin(self, name):
        """根据调用,返回插件实例"""
        raise RuntimeError('not implemented yet!')

    def get_total_plugin(self):
        """返回所有插件"""
        raise RuntimeError('not implemented yet!')

算法插件定义

每个插件需要实现公共接口方法。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class ApiBase(object):
    """API服务基类"""
    
    def __init__(self) -> None:
        "模型初始化加载"
    
    def predict(self, req_dict={}):
        """
        预测方法,推理算法入口
        :param req_dict:        请求参数,词典类型,eg:{"name":"value"}
        :return:                返回结果,词典类型,eg:{"name":"value"}
        """
        raise RuntimeError('not implemented \'predict\' method')

    def version(self):
        "版本方法"
        return '20220330'

配置文件定义

PluginManager通过配置文件进行可选插件指定加载配置,包括:插件名 、插件路径等信息。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
{
    "demo":{
        "infer_dir":"widget.demo.infer"
    },
    "digit_recognition":{
        "infer_dir":"widget.digit_recognition.infer"
    }
}

插件扩展

有些时候为了解耦,需要调用 C 代码,以 linux 平台为例,我们讲讲 python 如何调用 .so 文件进行扩展。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# -*- coding: utf-8 -*-
import json
import ctypes
from ctypes import c_float
from common.logger import LOG

class C_Type(object):
    def __init__(self) -> None:
        self.so = ctypes.CDLL('widget/ctype/lib/sum.so')
    
    def predict(self, req_dict={}):
        LOG.info("this is ctype infer, req_dict=%s" % json.dumps(req_dict))
        return req_dict

    def version(self):
        return '20220326'

def get_plugin_class():
    return C_Type

总结一下

高可扩展:通过插件机制,我们可以很方便进行服务扩展 部署灵活:也能通过配置发布单独服务或者多个服务 代码解耦:最主要能实现各个功能模块的开发解耦,方便进行业务定制

但插件的使用得注意,尽量选择依赖环境和处理性能基本一致的业务进行插件构建,否则会有性能问题,慢插件影响快插件的推理速度。

以上就是今天的全部内容,以上所有代码都汇总到github.com/codetodo-io/language-bootcamp,号友们可以下载演示。

❤️❤️❤️读者每一份热爱都是笔者前进的动力!我是三十一,感谢各位朋友

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2022-03-31,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 李三十一 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
一文看懂Python沙箱逃逸
让用户提交 Python 代码并在服务器上执行,是一些 OJ、量化网站重要的服务,很多 CTF 也有类似的题。为了不让恶意用户执行任意的 Python 代码,就需要确保 Python 运行在沙箱中。沙箱经常会禁用一些敏感的函数,例如 os,研究怎么逃逸、防护这类沙箱还是蛮有意思的。
FB客服
2019/05/23
3.2K0
python之模块和包
一 Python模块简介 1 模块化 一般来说,编程语言中,库,包,模块是同一种概念,是代码组织方式 python中只有一种模块对象类型,但是为了模块化组织的便利,提供了一个概念: 包 模块(module):指的是python的源代码文件 包(package):指的是模块组织在一起放入和包名同名的目录及相关文件 ---- 可以将代码量较大的程序分割成多个有组织,彼此间独立但又能互相交互的代码片段,这些自我包含的有组织的代码段就是模块 ---- 模块在物理形式上表现为以.py 结尾的代码文
py3study
2020/01/09
1.5K0
python之模块和包
Python 插件式程序设计与开发实践总结
如上,以user.py为程序入口脚本,运行该脚本时,需要创建一个user类对象,执行一系列动作(包含一系列动作的列表)。程序执行动作前,要求先获取动作名称,根据该名称,执行不同的操作。这些操作,对应不同的类函数。
授客
2021/01/05
7820
Python 插件式程序设计与开发实践总结
Python基础:反射
主要有四个成员。getattr、hasattr、setattr、delattr  获取成员、检查成员、设置成员、删除成员
py3study
2020/01/14
3140
CTF - Python 沙箱绕过与任意命令执行技巧
首先,您需要知道是否可以直接使用已导入的某些库执行代码,或者是否可以导入以下这些库:
井九
2024/10/12
1K0
Python中实现简单的插件框架
在系统设计中,经常我们希望设计一套插件机制,在不修改程序主体情况下,动态去加载附能。
py3study
2020/01/17
2.4K0
Python pickle 反序列化实例分析
之前 SUCTF 出了一题 pickle 反序列化的杂项题,就感觉相当有意思。后来 Balsn 一次性搞了三个,太强了,学到了很多,感谢这些师傅。下文记录了我的学习笔记以及踩过的坑,希望对大家理解 pickle 有点帮助。
wywwzjj
2023/05/09
8760
Python pickle 反序列化实例分析
__import__详解
当使用import导入Python模块的时候,默认调用的是__import__()函数。直接使用该函数的情况很少见,一般用于动态加载模块。
用户2936342
2018/08/27
6670
Python第九周 学习笔记(1)
描述器 ---- get(self, instance, owner) 访问属性时调用 set(self, instance, value) 当对属性赋值时调用 delete(self, instance) 删除属性时调用 self指代当前实例 instance是owner的实例 owner是属性的所属的类 描述器实现前提是描述器类实例作为类属性 当只实现get时 (非数据描述符),属性查找顺序是本实例优先,get方法次之 当实现get和set时(数据描述符) ,属性查找顺序是get方法优先 本
py3study
2020/01/10
5520
Python 骚操作,花式导包的 8 种方法
__import__ 函数可用于导入模块,import 语句也会调用函数。其定义为:
AirPython
2020/05/26
6680
python sys模块的常见用法汇总
python的内置模块sys,提供了系统相关的一些变量和函数,在实际开发中,常见的有以下几种用法
生信修炼手册
2020/05/25
1.9K0
Python 简单沙盒绕过
题目中过滤了 h、'、"和(,最终目的是要给一个QQ bot发送自己的代码让它执行,执行成功就可以得到flag
回天
2023/04/25
2K0
Python 简单沙盒绕过
Python内置函数——__import__ 的使用方法
__import__(name[, globals[, locals[, fromlist[, level]]]])
用户7886150
2020/12/19
7500
Python面试题之Python反射详解
解释Python的反射,先提一个简单的需求,现在我有一个简易的网站,由两个文件组成,一个是具体执行操作的commons.py文件,一个是入口文件index.py,现在我需要在入口文件中设置,让用户输入url,根据用户输入的url去后端执行相应的操作,内容如下:
Jetpropelledsnake21
2018/08/10
5330
Python 进阶:深入理解 import 机制与 importlib 的妙用
大家好,今天我们来深入探讨 Python 中的导入机制和 importlib 模块。相信不少朋友和我一样,平时写代码时可能只用过最基础的 import 语句,或者偶尔用 importlib.import_module 来做些动态导入。但其实这背后的机制非常有趣,而且 importlib 提供的功能远比我们想象的要丰富。
Piper破壳
2024/12/30
1990
Python相对、绝对导入浅析
这篇文章从另外一个不同的视角来分析一下Python的import机制,主要的目的是为了搞懂import中absolute、relative import遇到的几个报错。  这里不同的视角是指从Python import hooks这个方面来展开,当然本身关于Python import hooks有很多的文章,我这里不打算展开聊这个方面的内容,文章中主要会结合代码和PEP 302 – New Import Hooks这个PEP。  1. 几个跟import相关模块属性 首先我们需要了解几个跟import相关
庞小明
2018/03/09
1.3K0
理解python的import与__import__
import 和“__import__”都是用来导入module的,但是二者还是有所不同, 可以查看帮助文档来了解其不同. 先通过 help("import") 查看其帮助,可以找到如下的说明:
qsjs
2020/06/08
1.8K0
Python3.8 了解的差不多了吧,Python3.9 新特性了解一下!
原文有删改:https://docs.python.org/3.9/whatsnew/3.9.html
kbsc13
2019/10/24
1.8K0
python高级-模块(14)
有过C语言编程经验的朋友都知道在C语言中如果要引用sqrt函数,必须用语句#include <math.h>引入math.h这个头文件,否则是无法正常进行调用的。
Se7eN_HOU
2019/09/11
7360
python高级-模块(14)
你必须要了解了知识-python反射机制
反射机制就是在运行时,动态的确定对象的类型,并可以通过字符串调用对象属性、方法、导入模块,是一种基于字符串的事件驱动。
星星在线
2019/06/17
1.3K0
相关推荐
一文看懂Python沙箱逃逸
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验