之前想写nova的policy的实现, 但是发现网上,有人写的很不错了。

但是个人认为存在一些问题。 ref:  http://www.cnblogs.com/shaohef/p/4527436.html

希望 microversion 还没有人写。

microversion实现

microversion实现的机制,就是在http的头部增加一个请求的小版本, nova的serve大家 搜索 new, type, metaclass, 都会介绍。r 根据这个小版本号,做相应的action。
实在是没有什么好介绍了。

这个我想像的microversion有些gap。 我开始以为是在url中指定版本号,而不是在head中。 

microversion的值得研究的是如下这段代码。

 class subContorller(wsgi.Controller)
@wsgi.Controller.api_version("2.1", "2.3")
def my_api_method(self, req, id):
.... method_1 ... @wsgi.Controller.api_version("2.4") #noqa
def my_api_method(self, req, id):
.... method_2 ...

来自: http://docs.openstack.org/developer/nova/devref/api_microversions.html#changing-a-method-s-behaviour

这段代码,有点诡异,在一个类中实现了两个同名的属性,后面的那个将覆盖前面的那个。

如果是熟悉python的类和实例创建过程的,肯定认为很easy。

瞬间就能想到实现原理, 比如说 不正直的绅士, 他直接做过类似的代码。 

其实在IBM的kvm的team,做发行版本的同事,也能想到怎么实现的。

因为我之前给他们介绍个python的类和实例创建过程,其实google/so 一大堆。

我的介绍肯定不如他们自己学习效果好,我纯粹就是显摆而已。

我们要在magnum上,实现microversion,所以我我按照自己的思路尝试自己实现一个。

首先,在同一个类中,定义多个同名函数,最后一个函数, 会覆盖其他的。

怎么办呢?

跟大家一样,第一想法是重载 __getattribute__ 的类。

做法是重名的函数,想办法重新命名。 然后在__getattribute__中,知道期望的函数。

发现,没有找到一个合适的位置,来hack重名的函数。

下面代码中的第5行,是我找到的唯一可能问位置,但是,这个代码只有类的实例才会调用。  ~~~~

 def operater(min, max):
def operate(fn):
def wraper(self, *args, **kv):
if fn.func_name not in self.funs:
self.funs[fn.func_name] = [(fn.func_name+"_"+min+"_"+max, wraper)]
else:
self.funs[fn.func_name].append((fn.func_name+"_"+min+"_"+max, wraper))
print getattr(self, fn.func_name)
print "begin decorate"
return fn(self, *args, **kv)
print "end decorate"
type(self)
return wraper
return operate class Controller(object):
funs = {}
def __init__(self):
print self def __getattribute__(self, name):
if name in self.funs:
all_funs = self.funs[name]
fun = self.funs[0][1]
return fun
return object.__getattribute__(self, name) @operater("1.0", "1.5") # noqa
def fun1(self):
print self
print "inline fun1" @operater("1.0", "1.6") # noqa
def fun1(self):
print self
print "inline fun2" if __name__ == '__main__':
print "start main"
import pdb
pdb.set_trace()
i = Controller()
 print dir(i)
i.fun1()

大家可以尝试写一下, 不知道有没有实现的可能, 过程中可能会有不少坑。

实在是给大家做过太多的培训了,类合实例的创建过程,还是比较清楚的。

休息了一下,立马清醒。 发现自己比较傻逼,这个位置是创建类的时候。

大家 搜索 new, type, metaclass, 都会介绍。

查看nova的代码,果真如此。 

nova采用了six.add_metaclass 来构造类。把nova的相关代码摘取如下, 很简单,都不需要解释。

 import six

 class VersionedMethod(object):

     def __init__(self, name, start_version, end_version, func):
