Merge "Dynamic workload: Octavia LB's creation create_lbs creates 'N' LB's with specified numbers of pools and clients per LB"

This commit is contained in:
Zuul 2021-08-05 06:32:46 +00:00 committed by Gerrit Code Review
commit 27c1c3c97b
5 changed files with 299 additions and 6 deletions

View File

@ -551,14 +551,24 @@ workloads:
times: 1 times: 1
scenarios: scenarios:
- name: dynamic-workload - name: dynamic-workload
enabled: true enabled: false
image_name: cirros image_name: cirros
flavor_name: m1.small flavor_name: m1.small
ext_net_id: ext_net_id:
num_vms: 10 num_vms: 10
num_migrate_vms: 5 num_migrate_vms: 5
jump_host_ip:
octavia_image_name: custom-cirros
octavia_flavor_name: m1.tiny-cirros
num_lbs: 2
num_pools: 2
num_clients: 2
vip_subnet_id:
user: cirros
user_data_file:
# workloads can be 'all', a single workload(Eg. : create_delete_servers), # workloads can be 'all', a single workload(Eg. : create_delete_servers),
# or a comma separated string(Eg. : create_delete_servers,migrate_servers). # or a comma separated string(Eg. : create_delete_servers,migrate_servers).
# Currently supported workloads : create_delete_servers, migrate_servers # Currently supported workloads : create_delete_servers, migrate_servers
# create_loadbalancers
workloads: all workloads: all
file: rally/rally-plugins/dynamic-workloads/dynamic_workload.yml file: rally/rally-plugins/dynamic-workloads/dynamic_workload.yml

View File

@ -8,3 +8,9 @@ Functions:
- _create_router: Create neutron router. - _create_router: Create neutron router.
- get_servers_migration_list: Generate list of servers to migrate between computes. - get_servers_migration_list: Generate list of servers to migrate between computes.
- migrate_servers_with_fip: Migrate servers between computes - migrate_servers_with_fip: Migrate servers between computes
- create_loadbalancers: Create 'N' loadbalancers
- create_clients: Create 'N' clients
- create_listener: Create listener
- create_pool: Create pool
- create_member: Create member
- check_connection: Check the connection of LB

View File

@ -15,26 +15,35 @@ from rally.task import scenario
from rally.task import types from rally.task import types
from rally.task import validation from rally.task import validation
import base import base
import octavia
@types.convert(octavia_image={"type": "glance_image"}, octavia_flavor={"type": "nova_flavor"})
@validation.add("image_valid_on_flavor", flavor_param="octavia_flavor", image_param="octavia_image")
@types.convert(image={"type": "glance_image"}, flavor={"type": "nova_flavor"}) @types.convert(image={"type": "glance_image"}, flavor={"type": "nova_flavor"})
@validation.add("image_valid_on_flavor", flavor_param="flavor", image_param="image") @validation.add("image_valid_on_flavor", flavor_param="flavor", image_param="image")
@validation.add( @validation.add(
"required_services", services=[consts.Service.NEUTRON, consts.Service.NOVA] "required_services", services=[consts.Service.NEUTRON,
consts.Service.NOVA,
consts.Service.OCTAVIA]
) )
@validation.add("required_platform", platform="openstack", users=True) @validation.add("required_platform", platform="openstack", users=True)
@scenario.configure( @scenario.configure(
context={ context={
"cleanup@openstack": ["neutron", "nova"], "cleanup@openstack": ["neutron", "nova", "octavia"],
"keypair@openstack": {}, "keypair@openstack": {},
"allow_ssh@openstack": None, "allow_ssh@openstack": None,
}, },
name="BrowbeatPlugin.dynamic_workload", name="BrowbeatPlugin.dynamic_workload",
platform="openstack", platform="openstack",
) )
class DynamicWorkload(base.DynamicBase): class DynamicWorkload(base.DynamicBase, octavia.DynamicOctaviaBase):
def run(self, image, flavor, ext_net_id, num_migrate_vms, num_vms=1, workloads="all", def run(
router_create_args=None, network_create_args=None, subnet_create_args=None, **kwargs): self, image, flavor, ext_net_id, num_migrate_vms, jump_host_ip,
user_data_file, num_lbs, num_pools, vip_subnet_id, num_clients,
user, octavia_image, octavia_flavor, workloads="all", num_vms=1,
router_create_args=None, network_create_args=None,
subnet_create_args=None, **kwargs):
if workloads != "all": if workloads != "all":
workloads_list = workloads.split(",") workloads_list = workloads.split(",")
@ -47,3 +56,9 @@ class DynamicWorkload(base.DynamicBase):
self.server_boot_floatingip(image, flavor, ext_net_id, num_vms, router_create_args, self.server_boot_floatingip(image, flavor, ext_net_id, num_vms, router_create_args,
network_create_args, subnet_create_args, **kwargs) network_create_args, subnet_create_args, **kwargs)
self.migrate_servers_with_fip(num_migrate_vms) self.migrate_servers_with_fip(num_migrate_vms)
if "create_loadbalancers" in workloads_list:
self.create_loadbalancers(octavia_image, octavia_flavor, user, num_lbs,
jump_host_ip, vip_subnet_id, user_data_file, num_pools,
num_clients, router_create_args, network_create_args,
subnet_create_args, **kwargs)

