nova 创建虚机时的块设备处理
nova-api 侧的块设备参数
nova-api
侧参数如下例子:
1 |
|
source_type
表示待创建的块设备的初始数据来源,这一般用于将系统镜像数据拷贝到具体新块设备中,如果不需要数据,则标记类型为blank,创建一块空白块设备;destination_type
只支持volume和local两种形式。volume表示块设备由cinder提供,local表示通过计算节点(hypervisor所在节点)开辟一块空间提供。块设备分类
- 由于源类型(image,snapshot,volume,blank)有这么几种可选
- 目的类型(volume,local)有这两种可选
- 上面参数可以组合出几种形态,依据不同的形态有不同的含义:
dest type | source type | 说明 |
---|---|---|
volume | volume | 可直接挂载到虚拟机中。当 boot_index = 0, 相当于boot from volume |
snapshot | 调用 cinder 依据快照创建新卷,并挂载到虚拟机中。当 boot_index = 0, 相当于boot from snapshot | |
image | 调用cinder依据镜像创建新卷,将glance image的数据拷贝到新建卷中(由cinder访问glance完成),然后挂载到虚拟机中。当 boot_index = 0, 相当于boot from image | |
blank | 调用cinder依大小创建一个空卷并挂载到虚拟机中 | |
local | image | 在计算节点特定位置创建 ephemeral 分区,将 glance 的image 数据拷贝新建的分区中,并启动虚拟机 |
blank | guest_format=swap 时,创建 swap 分区,否则创建 ephemeral 分区,并挂载到虚拟机中 |
数据表
- 在
nova
的数据表中有block_device_mapping
conduct
在build_instance
的块信息数据就是从这张表查询出来的。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55nova@mariadb.cty.os 14:11: [nova]> select * from block_device_mapping where instance_uuid ='55888641-5569-4d50-a6c3-eeb8f6d51c34' \G;
*************************** 1. row ***************************
created_at: 2023-03-23 02:48:39
updated_at: 2023-03-23 02:48:45
deleted_at: NULL
id: 143
device_name: /dev/vda
delete_on_termination: 1
snapshot_id: NULL
volume_id: 9a53945a-b745-493e-8710-0f2c92e1fb5f
volume_size: 100
no_device: 0
connection_info: {"driver_volume_type": "rbd", "connector": {"platform": "x86_64", "host": "ytj-kvm-10e101e8e12", "do_local_attach": false, "ip": "10.101.8.12", "os_type": "linux2", "multipath": false, "initiator": "iqn.2012-01.com.openeuler:6a8eeeadc98"}, "serial": "9a53945a-b745-493e-8710-0f2c92e1fb5f", "data": {"secret_type": "ceph", "name": "volumes/volume-9a53945a-b745-493e-8710-0f2c92e1fb5f", "encrypted": false, "discard": true, "keyring": "[client.cinder]\n\tkey = AQBjifhjdv+0ORAAFWZTU30onDsqcmEJoNz+Tg==\n", "cluster_name": "ceph", "secret_uuid": "817cf80b-d3c5-47b2-9a70-aa578e539107", "qos_specs": null, "auth_enabled": true, "volume_id": "9a53945a-b745-493e-8710-0f2c92e1fb5f", "hosts": ["10.101.24.27", "10.101.24.28", "10.101.24.29"], "access_mode": "rw", "auth_username": "cinder", "ports": ["6789", "6789", "6789"]}}
instance_uuid: 55888641-5569-4d50-a6c3-eeb8f6d51c34
deleted: 0
source_type: image
destination_type: volume
guest_format: NULL
device_type: disk
disk_bus: virtio
boot_index: 0
image_id: dbbec6ed-017e-45b0-845d-45c73c718ac6
tag: NULL
attachment_id: NULL
uuid: cbe90cc3-c7fb-4fab-bd6e-72bbc87867c8
backup_id: NULL
volume_type: SAS-public
*************************** 2. row ***************************
created_at: 2023-03-23 02:48:57
updated_at: 2023-03-23 02:48:59
deleted_at: NULL
id: 146
device_name: /dev/vdb
delete_on_termination: 0
snapshot_id: NULL
volume_id: 56494430-5460-4582-a3d0-e22cb4ef0cd2
volume_size: 500
no_device: NULL
connection_info: {"status": "reserved", "detached_at": "", "volume_id": "56494430-5460-4582-a3d0-e22cb4ef0cd2", "attach_mode": null, "driver_volume_type": "rbd", "instance": "55888641-5569-4d50-a6c3-eeb8f6d51c34", "attached_at": "", "serial": "56494430-5460-4582-a3d0-e22cb4ef0cd2", "data": {"secret_type": "ceph", "name": "volumes/volume-56494430-5460-4582-a3d0-e22cb4ef0cd2", "encrypted": false, "discard": true, "keyring": "[client.cinder]\n\tkey = AQBjifhjdv+0ORAAFWZTU30onDsqcmEJoNz+Tg==\n", "cluster_name": "ceph", "secret_uuid": "817cf80b-d3c5-47b2-9a70-aa578e539107", "qos_specs": null, "auth_enabled": true, "volume_id": "56494430-5460-4582-a3d0-e22cb4ef0cd2", "hosts": ["10.101.24.27", "10.101.24.28", "10.101.24.29"], "access_mode": "rw", "auth_username": "cinder", "ports": ["6789", "6789", "6789"]}}
instance_uuid: 55888641-5569-4d50-a6c3-eeb8f6d51c34
deleted: 0
source_type: volume
destination_type: volume
guest_format: NULL
device_type: NULL
disk_bus: NULL
boot_index: NULL
image_id: NULL
tag: NULL
attachment_id: a0c38bca-1aff-4cf5-b25d-88b3d013d01d
uuid: 0b3f4fca-30d6-4262-ac44-482a6eea7f14
backup_id: NULL
volume_type: NULL
2 rows in set (0.00 sec)
conductor
1 |
|
nova-compute
nova compute 服务处理虚拟机块设备都主要集中在
_prep_block_device
函数中:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24def _prep_block_device(self, context, instance, bdms):
"""Set up the block device for an instance with error logging."""
try:
self._add_missing_dev_names(bdms, instance)
# 前文conductor查询出数据库中的 bdms 信息
# 然后这里将数据库类型的 bdms 封装成对应的blockdevice类
block_device_info = driver.get_block_device_info(instance, bdms)
mapping = driver.block_device_info_get_mapping(block_device_info)
multidisks = driver.block_device_info_get_local_disks(
block_device_info)
self._recheck_bare_disks_occupation_status(multidisks)
driver_block_device.attach_block_devices(
mapping, context, instance, self.volume_api, self.driver,
wait_func=self._await_block_device_map_created)
self._check_instance_backups(context, instance, mapping)
self._block_device_info_to_legacy(block_device_info)
return block_device_info
except exception.OverQuota as e:
# 省略多行将数据库类型的 bdms 封装成对应的 blockdevice 类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17def get_block_device_info(instance, block_device_mapping):
from nova.virt import block_device as virt_block_device
block_device_info = {
'root_device_name': instance.root_device_name,
'ephemerals': virt_block_device.convert_ephemerals(
block_device_mapping),
'block_device_mapping':
# 关键方法
virt_block_device.convert_all_volumes(*block_device_mapping),
'local_disks': virt_block_device.convert_local_disks(
block_device_mapping)
}
swap_list = virt_block_device.convert_swap(block_device_mapping)
block_device_info['swap'] = virt_block_device.get_swap(swap_list)
return block_device_info这里我们重点关注下
convert_all_volumes
这个方法,就是依次判断每个磁盘的类型,然后返回对应的对象1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30# 主调函数
def convert_all_volumes(*volume_bdms):
source_volume = convert_volumes(volume_bdms)
source_snapshot = convert_snapshots(volume_bdms)
source_image = convert_images(volume_bdms)
source_blank = convert_blanks(volume_bdms)
source_backup = convert_backups(volume_bdms)
return [vol for vol in
itertools.chain(source_volume, source_snapshot,
source_image, source_blank, source_backup)]
# 偏函数
convert_volumes = functools.partial(_convert_block_devices,
DriverVolumeBlockDevice)
# 转换主逻辑
def _convert_block_devices(device_type, block_device_mapping):
devices = []
for bdm in block_device_mapping:
try:
# 直接初始化对象调用 __ini__ 方法,后面会讲到
devices.append(device_type(bdm))
# 如果类型不符合,则pass
# _InvalidType 继承了_NotTransformable
except _NotTransformable:
pass
return devicesdevice_type
类型有多种,他们的继承关系如下:所有的类初始化,都会调用
DriveBlockDevice
的init
方法1
2
3
4
5
6
7
8def __init__(self, bdm):
self.__dict__['_bdm_obj'] = bdm
if self._bdm_obj.no_device:
raise _NotTransformable()
self.update({field: None for field in self._fields})
self._transform()init
方法最后会调用一下_transform
方法,其中一个实现如下:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16def _transform(self):
if (not self._bdm_obj.source_type == self._valid_source
or not self._bdm_obj.destination_type ==
self._valid_destination):
raise _InvalidType
self.update(
{k: v for k, v in self._bdm_obj.items()
if k in self._new_fields | set(['delete_on_termination'])}
)
self['mount_device'] = self._bdm_obj.device_name
try:
self['connection_info'] = jsonutils.loads(
self._bdm_obj.connection_info)
except TypeError:
self['connection_info'] = None类和
bdms
对应关系图,注意有些bdms
是没有对应的类的然后循环 block_device_mapping(列表) 中的类,根据不同的类去调用其
attach
方法,不同的attach
有不同的逻辑1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37def attach_block_devices(block_device_mapping, *attach_args, **attach_kwargs):
def _log_and_attach(bdm):
instance = attach_args[1]
if bdm.get('volume_id'):
LOG.info('Booting with volume %(volume_id)s at '
'%(mountpoint)s',
{'volume_id': bdm.volume_id,
'mountpoint': bdm['mount_device']},
instance=instance)
elif bdm.get('snapshot_id'):
LOG.info('Booting with volume snapshot %(snapshot_id)s at '
'%(mountpoint)s',
{'snapshot_id': bdm.snapshot_id,
'mountpoint': bdm['mount_device']},
instance=instance)
elif bdm.get('backup_id'):
LOG.info('Booting with volume backup %(backup_id)s at '
'%(mountpoint)s',
{'backup_id': bdm.backup_id,
'mountpoint': bdm['mount_device']},
instance=instance)
elif bdm.get('image_id'):
LOG.info('Booting with volume-backed-image %(image_id)s at '
'%(mountpoint)s',
{'image_id': bdm.image_id,
'mountpoint': bdm['mount_device']},
instance=instance)
else:
LOG.info('Booting with blank volume at %(mountpoint)s',
{'mountpoint': bdm['mount_device']},
instance=instance)
# 关键方法
bdm.attach(*attach_args, **attach_kwargs)
for device in block_device_mapping:
_log_and_attach(device)
return block_device_mapping
不同 attach 方法解析
DriverVolImageBlockDevice
:基于镜像启动虚机就是这个类型1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20class DriverVolImageBlockDevice( ):
_valid_source = 'image'
_proxy_as_attr_inherited = set(['image_id'])
def attach(self, context, instance, volume_api,
virt_driver, wait_func=None):
if not self.volume_id: # 如果 volume 不存在先调用 cinder-api 基于 image 创建一个 volume
vol = self._create_volume(
context, instance, volume_api, self.volume_size,
wait_func=wait_func, image_id=self.image_id)
self.volume_id = vol['id']
# TODO(mriedem): Create an attachment to reserve the volume and
# make us go down the new-style attach flow.
# 创建成功后,attach 和 DriverVolumeBlockDevice 一样了,直接调用父类方法
super(DriverVolImageBlockDevice, self).attach(
context, instance, volume_api, virt_driver)DriverVolumeBlockDevice
:- 1.从volume启动虚机(启动盘)
- 2.在启动时就给虚机附加一块volume(数据盘)
这两种情况都是这种类型。
1 |
|
DriverVolSnapshotBlockDevice
: 从快照启动虚机就是这种类型1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33class DriverVolSnapshotBlockDevice(DriverVolumeBlockDevice):
_valid_source = 'snapshot'
_proxy_as_attr_inherited = set(['snapshot_id'])
def attach(self, context, instance, volume_api,
virt_driver, wait_func=None):
if not self.volume_id:
# 先通过 snapshot_id 查询 snapshot
snapshot = volume_api.get_snapshot(context,
self.snapshot_id)
# NOTE(lyarwood): Try to use the original volume type if one isn't
# set against the bdm but is on the original volume.
if not self.volume_type and snapshot.get('volume_id'):
snap_volume_id = snapshot.get('volume_id')
orig_volume = volume_api.get(context, snap_volume_id)
self.volume_type = orig_volume.get('volume_type_id')
# 基于 snapshot 创建新 volume
vol = self._create_volume(
context, instance, volume_api, self.volume_size,
wait_func=wait_func, snapshot=snapshot)
self.volume_id = vol['id']
# TODO(mriedem): Create an attachment to reserve the volume and
# make us go down the new-style attach flow.
# 创建完成后 attach volume
# Call the volume attach now
super(DriverVolSnapshotBlockDevice, self).attach(
context, instance, volume_api, virt_driver)_do_attach 实现解析
日志分析
1、开始进行虚拟机的块设备处理
1 |
|
2、识别出块设备类型之后,和cinder-api进行交互,创建volume
1 |
|
3、同步的去attach_volume
1 |
|
nova 创建虚机时的块设备处理
http://mybestcheng.site/2023/03/28/openstack/nova/compute/block_device/