Networking Parent behavior methods

- Creating networking parent methods for  the create, update, get, list, delete and clean behavior methods
- Moving the use_over_limit_retry from the ports config to the parent config
- Updating the networks, subnets and ports behaviors to use parent methods

Change-Id: Ic71cb85c9520c457e43495ab6c9f6b662e3322f0
This commit is contained in:
Leonardo Maycotte 2015-06-01 21:59:46 -05:00
parent 1236b4d40d
commit acf96b4da0
7 changed files with 853 additions and 781 deletions

View File

@ -18,11 +18,14 @@ import requests
import time
from cafe.engine.behaviors import BaseBehavior
from cloudcafe.common.tools.datagen import rand_name
from cloudcafe.networking.networks.common.config import NetworkingBaseConfig
from cloudcafe.networking.networks.common.constants \
import NeutronResourceTypes, NeutronResponseCodes
import NeutronResponseCodes, NeutronResource
from cloudcafe.networking.networks.common.exceptions \
import UnhandledMethodCaseException
import ResourceBuildException, ResourceDeleteException, \
ResourceGetException, ResourceListException, ResourceUpdateException, \
UnhandledMethodCaseException
class NetworkingBaseBehaviors(BaseBehavior):
@ -155,13 +158,535 @@ class NetworkingBaseBehaviors(BaseBehavior):
id_list = [entity.id for entity in entity_list]
return id_list
def _delete_resources(self, resource_type, resource_list=None, name=None,
tenant_id=None, skip_delete=None):
def __over_limit_retry(self, resource_type, timeout, poll_interval,
status_code, resp, fn_name, fn_kwargs):
"""
@summary: deletes multiple resources (for ex. networks)
@summary: Retry mechanism for API rate limited calls
@param resource_type: type of resource for ex. networks, subnets, etc.
See NeutronResourceTypes in the networks constants
@type resource_type: str
@param timeout: timeout for over limit retries
@type timeout: int
@param poll_interval: sleep time interval between API retries
@type poll_interval: int
@param status_code: over limit API HTTP response code
@type: int
@param resp: API call response object
@type resp: Requests.response
@param fn_name: API function name
@type fn_name: str
@param fn_kwargs: API function arguments
@type fn_kwargs: dict
"""
endtime = time.time() + int(timeout)
retry_msg = ('OverLimit retry with a {timeout}s timeout '
'calling {resource_type} function {fn_name}').format(
timeout=timeout, resource_type=resource_type,
fn_name=fn_name)
self._log.info(retry_msg)
while (resp.status_code == status_code and
time.time() < endtime):
resp = getattr(self.client, fn_name)(**fn_kwargs)
time.sleep(poll_interval)
return resp
def _create_resource(self, resource, resource_build_attempts=None,
raise_exception=True, poll_interval=None,
has_name=True, use_exact_name=False,
attrs_kwargs=None, timeout=None,
use_over_limit_retry=False):
"""
@summary: Creates and verifies a resource is created as expected
@param resource: type of resource for ex. network, subnet, port, etc.
See NeutronResource in the networks constants
@type resource: resource instance with singular and plural forms
@param resource_build_attempts: number of API retries
@type resource_build_attempts: int
@param raise_exception: flag to raise an exception if the
resource was not created or to return None
@type raise_exception: bool
@param poll_interval: sleep time interval between API retries
@type poll_interval: int
@param has_name: if the resource has a name attribute
@type has_name: bool
@param use_exact_name: flag if the exact name given should be used
@type use_exact_name: bool
@param attrs_kwargs: resource attributes to create with, for ex. name
@type attrs_kwargs: dict
@param timeout: resource create timeout for over limit retries
@type timeout: int
@param use_over_limit_retry: flag to enable/disable the create
over limits retries
@type use_over_limit_retry: bool
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
# Defining the resource type in singular form (for ex. network)
resource_type = resource.singular
# If has_name is False name can be used as a reference for log messages
name = attrs_kwargs.get('name')
if has_name == True:
if name is None:
name = rand_name(self.config.starts_with_name)
elif not use_exact_name:
name = rand_name(name)
attrs_kwargs['name'] = name
else:
# In case name is NOT used as a reference for log messages
if name is None:
name = ''
poll_interval = poll_interval or self.config.api_poll_interval
resource_build_attempts = (resource_build_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_create_timeout
result = NetworkingResponse()
err_msg = '{0} Create failure'.format(resource_type)
for attempt in range(resource_build_attempts):
self._log.debug(
'Attempt {attempt_n} of {attempts} creating '
'{resource_type}'.format(attempt_n=attempt + 1,
attempts=resource_build_attempts,
resource_type=resource_type))
# Method uses resource type in singular form (slicing the ending s)
create_fn_name = 'create_{0}'.format(resource_type)
resp = getattr(self.client, create_fn_name)(**attrs_kwargs)
if use_over_limit_retry:
entity_too_large_status_code = (getattr(self.response_codes,
'REQUEST_ENTITY_TOO_LARGE'))
if resp.status_code == entity_too_large_status_code:
fn_kwargs = attrs_kwargs
resp = self.__over_limit_retry(
resource_type=resource_type, timeout=timeout,
poll_interval=poll_interval,
status_code=entity_too_large_status_code,
resp=resp, fn_name=create_fn_name,
fn_kwargs=fn_kwargs)
response_code = create_fn_name.upper()
status_code = getattr(self.response_codes, response_code)
resp_check = self.check_response(
resp=resp, status_code=status_code, label=name,
message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the create was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to CREATE {name} {resource_type} after '
'{attempts} attempts: {failures}').format(
name=name, resource_type=resource_type,
attempts=resource_build_attempts,
failures=result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceBuildException(err_msg)
return result
def _update_resource(self, resource, resource_id,
resource_update_attempts=None, raise_exception=False,
poll_interval=None, attrs_kwargs=None,
timeout=None, use_over_limit_retry=False):
"""
@summary: Updates and verifies a specified resource
@param resource: type of resource for ex. network, subnet, port, etc.
See NeutronResource in the networks constants
@type resource: resource instance with singular and plural forms
@param resource_id: The UUID for the resource
@type resource_id: str
@param resource_update_attempts: number of API retries
@type resource_update_attempts: int
@param raise_exception: flag to raise an exception if the
resource was not updated or to return None
@type raise_exception: bool
@param poll_interval: sleep time interval between API retries
@type poll_interval: int
@param attrs_kwargs: resource attributes to update
@type attrs_kwargs: dict
@param timeout: resource update timeout for over limit retries
@type timeout: int
@param use_over_limit_retry: flag to enable/disable the update
over limits retries
@type use_over_limit_retry: bool
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
# Defining the resource type in singular form (for ex. network)
resource_type = resource.singular
poll_interval = poll_interval or self.config.api_poll_interval
resource_update_attempts = (resource_update_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_update_timeout
result = NetworkingResponse()
err_msg = '{0} Update failure'.format(resource_type)
for attempt in range(resource_update_attempts):
self._log.debug(
'Attempt {attempt_n} of {attempts} updating {resource_type} '
'{resource_id}'.format(attempt_n=attempt + 1,
attempts=resource_update_attempts,
resource_type=resource_type,
resource_id=resource_id))
# Method uses resource type in singular form (slicing the ending s)
update_fn_name = 'update_{0}'.format(resource_type)
# Resource ID is expected to be the first client method parameter
resp = getattr(self.client, update_fn_name)(resource_id,
**attrs_kwargs)
if use_over_limit_retry:
entity_too_large_status_code = (getattr(self.response_codes,
'REQUEST_ENTITY_TOO_LARGE'))
if resp.status_code == entity_too_large_status_code:
fn_kwargs = attrs_kwargs
# Adding the resource id to the function kwargs
resource_id_name = '{0}_id'.format(resource_type)
fn_kwargs[resource_id_name] = resource_id
resp = self.__over_limit_retry(
resource_type=resource_type, timeout=timeout,
poll_interval=poll_interval,
status_code=entity_too_large_status_code,
resp=resp, fn_name=update_fn_name,
fn_kwargs=fn_kwargs)
response_code = update_fn_name.upper()
status_code = getattr(self.response_codes, response_code)
resp_check = self.check_response(
resp=resp, status_code=status_code, label=resource_id,
message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the update was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to UPDATE {resource_id} {resource_type} after '
'{attempts} attempts: {failures}').format(
resource_id=resource_id, resource_type=resource_type,
attempts=resource_update_attempts,
failures=result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceUpdateException(err_msg)
return result
def _get_resource(self, resource, resource_id,
resource_get_attempts=None, raise_exception=False,
poll_interval=None, timeout=None,
use_over_limit_retry=False):
"""
@summary: Shows and verifies a specified resource
@param resource: type of resource for ex. network, subnet, port, etc.
See NeutronResource in the networks constants
@type resource: resource instance with singular and plural forms
@param resource_id: The UUID for the resource
@type resource_id: str
@param resource_get_attempts: number of API retries
@type resource_get_attempts: int
@param raise_exception: flag to raise an exception if the get
resource was not as expected or to return None
@type raise_exception: bool
@param poll_interval: sleep time interval between API retries
@type poll_interval: int
@param timeout: resource get timeout for over limit retries
@type timeout: int
@param use_over_limit_retry: flag to enable/disable the get
over limits retries
@type use_over_limit_retry: bool
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
# Defining the resource type in singular form (for ex. network)
resource_type = resource.singular
poll_interval = poll_interval or self.config.api_poll_interval
resource_get_attempts = (resource_get_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_get_timeout
result = NetworkingResponse()
err_msg = '{0} Get failure'.format(resource_type)
for attempt in range(resource_get_attempts):
self._log.debug(
'Attempt {attempt_n} of {attempts} getting {resource_type} '
'{resource_id}'.format(attempt_n=attempt + 1,
attempts=resource_get_attempts,
resource_type=resource_type,
resource_id=resource_id))
# Method uses resource type in singular form (slicing the ending s)
get_fn_name = 'get_{0}'.format(resource_type)
resp = getattr(self.client, get_fn_name)(resource_id)
if use_over_limit_retry:
entity_too_large_status_code = (getattr(self.response_codes,
'REQUEST_ENTITY_TOO_LARGE'))
if resp.status_code == entity_too_large_status_code:
fn_kwargs = {}
# Adding the resource id to the function kwargs
resource_id_name = '{0}_id'.format(resource_type)
fn_kwargs[resource_id_name] = resource_id
resp = self.__over_limit_retry(
resource_type=resource_type, timeout=timeout,
poll_interval=poll_interval,
status_code=entity_too_large_status_code,
resp=resp, fn_name=get_fn_name,
fn_kwargs=fn_kwargs)
response_code = get_fn_name.upper()
status_code = getattr(self.response_codes, response_code)
resp_check = self.check_response(resp=resp,
status_code=status_code,
label=resource_id, message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the get was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to GET {resource_id} {resource_type} after {attempts} '
'attempts: {failures}').format(resource_id=resource_id,
resource_type=resource_type,
attempts=resource_get_attempts,
failures=result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceGetException(err_msg)
return result
def _list_resources(self, resource, resource_list_attempts=None,
raise_exception=False, poll_interval=None,
params_kwargs=None, timeout=None,
use_over_limit_retry=False):
"""
@summary: Lists resources and verifies the response is the expected
@param resource: type of resource for ex. network, subnet, port, etc.
See NeutronResource in the networks constants
@type resource: resource instance with singular and plural forms
@param resource_list_attempts: number of API retries
@type resource_list_attempts: int
@param raise_exception: flag to raise an exception if the resource list
was not as expected or to return None
@type raise_exception: bool
@param poll_interval: sleep time interval between API retries
@type poll_interval: int
@param params_kwargs: key value params to filter by the list results
@type params_kwargs: dict
@param timeout: resource list timeout for over limit retries
@type timeout: int
@param use_over_limit_retry: flag to enable/disable the list
over limits retries
@type use_over_limit_retry: bool
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
# Defining the resource type in plural form (for ex. networks)
resource_type = resource.plural
poll_interval = poll_interval or self.config.api_poll_interval
resource_list_attempts = (resource_list_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_get_timeout
result = NetworkingResponse()
err_msg = '{0} list failure'.format(resource_type)
for attempt in range(resource_list_attempts):
self._log.debug(
'Attempt {attempt_n} of {attempts} with {resource_type} '
'list'.format(attempt_n=attempt + 1,
attempts=resource_list_attempts,
resource_type=resource_type))
list_fn_name = 'list_{0}'.format(resource_type)
resp = getattr(self.client, list_fn_name)(**params_kwargs)
if use_over_limit_retry:
entity_too_large_status_code = (getattr(self.response_codes,
'REQUEST_ENTITY_TOO_LARGE'))
if resp.status_code == entity_too_large_status_code:
fn_kwargs = params_kwargs
resp = self.__over_limit_retry(
resource_type=resource_type, timeout=timeout,
poll_interval=poll_interval,
status_code=entity_too_large_status_code,
resp=resp, fn_name=list_fn_name,
fn_kwargs=fn_kwargs)
response_code = list_fn_name.upper()
status_code = getattr(self.response_codes, response_code)
resp_check = self.check_response(
resp=resp, status_code=status_code, label='', message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the list was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to LIST {resource_type} after {attempts} attempts: '
'{failures}').format(resource_type=resource_type,
attempts=resource_list_attempts,
failures=result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceListException(err_msg)
return result
def _delete_resource(self, resource, resource_id,
resource_delete_attempts=None, raise_exception=False,
poll_interval=None, timeout=None,
use_over_limit_retry=False):
"""
@summary: Deletes and verifies a specified resource is deleted
@param resource: type of resource for ex. network, subnet, port, etc.
See NeutronResource in the networks constants
@type resource: resource instance with singular and plural forms
@param resource_id: The UUID for the resource
@type resource_id: string
@param resource_delete_attempts: number of API retries
@type resource_delete_attempts: int
@param raise_exception: flag to raise an exception if the deleted
resource was not as expected or to return None
@type raise_exception: bool
@param poll_interval: sleep time interval between API retries
@type poll_interval: int
@param timeout: resource delete timeout for over limit retries
@type timeout: int
@param use_over_limit_retry: flag to enable/disable the delete
over limits retries
@type use_over_limit_retry: bool
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
# Defining the resource type in singular form (for ex. network)
resource_type = resource.singular
poll_interval = poll_interval or self.config.api_poll_interval
resource_delete_attempts = (resource_delete_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_get_timeout
result = NetworkingResponse()
for attempt in range(resource_delete_attempts):
self._log.debug(
'Attempt {attempt_n} of {attempts} deleting {resource_type} '
'{resource_id}'.format(attempt_n=attempt + 1,
attempts=resource_delete_attempts,
resource_type=resource_type,
resource_id=resource_id))
# Method uses resource type in singular form (slicing the ending s)
delete_fn_name = 'delete_{0}'.format(resource_type)
resp = getattr(self.client, delete_fn_name)(resource_id)
if use_over_limit_retry:
entity_too_large_status_code = (getattr(self.response_codes,
'REQUEST_ENTITY_TOO_LARGE'))
if resp.status_code == entity_too_large_status_code:
fn_kwargs = {}
# Adding the resource id to the function kwargs
resource_id_name = '{0}_id'.format(resource_type)
fn_kwargs[resource_id_name] = resource_id
resp = self.__over_limit_retry(
resource_type=resource_type, timeout=timeout,
poll_interval=poll_interval,
status_code=entity_too_large_status_code,
resp=resp, fn_name=delete_fn_name,
fn_kwargs=fn_kwargs)
result.response = resp
# Delete response is without entity so resp_check can not be used
response_code = delete_fn_name.upper()
status_code = getattr(self.response_codes, response_code)
if resp.ok and resp.status_code == status_code:
return result
err_msg = ('{resource_id} {resource_type} Delete failure, expected'
'status code: {expected_status}. Response: {status} '
'{reason} {content}').format(
resource_id=resource_id,
resource_type=resource_type,
expected_status=status_code,
status=resp.status_code,
reason=resp.reason,
content=resp.content)
self._log.error(err_msg)
result.failures.append(err_msg)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to DELETE {resource_id} {resource_type} after '
'{attempts} attempts: {failures}').format(
resource_id=resource_id, resource_type=resource_type,
attempts=resource_delete_attempts,
failures=result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceDeleteException(err_msg)
return result
def _delete_resources(self, resource, resource_list=None, name=None,
tenant_id=None, skip_delete=None):
"""
@summary: deletes multiple resources (for ex. networks)
@param resource: type of resource for ex. network, subnet, port, etc.
See NeutronResource in the networks constants
@type resource: resource instance with singular and plural forms
@param resource_list: list of resource UUIDs
@type resource_list: list(str)
@param name: resource name to filter by, asterisk can be used at the
@ -175,18 +700,23 @@ class NetworkingBaseBehaviors(BaseBehavior):
@return: failed delete list with resource UUIDs and failures
@rtype: list(dict)
"""
# Defining the resource type in singular and plural forms
resource_type_singular = resource.singular
resource_type_plural = resource.plural
# Getting the resource list based on the resource type (if not given)
if resource_list is None:
list_fn_name = 'list_{0}'.format(resource_type)
list_fn_name = 'list_{0}'.format(resource_type_plural)
resp = getattr(self, list_fn_name)(tenant_id=tenant_id)
# Getting the Neutron expected response based on the fn name
response_code = list_fn_name.upper()
status_code = getattr(NeutronResponseCodes, response_code)
status_code = getattr(self.response_codes, response_code)
if resp.response.status_code != status_code:
get_msg = 'Unable to get {0} for delete_{0} call'.format(
resource_type)
resource_type_plural)
self._log.info(get_msg)
return None
resources = resp.response.entity
@ -205,7 +735,7 @@ class NetworkingBaseBehaviors(BaseBehavior):
if skip_delete is not None:
do_not_delete.extend(skip_delete)
if resource_type == NeutronResourceTypes.NETWORKS:
if resource_type_plural == NeutronResource.NETWORKS:
property_list = ['public_network_id', 'service_network_id']
for prop in property_list:
if (hasattr(self.config, prop) and
@ -217,16 +747,114 @@ class NetworkingBaseBehaviors(BaseBehavior):
if resource_to_skip in resource_list:
resource_list.remove(resource_to_skip)
log_msg = 'Deleting {0}: {1}'.format(resource_type, resource_list)
log_msg = 'Deleting {0}: {1}'.format(resource_type_plural,
resource_list)
self._log.info(log_msg)
failed_deletes = []
delete_fn_name = 'delete_{0}'.format(resource_type[:-1])
delete_fn_name = 'delete_{0}'.format(resource_type_singular)
for resource_id in resource_list:
result = getattr(self, delete_fn_name)(resource_id)
if result.failures:
failed_deletes.append(result.failures)
return failed_deletes
def _clean_resource(self, resource, resource_id, timeout=None,
poll_interval=None):
"""
@summary: deletes a resource within a time out
@param resource: type of resource for ex. network, subnet, port, etc.
See NeutronResource in the networks constants
@type resource: resource instance with singular and plural forms
@param resource_id: The UUID for the for the resource
@type resource_id: str
@param timeout: seconds to wait for the resource to be deleted
@type timeout: int
@param poll_interval: sleep time interval between API delete/get calls
@type poll_interval: int
@return: None if delete was successful or the undeleted resource_id
@rtype: None or string
"""
# Defining the resource type in singular form (for ex. network)
resource_type = resource.singular
timeout = timeout or self.config.resource_delete_timeout
poll_interval = poll_interval or self.config.api_poll_interval
endtime = time.time() + int(timeout)
log_msg = ('Deleting {resource_id} {resource_type} within a {timeout}s'
' timeout').format(resource_id=resource_id,
resource_type=resource_type,
timeout=timeout)
self._log.info(log_msg)
# Method uses resource type in singular form (slicing the ending s)
delete_fn_name = 'delete_{0}'.format(resource_type)
get_fn_name = 'get_{0}'.format(resource_type)
resp = None
while time.time() < endtime:
try:
getattr(self.client, delete_fn_name)(resource_id)
resp = getattr(self.client, get_fn_name)(resource_id)
except Exception as err:
err_msg = ('Encountered an exception deleting a '
'{resource_type} within the _clean_resource method.'
' Exception: {error}').format(
resource_type=resource_type, error=err)
self._log.error(err_msg)
if (resp is not None and
resp.status_code == NeutronResponseCodes.NOT_FOUND):
return None
time.sleep(poll_interval)
err_msg = ('Unable to delete {resource_id} {resource_type} within a '
'{timeout}s timeout').format(resource_id=resource_id,
resource_type=resource_type,
timeout=timeout)
self._log.error(err_msg)
return resource_id
def _clean_resources(self, resource, resource_list, timeout=None,
poll_interval=None):
"""
@summary: deletes each resource from a list calling _clean_resource
@param resource: type of resource for ex. network, subnet, port, etc.
See NeutronResource in the networks constants
@type resource: resource instance with singular and plural forms
@param resource_list: list of resource UUIDs to delete
@type resource_list: list(str)
@param timeout: seconds to wait for the resource to be deleted
@type timeout: int
@param poll_interval: sleep time interval between API delete/get calls
@type poll_interval: int
@return: list of undeleted resource UUIDs
@rtype: list(str)
"""
# Defining the resource type in plural form (for ex. networks)
resource_type_plural = resource.plural
log_msg = 'Deleting {resource_type}: {resource_list}'.format(
resource_type=resource_type_plural, resource_list=resource_list)
self._log.info(log_msg)
undeleted_resources = []
for resource_id in resource_list:
# _cleanup_resource takes the resource obj
result = self._clean_resource(resource=resource,
resource_id=resource_id,
timeout=timeout,
poll_interval=poll_interval)
if result:
undeleted_resources.append(result)
if undeleted_resources:
err_msg = ('Unable to delete {resource_type}: '
'{undeleted_resources}').format(
resource_type=resource_type_plural,
undeleted_resources=undeleted_resources)
self._log.error(err_msg)
return undeleted_resources
class NetworkingResponse(object):
"""

View File

@ -60,6 +60,11 @@ class NetworkingBaseConfig(ConfigSectionInterface):
"""Number of times to retry an API call by a behavior method"""
return int(self.get("api_retries", 1))
@property
def use_over_limit_retry(self):
"""Flag to enable/disable retries due to over limits responses"""
return self.get_boolean("use_over_limit_retry", False)
@property
def resource_create_timeout(self):
"""Seconds to wait for creating a resource"""

View File

@ -15,13 +15,25 @@ limitations under the License.
"""
class NeutronResourceTypes(object):
class NeutronResource(object):
"""Neutron resource types"""
NETWORK = 'network'
NETWORKS = 'networks'
SUBNET = 'subnet'
SUBNETS = 'subnets'
PORT = 'port'
PORTS = 'ports'
PLURALS = {NETWORK: NETWORKS, SUBNET: SUBNETS, PORT: PORTS}
def __init__(self, singular_type):
self.singular = singular_type
@property
def plural(self):
return self.PLURALS.get(self.singular, self.singular)
class NeutronResponseCodes(object):
"""HTTP Neutron API Response codes"""

View File

@ -14,16 +14,10 @@ See the License for the specific language governing permissions and
limitations under the License.
"""
import time
from cloudcafe.common.tools.datagen import rand_name
from cloudcafe.networking.networks.common.behaviors \
import NetworkingBaseBehaviors, NetworkingResponse
import NetworkingBaseBehaviors
from cloudcafe.networking.networks.common.constants \
import NeutronResourceTypes, NeutronResponseCodes
from cloudcafe.networking.networks.common.exceptions \
import ResourceBuildException, ResourceDeleteException,\
ResourceGetException, ResourceListException, ResourceUpdateException
import NeutronResource, NeutronResponseCodes
class NetworksBehaviors(NetworkingBaseBehaviors):
@ -32,6 +26,8 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
super(NetworksBehaviors, self).__init__()
self.config = networks_config
self.client = networks_client
self.response_codes = NeutronResponseCodes
self.networks_resource = NeutronResource(NeutronResource.NETWORK)
def create_network(self, name=None, admin_state_up=None, shared=None,
tenant_id=None, resource_build_attempts=None,
@ -59,46 +55,16 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
if name is None:
name = rand_name(self.config.starts_with_name)
elif not use_exact_name:
name = rand_name(name)
attrs_kwargs = dict(name=name, admin_state_up=admin_state_up,
shared=shared, tenant_id=tenant_id)
poll_interval = poll_interval or self.config.api_poll_interval
resource_build_attempts = (resource_build_attempts or
self.config.api_retries)
result = self._create_resource(
resource=self.networks_resource,
resource_build_attempts=resource_build_attempts,
raise_exception=raise_exception, use_exact_name=use_exact_name,
poll_interval=poll_interval, attrs_kwargs=attrs_kwargs)
result = NetworkingResponse()
err_msg = 'Network Create failure'
for attempt in range(resource_build_attempts):
self._log.debug('Attempt {0} of {1} building network {2}'.format(
attempt + 1, resource_build_attempts, name))
resp = self.client.create_network(
name=name, admin_state_up=admin_state_up, shared=shared,
tenant_id=tenant_id)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.CREATE_NETWORK, label=name,
message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the create was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to create {0} network after {1} attempts: '
'{2}').format(name, resource_build_attempts, result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceBuildException(err_msg)
return result
return result
def update_network(self, network_id, name=None, admin_state_up=None,
shared=None, tenant_id=None,
@ -130,43 +96,17 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_update_attempts = (resource_update_attempts or
self.config.api_retries)
attrs_kwargs = dict(name=name, admin_state_up=admin_state_up,
shared=shared, tenant_id=tenant_id)
result = NetworkingResponse()
err_msg = 'Network Update failure'
for attempt in range(resource_update_attempts):
self._log.debug('Attempt {0} of {1} updating network {2}'.format(
attempt + 1, resource_update_attempts, network_id))
result = self._update_resource(
resource=self.networks_resource,
resource_id=network_id,
resource_update_attempts=resource_update_attempts,
raise_exception=raise_exception, poll_interval=poll_interval,
attrs_kwargs=attrs_kwargs)
resp = self.client.update_network(
network_id=network_id, name=name,
admin_state_up=admin_state_up, shared=shared,
tenant_id=tenant_id)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.UPDATE_NETWORK,
label=network_id, message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the update was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to update {0} network after {1} attempts: '
'{2}').format(network_id, resource_update_attempts,
result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceUpdateException(err_msg)
return result
return result
def get_network(self, network_id, resource_get_attempts=None,
raise_exception=False, poll_interval=None):
@ -184,40 +124,13 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_get_attempts = (resource_get_attempts or
self.config.api_retries)
result = self._get_resource(
resource=self.networks_resource,
resource_id=network_id,
resource_get_attempts=resource_get_attempts,
raise_exception=raise_exception, poll_interval=poll_interval)
result = NetworkingResponse()
err_msg = 'Network Get failure'
for attempt in range(resource_get_attempts):
self._log.debug('Attempt {0} of {1} getting network {2}'.format(
attempt + 1, resource_get_attempts, network_id))
resp = self.client.get_network(network_id=network_id)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.GET_NETWORK,
label=network_id, message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the get was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to GET {0} network after {1} attempts: '
'{2}').format(network_id, resource_get_attempts,
result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceGetException(err_msg)
return result
return result
def list_networks(self, network_id=None, name=None, status=None,
admin_state_up=None, shared=None, tenant_id=None,
@ -254,43 +167,18 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_list_attempts = (resource_list_attempts or
self.config.api_retries)
params_kwargs = dict(network_id=network_id, name=name, status=status,
admin_state_up=admin_state_up, shared=shared,
tenant_id=tenant_id, limit=limit, marker=marker,
page_reverse=page_reverse)
result = NetworkingResponse()
err_msg = 'Network List failure'
for attempt in range(resource_list_attempts):
self._log.debug('Attempt {0} of {1} with network list'.format(
attempt + 1, resource_list_attempts))
result = self._list_resources(
resource=self.networks_resource,
resource_list_attempts=resource_list_attempts,
raise_exception=raise_exception, poll_interval=poll_interval,
params_kwargs=params_kwargs)
resp = self.client.list_networks(
network_id=network_id, name=name, status=status,
admin_state_up=admin_state_up, shared=shared,
tenant_id=tenant_id, limit=limit, marker=marker,
page_reverse=page_reverse)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.LIST_NETWORKS,
label='', message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the list was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to LIST networks after {0} attempts: '
'{1}').format(resource_list_attempts, result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceListException(err_msg)
return result
return result
def delete_network(self, network_id, resource_delete_attempts=None,
raise_exception=False, poll_interval=None):
@ -308,43 +196,13 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_delete_attempts = (resource_delete_attempts or
self.config.api_retries)
result = self._delete_resource(
resource=self.networks_resource,
resource_id=network_id,
resource_delete_attempts=resource_delete_attempts,
raise_exception=raise_exception, poll_interval=poll_interval)
result = NetworkingResponse()
for attempt in range(resource_delete_attempts):
self._log.debug('Attempt {0} of {1} deleting network {2}'.format(
attempt + 1, resource_delete_attempts, network_id))
resp = self.client.delete_network(network_id=network_id)
result.response = resp
# Delete response is without entity so resp_check can not be used
if (resp.ok and
resp.status_code == NeutronResponseCodes.DELETE_NETWORK):
return result
err_msg = ('{network} Network Delete failure, expected status '
'code: {expected_status}. Response: {status} {reason} '
'{content}').format(
network=network_id,
expected_status=NeutronResponseCodes.DELETE_NETWORK,
status=resp.status_code, reason=resp.reason,
content=resp.content)
self._log.error(err_msg)
result.failures.append(err_msg)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to DELETE {0} network after {1} attempts: '
'{2}').format(network_id, resource_delete_attempts,
result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceDeleteException(err_msg)
return result
return result
def delete_networks(self, network_list=None, name=None, tenant_id=None,
skip_delete=None):
@ -366,7 +224,8 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
result = self._delete_resources(
resource_list=network_list, name=name,
tenant_id=tenant_id, skip_delete=skip_delete,
resource_type=NeutronResourceTypes.NETWORKS)
resource=self.networks_resource)
return result
def clean_network(self, network_id, timeout=None, poll_interval=None):
@ -381,31 +240,12 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
@return: None if delete was successful or the undeleted network_id
@rtype: None or string
"""
timeout = timeout or self.config.resource_delete_timeout
poll_interval = poll_interval or self.config.api_poll_interval
endtime = time.time() + int(timeout)
log_msg = 'Deleting {0} network within a {1}s timeout '.format(
network_id, timeout)
self._log.info(log_msg)
resp = None
while time.time() < endtime:
try:
self.client.delete_network(network_id=network_id)
resp = self.client.get_network(network_id=network_id)
except Exception as err:
err_msg = ('Encountered an exception deleting a network with'
'the clean_network method. Exception: {0}').format(err)
self._log.error(err_msg)
result = self._clean_resource(
resource=self.networks_resource,
resource_id=network_id,
timeout=timeout, poll_interval=poll_interval)
if (resp is not None and
resp.status_code == NeutronResponseCodes.NOT_FOUND):
return None
time.sleep(poll_interval)
err_msg = 'Unable to delete {0} network within a {1}s timeout'.format(
network_id, timeout)
self._log.error(err_msg)
return network_id
return result
def clean_networks(self, networks_list, timeout=None, poll_interval=None):
"""
@ -419,16 +259,9 @@ class NetworksBehaviors(NetworkingBaseBehaviors):
@return: list of undeleted networks UUIDs
@rtype: list(str)
"""
log_msg = 'Deleting networks: {0}'.format(networks_list)
self._log.info(log_msg)
undeleted_networks = []
for network in networks_list:
result = self.clean_network(network_id=network, timeout=timeout,
poll_interval=poll_interval)
if result:
undeleted_networks.append(result)
if undeleted_networks:
err_msg = 'Unable to delete networks: {0}'.format(
undeleted_networks)
self._log.error(err_msg)
return undeleted_networks
result = self._clean_resources(
resource=self.networks_resource,
resource_list=networks_list,
timeout=timeout, poll_interval=poll_interval)
return result

View File

@ -16,15 +16,12 @@ limitations under the License.
import netaddr
import time
from cloudcafe.common.tools.datagen import rand_name
from cloudcafe.networking.networks.common.behaviors \
import NetworkingBaseBehaviors, NetworkingResponse
import NetworkingBaseBehaviors
from cloudcafe.networking.networks.common.constants \
import NeutronResponseCodes, NeutronResourceTypes
import NeutronResponseCodes, NeutronResource
from cloudcafe.networking.networks.common.exceptions \
import NetworkIDMissingException, ResourceBuildException,\
ResourceDeleteException, ResourceGetException, ResourceListException,\
ResourceUpdateException
import NetworkIDMissingException
class PortsBehaviors(NetworkingBaseBehaviors):
@ -33,6 +30,8 @@ class PortsBehaviors(NetworkingBaseBehaviors):
super(PortsBehaviors, self).__init__()
self.config = ports_config
self.client = ports_client
self.response_codes = NeutronResponseCodes
self.ports_resource = NeutronResource(NeutronResource.PORT)
def get_subnet_ids_from_fixed_ips(self, fixed_ips):
"""
@ -70,7 +69,7 @@ class PortsBehaviors(NetworkingBaseBehaviors):
device_owner=None, tenant_id=None, security_groups=None,
resource_build_attempts=None, raise_exception=True,
use_exact_name=False, poll_interval=None,
timeout=None, use_over_limit_retry=None):
timeout=None, use_over_limit_retry=False):
"""
@summary: Creates and verifies a Port is created as expected
@param network_id: network port is associated with (CRUD: CR)
@ -104,9 +103,9 @@ class PortsBehaviors(NetworkingBaseBehaviors):
@type use_exact_name: bool
@param poll_interval: sleep time interval between API retries
@type poll_interval: int
@param timeout: port update timeout for over limit retries
@param timeout: port create timeout for over limit retries
@type timeout: int
@param use_over_limit_retry: flag to enable/disable the port update
@param use_over_limit_retry: flag to enable/disable the port create
over limits retries
@type use_over_limit_retry: bool
@return: NetworkingResponse object with api response and failure list
@ -115,74 +114,27 @@ class PortsBehaviors(NetworkingBaseBehaviors):
if not network_id:
raise NetworkIDMissingException
if name is None:
name = rand_name(self.config.starts_with_name)
elif not use_exact_name:
name = rand_name(name)
attrs_kwargs = dict(
network_id=network_id, name=name,
admin_state_up=admin_state_up, mac_address=mac_address,
fixed_ips=fixed_ips, device_id=device_id,
device_owner=device_owner, tenant_id=tenant_id,
security_groups=security_groups)
poll_interval = poll_interval or self.config.api_poll_interval
resource_build_attempts = (resource_build_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_create_timeout
result = self._create_resource(
resource=self.ports_resource,
resource_build_attempts=resource_build_attempts,
raise_exception=raise_exception, use_exact_name=use_exact_name,
poll_interval=poll_interval, attrs_kwargs=attrs_kwargs,
timeout=timeout, use_over_limit_retry=use_over_limit_retry)
result = NetworkingResponse()
err_msg = 'Port Create failure'
for attempt in range(resource_build_attempts):
self._log.debug('Attempt {0} of {1} building port {2}'.format(
attempt + 1, resource_build_attempts, name))
resp = self.client.create_port(
network_id=network_id, name=name,
admin_state_up=admin_state_up, mac_address=mac_address,
fixed_ips=fixed_ips, device_id=device_id,
device_owner=device_owner, tenant_id=tenant_id,
security_groups=security_groups)
if use_over_limit_retry:
endtime = time.time() + int(timeout)
retry_msg = ('OverLimit retry with a {0}s timeout creating a '
'port on network {1}').format(timeout, network_id)
self._log.info(retry_msg)
while (resp.status_code ==
NeutronResponseCodes.REQUEST_ENTITY_TOO_LARGE and
time.time() < endtime):
resp = self.client.create_port(
network_id=network_id, name=name,
admin_state_up=admin_state_up, mac_address=mac_address,
fixed_ips=fixed_ips, device_id=device_id,
device_owner=device_owner, tenant_id=tenant_id,
security_groups=security_groups)
time.sleep(poll_interval)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.CREATE_PORT, label=name,
message=err_msg, network_id=network_id)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the create was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to create {0} port after {1} attempts: '
'{2}').format(name, resource_build_attempts, result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceBuildException(err_msg)
return result
return result
def update_port(self, port_id, name=None, admin_state_up=None,
fixed_ips=None, device_id=None, device_owner=None,
security_groups=None, resource_update_attempts=None,
raise_exception=False, poll_interval=None,
timeout=None, use_over_limit_retry=None):
timeout=None, use_over_limit_retry=False):
"""
@summary: Updates and verifies a specified Port
@param port_id: The UUID for the port
@ -219,66 +171,24 @@ class PortsBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_update_attempts = (resource_update_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_update_timeout
result = NetworkingResponse()
err_msg = 'Port Update failure'
for attempt in range(resource_update_attempts):
self._log.debug('Attempt {0} of {1} updating port {2}'.format(
attempt + 1, resource_update_attempts, port_id))
resp = self.client.update_port(
port_id=port_id, name=name, admin_state_up=admin_state_up,
attrs_kwargs = dict(
name=name, admin_state_up=admin_state_up,
fixed_ips=fixed_ips, device_id=device_id,
device_owner=device_owner, security_groups=security_groups)
if use_over_limit_retry:
endtime = time.time() + int(timeout)
retry_msg = ('OverLimit retry with a {0}s timeout updating '
'port {1}').format(timeout, port_id)
self._log.info(retry_msg)
while (resp.status_code ==
NeutronResponseCodes.REQUEST_ENTITY_TOO_LARGE and
time.time() < endtime):
resp = self.client.update_port(
port_id=port_id, name=name,
admin_state_up=admin_state_up,
fixed_ips=fixed_ips, device_id=device_id,
device_owner=device_owner,
security_groups=security_groups)
time.sleep(poll_interval)
result = self._update_resource(
resource=self.ports_resource,
resource_id=port_id,
resource_update_attempts=resource_update_attempts,
raise_exception=raise_exception, poll_interval=poll_interval,
attrs_kwargs=attrs_kwargs, timeout=timeout,
use_over_limit_retry=use_over_limit_retry)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.UPDATE_PORT,
label=port_id, message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the update was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to update {0} port after {1} attempts: '
'{2}').format(port_id, resource_update_attempts,
result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceUpdateException(err_msg)
return result
return result
def get_port(self, port_id, resource_get_attempts=None,
raise_exception=False, poll_interval=None,
timeout=None, use_over_limit_retry=None):
timeout=None, use_over_limit_retry=False):
"""
@summary: Shows and verifies a specified port
@param port_id: The UUID for the port
@ -292,67 +202,27 @@ class PortsBehaviors(NetworkingBaseBehaviors):
@type poll_interval: int
@param timeout: port get timeout for over limit retries
@type timeout: int
@param use_over_limit_retry: flag to enable/disable the port update
@param use_over_limit_retry: flag to enable/disable the port get
over limits retries
@type use_over_limit_retry: bool
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_get_attempts = (resource_get_attempts or
self.config.api_retries)
poll_interval = poll_interval or self.config.api_poll_interval
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_get_timeout
result = self._get_resource(
resource=self.ports_resource,
resource_id=port_id,
resource_get_attempts=resource_get_attempts,
raise_exception=raise_exception, poll_interval=poll_interval,
timeout=timeout, use_over_limit_retry=use_over_limit_retry)
result = NetworkingResponse()
err_msg = 'Port Get failure'
for attempt in range(resource_get_attempts):
self._log.debug('Attempt {0} of {1} getting network {2}'.format(
attempt + 1, resource_get_attempts, port_id))
resp = self.client.get_port(port_id=port_id)
if use_over_limit_retry:
endtime = time.time() + int(timeout)
retry_msg = ('OverLimit retry with a {0}s timeout getting '
'port {1}').format(timeout, port_id)
self._log.info(retry_msg)
while (resp.status_code ==
NeutronResponseCodes.REQUEST_ENTITY_TOO_LARGE and
time.time() < endtime):
resp = self.client.get_port(port_id=port_id)
time.sleep(poll_interval)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.GET_PORT,
label=port_id, message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the get was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to GET {0} port after {1} attempts: '
'{2}').format(port_id, resource_get_attempts, result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceGetException(err_msg)
return result
return result
def list_ports(self, port_id=None, network_id=None, name=None, status=None,
admin_state_up=None, device_id=None, tenant_id=None,
device_owner=None, mac_address=None, limit=None,
marker=None, page_reverse=None, resource_list_attempts=None,
raise_exception=False, poll_interval=None, timeout=None,
use_over_limit_retry=None):
use_over_limit_retry=False):
"""
@summary: Lists ports and verifies the response is the expected
@param port_id: The UUID for the port to filter by
@ -394,67 +264,25 @@ class PortsBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_list_attempts = (resource_list_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_get_timeout
params_kwargs = dict(
port_id=port_id, network_id=network_id, name=name,
status=status, admin_state_up=admin_state_up,
device_id=device_id, tenant_id=tenant_id,
device_owner=device_owner, mac_address=mac_address,
limit=limit, marker=marker, page_reverse=page_reverse)
result = NetworkingResponse()
err_msg = 'Port List failure'
for attempt in range(resource_list_attempts):
self._log.debug('Attempt {0} of {1} with port list'.format(
attempt + 1, resource_list_attempts))
result = self._list_resources(
resource=self.ports_resource,
resource_list_attempts=resource_list_attempts,
raise_exception=raise_exception, poll_interval=poll_interval,
params_kwargs=params_kwargs, timeout=timeout,
use_over_limit_retry=use_over_limit_retry)
resp = self.client.list_ports(
port_id=port_id, network_id=network_id, name=name,
status=status, admin_state_up=admin_state_up,
device_id=device_id, tenant_id=tenant_id,
device_owner=device_owner, mac_address=mac_address,
limit=limit, marker=marker, page_reverse=page_reverse)
if use_over_limit_retry:
endtime = time.time() + int(timeout)
retry_msg = ('OverLimit retry with a {0}s timeout listing '
'ports').format(timeout, port_id)
self._log.info(retry_msg)
while (resp.status_code ==
NeutronResponseCodes.REQUEST_ENTITY_TOO_LARGE and
time.time() < endtime):
resp = self.client.list_ports(
port_id=port_id, network_id=network_id, name=name,
status=status, admin_state_up=admin_state_up,
device_id=device_id, tenant_id=tenant_id,
device_owner=device_owner, mac_address=mac_address,
limit=limit, marker=marker, page_reverse=page_reverse)
time.sleep(poll_interval)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.LIST_PORTS,
label='', message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the list was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to LIST ports after {0} attempts: '
'{1}').format(resource_list_attempts, result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceListException(err_msg)
return result
return result
def delete_port(self, port_id, resource_delete_attempts=None,
raise_exception=False, poll_interval=None,
timeout=None, use_over_limit_retry=None):
timeout=None, use_over_limit_retry=False):
"""
@summary: Deletes and verifies a specified port is deleted
@param string port_id: The UUID for the port
@ -474,58 +302,14 @@ class PortsBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_delete_attempts = (resource_delete_attempts or
self.config.api_retries)
use_over_limit_retry = (use_over_limit_retry or
self.config.use_over_limit_retry)
timeout = timeout or self.config.resource_delete_timeout
result = self._delete_resource(
resource=self.ports_resource,
resource_id=port_id,
resource_delete_attempts=resource_delete_attempts,
raise_exception=raise_exception, poll_interval=poll_interval,
timeout=timeout, use_over_limit_retry=use_over_limit_retry)
result = NetworkingResponse()
for attempt in range(resource_delete_attempts):
self._log.debug('Attempt {0} of {1} deleting port {2}'.format(
attempt + 1, resource_delete_attempts, port_id))
resp = self.client.delete_port(port_id=port_id)
if use_over_limit_retry:
endtime = time.time() + int(timeout)
retry_msg = ('OverLimit retry with a {0}s timeout deleting '
'port {1}').format(timeout, port_id)
self._log.info(retry_msg)
while (resp.status_code ==
NeutronResponseCodes.REQUEST_ENTITY_TOO_LARGE and
time.time() < endtime):
resp = self.client.delete_port(port_id=port_id)
time.sleep(poll_interval)
result.response = resp
# Delete response is without entity so resp_check can not be used
if (resp.ok and
resp.status_code == NeutronResponseCodes.DELETE_PORT):
return result
err_msg = ('{port} Port Delete failure, expected status '
'code: {expected_status}. Response: {status} {reason} '
'{content}').format(
port=port_id,
expected_status=NeutronResponseCodes.DELETE_PORT,
status=resp.status_code, reason=resp.reason,
content=resp.content)
self._log.error(err_msg)
result.failures.append(err_msg)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to DELETE {0} port after {1} attempts: '
'{2}').format(port_id, resource_delete_attempts,
result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceDeleteException(err_msg)
return result
return result
def delete_ports(self, port_list=None, name=None, tenant_id=None,
skip_delete=None):
@ -547,7 +331,7 @@ class PortsBehaviors(NetworkingBaseBehaviors):
result = self._delete_resources(
resource_list=port_list, name=name,
tenant_id=tenant_id, skip_delete=skip_delete,
resource_type=NeutronResourceTypes.PORTS)
resource=self.ports_resource)
return result
def clean_port(self, port_id, timeout=None, poll_interval=None):
@ -562,32 +346,12 @@ class PortsBehaviors(NetworkingBaseBehaviors):
@return: None if delete was successful or the undeleted port_id
@rtype: None or string
"""
timeout = timeout or self.config.resource_delete_timeout
poll_interval = poll_interval or self.config.api_poll_interval
result = self._clean_resource(
resource=self.ports_resource,
resource_id=port_id,
timeout=timeout, poll_interval=poll_interval)
endtime = time.time() + int(timeout)
log_msg = 'Deleting {0} port within a {1}s timeout '.format(
port_id, timeout)
self._log.info(log_msg)
resp = None
while time.time() < endtime:
try:
self.client.delete_port(port_id=port_id)
resp = self.client.get_port(port_id=port_id)
except Exception as err:
err_msg = ('Encountered an exception deleting a port with'
'the clean_network method. Exception: {0}').format(err)
self._log.error(err_msg)
if (resp is not None and
resp.status_code == NeutronResponseCodes.NOT_FOUND):
return None
time.sleep(poll_interval)
err_msg = 'Unable to delete {0} port within a {1}s timeout'.format(
port_id, timeout)
self._log.error(err_msg)
return port_id
return result
def clean_ports(self, ports_list, timeout=None, poll_interval=None):
"""
@ -601,16 +365,9 @@ class PortsBehaviors(NetworkingBaseBehaviors):
@return: list of undeleted ports UUIDs
@rtype: list(str)
"""
log_msg = 'Deleting ports: {0}'.format(ports_list)
self._log.info(log_msg)
undeleted_ports = []
for port in ports_list:
result = self.clean_port(port_id=port, timeout=timeout,
poll_interval=poll_interval)
if result:
undeleted_ports.append(result)
if undeleted_ports:
err_msg = 'Unable to delete ports: {0}'.format(
undeleted_ports)
self._log.error(err_msg)
return undeleted_ports
result = self._clean_resources(
resource=self.ports_resource,
resource_list=ports_list,
timeout=timeout, poll_interval=poll_interval)
return result

View File

@ -47,11 +47,6 @@ class PortsConfig(NetworkingBaseConfig):
"""Ports fixed IPs quota"""
return int(self.get("fixed_ips_per_port", 5))
@property
def use_over_limit_retry(self):
"""Flag to enable/disable retries due to over limits responses"""
return self.get_boolean("use_over_limit_retry", False)
@property
def api_poll_interval(self):
"""Time interval for api calls on retry loops"""

View File

@ -21,15 +21,13 @@ import IPy
import netaddr
from cloudcafe.common.tools.datagen import rand_name, random_cidr
from cloudcafe.common.tools.datagen import random_cidr
from cloudcafe.networking.networks.common.behaviors \
import NetworkingBaseBehaviors, NetworkingResponse
import NetworkingBaseBehaviors
from cloudcafe.networking.networks.common.constants \
import NeutronResponseCodes, NeutronResourceTypes
import NeutronResponseCodes, NeutronResource
from cloudcafe.networking.networks.common.exceptions \
import InvalidIPException, NetworkIDMissingException,\
ResourceBuildException, ResourceDeleteException, ResourceGetException,\
ResourceListException, ResourceUpdateException
import InvalidIPException, NetworkIDMissingException
class SubnetsBehaviors(NetworkingBaseBehaviors):
@ -38,6 +36,8 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
super(SubnetsBehaviors, self).__init__()
self.config = subnets_config
self.client = subnets_client
self.response_codes = NeutronResponseCodes
self.subnets_resource = NeutronResource(NeutronResource.SUBNET)
def verify_ip(self, ip_cidr, ip_range=None):
"""
@ -517,49 +517,20 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
ip_version = 4
cidr = self.create_ipv4_cidr()
if name is None:
name = rand_name(self.config.starts_with_name)
elif not use_exact_name:
name = rand_name(name)
attrs_kwargs = dict(
network_id=network_id, ip_version=ip_version, cidr=cidr,
name=name, tenant_id=tenant_id, gateway_ip=gateway_ip,
dns_nameservers=dns_nameservers,
allocation_pools=allocation_pools, host_routes=host_routes,
enable_dhcp=enable_dhcp)
poll_interval = poll_interval or self.config.api_poll_interval
resource_build_attempts = (resource_build_attempts or
self.config.api_retries)
result = self._create_resource(
resource=self.subnets_resource,
resource_build_attempts=resource_build_attempts,
raise_exception=raise_exception, use_exact_name=use_exact_name,
poll_interval=poll_interval, attrs_kwargs=attrs_kwargs)
result = NetworkingResponse()
err_msg = 'Subnet Create failure'
for attempt in range(resource_build_attempts):
self._log.debug('Attempt {0} of {1} building subnet {2}'.format(
attempt + 1, resource_build_attempts, name))
resp = self.client.create_subnet(
network_id=network_id, ip_version=ip_version, cidr=cidr,
name=name, tenant_id=tenant_id, gateway_ip=gateway_ip,
dns_nameservers=dns_nameservers,
allocation_pools=allocation_pools, host_routes=host_routes,
enable_dhcp=enable_dhcp)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.CREATE_SUBNET, label=name,
message=err_msg, network_id=network_id)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the update was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to create {0} subnet after {1} attempts: '
'{2}').format(name, resource_build_attempts, result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceBuildException(err_msg)
return result
return result
def update_subnet(self, subnet_id, name=None, gateway_ip=None,
dns_nameservers=None, host_routes=None,
@ -597,43 +568,19 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_update_attempts = (resource_update_attempts or
self.config.api_retries)
attrs_kwargs = dict(
name=name, gateway_ip=gateway_ip,
dns_nameservers=dns_nameservers, host_routes=host_routes,
enable_dhcp=enable_dhcp, allocation_pools=allocation_pools)
result = NetworkingResponse()
err_msg = 'Subnet Update failure'
for attempt in range(resource_update_attempts):
self._log.debug('Attempt {0} of {1} updating subnet {2}'.format(
attempt + 1, resource_update_attempts, subnet_id))
result = self._update_resource(
resource=self.subnets_resource,
resource_id=subnet_id,
resource_update_attempts=resource_update_attempts,
raise_exception=raise_exception, poll_interval=poll_interval,
attrs_kwargs=attrs_kwargs)
resp = self.client.update_subnet(
subnet_id=subnet_id, name=name, gateway_ip=gateway_ip,
dns_nameservers=dns_nameservers, host_routes=host_routes,
enable_dhcp=enable_dhcp, allocation_pools=allocation_pools)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.UPDATE_SUBNET,
label=subnet_id, message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the update was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to update {0} subnet after {1} attempts: '
'{2}').format(subnet_id, resource_update_attempts,
result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceUpdateException(err_msg)
return result
return result
def get_subnet(self, subnet_id, resource_get_attempts=None,
raise_exception=False, poll_interval=None):
@ -651,40 +598,13 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_get_attempts = (resource_get_attempts or
self.config.api_retries)
result = self._get_resource(
resource=self.subnets_resource,
resource_id=subnet_id,
resource_get_attempts=resource_get_attempts,
raise_exception=raise_exception, poll_interval=poll_interval)
result = NetworkingResponse()
err_msg = 'Subnet Get failure'
for attempt in range(resource_get_attempts):
self._log.debug('Attempt {0} of {1} getting subnet {2}'.format(
attempt + 1, resource_get_attempts, subnet_id))
resp = self.client.get_subnet(subnet_id=subnet_id)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.GET_SUBNET,
label=subnet_id, message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the get was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to GET {0} subnet after {1} attempts: '
'{2}').format(subnet_id, resource_get_attempts,
result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceGetException(err_msg)
return result
return result
def list_subnets(self, subnet_id=None, network_id=None, cidr=None,
tenant_id=None, gateway_ip=None, ip_version=None,
@ -725,43 +645,19 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_list_attempts = (resource_list_attempts or
self.config.api_retries)
params_kwargs = dict(
subnet_id=subnet_id, network_id=network_id, cidr=cidr,
tenant_id=tenant_id, gateway_ip=gateway_ip,
ip_version=ip_version, enable_dhcp=enable_dhcp, name=name,
limit=limit, marker=marker, page_reverse=page_reverse)
result = NetworkingResponse()
err_msg = 'Subnet List failure'
for attempt in range(resource_list_attempts):
self._log.debug('Attempt {0} of {1} with subnet list'.format(
attempt + 1, resource_list_attempts))
result = self._list_resources(
resource=self.subnets_resource,
resource_list_attempts=resource_list_attempts,
raise_exception=raise_exception, poll_interval=poll_interval,
params_kwargs=params_kwargs)
resp = self.client.list_subnets(
subnet_id=subnet_id, network_id=network_id, cidr=cidr,
tenant_id=tenant_id, gateway_ip=gateway_ip,
ip_version=ip_version, enable_dhcp=enable_dhcp, name=name,
limit=limit, marker=marker, page_reverse=page_reverse)
resp_check = self.check_response(resp=resp,
status_code=NeutronResponseCodes.LIST_SUBNETS,
label='', message=err_msg)
result.response = resp
if not resp_check:
return result
# Failures will be an empty list if the list was successful the
# first time
result.failures.append(resp_check)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to LIST subnets after {0} attempts: '
'{1}').format(resource_list_attempts, result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceListException(err_msg)
return result
return result
def delete_subnet(self, subnet_id, resource_delete_attempts=None,
raise_exception=False, poll_interval=None):
@ -779,43 +675,13 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
@return: NetworkingResponse object with api response and failure list
@rtype: common.behaviors.NetworkingResponse
"""
poll_interval = poll_interval or self.config.api_poll_interval
resource_delete_attempts = (resource_delete_attempts or
self.config.api_retries)
result = self._delete_resource(
resource=self.subnets_resource,
resource_id=subnet_id,
resource_delete_attempts=resource_delete_attempts,
raise_exception=raise_exception, poll_interval=poll_interval)
result = NetworkingResponse()
for attempt in range(resource_delete_attempts):
self._log.debug('Attempt {0} of {1} deleting subnet {2}'.format(
attempt + 1, resource_delete_attempts, subnet_id))
resp = self.client.delete_subnet(subnet_id=subnet_id)
result.response = resp
# Delete response is without entity so resp_check can not be used
if (resp.ok and
resp.status_code == NeutronResponseCodes.DELETE_SUBNET):
return result
err_msg = ('{subnet} Subnet Delete failure, expected status '
'code: {expected_status}. Response: {status} {reason} '
'{content}').format(
subnet=subnet_id,
expected_status=NeutronResponseCodes.DELETE_SUBNET,
status=resp.status_code, reason=resp.reason,
content=resp.content)
self._log.error(err_msg)
result.failures.append(err_msg)
time.sleep(poll_interval)
else:
err_msg = (
'Unable to DELETE {0} subnet after {1} attempts: '
'{2}').format(subnet_id, resource_delete_attempts,
result.failures)
self._log.error(err_msg)
if raise_exception:
raise ResourceDeleteException(err_msg)
return result
return result
def delete_subnets(self, subnet_list=None, name=None, tenant_id=None,
skip_delete=None):
@ -837,7 +703,8 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
result = self._delete_resources(
resource_list=subnet_list, name=name,
tenant_id=tenant_id, skip_delete=skip_delete,
resource_type=NeutronResourceTypes.SUBNETS)
resource=self.subnets_resource)
return result
def clean_subnet(self, subnet_id, timeout=None, poll_interval=None):
@ -852,30 +719,12 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
@return: None if delete was successful or the undeleted subnet_id
@rtype: None or string
"""
timeout = timeout or self.config.resource_delete_timeout
poll_interval = poll_interval or self.config.api_poll_interval
endtime = time.time() + int(timeout)
log_msg = 'Deleting {0} subnet within a {1}s timeout '.format(
subnet_id, timeout)
self._log.info(log_msg)
resp = None
while time.time() < endtime:
try:
self.client.delete_subnet(subnet_id=subnet_id)
resp = self.client.get_subnet(subnet_id=subnet_id)
except Exception as err:
err_msg = ('Encountered an exception deleting a subnet with'
'the clean_subnet method. Exception: {0}').format(err)
self._log.error(err_msg)
if (resp is not None and
resp.status_code == NeutronResponseCodes.NOT_FOUND):
return None
time.sleep(poll_interval)
result = self._clean_resource(
resource=self.subnets_resource,
resource_id=subnet_id,
timeout=timeout, poll_interval=poll_interval)
err_msg = 'Unable to delete {0} subnet within a {1}s timeout'.format(
subnet_id, timeout)
self._log.error(err_msg)
return subnet_id
return result
def clean_subnets(self, subnets_list, timeout=None, poll_interval=None):
"""
@ -889,16 +738,9 @@ class SubnetsBehaviors(NetworkingBaseBehaviors):
@return: list of undeleted subnets UUIDs
@rtype: list(str)
"""
log_msg = 'Deleting subnets: {0}'.format(subnets_list)
self._log.info(log_msg)
undeleted_subnets = []
for subnet in subnets_list:
result = self.clean_subnet(subnet_id=subnet, timeout=timeout,
poll_interval=poll_interval)
if result:
undeleted_subnets.append(result)
if undeleted_subnets:
err_msg = 'Unable to delete subnets: {0}'.format(
undeleted_subnets)
self._log.error(err_msg)
return undeleted_subnets
result = self._clean_resources(
resource=self.subnets_resource,
resource_list=subnets_list,
timeout=timeout, poll_interval=poll_interval)
return result