View File

@ -3,6 +3,12 @@
{% set num_vms = num_vms or 1 %} {% set num_vms = num_vms or 1 %}
{% set num_migrate_vms = num_migrate_vms or 1 %} {% set num_migrate_vms = num_migrate_vms or 1 %}
{% set workloads = workloads or 'all' %} {% set workloads = workloads or 'all' %}
{% set octavia_image_name = octavia_image_name or 'custom-cirros' %}
{% set octavia_flavor_name = octavia_flavor_name or 'm1.tiny-cirros' %}
{% set num_lbs = num_lbs or 2 %}
{% set num_pools = num_pools or 2 %}
{% set num_clients = num_clients or 2 %}
{% set user = user or 'cirros' %}
{% set sla_max_avg_duration = sla_max_avg_duration or 60 %} {% set sla_max_avg_duration = sla_max_avg_duration or 60 %}
{% set sla_max_failure = sla_max_failure or 0 %} {% set sla_max_failure = sla_max_failure or 0 %}
{% set sla_max_seconds = sla_max_seconds or 60 %} {% set sla_max_seconds = sla_max_seconds or 60 %}
@ -15,6 +21,17 @@ BrowbeatPlugin.dynamic_workload:
name: '{{flavor_name}}' name: '{{flavor_name}}'
image: image:
name: '{{image_name}}' name: '{{image_name}}'
user: '{{ user }}'
jump_host_ip: '{{ jump_host_ip }}'
octavia_image:
name: '{{ octavia_image_name }}'
octavia_flavor:
name: '{{ octavia_flavor_name }}'
num_lbs: {{ num_lbs }}
num_pools: {{ num_pools }}
num_clients: {{ num_clients }}
vip_subnet_id: '{{ vip_subnet_id }}'
user_data_file: '{{ user_data_file }}'
network_create_args: {} network_create_args: {}
router_create_args: {} router_create_args: {}
subnet_create_args: {} subnet_create_args: {}

View File

