A lightweight IOC container framework


Keywords
ioc, web, dependency, dependencies
License
MIT
Install
pip install heeframework==1.0.28

Documentation

【HeeFramework】


1 概要

HeeFramewrok(以下简称Hee) 是一款面向模块的低侵入式的IOC容器框架,为解决构建企业级复杂软件而创建的。 Hee提供了一个容器来管理各个组件对象的生命周期和组件依赖性,并定义了一个新的模块组织方式。

2 主要功能

1 容器
Hee提供一个容器来管理组件对象的生命周期,为对象的管理提供提供统一的注册容器。

2 控制器模块的自动注册
您只需要创建您的控制器模块,Hee会自动将您创建的控制器模块进行加载和注册。

3 动态发现和导入用户自定义的模块
提供动态扫描机制,能够从默认或指定的根路径下递归扫描各个模块,自动发现模块中标注的类,并自动创建该类的示例。

4 控制反转
Hee通过控制反转技术自动注入对象依赖,包括单向依赖的和双向依赖。不需要您查找或者去创建您所需要的依赖,

5 内置组件
Hee会为每个类自动创建通用的对象,比如说log对象。

6 自动检查和导入依赖的内建模块
Hee在启动时会自动扫描所框架自身依赖的模块,并自动导入。当然,导入哪些取决于您将要用到框架的哪些功能。

7 提供Web组件
Hee启动时会给需要的控制器组件自动注入web对象,web对象可支持大多数的web操作,比如文件上传、下载, 方便web项目的开发。

8 提供数据库访问组件
提供SQL模板的方式来方便数据库访问。在编写SQL时,可以先编写一个SQL模板,并用占位参数进行占位, 之后可以通过dict将参数进行传入。

9 定时调度
在任何子模块文件中的方法上面可以加一个@heejob注解,即可标识该方法为定时调度方法,框架会按照cron进行定时调度。

3 特色

易于使用的
通过之后"使用说明章节"中的案例,您将会发现很快您就能上手使用该框架,并喜欢上它。

轻量的
无论从体积上还是设计上Hee是轻量的,它通过动态内建模块机制来给自己瘦身。

面向子模块的 面向子模块的设计,比如依赖注入的目标为子模块本身,而不是一个自定类,这样能够方便程序编写。

低侵入性
除了修饰器来标识您需要让容器管理的类外,几乎不会增加其他任何额外内容。

4 涉及概念

4.1 动态模块

动态模块是一种按需加载的内置模块,如果你在配置文件中出现了该模块的相关段落,则自动下载该动态模块的依赖,并初始化该动态模块。
如果没有配置该模块,则不会加载该模块的任何内容

5 快速开始

5.1 安装Hee

    pip3 install heeframework -i https://pypi.org/simple

5.2 Hee应用

5.2.1 Hee IOC普通应用(HeeApplication)

Hee IOC普通应用即为一个普通的应用程序,它与原生的Python程序的不同点在于提供了模块自动发现,对象容器和基于控制反转的依赖注入能力。 当然,普通应用亦能够使用Hee提供的丰富内置组件。

在你的工程源码根路径下创建一个application.py(文件名不是固定的,你完全自己可以取名字),然后写入以下内容。

from hee import HeeApplication

class Application(HeeApplication):
    # your code
    pass


if __name__ == '__main__':
    app = Application()
    # your logic
    input()   # replace with other

完成之后,执行 python3 application.py 即可启动该项目。

5.2.3. 创建一个Hee Web应用

在工程下的源码根路径下创建一个application.py(文件名不是固定的,你完全自己可以取名字)。

目录结构:

your_project (PyCharm根目录)    
┗main
 ┗application.py   

application.py文件内容:

from hee import HeeWebApplication

class Application(HeeWebApplication):
    # your code
    pass

if __name__ == '__main__':
    app = Application()
    app.start()

写完之后,python3 application.py 即可启动该项目。 启动之后会自动生成以下目录:

your_project (PyCharm根目录)    
┗main   
 ┗config  
  ┗app.conf  
  ┗log4p.conf  
 ┗modules    
   ┗controller  
   ┗service  
   ┗dao
 ┗static
  ┗index.html
 ┗template
 ┗application.py    

基础框架搭建完成 。

5.3 编写控制器

当你构建的是一个HeeRestApplication或者HeeWebApplication时,控制器将会是一个必不可少的子模块组件。 控制器是处理来自互联网的http请求的第一道屏障,主要用于各种控制能力,包括参数认证控制、权限控制、校验控制、流程控制,返回控制等等。

5.3.1 创建一个控制器模块

在您的根目录下创建一个controller文件夹,创建 foo_controller.py 文件,并写入以下内容。控制器将会被自动注册到应用程序,您无需任何其他操作。 目录结构:

your_project (PyCharm根目录)    
┗main
 ┗controller
  ┗foo_controller.py
 ┗application.py   

foo_controller.py文件内容

from hee import HeeMapping

mapping:HeeMapping = HeeMapping("/foo")

@mapping.route("/find")
def find():
    print("finding bar!")
    return "bar"

当以上完成后,像之前一样启动您的的Application,随后就可以在浏览器中输入:http://localhost:5000/foo/find ,如果正确返回bar,恭喜您,您成功编写了您的第一个controller!

5.4 编写业务子模块

在你HeeApplication所在的目录创建子文件夹service,service文件夹中创建的任何.py文件将会视为子模块(submod),子模块将会自动导入容器进行管理。 子模块会被Hee扫描,并加入Hee容器。 我们现在在service目录创建一个foo_service.py,并写入以下内容:

