Add validator for fixed_subnet

Validate the existance of the subnet referenced by fixed_subnet. It's
not checked if the subnet is associated to the fixed_network.

Closes-Bug: #2038109
Change-Id: Ia75f0ae525b768ad5b965d22b522cca6f80dcab2
This commit is contained in:
Felipe Reyes 2023-10-05 18:45:55 -03:00
parent 537e69aeb8
commit a8bce0bfee
2 changed files with 110 additions and 6 deletions

@ -118,6 +118,28 @@ def validate_fixed_network(cli, fixed_network):
return network_id return network_id
def validate_fixed_subnet(cli, fixed_subnet):
"""Validate fixed subnet"""
count = 0
subnet_id = None
subnets = cli.neutron().list_subnets()
for subnet in subnets.get('subnets'):
if fixed_subnet in [subnet.get('name'), subnet.get('id')]:
count += 1
subnet_id = subnet.get('id')
if count == 0:
# Unable to find the configured fixed_subnet.
raise exception.FixedSubnetNotFound(subnet=fixed_subnet)
elif count > 1:
msg = _("Multiple subnets exist with same name '%s'. "
"Please use the subnet ID instead.")
raise exception.Conflict(msg % fixed_subnet)
else:
return subnet_id
def validate_labels(labels): def validate_labels(labels):
""""Validate labels""" """"Validate labels"""
@ -148,15 +170,15 @@ def validate_os_resources(context, cluster_template, cluster=None):
for attr, validate_method in validators.items(): for attr, validate_method in validators.items():
if cluster and attr in cluster and cluster[attr]: if cluster and attr in cluster and cluster[attr]:
if attr != 'labels': if attr == 'labels':
validate_method(cli, cluster[attr])
else:
validate_method(cluster[attr]) validate_method(cluster[attr])
elif attr in cluster_template and cluster_template[attr] is not None:
if attr != 'labels':
validate_method(cli, cluster_template[attr])
else: else:
validate_method(cli, cluster[attr])
elif attr in cluster_template and cluster_template[attr] is not None:
if attr == 'labels':
validate_method(cluster_template[attr]) validate_method(cluster_template[attr])
else:
validate_method(cli, cluster_template[attr])
if cluster: if cluster:
validate_keypair(cli, cluster['keypair']) validate_keypair(cli, cluster['keypair'])
@ -202,6 +224,7 @@ validators = {'image_id': validate_image,
'master_flavor_id': validate_flavor, 'master_flavor_id': validate_flavor,
'external_network_id': validate_external_network, 'external_network_id': validate_external_network,
'fixed_network': validate_fixed_network, 'fixed_network': validate_fixed_network,
'fixed_subnet': validate_fixed_subnet,
'labels': validate_labels} 'labels': validate_labels}
labels_validators = {'swarm_strategy': validate_labels_strategy} labels_validators = {'swarm_strategy': validate_labels_strategy}

@ -132,6 +132,46 @@ class TestAttrValidator(base.BaseTestCase):
attr_validator.validate_fixed_network, attr_validator.validate_fixed_network,
mock_os_cli, 'test_net') mock_os_cli, 'test_net')
def test_validate_fixed_subnet_with_valid_subnet(self):
mock_neutron = mock.MagicMock()
mock_subnets = {'subnets': [{'name': 'test_subnet',
'id': 'test_subnet_id',
'network_id': 'test_net_id'}]}
mock_neutron.list_subnets.return_value = mock_subnets
mock_os_cli = mock.MagicMock()
mock_os_cli.neutron.return_value = mock_neutron
self.assertEqual('test_subnet_id',
attr_validator.validate_fixed_subnet(mock_os_cli,
'test_subnet'))
mock_neutron.list_subnets.assert_called_with()
def test_validate_fixed_subnet_with_invalid_subnet(self):
mock_neutron = mock.MagicMock()
mock_subnets = {'subnets': [{'name': 'test_subnet',
'id': 'test_subnet_id',
'network_id': 'test_net_id'}]}
mock_neutron.list_subnets.return_value = mock_subnets
mock_os_cli = mock.MagicMock()
mock_os_cli.neutron.return_value = mock_neutron
self.assertRaises(exception.FixedSubnetNotFound,
attr_validator.validate_fixed_subnet,
mock_os_cli, 'test_subnet_not_found')
def test_validate_fixed_subnet_with_multiple_valid_subnet(self):
mock_neutron = mock.MagicMock()
mock_subnets = {'subnets': [{'name': 'test_subnet',
'id': 'test_subnet_id',
'network_id': 'test_net_id'},
{'name': 'test_subnet',
'id': 'test_subnet_id2',
'network_id': 'test_net_id2'}]}
mock_neutron.list_subnets.return_value = mock_subnets
mock_os_cli = mock.MagicMock()
mock_os_cli.neutron.return_value = mock_neutron
self.assertRaises(exception.Conflict,
attr_validator.validate_fixed_subnet,
mock_os_cli, 'test_subnet')
def test_validate_keypair_with_no_keypair(self): def test_validate_keypair_with_no_keypair(self):
mock_keypair = mock.MagicMock() mock_keypair = mock.MagicMock()
mock_keypair.id = None mock_keypair.id = None
@ -289,6 +329,47 @@ class TestAttrValidator(base.BaseTestCase):
attr_validator.validate_os_resources(mock_context, attr_validator.validate_os_resources(mock_context,
mock_cluster_template) mock_cluster_template)
@mock.patch('magnum.common.clients.OpenStackClients')
def test_validate_os_resources_with_valid_fixed_subnet(self,
os_clients_klass):
mock_cluster_template = {'fixed_network': 'test_net',
'fixed_subnet': 'test_subnet'}
mock_context = mock.MagicMock()
mock_os_cli = mock.MagicMock()
os_clients_klass.return_value = mock_os_cli
mock_neutron = mock.MagicMock()
mock_networks = {'networks': [{'name': 'test_net',
'id': 'test_net_id'}]}
mock_neutron.list_networks.return_value = mock_networks
mock_subnets = {'subnets': [{'name': 'test_subnet',
'id': 'test_subnet_id',
'network_id': 'test_net_id'}]}
mock_neutron.list_subnets.return_value = mock_subnets
mock_os_cli.neutron.return_value = mock_neutron
attr_validator.validate_os_resources(mock_context,
mock_cluster_template)
@mock.patch('magnum.common.clients.OpenStackClients')
def test_validate_os_resources_with_invalid_fixed_subnet(self,
os_clients_klass):
mock_cluster_template = {'fixed_network': 'test_net',
'fixed_subnet': 'test_subnet2'}
mock_context = mock.MagicMock()
mock_os_cli = mock.MagicMock()
os_clients_klass.return_value = mock_os_cli
mock_neutron = mock.MagicMock()
mock_networks = {'networks': [{'name': 'test_net',
'id': 'test_net_id'}]}
mock_neutron.list_networks.return_value = mock_networks
mock_subnets = {'subnets': [{'name': 'test_subnet',
'id': 'test_subnet_id',
'network_id': 'test_net_id'}]}
mock_neutron.list_subnets.return_value = mock_subnets
mock_os_cli.neutron.return_value = mock_neutron
self.assertRaises(exception.FixedSubnetNotFound,
attr_validator.validate_os_resources, mock_context,
mock_cluster_template)
@mock.patch('magnum.common.clients.OpenStackClients') @mock.patch('magnum.common.clients.OpenStackClients')
def test_validate_os_resources_with_cluster(self, mock_os_cli): def test_validate_os_resources_with_cluster(self, mock_os_cli):
mock_cluster_template = {} mock_cluster_template = {}