文章以ocata版本进行分析

cinder   api 的创建和启动,和 nova   api 类似,都是通过在 api-paste.ini 中定义 app ,然后将 app 加载之后,启动 wsgi 服务

走 keystone 验证的 cinder 服务,最后走 apiv3

由此, cinder   api 的入口就找到了。

api 请求的 router ,指定了 cinder/api/v3/volumes.py:VolumeController   作为实现 volume 增删改查的控制器。其中包括了 create /delete/get 等方法。

以 create 方法举例,先取出 request 的 body 数据,然后对 body 中的数据进行验证,重组,然后调用 volume_api

这里的 volume_api 指的是 cinder/volume/api.py:create   方法还是针对数据进行进一步格式化,然后使用 taskflow ,完成复杂的 volume 的创建。这个taskflow工具的大致工作方式是,创建一个 flow ,然后再 flow 中添加task ,每一个 task 要实现一个 execute 方法,作为执行函数,然后按步骤执行 task ,每一个 task 必须在前一个 task 成功执行的前提下才能执行,并支持回滚,需要继承指定类,重写 execute 执行方法和 revert 回滚方法。

这里的 api_flow 使用了 linear_flow ,也就是线形顺序依次执行,回滚也是按照倒叙依次回滚, revert 回滚方法会接收到 execute 方法执行的结果作为参数,以便进行相应的回滚操作。

首先第一个 task 是: ExtractVolumeRequestTask

这个 task 主要是完成对数据的检查,权限的验证等等,最后返回的还是创建参数 dict :

然后进入下一个 task : QuotareserveTask 。

这个 task 主要是对 quota (配额)进行预展,保证创建的资源不会超过 quota 配额。

查看这个 task 下的 execute 方法:

先调用 QUOTAS 模块( cinder/quotas.py )的 limit_check   检查 values 参数中的 quota 是否满足条件,这里可以看到检查的是 per_volume_gigabytes 。这个 quota 的含义是每一个 volume 的最大容量上限。其他的 quota 项可以通过 cinder 命令查看:

通过 quota 验证之后,再进行预占资源:

还是调用 QUOTAS 模块中的方法,这里预占的资源是, volume : 1 指占用 volume 一个, gigabytes : size 指占用资源 size 。

QUOTAS.reserve 方法将要预占的资源项目写入到数据库中,对应的相关数据可以查看 mysql cinder 库中的 reservations 表:

这里的数据表示了每一次的资源预占情况,之后 满足条件的情况下 ,则预占资源会更新到实际的 quota_usages 表中(下图),然后将 reservations 中的数据逻辑删除(上图: deleted=1 )

回到 task 中。

预占资源成功之后会返回对应的数据结果 :

如果失败,会调用 revert 方法:

可以看到这里会获取 reservations ,然后做 quota 的 rollback 。其他的 task 也是类似,不再赘述。

进入下一个 task:EntryCreateTask:

这个 task 的任务是生成 volume 对象,然后写入数据库:

再下一个 task 是: QuotaCommitTask :这个 task 将占用的 quota ( reservations )   更新到   正式的 quota 表中:

这时,数据验证和数据库方面的工作就完成了,接下来的工作就是下发创建 volume 的实际任务给 cinder   volume :

可以看到后续的 VolumeCastTask 包括 scheduler 和 volume ,一个是调度,一个是创建。

这个 task 的 execute 方法:

继续调用 cast_create_volume 方法:

这块根据传参决定是由 cinder   scheduler 调度选择节点创建 volume   还是   指定 host 创建。

这里按照普通的步骤,从 scheduler 调度开始往下跟踪,因为最终还是会调用 volume_rpcapi.create_volume 的。

进入 cinder/scheduler/rpcapi.py : create_volume :

这里先创建一个 worker ,用来记录工作的开始。

然后向 scheduler 发送消息: create_volume

接收这个消息的是 cinder/scheduler/manager.py:create_volume:

这里有个装饰器,对应上面的 worker 。