self.name = name
self.start_version = start_version
self.end_version = end_version
self.func = func def __str__(self):
return ("Version Method %s: min: %s, max: %s"
% (self.name, self.start_version, self.end_version)) VER_METHOD_ATTR = 'versioned_methods'
obj_min_ver = "2.0"
obj_max_ver = "2.3" class ControllerMetaclass(type):
def __new__(mcs, name, bases, cls_dict):
versioned_methods = None
# start with wsgi actions from base classes
for base in bases:
# actions.update(getattr(base, 'wsgi_actions', {})) if base.__name__ == "Controller":
# NOTE(cyeoh): This resets the VER_METHOD_ATTR attribute
# between API controller class creations. This allows us
# to use a class decorator on the API methods that doesn't
# require naming explicitly what method is being versioned as
# it can be implicit based on the method decorated. It is a bit
# ugly.
print "+" * 80
print base.__dict__
if VER_METHOD_ATTR in base.__dict__:
versioned_methods = getattr(base, VER_METHOD_ATTR)
delattr(base, VER_METHOD_ATTR) for key, value in cls_dict.items():
if not callable(value):
continue
if versioned_methods:
cls_dict[VER_METHOD_ATTR] = versioned_methods return super(ControllerMetaclass, mcs).__new__(mcs, name, bases,
cls_dict) class abc(object):
pass @six.add_metaclass(ControllerMetaclass)
class Controller(abc): def __getattribute__(self, key):
def version_select(*args, **kwargs):
# The first arg to all versioned methods is always the request
# object. The version for the request is attached to the
# request object
func_list = self.versioned_methods[key]
print func_list
for func in func_list:
print "^" * 80
return func.func(self, *args, **kwargs)
# print func.func_name, func.obj_min_ver, func.obj_max_ver
return func.func(self, *args, **kwargs)
# No version match
raise exception.VersionNotFoundForAPIMethod(version=ver) try:
# super(LockerDecorator, self).__getattribute__(self, name)
version_meth_dict = abc.__getattribute__(self, VER_METHOD_ATTR)
except AttributeError:
# No versioning on this class
return abc.__getattribute__(self, key) if version_meth_dict and \
key in abc.__getattribute__(self, VER_METHOD_ATTR):
return version_select return abc.__getattribute__(self, key) # NOTE(cyeoh): This decorator MUST appear first (the outermost
# decorator) on an API method for it to work correctly
@classmethod
def api_version(cls, min_ver, max_ver=None):
def decorator(f): # Add to list of versioned methods registered
func_name = f.__name__
new_func = VersionedMethod(func_name, obj_min_ver, obj_max_ver, f) func_dict = getattr(cls, VER_METHOD_ATTR, {})
if not func_dict:
setattr(cls, VER_METHOD_ATTR, func_dict) # global func_list
func_list = func_dict.get(func_name, [])
if not func_list:
func_dict[func_name] = func_list
func_list.append(new_func)
func_list.sort(key=lambda f: f.start_version, reverse=True) return f return decorator class MyController(Controller):
@Controller.api_version("2.2")
def create(self, req, body):
print "create v2.2" @Controller.api_version("2.1", "2.1") # noqa
def create(self, req, body):
print "create v2.1" def delete(self, req, body):
print "delete no version" if __name__ == "__main__":
i = MyController()
i.create("", "")
i.delete("", "")
print "exit"

