-
Notifications
You must be signed in to change notification settings - Fork 10
Expand file tree
/
Copy pathtest_lvm_sr.py
More file actions
173 lines (147 loc) · 6.63 KB
/
Copy pathtest_lvm_sr.py
File metadata and controls
173 lines (147 loc) · 6.63 KB
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
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
from __future__ import annotations
import pytest
import logging
from lib.commands import SSHCommandFailed
from lib.common import Defer, vm_image, wait_for
from lib.fistpoint import FistPoint
from lib.host import Host
from lib.sr import SR
from lib.vdi import VDI
from lib.vm import VM
from tests.storage import (
CoalesceOperation,
ImageFormat,
XVACompression,
coalesce_integrity,
try_to_create_sr_with_missing_device,
vdi_export_import,
vdi_is_open,
xva_export_import,
)
# Requirements:
# - one XCP-ng host with an additional unused disk for the SR
class TestLVMSRCreateDestroy:
"""
Tests that do not use fixtures that setup the SR or import VMs,
because they precisely need to test SR creation and destruction,
and VM import.
"""
def test_create_sr_with_missing_device(self, host: Host) -> None:
try_to_create_sr_with_missing_device('lvm', 'LVM-local-SR-test', host)
def test_create_and_destroy_sr(self, host: Host,
unused_512B_disks: dict[Host, list[Host.BlockDeviceInfo]],
image_format: ImageFormat
) -> None:
sr_disk = unused_512B_disks[host][0]["name"]
# Create and destroy tested in the same test to leave the host as unchanged as possible
sr = host.sr_create('lvm', "LVM-local-SR-test", {
'device': '/dev/' + sr_disk,
'preferred-image-formats': image_format
}, verify=True)
# import a VM in order to detect vm import issues here rather than in the vm_on_xfs_fixture used in
# the next tests, because errors in fixtures break teardown
vm = host.import_vm(vm_image('mini-linux-x86_64-bios'), sr_uuid=sr.uuid)
vm.destroy(verify=True)
sr.destroy(verify=True)
@pytest.mark.usefixtures("lvm_sr")
class TestLVMSR:
@pytest.mark.quicktest
def test_quicktest(self, lvm_sr: SR) -> None:
lvm_sr.run_quicktest()
def test_vdi_is_not_open(self, vdi_on_lvm_sr: VDI) -> None:
assert not vdi_is_open(vdi_on_lvm_sr)
def test_vdi_image_format(self, vdi_on_lvm_sr: VDI, image_format: ImageFormat) -> None:
fmt = vdi_on_lvm_sr.get_image_format()
# feature-detect: if the SM doesn't report image-format, skip this check
if not fmt:
pytest.skip("SM does not report sm-config:image-format; skipping format check")
assert fmt == image_format
@pytest.mark.small_vm # run with a small VM to test the features
@pytest.mark.big_vm # and ideally with a big VM to test it scales
def test_start_and_shutdown_VM(self, vm_on_lvm_sr: VM) -> None:
vm = vm_on_lvm_sr
vm.start()
vm.wait_for_os_booted()
vm.shutdown(verify=True)
@pytest.mark.small_vm
@pytest.mark.big_vm
def test_snapshot(self, vm_on_lvm_sr: VM) -> None:
vm = vm_on_lvm_sr
vm.start()
try:
vm.wait_for_os_booted()
vm.test_snapshot_on_running_vm()
finally:
vm.shutdown(verify=True)
@pytest.mark.small_vm
@pytest.mark.big_vm
def test_failing_resize_on_inflate_after_setSize(
self, host: Host, lvm_sr: SR, vm_on_lvm_sr: VM, exit_on_fistpoint: None
) -> None:
vm = vm_on_lvm_sr
lvinflate = ""
vdi = VDI(vm.vdi_uuids()[0], host=vm.host)
new_size = vdi.get_virtual_size() + (1 * 1024 * 1024 * 1024) # Adding a 1GiB to size
with FistPoint(vm.host, "LVHDRT_inflate_after_setSize"), pytest.raises(SSHCommandFailed) as exc_info:
vdi.resize(new_size)
logging.info(exc_info)
lvlist = host.lvs(f"VG_XenStorage-{lvm_sr.uuid}")
for lv in lvlist:
if lv.startswith("inflate_"):
logging.info(f"Found inflate journal following error: {lv}")
lvinflate = lv
logging.info(f"Launching SR scan for {lvm_sr.uuid} to resolve failure")
try:
host.xe("sr-scan", {"uuid": lvm_sr.uuid})
except SSHCommandFailed as e:
assert False, f"Failing to scan following a inflate error {e}"
assert lvinflate not in host.lvs(f"VG_XenStorage-{lvm_sr.uuid}"), \
"Inflate journal still exist following the scan"
@pytest.mark.small_vm
@pytest.mark.big_vm
def test_failing_resize_on_inflate_after_setSizePhys(
self, host: Host, lvm_sr: SR, vm_on_lvm_sr: VM, exit_on_fistpoint: None
) -> None:
vm = vm_on_lvm_sr
lvinflate = ""
vdi = VDI(vm.vdi_uuids()[0], host=vm.host)
new_size = vdi.get_virtual_size() + (1 * 1024 * 1024 * 1024) # Adding a 1GiB to size
with FistPoint(vm.host, "LVHDRT_inflate_after_setSizePhys"), pytest.raises(SSHCommandFailed) as exc_info:
vdi.resize(new_size)
logging.info(exc_info)
lvlist = host.lvs(f"VG_XenStorage-{lvm_sr.uuid}")
for lv in lvlist:
if lv.startswith("inflate_"):
logging.info(f"Found inflate journal following error: {lv}")
lvinflate = lv
logging.info(f"Launching SR scan for {lvm_sr.uuid} to resolve failure")
try:
host.xe("sr-scan", {"uuid": lvm_sr.uuid})
except SSHCommandFailed as e:
assert False, f"Failing to scan following a inflate error {e}"
assert lvinflate not in host.lvs(f"VG_XenStorage-{lvm_sr.uuid}"), \
"Inflate journal still exist following the scan"
@pytest.mark.small_vm
@pytest.mark.parametrize("vdi_op", ["snapshot", "clone"])
def test_coalesce(self, storage_test_vm: VM, vdi_on_lvm_sr: VDI, vdi_op: CoalesceOperation, defer: Defer) -> None:
coalesce_integrity(storage_test_vm, vdi_on_lvm_sr, vdi_op, defer)
@pytest.mark.small_vm
@pytest.mark.parametrize("compression", ["none", "gzip", "zstd"])
def test_xva_export_import(self, vm_on_lvm_sr: VM, compression: XVACompression, defer: Defer) -> None:
xva_export_import(vm_on_lvm_sr, compression, defer)
@pytest.mark.small_vm
def test_vdi_export_import(self, storage_test_vm: VM, lvm_sr: SR, image_format: ImageFormat, defer: Defer) -> None:
vdi_export_import(storage_test_vm, lvm_sr, image_format, defer)
# *** tests with reboots (longer tests).
@pytest.mark.reboot
@pytest.mark.small_vm
def test_reboot(self, host: Host, lvm_sr: SR, vm_on_lvm_sr: VM) -> None:
sr = lvm_sr
vm = vm_on_lvm_sr
host.reboot(verify=True)
wait_for(sr.all_pbds_attached, "Wait for PBD attached")
# start the VM as a way to check that the underlying SR is operational
vm.start()
vm.wait_for_os_booted()
vm.shutdown(verify=True)
# *** End of tests with reboots