Add features to support VTS
Change-Id: I425ffb5095646271c8bf408ee3df480f4b277747
This commit is contained in:
parent
ccca3bfabc
commit
6cfbc4d5e8
@ -147,13 +147,13 @@ class Compute(object):
|
|||||||
# and check that it gets into the ACTIVE state
|
# and check that it gets into the ACTIVE state
|
||||||
def create_server(self, vmname, image, flavor, key_name,
|
def create_server(self, vmname, image, flavor, key_name,
|
||||||
nic, sec_group, avail_zone=None, user_data=None,
|
nic, sec_group, avail_zone=None, user_data=None,
|
||||||
config_drive=None,
|
config_drive=None, files=None, retry_count=10):
|
||||||
retry_count=10):
|
|
||||||
|
|
||||||
if sec_group:
|
if sec_group:
|
||||||
security_groups = [sec_group.name]
|
security_groups = [sec_group.name]
|
||||||
else:
|
else:
|
||||||
security_groups = None
|
security_groups = None
|
||||||
|
|
||||||
# Also attach the created security group for the test
|
# Also attach the created security group for the test
|
||||||
instance = self.novaclient.servers.create(name=vmname,
|
instance = self.novaclient.servers.create(name=vmname,
|
||||||
image=image,
|
image=image,
|
||||||
@ -163,6 +163,7 @@ class Compute(object):
|
|||||||
availability_zone=avail_zone,
|
availability_zone=avail_zone,
|
||||||
userdata=user_data,
|
userdata=user_data,
|
||||||
config_drive=config_drive,
|
config_drive=config_drive,
|
||||||
|
files=files,
|
||||||
security_groups=security_groups)
|
security_groups=security_groups)
|
||||||
if not instance:
|
if not instance:
|
||||||
return None
|
return None
|
||||||
|
@ -17,7 +17,7 @@ import re
|
|||||||
|
|
||||||
from log import LOG
|
from log import LOG
|
||||||
import monitor
|
import monitor
|
||||||
from netaddr import IPAddress
|
import netaddr
|
||||||
import sshutils
|
import sshutils
|
||||||
|
|
||||||
|
|
||||||
@ -55,7 +55,8 @@ class Instance(object):
|
|||||||
self.gmond_port = int(config.gmond_svr_port)
|
self.gmond_port = int(config.gmond_svr_port)
|
||||||
else:
|
else:
|
||||||
self.gmond_port = 0
|
self.gmond_port = 0
|
||||||
self.config_drive = None
|
self.config_drive = config.config_drive
|
||||||
|
self.no_floatingip = config.no_floatingip
|
||||||
|
|
||||||
# Setup the ssh connectivity
|
# Setup the ssh connectivity
|
||||||
# this function is only used for native hosts
|
# this function is only used for native hosts
|
||||||
@ -70,6 +71,42 @@ class Instance(object):
|
|||||||
connect_retry_count=self.config.ssh_retry_count)
|
connect_retry_count=self.config.ssh_retry_count)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
def get_network_interface(self, port_info):
|
||||||
|
ip_address = port_info['fixed_ips'][0]['ip_address']
|
||||||
|
subnet_id = port_info['fixed_ips'][0]['subnet_id']
|
||||||
|
subnet_info = self.net.neutron_client.show_subnet(subnet_id)['subnet']
|
||||||
|
|
||||||
|
cidr = netaddr.IPNetwork(subnet_info['cidr'])
|
||||||
|
network_dict = {
|
||||||
|
'ip_address': ip_address,
|
||||||
|
'netmask': str(cidr.netmask),
|
||||||
|
'network': str(cidr.network),
|
||||||
|
'broadcast': str(cidr.broadcast),
|
||||||
|
'gateway': subnet_info['gateway_ip'],
|
||||||
|
'dns': ''
|
||||||
|
}
|
||||||
|
if subnet_info['dns_nameservers']:
|
||||||
|
dns_servers = ','.join(subnet_info['dns_nameservers'])
|
||||||
|
network_dict['dns'] = 'dns-nameservers %s\n' % dns_servers
|
||||||
|
|
||||||
|
retStr = (
|
||||||
|
'# The loopback network interface\n'
|
||||||
|
'auto lo\n'
|
||||||
|
'iface lo inet loopback\n'
|
||||||
|
'\n'
|
||||||
|
'# The primary network interface\n'
|
||||||
|
'auto eth0\n'
|
||||||
|
'iface eth0 inet static\n'
|
||||||
|
' address %(ip_address)s\n'
|
||||||
|
' netmask %(netmask)s\n'
|
||||||
|
' network %(network)s\n'
|
||||||
|
' broadcast %(broadcast)s\n'
|
||||||
|
' gateway %(gateway)s\n'
|
||||||
|
' %(dns)s'
|
||||||
|
) % network_dict
|
||||||
|
|
||||||
|
return retStr
|
||||||
|
|
||||||
# Create a new VM instance, associate a floating IP for ssh access
|
# Create a new VM instance, associate a floating IP for ssh access
|
||||||
# and extract internal network IP
|
# and extract internal network IP
|
||||||
# Retruns True if success, False otherwise
|
# Retruns True if success, False otherwise
|
||||||
@ -90,7 +127,7 @@ class Instance(object):
|
|||||||
else:
|
else:
|
||||||
user_data = None
|
user_data = None
|
||||||
|
|
||||||
if self.config.vnic_type:
|
if self.config.vnic_type or self.config.no_dhcp:
|
||||||
# create the VM by passing a port ID instead of a net ID
|
# create the VM by passing a port ID instead of a net ID
|
||||||
self.port = self.net.create_port(int_net['id'],
|
self.port = self.net.create_port(int_net['id'],
|
||||||
[sec_group.id],
|
[sec_group.id],
|
||||||
@ -103,6 +140,11 @@ class Instance(object):
|
|||||||
# create the VM by passing a net ID
|
# create the VM by passing a net ID
|
||||||
nics = [{'net-id': int_net['id']}]
|
nics = [{'net-id': int_net['id']}]
|
||||||
|
|
||||||
|
files = None
|
||||||
|
if self.config.no_dhcp:
|
||||||
|
network_interface = self.get_network_interface(self.port)
|
||||||
|
files = {'/etc/network/interfaces': network_interface}
|
||||||
|
|
||||||
self.instance = self.comp.create_server(self.name,
|
self.instance = self.comp.create_server(self.name,
|
||||||
image,
|
image,
|
||||||
flavor_type,
|
flavor_type,
|
||||||
@ -112,6 +154,7 @@ class Instance(object):
|
|||||||
az,
|
az,
|
||||||
user_data,
|
user_data,
|
||||||
self.config_drive,
|
self.config_drive,
|
||||||
|
files,
|
||||||
self.config.generic_retry_count)
|
self.config.generic_retry_count)
|
||||||
if user_data:
|
if user_data:
|
||||||
user_data.close()
|
user_data.close()
|
||||||
@ -130,7 +173,7 @@ class Instance(object):
|
|||||||
else:
|
else:
|
||||||
# Set the internal ip to the correct ip for v4 and v6
|
# Set the internal ip to the correct ip for v4 and v6
|
||||||
for ip_address in self.instance.networks[internal_network_name]:
|
for ip_address in self.instance.networks[internal_network_name]:
|
||||||
ip = IPAddress(ip_address)
|
ip = netaddr.IPAddress(ip_address)
|
||||||
if self.config.ipv6_mode:
|
if self.config.ipv6_mode:
|
||||||
if ip.version == 6:
|
if ip.version == 6:
|
||||||
self.internal_ip = ip_address
|
self.internal_ip = ip_address
|
||||||
@ -140,14 +183,17 @@ class Instance(object):
|
|||||||
if ip.version == 4:
|
if ip.version == 4:
|
||||||
self.internal_ip = ip_address
|
self.internal_ip = ip_address
|
||||||
ipv4_fixed_address = ip_address
|
ipv4_fixed_address = ip_address
|
||||||
fip = self.net.create_floating_ip()
|
if self.no_floatingip:
|
||||||
if not fip:
|
self.ssh_access.host = self.internal_ip
|
||||||
self.display('Floating ip creation failed')
|
else:
|
||||||
return False
|
fip = self.net.create_floating_ip()
|
||||||
self.ssh_access.host = fip['floatingip']['floating_ip_address']
|
if not fip:
|
||||||
self.ssh_ip_id = fip['floatingip']['id']
|
self.display('Floating ip creation failed.')
|
||||||
self.display('Associating floating IP %s', self.ssh_access.host)
|
return False
|
||||||
self.instance.add_floating_ip(self.ssh_access.host, ipv4_fixed_address)
|
self.ssh_access.host = fip['floatingip']['floating_ip_address']
|
||||||
|
self.ssh_ip_id = fip['floatingip']['id']
|
||||||
|
self.display('Associating floating IP %s', self.ssh_access.host)
|
||||||
|
self.instance.add_floating_ip(self.ssh_access.host, ipv4_fixed_address)
|
||||||
|
|
||||||
# extract the IP for the data network
|
# extract the IP for the data network
|
||||||
self.display('Internal network IP: %s', self.internal_ip)
|
self.display('Internal network IP: %s', self.internal_ip)
|
||||||
|
@ -75,35 +75,34 @@ class Network(object):
|
|||||||
self.ext_net = network
|
self.ext_net = network
|
||||||
break
|
break
|
||||||
|
|
||||||
if not self.ext_net:
|
if self.ext_net:
|
||||||
LOG.error("No external network found.")
|
LOG.info("Using external network: " + self.ext_net['name'])
|
||||||
return
|
# Find or create the router to the external network
|
||||||
|
ext_net_id = self.ext_net['id']
|
||||||
|
routers = neutron_client.list_routers()['routers']
|
||||||
|
for router in routers:
|
||||||
|
external_gw_info = router['external_gateway_info']
|
||||||
|
if external_gw_info:
|
||||||
|
if external_gw_info['network_id'] == ext_net_id:
|
||||||
|
self.ext_router = router
|
||||||
|
LOG.info('Found external router: %s' % (self.ext_router['name']))
|
||||||
|
break
|
||||||
|
|
||||||
LOG.info("Using external network: " + self.ext_net['name'])
|
# create a new external router if none found and a name was given
|
||||||
|
self.ext_router_name = config.router_name
|
||||||
# Find or create the router to the external network
|
if (not self.ext_router) and self.ext_router_name:
|
||||||
ext_net_id = self.ext_net['id']
|
self.ext_router = self.create_router(self.ext_router_name,
|
||||||
routers = neutron_client.list_routers()['routers']
|
self.ext_net['id'])
|
||||||
for router in routers:
|
LOG.info('Created ext router %s.' % (self.ext_router_name))
|
||||||
external_gw_info = router['external_gateway_info']
|
self.ext_router_created = True
|
||||||
if external_gw_info:
|
else:
|
||||||
if external_gw_info['network_id'] == ext_net_id:
|
LOG.warning("No external network found.")
|
||||||
self.ext_router = router
|
|
||||||
LOG.info('Found external router: %s' % (self.ext_router['name']))
|
|
||||||
break
|
|
||||||
|
|
||||||
# create a new external router if none found and a name was given
|
|
||||||
self.ext_router_name = config.router_name
|
|
||||||
if (not self.ext_router) and self.ext_router_name:
|
|
||||||
self.ext_router = self.create_router(self.ext_router_name,
|
|
||||||
self.ext_net['id'])
|
|
||||||
LOG.info('Created ext router %s.' % (self.ext_router_name))
|
|
||||||
self.ext_router_created = True
|
|
||||||
|
|
||||||
if config.ipv6_mode:
|
if config.ipv6_mode:
|
||||||
self.ipv6_enabled = True
|
self.ipv6_enabled = True
|
||||||
|
|
||||||
# Create the networks and subnets depending on v4 or v6
|
# Create the networks and subnets depending on v4 or v6
|
||||||
|
enable_dhcp = not config.no_dhcp
|
||||||
if config.ipv6_mode:
|
if config.ipv6_mode:
|
||||||
for (net, subnet, cidr, subnet_v6, cidr_v6) in zip(config.internal_network_name,
|
for (net, subnet, cidr, subnet_v6, cidr_v6) in zip(config.internal_network_name,
|
||||||
config.internal_subnet_name,
|
config.internal_subnet_name,
|
||||||
@ -112,7 +111,8 @@ class Network(object):
|
|||||||
config.internal_cidr_v6):
|
config.internal_cidr_v6):
|
||||||
int_net = self.create_net(net, subnet, cidr,
|
int_net = self.create_net(net, subnet, cidr,
|
||||||
config.dns_nameservers,
|
config.dns_nameservers,
|
||||||
subnet_v6, cidr_v6, config.ipv6_mode)
|
subnet_v6, cidr_v6, config.ipv6_mode,
|
||||||
|
enable_dhcp=enable_dhcp)
|
||||||
self.vm_int_net.append(int_net)
|
self.vm_int_net.append(int_net)
|
||||||
if config.same_network_only:
|
if config.same_network_only:
|
||||||
break
|
break
|
||||||
@ -121,14 +121,16 @@ class Network(object):
|
|||||||
config.internal_subnet_name,
|
config.internal_subnet_name,
|
||||||
config.internal_cidr):
|
config.internal_cidr):
|
||||||
int_net = self.create_net(net, subnet, cidr,
|
int_net = self.create_net(net, subnet, cidr,
|
||||||
config.dns_nameservers)
|
config.dns_nameservers,
|
||||||
|
enable_dhcp=enable_dhcp)
|
||||||
self.vm_int_net.append(int_net)
|
self.vm_int_net.append(int_net)
|
||||||
if config.same_network_only:
|
if config.same_network_only:
|
||||||
break
|
break
|
||||||
|
|
||||||
# Add both internal networks to router interface to enable
|
# Add both internal networks to router interface to enable
|
||||||
# network to network connectivity
|
# network to network connectivity
|
||||||
self.__add_router_interface()
|
if self.ext_net:
|
||||||
|
self.__add_router_interface()
|
||||||
|
|
||||||
self.l2agent_type = self._get_l2agent_type()
|
self.l2agent_type = self._get_l2agent_type()
|
||||||
self.internal_iface_dict = self._get_internal_iface_dict()
|
self.internal_iface_dict = self._get_internal_iface_dict()
|
||||||
@ -138,7 +140,8 @@ class Network(object):
|
|||||||
# return that network.
|
# return that network.
|
||||||
# dns_nameservers: a list of name servers e.g. ['8.8.8.8']
|
# dns_nameservers: a list of name servers e.g. ['8.8.8.8']
|
||||||
def create_net(self, network_name, subnet_name, cidr, dns_nameservers,
|
def create_net(self, network_name, subnet_name, cidr, dns_nameservers,
|
||||||
subnet_name_ipv6=None, cidr_ipv6=None, ipv6_mode=None):
|
subnet_name_ipv6=None, cidr_ipv6=None, ipv6_mode=None,
|
||||||
|
enable_dhcp=True):
|
||||||
|
|
||||||
for network in self.networks:
|
for network in self.networks:
|
||||||
if network['name'] == network_name:
|
if network['name'] == network_name:
|
||||||
@ -162,6 +165,9 @@ class Network(object):
|
|||||||
'dns_nameservers': dns_nameservers
|
'dns_nameservers': dns_nameservers
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if not enable_dhcp:
|
||||||
|
body['subnet']['enable_dhcp'] = False
|
||||||
|
|
||||||
subnet = self.neutron_client.create_subnet(body)['subnet']
|
subnet = self.neutron_client.create_subnet(body)['subnet']
|
||||||
# add subnet id to the network dict since it has just been added
|
# add subnet id to the network dict since it has just been added
|
||||||
network['subnets'] = [subnet['id']]
|
network['subnets'] = [subnet['id']]
|
||||||
@ -178,6 +184,8 @@ class Network(object):
|
|||||||
'ipv6_address_mode': ipv6_mode
|
'ipv6_address_mode': ipv6_mode
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if not enable_dhcp:
|
||||||
|
body['subnet']['enable_dhcp'] = False
|
||||||
subnet = self.neutron_client.create_subnet(body)['subnet']
|
subnet = self.neutron_client.create_subnet(body)['subnet']
|
||||||
# add the subnet id to the network dict
|
# add the subnet id to the network dict
|
||||||
network['subnets'].append(subnet['id'])
|
network['subnets'].append(subnet['id'])
|
||||||
@ -253,6 +261,9 @@ class Network(object):
|
|||||||
# May fail with neutronclient.common.exceptions.Conflict
|
# May fail with neutronclient.common.exceptions.Conflict
|
||||||
# if there are floating IP in use - just ignore
|
# if there are floating IP in use - just ignore
|
||||||
LOG.warning('Router interface may have floating IP in use: not deleted')
|
LOG.warning('Router interface may have floating IP in use: not deleted')
|
||||||
|
except TypeError:
|
||||||
|
# Externel router is not existed, so let's just continue
|
||||||
|
pass
|
||||||
|
|
||||||
# Lookup network given network name
|
# Lookup network given network name
|
||||||
def lookup_network(self, network_name):
|
def lookup_network(self, network_name):
|
||||||
@ -304,7 +315,7 @@ class Network(object):
|
|||||||
body['port']['binding:vnic_type'] = vnic_type
|
body['port']['binding:vnic_type'] = vnic_type
|
||||||
port = self.neutron_client.create_port(body)
|
port = self.neutron_client.create_port(body)
|
||||||
if self.config.debug:
|
if self.config.debug:
|
||||||
LOG.info('Created port ' + port['port']['id'])
|
LOG.debug('Created port ' + port['port']['id'])
|
||||||
return port['port']
|
return port['port']
|
||||||
|
|
||||||
def delete_port(self, port):
|
def delete_port(self, port):
|
||||||
|
@ -30,11 +30,6 @@ class PerfInstance(Instance):
|
|||||||
self.tp_tool = config.tp_tool(self)
|
self.tp_tool = config.tp_tool(self)
|
||||||
else:
|
else:
|
||||||
self.tp_tool = None
|
self.tp_tool = None
|
||||||
# Override the config drive option to save in instance
|
|
||||||
if config.config_drive:
|
|
||||||
self.config_drive = True
|
|
||||||
else:
|
|
||||||
self.config_drive = None
|
|
||||||
|
|
||||||
# No args is reserved for native host server
|
# No args is reserved for native host server
|
||||||
def create(self, image=None, flavor_type=None,
|
def create(self, image=None, flavor_type=None,
|
||||||
|
23
vmtp/vmtp.py
23
vmtp/vmtp.py
@ -270,8 +270,8 @@ class VmtpTest(object):
|
|||||||
self.config.internal_network_name = int_net_name
|
self.config.internal_network_name = int_net_name
|
||||||
else:
|
else:
|
||||||
# Make sure we have an external network and an external router
|
# Make sure we have an external network and an external router
|
||||||
self.assert_true(self.net.ext_net)
|
# self.assert_true(self.net.ext_net)
|
||||||
self.assert_true(self.net.ext_router)
|
# self.assert_true(self.net.ext_router)
|
||||||
self.assert_true(self.net.vm_int_net)
|
self.assert_true(self.net.vm_int_net)
|
||||||
|
|
||||||
# Get hosts for the availability zone to use
|
# Get hosts for the availability zone to use
|
||||||
@ -826,6 +826,22 @@ def parse_opts_from_cli():
|
|||||||
help='binding vnic type for test VMs',
|
help='binding vnic type for test VMs',
|
||||||
metavar='<direct|macvtap|normal>')
|
metavar='<direct|macvtap|normal>')
|
||||||
|
|
||||||
|
parser.add_argument('--no-dhcp', dest='no_dhcp',
|
||||||
|
default=False,
|
||||||
|
action='store_true',
|
||||||
|
help='Assign IP address to guest instance')
|
||||||
|
|
||||||
|
parser.add_argument('--no-floatingip', dest='no_floatingip',
|
||||||
|
default=False,
|
||||||
|
action='store_true',
|
||||||
|
help='Do not assign floating IP to guest instance')
|
||||||
|
|
||||||
|
parser.add_argument('--use-config-drive', dest='config_drive',
|
||||||
|
default=False,
|
||||||
|
action='store_true',
|
||||||
|
help='Use config drive to configure guest instance. Enable this option '
|
||||||
|
'when metadata service is not available')
|
||||||
|
|
||||||
parser.add_argument('-d', '--debug', dest='debug',
|
parser.add_argument('-d', '--debug', dest='debug',
|
||||||
default=False,
|
default=False,
|
||||||
action='store_true',
|
action='store_true',
|
||||||
@ -979,6 +995,9 @@ def merge_opts_to_configs(opts):
|
|||||||
if opts.os_dataplane_network:
|
if opts.os_dataplane_network:
|
||||||
config.os_dataplane_network = opts.os_dataplane_network
|
config.os_dataplane_network = opts.os_dataplane_network
|
||||||
|
|
||||||
|
config.config_drive = opts.config_drive
|
||||||
|
config.no_floatingip = opts.no_floatingip
|
||||||
|
config.no_dhcp = opts.no_dhcp
|
||||||
config.delete_image_after_run = opts.delete_image_after_run
|
config.delete_image_after_run = opts.delete_image_after_run
|
||||||
|
|
||||||
#####################################################
|
#####################################################
|
||||||
|
Loading…
x
Reference in New Issue
Block a user