方法调用了 create_volume.get_flow 方法,这个方法也是一个 taskflow 实现的工作流:

这个方法是 cinder   scheduler 的工作流实现:

这个 flow 也是一个线性 flow : linear   flow ,里面添加了两个 task:ExtractSchedulerSpecTask   和 SchedulerCreateVolumeTask

ExtractSchedulerSpecTask :   整合 volume 的数据

具体细看下面的 task

SchedulerCreateVolumeTask :调用对应 driver ,进行创建 volume 的动作。

execute 方法,直接调用了 driver 的 scheduler_create_volume 方法:

这里的 self.driver_api 是在 schedulerManager 的 init 方法中初始化的

默认配置使用的是: cinder.scheduler.filter_scheduler.FilterScheduler

先调用 scheduler 方法,选择一个调度创建的存储后端 backend ,然后将这个后端信息更新到 volume 的数据库对象中,

这里有一步 post   select   populate   filter   properties ,好像是将此次调度到的存储节点保存,以确保当此次失败之后,重试操作不会再一次调度到这个节点上。

最后使用 rpc 发送创建消息,这里的 volume_rpcapi 是 cinder   volume 的 rpc   client ,这个 client 负责向 cinder   volume 服务发送 create_volume 的消息,接受并处理此消息

的是 cinder/volume/manager.py : create_volume 。跟踪此方法,还是使用了 taskflow 实现的

这个 flow 里添加的 task 包括

ExtractVolumeRefTask :刷新数据库 volume 的数据

OnFailureRescheduleTask:   主要是 task 失败 revert 的相关处理

ExtractVolumeSpecTask: 针对创建的参数进行数据重组,例如从 snapshot 创建, clone 等

NotifyVolumeActionTask: 想对应的 host 发送 create.start   的 notifier

CreateVolumeFromSpecTask: 根据对应的 create_type ,调用对应的方法,比如:创建 raw 类型的 volume ,会调用对应 driver 的创建方法:

CreateVolumeOnFinishTask: 这一步根据上一步的创建结果,更新数据库的 volume 状态

接下来深入到 driver 的创建方法中:

以 lvm 为例, lvm 的 driver 为例。

在 cinder/volume/manager.py   : VolumeManager   中,初始化了

跟踪代码到 LVMVolumeDriver 下的 create_volume 中:

这里使用了 self.vg, 这个 vg 是在哪里初始化的呢?

回到最初的 cinder   service 的启动代码里: cinder/service.py

这里有个 self.manager.init_host 调用,这个 manager 也就是上面的 VolumeManager ,这里调用了 init_host 方法,那 VolumeManager 的 init_host 有哪些内容?

可以看到调用了 driver 的 check_for_setup_error()

这个方法的实现,在对应的 lvm   driver 中:

在这里初始化了 self.vg 。

所以这个 self.vg 就跟踪到了 cinder/brick/local_dev/lvm.py 。

也就是说 cinder/volume/drivers/lvm.py 中的方法最终调用到 cinder/brick/local_dev/lvm.py

当前分析的 create_volume 如下:

可以看到调用了 lvcreate 命令,进行了最终的 volume 的创建