from logging import Logger
from hee import component

# Automatic injection
log: Logger = None

@component
class FooService:
    def __init__(self):
        pass

    def do_something(self):
        log.info("do my business!")
        return "my business done"

提示1:FooService类有一个@component装饰,这是一个非常重要的装饰器,表示容器会创建一个component并交由容器进行管理。
提示2:该模块有一个为None的log成员,前面提到过,该log是Hee内置对象,Hee会将其自动注入,不需要对依赖进行手工查找。
提示3:我们可以将组件按职责划分为不同的层级。

5.5 控制反转与依赖注入

控制反转的核心思想是,你不要从容器查找或者自己去创建依赖的对象,而是通过被动的方式传递给你。 这里我们让Hee给FooController自动注入FooService的实例。我们修改foo_controller为以下:

from logging import Logger

from hee import HeeMapping
from modules.service.foo_service import FooService

mapping = HeeMapping("/foo")

# log
log: Logger = None

# foo service, auto injection
foo_service: FooService = None

@mapping.route("/find")
def find():
    log.info(foo_service.do_something())
    return "bar"

OK, 修改完成之后,我们重启应用,然后浏览器中输入:http://localhost:5000/foo/find,您在日志中将会看到:

2020-11-13 15:02:27,562 - foo_service.py line+18 - INFO - do my business!
2020-11-13 15:02:27,563 - foo_controller.py line+21 - INFO - my business done

恭喜您,成功完成!

6 开发指南

6.1 构建一个WEB应用

1 编写入口应用程序入口
新建一个工程,并新建一个源码目录,创建一个application.py的文件

your_project (PyCharm根目录)  
┗main  
   ┗application.py
from hee import HeeWebApplication

class Application(HeeWebApplication):
    pass

if __name__ == '__main__':
    app = Application()
    app.start()

如果是pycharm,右键点击main目录,并将其设置为源码根目录 执行python3 application.py 或直接在右键运行 application.py,然后刷新一下工程目录。 你将会看到hee已经生成好了标准目录,以及配置文件等。

your_project (PyCharm根目录)    
┗main   
 ┗config  
  ┗app.conf  
  ┗log4p.conf  
 ┗modules    
   ┗controller  
   ┗service  
   ┗dao
 ┗static
  ┗index.html
 ┗template
 ┗application.py    

这是hee推荐的标准目录,随后就可以将控制器写在controller中,业务逻辑写在service中,数据访问层写在dao中。

2 编写控制器
在modules/controller下创建test_controller.py文件,然后在其中初始化HeeMapping和一个接口,如下:

from logging import Logger

from hee import HeeMapping
from hee.heeframework import Web

mapping = HeeMapping("/test")
web: Web = None   # 自动注入
log: Logger = None  # 自动注入测试

@mapping.route("/find")
def find():
   return "success"

3 读取请求参数
直接在方法中编写:

params = web.request_params()

4 读取json请求数据格式

data = web.request_json()

5 读取文本数据

 data = web.request_data()

6 文件上传

  files = web.request_files()

7 用户对象以json格式返回

@mapping.route("/find")
def find():
  data = MyObject()
  return web.resp_json(data)

8 返回静态文件

  @mapping.route("/find")
  def find():
      return web.resp_static_file("/static/home.html")

6.2 构建一个Restful应用

6.2.1. Hee Restful应用(HeeRestApplication)

构建web应用与restful应用方法一致,不同的是继承于HeeRestApplication,rest应用不会自动创建static和template目录。

6.3 构建一个定时调度应用

步骤同上,然后在modules目录下创建一个jobs目录来存放所有的定时调度任务模块,创建一个test_job.py文件,然后写入以下内容。

@heejob(job_name="测试job", cron="0/5 * * * * *")
def test_job():
    print('hello')

启动application.py。

6.4 编写业务层代码

from logging import Logger

from hee.rdb import RDB
from hee import component
from modules.dao.test_dao import TestDao

log: Logger = None  # 自动注入
dao: TestDao = None      # 自动注入
db: RDB = None  # 自动注入

@component
class MonitorService:

    def find(self, mdata):
        # 查询原有数据
       dao.test_select(mdata.entity, mdata.node_code)

6.5 编写数据访问层代码

数据库访问目前支持MySQL,使用前需要配置MySQL连接信息
1 在app.conf配置

[MYSQL]
host=127.0.0.1
port=3306
user=root
pass=123456
db=ambition
pool_max=1000
pool_init=5
pool_idle=5

2 编写数据访问

from hee.rdb import RDB
from hee import component

db: RDB = None   # 自动注入

@component
class TestDao:
    def __init__(self):
        pass

    def test_select(self, entity, node_code):
        sql: str = """
                    select * 
                    from monitor_data 
                    where entity=#{entity} 
                    and node_code=#{node_code}
                    """

        return db.select_one(sql, {
            "entity": entity,
            "node_code": node_code}
        )

7 常见问题

本章节主要介绍介绍开发过程中极其常见的问题以及对应的解决方案,方便研发迅速定位问题并解决问题。

  1. 注入内建对象log, config时,发现注入不成功为None
    这个问题一般在使用pycharm开发时出现,pycharm会将工程的顶级目录作为源码根目录。
    在你的源码根目录上右键单击,然后标记为源码根目录即可解决该问题。

  2. 定时job不执行 一个可能的原因是你写的模块中存在一个属性,该属性的名字与job方法的名字重复,举例:

test_job: DemoService = None

@heejob(job_name="测试job", cron="0/5 * * * * *")
def test_job():
    print('hello')