OpenStack的源码分析在网上已经非常多了,针对各个部分的解读亦是非常详尽。这里我根据自己的理解把之前读过的Swift源码的一些要点记录一下,希望给需要的同学能带来一些帮助。

一、Swift的整体框架图

如上图,Swift的源码目录结构。其中proxy是前端的业务接入进程。account、container和object目录分别是账户、容器 和对象的业务处理逻辑进程。common目录是一些通用工具代码。common中比较重要的有:哈希环的处理逻辑。接下来会依次介绍各个进程的源码逻辑和 一些关键点机制。

各个业务进程或模块之间的逻辑关系可以参考《Openstack Swift简介》文中的架构图。

二、Proxy进程的业务处理

首先需要掌握基于PasteDeploy的堆栈式Wsgi架构。根据PasteDeploy定义的各个层,可以很快理清配置文件定义的代码流程,从 middleware到server。找到最外层的middleware,即是业务的入口。对于proxy进程,可以简单给出业务时序图:

每一层的分工非常清晰,如在proxy进程默认配置文件中,最上层是做异常处理,所有的业务流程抛出的未处理的异常,在这里都将得到处理。

Proxy进程会分析请求的URI(account、container和object组成的资源路径)和请求方法(put、del等)来分析当前 请求的资源的具体类型,然后分贝找到控制该资源的controller,由controller来分发请求到具体的资源server。分发到原则是一致性 哈希环。一致性哈希环在系统初始化时由工具生成,在《Swift 和 Keystone单机安装总结》一文中有具体的操作步骤。

在《Openstack Swift简介》从理论上面介绍了具体的节点寻找过程。采用md5值加移位的方式来确定part,然后找到所有的虚拟节点。具体的代码为:

 
 
  1. container_partition,containers=self.app.container_ring.get_nodes(
  2. self.account_name,self.container_name)
  3. defget_nodes(self,account,container=None,obj=None):
  4. """
  5. Getthepartitionandnodes
  6. foranaccount/container/object.
  7. Ifanodeisresponsible
  8. formorethanonereplica,itwill
  9. onlyappearinthe
  10. outputonce.
  11. :paramaccount:accountname
  12. :param
  13. container:containername
  14. :paramobj:objectname
  15. :returns:atupleof(partition,listofnodedicts)
  16. Eachnodedictwillhaveatleastthefollowingkeys:
  17. ======
  18. ===============================================================
  19. iduniqueinteger
  20. identifieramongstdevices
  21. weightafloatofthe
  22. relativeweightofthisdeviceascomparedto
  23. others;
  24. thisindicateshowmanypartitionsthebuilderwilltry
  25. toassign
  26. tothisdevice
  27. zoneintegerindicating
  28. whichzonethedeviceisin;agiven
  29. partition
  30. willnotbeassignedtomultipledeviceswithinthe
  31. samezone
  32. iptheipaddressofthe
  33. device
  34. portthetcpportofthedevice
  35. devicethedevice'snameondisk(sdb1,for
  36. example)
  37. Metageneraluse'extra'
  38. field;forexample:theonlinedate,the
  39. hardware
  40. description
  41. ======
  42. ===============================================================
  43. """
  44. part=self.get_part(account,
  45. container,obj)
  46. returnpart,
  47. self._get_part_nodes(part)
  48. defget_part(self,obj=None):
  49. """
  50. Getthepartitionforan
  51. account/container/object.
  52. :paramaccount:accountname
  53. :param
  54. container:containername
  55. :paramobj:objectname
  56. :returns:thepartitionnumber
  57. """
  58. key=hash_path(account,container,obj,raw_digest=True)
  59. iftime()>;self._rtime:
  60. self._reload()
  61. part=struct.unpack_from('>;I',key)[0]>>
  62. self._part_shift
  63. returnpart
  64. def_get_part_nodes(self,part):
  65. part_nodes=[]
  66. seen_ids=set()
  67. forr2p2din
  68. self._replica2part2dev_id:
  69. if
  70. part<;len(r2p2d):
  71. dev_id=
  72. r2p2d[part]
  73. ifdev_id
  74. notinseen_ids:
  75. part_nodes.append(self.devs[dev_id])
  76. seen_ids.add(dev_id)
  77. returnpart_nodes