cinder 服务启动与请求流程源码分析的更多相关文章

  1. SpringMVC请求流程源码分析

    一.SpringMVC使用 1.工程创建 创建maven工程. 添加java.resources目录. 引入Spring-webmvc 依赖. <dependency> <group ...

  2. CBV请求流程源码分析

    一.CBV流程解析 urls.py urlpatterns = [ url(r'^admin/', admin.site.urls), url(r'^book/', views.BookView.as ...

  3. [Android]从Launcher开始启动App流程源码分析

    以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5017056.html 从Launcher开始启动App流程源码 ...

  4. Spark(五十一):Spark On YARN(Yarn-Cluster模式)启动流程源码分析(二)

    上篇<Spark(四十九):Spark On YARN启动流程源码分析(一)>我们讲到启动SparkContext初始化,ApplicationMaster启动资源中,讲解的内容明显不完整 ...

  5. Spark(四十九):Spark On YARN启动流程源码分析(一)

    引导: 该篇章主要讲解执行spark-submit.sh提交到将任务提交给Yarn阶段代码分析. spark-submit的入口函数 一般提交一个spark作业的方式采用spark-submit来提交 ...

  6. [Android]Android系统启动流程源码分析

    以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5013863.html Android系统启动流程源码分析 首先 ...

  7. Spring加载流程源码分析03【refresh】

      前面两篇文章分析了super(this)和setConfigLocations(configLocations)的源代码,本文来分析下refresh的源码, Spring加载流程源码分析01[su ...

  8. spring boot 加载web容器tomcat流程源码分析

    spring boot 加载web容器tomcat流程源码分析 我本地的springboot版本是2.5.1,后面的分析都是基于这个版本 <parent> <groupId>o ...

  9. Android笔记--View绘制流程源码分析(二)

    Android笔记--View绘制流程源码分析二 通过上一篇View绘制流程源码分析一可以知晓整个绘制流程之前,在activity启动过程中: Window的建立(activit.attach生成), ...

随机推荐

  1. php: xampp安装对应的phalcon版本(3.2.2-php5.6):比如redis-php5.6, php_igbinary-5.6

    php: xampp安装对应的phalcon版本(3.2.2-php5.6):比如redis-php5.6, php_igbinary-5.6 一.php安装redis扩展   1.使用phpinfo ...

  2. PHP面试题,自己几斤几两,看看就知道了

    0.简单做一下自我介绍,? 然后谈一下近三年来你的得意之作? 1.面试官看过你的简历,会问一些你做的项目的用户量.pv.吞吐量.相关难点和解决方法等 2.数据库设计经验,为什么进行分表? 分库? 一般 ...

  3. 山东省第七届ACM省赛

    ID Title Hint A Julyed 无 B Fibonacci 打表 C Proxy 最短路径 D Swiss-system tournament 归并排序 E The Binding of ...

  4. hash算法打散存储文件

    1.首先,为防止一个目录下面出现太多文件,所以使用hash算法打散存储 举例代码: int hashcode = filename.hashCode();//得到hashCode int dir1 = ...

  5. Netty5.x中新增和值得注意的点(转载http://www.coderli.com/netty-5-new-and-noteworthy/)

    该文档会列出在Netty新版本中值得注意变化和新特性列表.帮助你的应用更好的适应新的版本.   不像Netty3.x和4.x之间的变化,5.x没有那么大的变化,不过也取得了其简化设计中的一些突破性进展 ...

  6. MFC实现普通DLL

    库有两种:动态链接库和静态链接库. 一,使用动态链接库: 通过项目——属性——配置属性——常规——项目默认值——配置类型下,选择动态库(.dll)选项 这样会生成.lib和.dll两种文件. 只是该. ...

  7. sphinx:python项目文档自动生成

    Sphinx: 发音: DJ音标发音: [sfiŋks] KK音标发音: [sfɪŋks] 单词本身释义: an ancient imaginary creature with a lion's bo ...

  8. 机器学习 Logistic Regression

    Logistic Regression 之前我们讨论过回归问题,并且讨论了线性回归模型.现在我们来看看分类问题,分类问题与回归问题类似,只不过输出变量一个是离散的,一个是连续的.我们先关注二分类问题, ...

  9. php中socket的使用

    php中使用socket在服务器端主要使用这么几个函数: 1/$socket = socket_create(AF_INET,SOCK_STREAM,SOL_TCP)创建一个socket 2/sock ...

  10. Gym 101142G : Gangsters in Central City(DFS序+LCA+set)

    题意:现在有一棵树,1号节点是水源,叶子节点是村庄,现在有些怪兽会占领一些村庄(即只占领叶子节点),现在要割去一些边,使得怪兽到不了水源.给出怪兽占领和离开的情况,现在要割每次回答最小的割,使得怪兽不 ...