Nova的代码中支持Hook机制,也就是在某些函数的前后,可以加入自己的代码逻辑。Hook代码可以完全独立于Nova开发,本质上使用setuptools的entry points机制。K版本的OpenStack  Nova中支持Hook的流程有:

nova.compute .api.API:create

nova.compute.manager.ComputeManager:_do_build_and_run_instance

nova.compute.manager.ComputeManager:_delete_instance

nova.network.base_api:update_instance_cache_with_nw_info

一:Nova中的Hook原理

以nova.compute.api.API: create方法为例,该方法是nova-api收到创建实例的命令之后要调用的方法,具体代码如下:

@hooks.add_hook("create_instance")
def create(self, context, instance_type,
image_href, kernel_id=None, ramdisk_id=None,
min_count=None, max_count=None,
display_name=None, display_description=None,
key_name=None, key_data=None, security_group=None,
availability_zone=None, user_data=None, metadata=None,
injected_files=None, admin_password=None,
block_device_mapping=None, access_ip_v4=None,
access_ip_v6=None, requested_networks=None, config_drive=None,
auto_disk_config=None, scheduler_hints=None, legacy_bdm=True,
shutdown_terminate=False, check_server_group_quota=False):
"""Provision instances, sending instance information to the
scheduler. The scheduler will determine where the instance(s)
go and will handle creating the DB entries. Returns a tuple of (instances, reservation_id)
"""
...

可见该方法被装饰器hooks.add_hook所修饰了,这就是加载钩子的方法,看一下hooks.add_hook代码:

def add_hook(name, pass_function=False):
"""Execute optional pre and post methods around the decorated
function. This is useful for customization around callables.
""" def outer(f):
f.__hook_name__ = name @functools.wraps(f)
def inner(*args, **kwargs):
manager = _HOOKS.setdefault(name, HookManager(name)) function = None
if pass_function:
function = f manager.run_pre(name, args, kwargs, f=function)
rv = f(*args, **kwargs)
manager.run_post(name, rv, args, kwargs, f=function) return rv return inner
return outer

经过该装饰器的修饰之后,调用create(*args, **kwargs)方法时,实际上就是调用的inner(*args,**kwargs)函数。

inner函数比较简单,首先在全局变量_HOOKS中根据name寻找该方法对应的HookManager。这里的name就是add_hook的入参:"create_instance"。若第一次调用create方法,则创建一个针对该name的HookManager。然后首先调用manager.run_pre,接着是create方法本身,最后调用manager.run_post方法。

这样就相当于在create方法的前后插入了代码。看一下HookManager的实现:

class HookManager(stevedore.hook.HookManager):
def __init__(self, name):
"""Invoke_on_load creates an instance of the Hook class :param name: The name of the hooks to load.
:type name: str
"""
super(HookManager, self).__init__(NS, name, invoke_on_load=True) def _run(self, name, method_type, args, kwargs, func=None):
if method_type not in ('pre', 'post'):
msg = _("Wrong type of hook method. "
"Only 'pre' and 'post' type allowed")
raise ValueError(msg) for e in self.extensions:
obj = e.obj
hook_method = getattr(obj, method_type, None)
if hook_method:
LOG.debug("Running %(name)s %(type)s-hook: %(obj)s",
{'name': name, 'type': method_type, 'obj': obj})
try:
if func:
hook_method(func, *args, **kwargs)
else:
hook_method(*args, **kwargs)
except FatalHookException:
msg = _LE("Fatal Exception running %(name)s "
"%(type)s-hook: %(obj)s")
LOG.exception(msg, {'name': name, 'type': method_type,
'obj': obj})
raise
except Exception:
msg = _LE("Exception running %(name)s "
"%(type)s-hook: %(obj)s")
LOG.exception(msg, {'name': name, 'type': method_type,
'obj': obj}) def run_pre(self, name, args, kwargs, f=None):
"""Execute optional pre methods of loaded hooks. :param name: The name of the loaded hooks.
:param args: Positional arguments which would be transmitted into
all pre methods of loaded hooks.
:param kwargs: Keyword args which would be transmitted into all pre
methods of loaded hooks.
:param f: Target function.
"""
self._run(name=name, method_type='pre', args=args, kwargs=kwargs,
func=f) def run_post(self, name, rv, args, kwargs, f=None):
"""Execute optional post methods of loaded hooks. :param name: The name of the loaded hooks.
:param rv: Return values of target method call.
:param args: Positional arguments which would be transmitted into
all post methods of loaded hooks.
:param kwargs: Keyword args which would be transmitted into all post
methods of loaded hooks.
:param f: Target function.
"""
self._run(name=name, method_type='post', args=(rv,) + args,
kwargs=kwargs, func=f)