然后根据quorum原则来决定当前请求至少需要几个节点成功即可返回。如NWR分别为322,则至少需要2个节点写成功,才能确保此次写成功。体现在公用的make_request方法中:

  
  
  • defmake_requests(self,req,ring,part,method,path,headers,
  • query_string=''):
  • """
  • Sendsan
  • HTTPrequesttomultiplenodesandaggregatestheresults.
  • Itattemptstheprimarynodesconcurrently,theniterates
  • overthe
  • handoffnodesasneeded.
  • :paramreq:arequestsentbytheclient
  • :paramring:theringusedforfindingbackendservers
  • :parampart:thepartitionnumber
  • :parammethod:themethodtosendtothebackend
  • :param
  • path:thepathtosendtothebackend
  • (fullpathendsupbeing/<$device>/<$part>/<$path>)
  • :paramheaders:alistofdicts,whereeachdict
  • representsone
  • backendrequestthatshouldbemade.
  • :paramquery_string:
  • optionalquerystringtosendtothebackend
  • :returns:a
  • swob.ResponSEObject
  • """
  • start_nodes=ring.get_part_nodes(part)
  • nodes=
  • GreenthreadSafeIterator(self.app.iter_nodes(ring,part))
  • pile=GreenAsyncPile(len(start_nodes))
  • forheadin
  • headers:
  • pile.spawn(self._make_request,nodes,
  • head,query_string,self.app.logger.thread_locals)
  • response=[]
  • statuses=[]
  • for
  • respinpile:
  • ifnotresp:
  • continue
  • response.append(resp)
  • statuses.append(resp[0])
  • ifself.have_quorum(statuses,
  • len(start_nodes)):
  • break
  • #giveanypendingrequests*some*chancetofinish
  • pile.waitall(self.app.post_quorum_timeout)
  • whilelen(response)<;len(start_nodes):
  • response.append((HTTP_SERVICE_UNAVAILABLE,'','',''))
  • statuses,reasons,resp_headers,bodies=zip(*response)
  • returnself.best_response(req,statuses,bodies,
  • '%s%s'%(self.server_type,req.method),
  • headers=resp_headers)


  • 原文链接:http://mobile.51cto.com/hot-448613.htm

    OpenStack Swift源码导读:业务整体架构和Proxy进程的更多相关文章

    1. HTML实现代码雨源码及效果示例

      这篇文章主要介绍了HTML实现代码雨源码及效果示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

    2. 源码推荐:简化Swift编写的iOS动画,iOS Material Design库

      本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

    3. openstack swift ring文件解析

      一直以为对swift的ring文件充满好奇,这段时间重新研究swift,正好借此机会好好研究下。

    4. swift皮筋弹动发射飞机ios源码

      这是一个款采用swift实现的皮筋弹动发射飞机游戏源码,游戏源码比较详细,大家可以研究学习一下吧。

    5. KeyStone安装部署笔记

      KeyStone客户端:OpenStack身份服务API的命令行接口。OpenStack出于扩展性的考虑也支持多个region。下面的命令在regionOne创建了keystone的三种端点:为admin租户和用户申请令牌:最后验证admin租户和用户:输出结果验证了身份服务工作正常,Keystone安装部署成功。

    6. OpenStack对象存储――Swift

      Swift前身是RackspaceCloudFiles项目,随着Rackspace加入到OpenStack社区,于2010年7月贡献给OpenStack,作为该开源项目的一部分。Swift目前的最新版本是OpenStackEssex1.5.1。Swift特性在OpenStack官网中,列举了Swift的20多个特性,其中最引人关注的是以下几点。在OpenStack中还可以与镜像服务Glance结合,为其存储镜像文件。Auth认证服务目前已从Swift中剥离出来,使用OpenStack的认证服务Keysto

    7. OpenStack之swift安装笔记

      安装过程依然参考的是OpenStack的官方安装手册,但补充了一些细节。OpenStack的对象存储是一个多租户的、高可扩展的对象存储系统,通过RESTfulHTTPAPI管理着大量的非结构化数据,在部署对象存储之前必须至少安装了身份服务。首先在KeyStone中创建Swift的用户、服务及endpoint,具体命令及结果如下:然后安装proxy-server,swift客户端、memcached等:yuminstallopenstack-swift-proxypython-swiftclientpyth

    8. swift 写的app 源码,保存一下下

      http://www.topthink.com/topic/3345.htmlhttp://www.csdn.net/article/2015-01-09/2823502-swift-open-source-libs

    9. OpenStack入门到实战视频教程全集下载罗勇老师经典教程系列

      全网最经典的openstack视频教程,现在免费了!罗老师讲得实在是太好了,目前绝对是全网最好的openstack入门视频,而且讲得特别细,非常适合初学者学习,收集了好久终于齐了,现在贡献给大家。

    10. swift 源码网站 code4app

      http://code4app.com/ios/HTHorizontalSelectionList/54cb2c94933bf0883a8b4583http://123.th7.cn/code/DMPagerViewController_2522.html

    随机推荐

    1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

      下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

    2. swift UISlider,UIStepper

      我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

    3. preferredFontForTextStyle字体设置之更改

      即:

    4. Swift没有异常处理,遇到功能性错误怎么办?

      本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

    5. 字典实战和UIKit初探

      ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

    6. swift语言IOS8开发战记21 Core Data2

      上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

    7. swift语言IOS8开发战记22 Core Data3

      上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

    8. swift实战小程序1天气预报

      在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

    9. 【iOS学习01】swift ? and !  的学习

      如果不初始化就会报错。

    10. swift语言IOS8开发战记23 Core Data4

      接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

    返回
    顶部