forked from firecracker-microvm/firecracker
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtest_boottime.py
161 lines (132 loc) · 5.15 KB
/
test_boottime.py
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
# Copyright 2018 Amazon.com, Inc. or its affiliates. All Rights Reserved.
# SPDX-License-Identifier: Apache-2.0
"""Tests that ensure the boot time to init process is within spec.
The boot time tests use the minimal kernel image and root file system found in
the S3 bucket at https://s3.amazonaws.com/spec.ccfc.min/img/minimal.
For boot time testing the serial console is not enabled.
"""
import os
import re
import time
import platform
from framework import decorators
import host_tools.logging as log_tools
# The maximum acceptable boot time in us.
MAX_BOOT_TIME_US = 150000
# NOTE: for aarch64 most of the boot time is spent by the kernel to unpack the
# initramfs in RAM. This time is influenced by the size and the compression
# method of the used initrd image.
INITRD_BOOT_TIME_US = 160000 if platform.machine() == "x86_64" else 500000
# TODO: Keep a `current` boot time in S3 and validate we don't regress
# Regex for obtaining boot time from some string.
TIMESTAMP_LOG_REGEX = r'Guest-boot-time\s+\=\s+(\d+)\s+us'
NO_OF_MICROVMS = 10
def test_single_microvm_boottime_no_network(test_microvm_with_boottime):
"""Check guest boottime of microvm without network."""
log_fifo, _ = _configure_vm(test_microvm_with_boottime)
time.sleep(0.4)
boottime_us = _test_microvm_boottime(log_fifo)
print("Boot time with no network is: " + str(boottime_us) + " us")
@decorators.test_context('boottime', NO_OF_MICROVMS)
def test_multiple_microvm_boottime_no_network(test_multiple_microvms):
"""Check guest boottime without network when spawning multiple microvms."""
microvms = test_multiple_microvms
assert microvms
assert len(microvms) == NO_OF_MICROVMS
log_fifos = []
for i in range(NO_OF_MICROVMS):
log_fifo, _ = _configure_vm(microvms[i])
log_fifos.append(log_fifo)
time.sleep(0.4)
for i in range(NO_OF_MICROVMS):
_ = _test_microvm_boottime(log_fifos[i])
@decorators.test_context('boottime', NO_OF_MICROVMS)
def test_multiple_microvm_boottime_with_network(
test_multiple_microvms,
network_config
):
"""Check guest boottime with network when spawning multiple microvms."""
microvms = test_multiple_microvms
assert microvms
assert len(microvms) == NO_OF_MICROVMS
log_fifos = []
_taps = []
for i in range(NO_OF_MICROVMS):
log_fifo, _tap = _configure_vm(microvms[i], {
"config": network_config, "iface_id": str(i)
})
log_fifos.append(log_fifo)
_taps.append(_tap)
time.sleep(0.4)
for i in range(NO_OF_MICROVMS):
_ = _test_microvm_boottime(log_fifos[i])
def test_single_microvm_boottime_with_network(
test_microvm_with_boottime,
network_config
):
"""Check guest boottime of microvm with network."""
log_fifo, _tap = _configure_vm(test_microvm_with_boottime, {
"config": network_config, "iface_id": "1"
})
time.sleep(0.4)
boottime_us = _test_microvm_boottime(log_fifo)
print("Boot time with network configured is: " + str(boottime_us) + " us")
def test_single_microvm_initrd_boottime(
test_microvm_with_initrd):
"""Check guest boottime of microvm with initrd."""
log_fifo, _tap = _configure_vm(test_microvm_with_initrd, initrd=True)
time.sleep(0.8)
boottime_us = _test_microvm_boottime(
log_fifo, max_time_us=INITRD_BOOT_TIME_US)
print("Boot time with initrd is: " + str(boottime_us) + " us")
def _test_microvm_boottime(log_fifo, max_time_us=MAX_BOOT_TIME_US):
"""Assert that we meet the minimum boot time.
TODO: Should use a microVM with the `boottime` capability.
"""
lines = log_fifo.sequential_reader(20)
boot_time_us = 0
for line in lines:
timestamps = re.findall(TIMESTAMP_LOG_REGEX, line)
if timestamps:
boot_time_us = int(timestamps[0])
assert boot_time_us > 0
assert boot_time_us < max_time_us
return boot_time_us
def _configure_vm(microvm, network_info=None, initrd=False):
"""Auxiliary function for preparing microvm before measuring boottime."""
microvm.spawn()
# Machine configuration specified in the SLA.
config = {
'vcpu_count': 1,
'mem_size_mib': 128
}
if initrd:
config['add_root_device'] = False
config['use_initrd'] = True
microvm.basic_config(**config)
if network_info:
_tap, _, _ = microvm.ssh_network_config(
network_info["config"],
network_info["iface_id"]
)
# Configure logging.
log_fifo_path = os.path.join(
microvm.path,
'log_fifo' + microvm.id.split('-')[0]
)
metrics_fifo_path = os.path.join(
microvm.path,
'metrics_fifo' + microvm.id.split('-')[0]
)
log_fifo = log_tools.Fifo(log_fifo_path)
metrics_fifo = log_tools.Fifo(metrics_fifo_path)
response = microvm.logger.put(
log_fifo=microvm.create_jailed_resource(log_fifo.path),
metrics_fifo=microvm.create_jailed_resource(metrics_fifo.path),
level='Info',
show_level=False,
show_log_origin=False
)
assert microvm.api_session.is_status_no_content(response.status_code)
microvm.start()
return log_fifo, _tap if network_info else None