HookManager继承自stevedore.hook.HookManager。该父类就是stevedore用来管理hook类插件的基本类,关于stevedore可以参考《stevedore简介》。实例化HookManager时,传入的插件的namespace是NS =”nova.hooks”, 插件名是name="create_instance"。

HookManager.run_pre和HookManager.run_post方法都是调用的HookManager._run方法,只不过参数method_type分别是”pre”和”post”。

在HookManager._run方法中,轮训找到的插件实例,然后根据method_type寻找对应的实例属性。这样,只要插件类实现了pre和post方法,这里就可以调用这些方法。调用Hook方法时,传入的参数就是调用create方法的参数,还可以加入一个函数参数,这里忽略。

二:创建注册Hook

下面看一下如何创建并注册一个Hook。按照上面的梳理,只要定义一个实现了pre和post方法的类,然后在setup.py中,使用”nova.hooks”作为entry  points组名,以” create_instance”为entry  point名注册该类即可。下面是类实现:

class SimpleHookCreate (object):
def pre(self, *args, **kwargs):
logger.error("[PRE]this is hook1") def post(self, *args, **kwargs):
logger.error("[POST]this is hook1")

这里pre和post方法,只是在日志中打印简单语句而已。下面是该模块的setup.py脚本:

import setuptools

setuptools.setup(
name="demo_nova_hooks",
packages=['demo_nova_hooks'],
entry_points={
'nova.hooks': [
'create_instance=demo_nova_hooks.simple:SimpleHookCreate'
]
},
)

有关setuptools、entry point的相关知识,参阅《Distutils发布Python模块》、《setuptools简介》等相关文章。

该Hook模块的源码树如下:

setup.py
demo_nova_hooks\
__init__.py
simple.py
    使用 python  setup.py  install命令安装该Hook后,这样在Openstack中创建实例的时候,就会在日志中打印出相应的信息。

另外,因为使用的是stevedore中的Hook方式加载的插件,因此针对同一个entry point组名下的同一个entry point名,可以安装注册多个Hook类。

参考:

http://blog.oddbit.com/2014/09/27/integrating-custom-code-with-n/

http://blog.csdn.net/gqtcgq/article/details/49255995

http://blog.csdn.net/gqtcgq/article/details/49519685

http://blog.csdn.net/gqtcgq/article/details/49620279

