Sanjay Chari f55cd6c8eb Server Migration Dynamic workload
This patch introduces a dynamic workload which creates num_vms servers with floating IPs,
and migrates num_migrate_vms servers between computes.
It also adds a parameter in browbeat-config.yaml to specify the dynamic workloads which
the user wants to run in a given iteration.

Change-Id: Ib7eff2e12c6d256ecd1f36ad30f1b19841cb1549
2021-07-29 17:58:27 +05:30

149 lines
6.5 KiB
Python

# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import logging
import random
from rally_openstack.scenarios.neutron import utils as neutron_utils
from rally_openstack.scenarios.vm import utils as vm_utils
from rally.task import atomic
LOG = logging.getLogger(__name__)
class DynamicBase(vm_utils.VMScenario, neutron_utils.NeutronScenario):
def create_delete_servers(self, image, flavor, num_vms=1, min_sleep=0, max_sleep=10,
network_create_args=None, subnet_create_args=None):
"""Create <num_vms> servers and delete <num_vms//2> servers.
:param image: image ID or instance for server creation
:param flavor: int, flavor ID or instance for server creation
:param num_vms: int, number of servers to create
:param min_sleep: int, minimum duration to sleep before deleting a server
:param max_sleep: int, maximum duration to sleep before deleting a server
:param network_create_args: dict, arguments for network creation
:param subnet_create_args: dict, arguments for subnet creation
"""
network = self._create_network(network_create_args or {})
self._create_subnet(network, subnet_create_args or {})
kwargs = {}
kwargs["nics"] = [{"net-id": network["network"]["id"]}]
servers = []
for i in range(num_vms):
server = self._boot_server(image, flavor, **kwargs)
LOG.info("Created server {} when i = {}".format(server,i))
servers.append(server)
# Delete least recently created server from list when i=1,3,5,7.....
if i % 2 == 1:
self.sleep_between(min_sleep, max_sleep)
server_to_delete = servers[0]
self._delete_server(server_to_delete, force=True)
LOG.info("Deleted server {} when i = {}".format(server_to_delete,i))
servers.pop(0)
def server_boot_floatingip(self, image, flavor, ext_net_id, num_vms=1, router_create_args=None,
network_create_args=None, subnet_create_args=None, **kwargs):
"""Create <num_vms> servers with floating IPs
:param image: image ID or instance for server creation
:param flavor: int, flavor ID or instance for server creation
:param ext_net_id: external network ID for floating IP creation
:param num_vms: int, number of servers to create
:param router_create_args: dict, arguments for router creation
:param network_create_args: dict, arguments for network creation
:param subnet_create_args: dict, arguments for subnet creation
:param kwargs: dict, Keyword arguments to function
"""
ext_net_name = None
if ext_net_id:
ext_net_name = self.clients("neutron").show_network(ext_net_id)["network"][
"name"
]
router_create_args["name"] = self.generate_random_name()
router_create_args["tenant_id"] = self.context["tenant"]["id"]
router_create_args.setdefault(
"external_gateway_info", {"network_id": ext_net_id, "enable_snat": True}
)
router = self._create_router(router_create_args)
network = self._create_network(network_create_args or {})
subnet = self._create_subnet(network, subnet_create_args or {})
self._add_interface_router(subnet["subnet"], router["router"])
for i in range(num_vms):
kwargs["nics"] = [{"net-id": network["network"]["id"]}]
guest = self._boot_server_with_fip(
image, flavor, True, ext_net_name, **kwargs
)
self._wait_for_ping(guest[1]["ip"])
@atomic.action_timer("neutron.create_router")
def _create_router(self, router_create_args):
"""Create neutron router.
:param router_create_args: POST /v2.0/routers request options
:returns: neutron router dict
"""
return self.admin_clients("neutron").create_router(
{"router": router_create_args}
)
def get_servers_migration_list(self, num_migrate_vms):
"""Generate list of servers to migrate between computes
:param num_migrate_vms: int, number of servers to migrate between computes
:returns: list of server objects to migrate between computes
"""
servers = self._list_servers(True)
trunks = self._list_trunks()
eligible_servers = []
for server in servers:
has_floating_ip = False
is_not_trunk_network_server = True
for network, addr_list in server.addresses.items():
if len(addr_list) > 1:
has_floating_ip = True
for interface in self._list_interfaces(server):
for trunk in trunks:
if interface._info['port_id'] == trunk['port_id']:
is_not_trunk_network_server = False
break
if not(is_not_trunk_network_server):
break
if has_floating_ip and is_not_trunk_network_server:
eligible_servers.append(server)
random.shuffle(eligible_servers)
num_servers_to_migrate = min(num_migrate_vms, len(eligible_servers))
list_of_servers_to_migrate = []
for i in range(num_servers_to_migrate):
list_of_servers_to_migrate.append(eligible_servers[i])
return list_of_servers_to_migrate
def migrate_servers_with_fip(self, num_migrate_vms):
"""Migrate servers between computes
:param num_migrate_vms: int, number of servers to migrate between computes
"""
for server in self.get_servers_migration_list(num_migrate_vms):
fip = list(server.addresses.values())[0][1]['addr']
LOG.info("ping {} before server migration".format(fip))
self._wait_for_ping(fip)
self._migrate(server)
self._resize_confirm(server, status="ACTIVE")
LOG.info("ping {} after server migration".format(fip))
self._wait_for_ping(fip)