nova的 microversion 实现的更多相关文章

  1. Nova PhoneGap框架 第一章 前言

    Nova PhoneGap Framework诞生于2012年11月,从第一个版本的发布到现在,这个框架经历了多个项目的考验.一直以来我们也持续更新这个框架,使其不断完善.到现在,这个框架已比较稳定了 ...

  2. Nova PhoneGap框架 总结

    Nova PhoneGap Framework 是完全针对PhoneGap应用程序量身定做的,在这个框架下开发的应用程序很容易实现高质量的代码,很容易让程序拥有很好的性能和用户体验. 在经历了多个项目 ...

  3. nova instance出错:"message": "Proxy error: 502 Read from server failed

    执行 $ nova resize instance1 时候出错: {, "details": " File \"/opt/stack/nova/nova/com ...

  4. Nova PhoneGap框架 第二章 理解index.html

    跟绝大多数PhoneGap程序一样,Index.html是程序的入口.这个页面应该完成应用程序的初始化工作. 首先,让我们来看看这个页面通常都长什么样子: 下面我将一一解释这个页面都做了哪些初始化工作 ...

  5. Nova PhoneGap框架 第三章 页面

    页面在项目架构中是一个很重要的概念,它让我们能够将一个功能复杂的项目拆分成一个一个功能比较独立的小区域,这极大的提高了代码的可读性和可维护性. 在我们这个框架中,一个页面由JS和HTML两部分组成,首 ...

  6. Nova PhoneGap框架 第四章 本地数据库

    我一直想把EntityFramework(简称EF)的那一套搬过来,应用于HTML5 SQLite. 幸运的是,我几乎做到了,有些功能无法完成的那是因为SQLite本身不支持.至少从现在已经完成的功能 ...

  7. Nova PhoneGap框架 第七章 设备事件处理

    我们的框架包含了几种设备事件的处理,目的是为了让我们的程序员更容易的完成代码.这些事件包括:回退键(Android)和横竖屏切换事件. 7.1 Android回退键 首先来说说回退键的事件处理.当用户 ...

  8. Nova PhoneGap框架 第八章 滚动条

    你可能会疑惑为什么滚动条这么常见的功能会在这里单独列出,但如果你有过PhoneGap开发经验的话,你就会发现要在Android 2.3 里面实现滚动条那真不是一件容易的事. 8.1 概述 目前主流的P ...

  9. Nova PhoneGap框架 第九章 控件

    我们的框架中也提供了一些常用的控件,这些控件大多都依赖于我们的框架,也正是在我们的框架下才使得实现这些控件的变得更简单.但是我们的框架是不依赖与这些控件的,如果你用不上这些控件,你完全可以把相关的代码 ...

随机推荐

  1. linux日志(常用命令)

    LinuxIDERedHat防火墙活动  cat tail -f 日 志 文 件 说    明 /var/log/message 系统启动后的信息和错误日志,是Red Hat Linux中最常用的日志 ...

  2. EditPlus3.3 集成 SVN

    今天在玩EditPlus的时候,由于自己想上传文件至SVN,本机已经安装了TSVN,听说近期EditPlus支持了SVN操作,于是自己便带着好奇的心试试了.     已有的环境: EditPlus 3 ...

  3. MAC xampp 启动失败

    原文地址: http://meiyitianabc.blog.163.com/blog/static/1050221272013116232752/ 问题:80port被暂用,导致server无法启动 ...

  4. 使用SuperWebSocket 构建实时 Web 应用

    Web 应用的信息交互过程通常是客户端通过浏览器发出一个请求,服务器端接收和审核完请求后进行处理并返回结果给客户端,然后客户端浏览器将信息呈现出来,这种机制对于信息变化不是特别频繁的应用尚能相安无事, ...

  5. JavaScript可以这样用

    javascript:Qrlink(<%#Eval("ActivityType")%>,<%#Eval("ID")%>,<%#Ev ...

  6. 查询SQL中某表里有多少列包含某字段

    select c.name from SYSCOLUMNS as c left join SYSOBJECTS as t on c.id=t.id where c.name like '这里是某个字段 ...

  7. AngularJs 常用函数

    /** * [intersect 取两个数组的交集] var firstArray = [1,3,5]; var secondArray = [2,5,8]; var result */ .filte ...

  8. 20160115--Hibernate

    package com.hanqi.dao; import static org.junit.Assert.*; import java.util.*; import org.hibernate.se ...

  9. Openlayers 自定义控件

    OpenLayers.Control.YourControl = OpenLayers.Class(OpenLayers.Control, { // may private properties he ...

  10. C++自定义命名空间

    关于C++自定义命名空间,今天验证了一下命名空间如何使用,和嵌套命名空间以及出现的bug. 如何自定义命名空间,实例如下: insertion_sort.h和insertion_sort.cpp #p ...