@ -0,0 +1,245 @@
# 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 time
import io
from rally.common import sshutils
from rally_openstack.scenarios.octavia import utils as octavia_utils
from octaviaclient.api import exceptions
LOG = logging.getLogger(__name__)
class DynamicOctaviaBase(octavia_utils.OctaviaBase):
def create_clients(self, num_clients, user, user_data_file, image, flavor, **kwargs):
"""Create <num_clients> clients
:param num_clients: int, number of clients to create
:param image: image ID or instance for server creation
:param flavor: int, flavor ID or instance for server creation
:param user: user to ssh
:param user_data_file: user data file to serve from the metadata server
:param kwargs: dict, Keyword arguments to function
"""
_clients = []
for i in range(num_clients):
try:
userdata = io.open(user_data_file, "r")
kwargs["userdata"] = userdata
LOG.info("Added user data")
except Exception as e:
LOG.info("couldn't add user data %s", e)
LOG.info("Launching Client : {}".format(i))
server = self._boot_server(
image,
flavor,
key_name=self.context["user"]["keypair"]["name"],
**kwargs)
if hasattr(userdata, 'close'):
userdata.close()
for net in server.addresses:
network_name = net
break
if network_name is None:
return False
# IP Address
_clients.append(
str(server.addresses[network_name][0]["addr"]))
LOG.info(_clients)
return _clients
def create_listener(self, lb_id, protocol_port, max_attempts=10):
"""Create a listener.
:param lb_id: ID of loadbalancer
:param protocol_port: protocol port number for the listener
:param max_attempts: max retries
"""
listener_args = {
"name": self.generate_random_name(),
"loadbalancer_id": lb_id,
"protocol": 'HTTP',
"protocol_port": protocol_port,
"connection_limit": -1,
"admin_state_up": True,
}
LOG.info("Creating a listener for lb {}".format(lb_id))
attempts = 0
# Retry to avoid HTTP 409 errors like "Load Balancer
# is immutable and cannot be updated"
while attempts < max_attempts:
try:
listener = self.octavia.listener_create(json={"listener": listener_args})
break
except exceptions.OctaviaClientException as e:
# retry for 409 return code
if e.code == 409:
attempts += 1
time.sleep(60)
continue
break
LOG.info("Listener created {}".format(listener))
LOG.info("Waiting for the lb {} to be active, after listener_create"
.format(lb_id))
return listener
def create_pool(self, lb_id, listener_id, max_attempts=10):
"""Create a pool.
:param lb_id: ID of loadbalancer
:param listener_id: ID of listener
:param max_attempts: max retries
"""
LOG.info("Creating a pool for lb {}".format(lb_id))
attempts = 0
# Retry to avoid HTTP 409 errors like "Load Balancer
# is immutable and cannot be updated"
while attempts < max_attempts:
try:
# internally pool_create will wait for active state
pool = self.octavia.pool_create(
lb_id=lb_id,
protocol='HTTP',
lb_algorithm='ROUND_ROBIN',
listener_id=listener_id,
admin_state_up=True)
break
except exceptions.OctaviaClientException as e:
# retry for 409 return code
if e.code == 409:
attempts += 1
time.sleep(120)
continue
break
return pool
def create_member(self, client_ip, pool_id, protocol_port, subnet_id, lb_id, max_attempts=10):
"""Create a member.
:param client_ip: client ip address
:param pool_id: ID of pool
:param subnet_id: subnet ID for pool
:param lb_id: ID of loadbalancer
:param max_attempts: max retries
"""
member_args = {
"address": client_ip,
"protocol_port": protocol_port,
"subnet_id": subnet_id,
"admin_state_up": True,
"name": self.generate_random_name(),
}
LOG.info("Adding member : {} to the pool {} lb {}"
.format(client_ip, pool_id, lb_id))
attempts = 0
# Retry to avoid "Load Balancer is immutable and cannot be updated"
while attempts < max_attempts:
try:
self.octavia.member_create(pool_id,
json={"member": member_args})
break
except exceptions.OctaviaClientException as e:
# retry for 409 return code
if e.code == 409:
attempts += 1
time.sleep(120)
LOG.info("mem_create exception: Waiting for the lb {} to be active"
.format(lb_id))
continue
break
time.sleep(30)
LOG.info("Waiting for the lb {} to be active, after member_create"
.format(lb_id))
def check_connection(self, lb, user, jump_host_ip, num_pools, num_clients, max_attempts=10):
"""Checks the connection
:param lb: loadbalancer
:param user: ssh user
:param jump_host_ip: Floating IP of jumphost
:param num_pools: number of pools per loadbalancer
:param num_clients: number of clients per loadbalancer
:param max_attempts: max retries
"""
port = 80
lb_ip = lb["vip_address"]
LOG.info("Load balancer IP: {}".format(lb_ip))
jump_ssh = sshutils.SSH(user, jump_host_ip, 22, None, None)
# check for connectivity
self._wait_for_ssh(jump_ssh)
for i in range(num_pools):
for j in range(num_clients):
cmd = "curl -s {}:{}".format(lb_ip, port)
attempts = 0
while attempts < max_attempts:
test_exitcode, stdout_test, stderr = jump_ssh.execute(cmd, timeout=60)
LOG.info("cmd: {}, stdout:{}".format(cmd, stdout_test))
if stdout_test != '1':
LOG.error("ERROR with HTTP response {}".format(cmd))
attempts += 1
time.sleep(30)
else:
LOG.info("cmd: {} successful".format(cmd))
break
port = port + 1
def create_loadbalancers(
self, octavia_image, octavia_flavor, user, num_lbs, jump_host_ip, vip_subnet_id,
user_data_file, num_pools, num_clients, router_create_args=None,
network_create_args=None, subnet_create_args=None, **kwargs):
"""Create <num_lbs> loadbalancers with specified <num_pools> <num_clients> per Loadbalancer.
:param octavia_image: image ID or instance for server creation
:param octavia_flavor: int, flavor ID or instance for server creation
:param user: ssh user
:param num_lbs: int, number of loadbalancers to create
:param jump_host_ip: floating ip of jumphost
:param vip_subnet_id: Set subnet for the load balancer (name or ID)
:param user_data_file: pass user-data file for clients
:param num_pools: int, number of pools to create
:param num_clients: int, number of clients 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
"""
network = self._create_network(network_create_args or {})
subnet = self._create_subnet(network, subnet_create_args or {})
kwargs["nics"] = [{"net-id": network['network']['id']}]
subnet_id = subnet['subnet']['id']
_clients = self.create_clients(num_clients, user, user_data_file, octavia_image,
octavia_flavor, **kwargs)
LOG.info("Creating a load balancer")
for _ in range(num_lbs):
protocol_port = 80
lb = self.octavia.load_balancer_create(subnet_id=vip_subnet_id, admin_state=True)
lb_id = lb["id"]
LOG.info("Waiting for the lb {} to be active".format(lb["id"]))
self.octavia.wait_for_loadbalancer_prov_status(lb)
time.sleep(90)
for _ in range(num_pools):
listener = self.create_listener(lb_id, protocol_port)
self.octavia.wait_for_loadbalancer_prov_status(lb)
pool = self.create_pool(lb_id, listener["listener"]["id"])
self.octavia.wait_for_loadbalancer_prov_status(lb)
for client_ip in _clients:
self.create_member(client_ip, pool["id"], protocol_port, subnet_id, lb_id)
protocol_port = protocol_port + 1
self.check_connection(lb, user, jump_host_ip, num_pools, num_clients)