Nova中的Hook机制的更多相关文章

  1. 【repost】JS中的hook机制

    hook机制也就是钩子机制,由表驱动实现,常用来处理多种特殊情况的处理.我们预定义了一些钩子,在常用的代码逻辑中去适配一些特殊的事件,这样可以让我们少些很多if else语句.举个高考加分的例子,比如 ...

  2. [转] js中的钩子机制(hook)

    什么是钩子机制?使用钩子机制有什么好处? 钩子机制也叫hook机制,或者你可以把它理解成一种匹配机制,就是我们在代码中设置一些钩子,然后程序执行时自动去匹配这些钩子:这样做的好处就是提高了程序的执行效 ...

  3. Hook机制里登场的角色

    稍有接触过 WordPress 主题或插件制作修改的朋友,对 WordPress 的Hook机制应该不陌生,但通常刚接触WordPress Hook 的新手,对其运作原理可能会有点混乱或模糊.本文针对 ...

  4. 黄聪:WordPress 的 Hook 机制与原理(add_action、add_filter)

    稍有接触过 WordPress 主题或插件制作修改的朋友,对 WordPress 的Hook机制应该不陌生,但通常刚接触WordPress Hook 的新手,对其运作原理可能会有点混乱或模糊.本文针对 ...

  5. MFC中的HOOK编程

    HOOK,n.钩, 吊钩,通常称钩子. 在计算机中,是Windows消息处理机制的一个平台,应用程序能够在上面设置子程以监视指定窗体的某种消息,并且所监视的窗体能够是其它进程所创建的.当消息到达后,在 ...

  6. 插件开发之360 DroidPlugin源码分析(二)Hook机制

    转载请注明出处:http://blog.csdn.net/hejjunlin/article/details/52124397 前言:新插件的开发,可以说是为插件开发者带来了福音,虽然还很多坑要填补, ...

  7. PHP中的插件机制原理和实例

    PHP项目中很多用到插件的地方,更尤其是基础程序写成之后很多功能由第三方完善开发的时候,更能用到插件机制,现在说一下插件的实现.特点是无论你是否激活,都不影响主程序的运行,即使是删除也不会影响. 从一 ...

  8. PostgreSQL的hook机制初步学习

    磨砺技术珠矶,践行数据之道,追求卓越价值 回到上一级页面:PostgreSQL内部结构与源代码研究索引页    回到顶级页面:PostgreSQL索引页 本文的目的一是为了备忘,二是为了抛砖引玉,希望 ...

  9. 浅谈svn的hook机制

    一.什么是钩子 所谓svn的hook机制,就是用户在管理数据仓库的时候,当特定的事件发生时,相应的hook会被调用,hook 其实就相当于特定事件的处理函数. 当前 Subversion 提供了5种可 ...

随机推荐

  1. E浏览器常见的9个css Bug以及解决办法

    我们在浏览网页的时候经常看见这样的现象:某个网页在IE6浏览器中打开很正常,但是在IE8里面打开可能完全变形了.或者也有可能出现完全相反的现象.这让Web程序员及设计师往往为了其CSS在各个IE版本下 ...

  2. Spring事务_注解_特性

    Spring 是一个 IOC 和 AOP 容器框架. ## 控制反转(IOC) ## 传统的 java 开发模式中,当需要一个对象时,我们会自己使用 new 或者 getInstance 等直接或者间 ...

  3. oracle习题-emp表查询练习

    emp表查询练习 1 查询emp表的全部记录 Select * from emp; 2 查询出每个雇员的编号.姓名.基本工资 Select empno,ename,sal from emp; 3 查询 ...

  4. Python科学计算生态圈--Scipy

  5. fileinput URL携带参数的问题,uploadExtraData,Bootstrap

    因为无法在URL中执行其他代码,通过阅读文档可知可以由uploadExtraData参数携带数据.于是在这里携带,如果uploadExtraData 无法接收到数据,一般是格式有误, 先尝试这个简单的 ...

  6. Vue--使用watch、computed、filter方法来监控

    watch与computed.filter: watch:监控已有属性,一旦属性发生了改变就去自动调用对应的方法 computed:监控已有的属性,一旦属性的依赖发生了改变,就去自动调用对应的方法 f ...

  7. webpack学习之—— 依赖图(Dependency Graph) 及 构建目标(Targets)

    Dependency Graph 任何时候,一个文件依赖于另一个文件,webpack 就把此视为文件之间有依赖关系.这使得 webpack 可以接收非代码资源(non-code asset)(例如图像 ...

  8. FPGA按键功能

    1.如何判断按键成功按下? 2.在什么时候采集数据? 按键在按下的过程中会产生大约2ms-3ms抖动,如果此时此刻采集数据来判断按键是不准确的,那么为了采集到准确的数据需要设置一个大约10ms左右的计 ...

  9. 几种支持REST的Java框架

    目前宣称支持REST的Java框架包括以下这些: Restlet(http://www.restlet.org/) Cetia4(https://cetia4.dev.java.net/) Apach ...

  10. ajax成功请求到后台,但是前端报404错误

    记录下今天遇见的一个问题,一个删除的ajax请求,传递一个主键到后台,后台成功接受并执行删除,但是前端报POST404错误. 查找原因是由于Controller忘记写返回状态码 @Responsebo ...