# Licensed to the Apache Software Foundation (ASF) under one # or more contributor license agreements. See the NOTICE file # distributed with this work for additional information # regarding copyright ownership. The ASF licenses this file # to you under the Apache License, Version 2.0 (the # "License"); you may not use this file except in compliance # with the License. You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, # software distributed under the License is distributed on an # "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY # KIND, either express or implied. See the License for the # specific language governing permissions and limitations # under the License. """ P1 tests for Scaling up Vm """ # Import Local Modules from marvin.codes import FAILED from marvin.cloudstackTestCase import cloudstackTestCase from marvin.cloudstackAPI import scaleVirtualMachine from marvin.lib.utils import cleanup_resources from marvin.lib.base import (Account, Host, VirtualMachine, ServiceOffering, Template, Configurations) from marvin.lib.common import (get_zone, get_template, get_domain) from nose.plugins.attrib import attr from marvin.sshClient import SshClient import time _multiprocess_shared_ = True class TestScaleVm(cloudstackTestCase): @classmethod def setUpClass(cls): testClient = super(TestScaleVm, cls).getClsTestClient() cls.apiclient = testClient.getApiClient() cls.services = testClient.getParsedTestDataConfig() cls.hostConfig = cls.config.__dict__["zones"][0].__dict__["pods"][0].__dict__["clusters"][0].__dict__["hosts"][ 0].__dict__ cls._cleanup = [] cls.unsupportedHypervisor = False cls.hypervisor = cls.testClient.getHypervisorInfo() if cls.hypervisor.lower() in ('kvm', 'hyperv', 'lxc'): cls.unsupportedHypervisor = True return # Get Zone, Domain and templates domain = get_domain(cls.apiclient) cls.zone = get_zone(cls.apiclient, testClient.getZoneForTests()) cls.services['mode'] = cls.zone.networktype if cls.hypervisor.lower() in ['simulator', 'vmware']: cls.template = get_template( cls.apiclient, cls.zone.id, cls.services["ostype"] ) if cls.template == FAILED: assert False, "get_template() failed to return template\ with description %s" % cls.services["ostype"] cls.template = Template.update( cls.template, cls.apiclient, isdynamicallyscalable='true' ) else: cls.template = Template.register( cls.apiclient, cls.services["CentOS7template"], zoneid=cls.zone.id ) cls._cleanup.append(cls.template) cls.template.download(cls.apiclient) time.sleep(60) # Set Zones and disk offerings cls.services["small"]["zoneid"] = cls.zone.id cls.services["small"]["template"] = cls.template.id # Create account, service offerings, vm. cls.account = Account.create( cls.apiclient, cls.services["account"], domainid=domain.id ) cls._cleanup.append(cls.account) cls.small_offering = ServiceOffering.create( cls.apiclient, cls.services["service_offerings"]["small"] ) cls._cleanup.append(cls.small_offering) cls.big_offering = ServiceOffering.create( cls.apiclient, cls.services["service_offerings"]["big"] ) cls._cleanup.append(cls.big_offering) Configurations.update( cls.apiclient, name="enable.dynamic.scale.vm", value="true" ) cls.small_offering_dynamic_scaling_disabled = ServiceOffering.create( cls.apiclient, cls.services["service_offerings"]["small"], dynamicscalingenabled=False ) cls.big_offering_dynamic_scaling_disabled = ServiceOffering.create( cls.apiclient, cls.services["service_offerings"]["small"], dynamicscalingenabled=False ) # create a virtual machine cls.virtual_machine = VirtualMachine.create( cls.apiclient, cls.services["small"], accountid=cls.account.name, domainid=cls.account.domainid, serviceofferingid=cls.small_offering.id, mode=cls.services["mode"] ) # create a virtual machine which cannot be dynamically scalable cls.virtual_machine_with_service_offering_dynamic_scaling_disabled = VirtualMachine.create( cls.apiclient, cls.services["small"], accountid=cls.account.name, domainid=cls.account.domainid, serviceofferingid=cls.small_offering_dynamic_scaling_disabled.id, mode=cls.services["mode"] ) # create a virtual machine which cannot be dynamically scalable cls.virtual_machine_not_dynamically_scalable = VirtualMachine.create( cls.apiclient, cls.services["small"], accountid=cls.account.name, domainid=cls.account.domainid, serviceofferingid=cls.small_offering.id, mode=cls.services["mode"], dynamicscalingenabled=False ) cls._cleanup = [ cls.small_offering, cls.big_offering, cls.small_offering_dynamic_scaling_disabled, cls.big_offering_dynamic_scaling_disabled, cls.account ] @classmethod def tearDownClass(cls): Configurations.update( cls.apiclient, name="enable.dynamic.scale.vm", value="false" ) super(TestScaleVm,cls).tearDownClass() return def setUp(self): self.apiclient = self.testClient.getApiClient() self.dbclient = self.testClient.getDbConnection() self.cleanup = [] if self.unsupportedHypervisor: self.skipTest("Skipping test because unsupported hypervisor\ %s" % self.hypervisor) def tearDown(self): # Clean up, terminate the created ISOs super(TestScaleVm,self).tearDown() return def get_ssh_client(self, ip, username, password, retries=10): """ Setup ssh client connection and return connection """ try: ssh_client = SshClient(ip, 22, username, password, retries) except Exception as e: raise self.skipTest("Unable to create ssh connection: " % e) self.assertIsNotNone( ssh_client, "Failed to setup ssh connection to ip=%s" % ip) return ssh_client @attr(hypervisor="xenserver") @attr(tags=["advanced", "basic"], required_hardware="false") def test_01_scale_vm(self): """Test scale virtual machine """ # Validate the following # Scale up the vm and see if it scales to the new svc offering and is # finally in running state # VirtualMachine should be updated to tell cloudstack # it has PV tools # available and successfully scaled. We will only mock # that behaviour # here but it is not expected in production since the VM # scaling is not # guaranteed until tools are installed, vm rebooted # If hypervisor is Vmware, then check if # the vmware tools are installed and the process is running # Vmware tools are necessary for scale VM operation if self.hypervisor.lower() == "vmware": sshClient = self.virtual_machine.get_ssh_client() result = str( sshClient.execute("service vmware-tools status")).lower() self.debug("and result is: %s" % result) if not "running" in result: self.skipTest("Skipping scale VM operation because\ VMware tools are not installed on the VM") if self.hypervisor.lower() != 'simulator': hostid = self.virtual_machine.hostid host = Host.list( self.apiclient, zoneid=self.zone.id, hostid=hostid, type='Routing' )[0] try: username = self.hostConfig["username"] password = self.hostConfig["password"] ssh_client = self.get_ssh_client(host.ipaddress, username, password) res = ssh_client.execute("hostnamectl | grep 'Operating System' | cut -d':' -f2") except Exception as e: pass if 'XenServer' in res[0]: self.skipTest("Skipping test for XenServer as it's License does not allow scaling") self.virtual_machine.update( self.apiclient, isdynamicallyscalable='true') self.debug("Scaling VM-ID: %s to service offering: %s and state %s" % ( self.virtual_machine.id, self.big_offering.id, self.virtual_machine.state )) cmd = scaleVirtualMachine.scaleVirtualMachineCmd() cmd.serviceofferingid = self.big_offering.id cmd.id = self.virtual_machine.id try: self.apiclient.scaleVirtualMachine(cmd) except Exception as e: if "LicenceRestriction" in str(e): self.skipTest("Your XenServer License does not allow scaling") else: self.fail("Scaling failed with the following exception: " + str(e)) list_vm_response = VirtualMachine.list( self.apiclient, id=self.virtual_machine.id ) self.assertEqual( isinstance(list_vm_response, list), True, "Check list response returns a valid list" ) self.assertNotEqual( list_vm_response, None, "Check virtual machine is in listVirtualMachines" ) vm_response = list_vm_response[0] self.assertEqual( vm_response.id, self.virtual_machine.id, "Check virtual machine ID of scaled VM" ) self.debug( "Scaling VM-ID: %s from service offering: %s to new service\ offering %s and the response says %s" % (self.virtual_machine.id, self.virtual_machine.serviceofferingid, self.big_offering.id, vm_response.serviceofferingid)) self.assertEqual( vm_response.serviceofferingid, self.big_offering.id, "Check service offering of the VM" ) self.assertEqual( vm_response.state, 'Running', "Check the state of VM" ) return @attr(tags=["advanced", "basic"], required_hardware="false") def test_02_scale_vm(self): """Test scale virtual machine which is created from a service offering for which dynamicscalingenabled is false. Scaling operation should fail. """ # VirtualMachine should be updated to tell cloudstack # it has PV tools # available and successfully scaled. We will only mock # that behaviour # here but it is not expected in production since the VM # scaling is not # guaranteed until tools are installed, vm rebooted # If hypervisor is Vmware, then check if # the vmware tools are installed and the process is running # Vmware tools are necessary for scale VM operation if self.hypervisor.lower() == "vmware": sshClient = self.virtual_machine_with_service_offering_dynamic_scaling_disabled.get_ssh_client() result = str( sshClient.execute("service vmware-tools status")).lower() self.debug("and result is: %s" % result) if not "running" in result: self.skipTest("Skipping scale VM operation because\ VMware tools are not installed on the VM") list_vm_response = VirtualMachine.list( self.apiclient, id=self.virtual_machine_with_service_offering_dynamic_scaling_disabled.id ) self.assertEqual( isinstance(list_vm_response, list), True, "Check list response returns a valid list" ) self.assertNotEqual( list_vm_response, None, "Check virtual machine is in listVirtualMachines" ) vm_response = list_vm_response[0] self.assertEqual( vm_response.id, self.virtual_machine_with_service_offering_dynamic_scaling_disabled.id, "Check virtual machine ID of scaled VM" ) self.assertEqual( vm_response.isdynamicallyscalable, False, "Check if VM is not dynamically scalable" ) self.debug("Scaling VM-ID: %s to service offering: %s for which dynamic scaling is disabled and VM state %s" % ( self.virtual_machine_with_service_offering_dynamic_scaling_disabled.id, self.big_offering_dynamic_scaling_disabled.id, self.virtual_machine.state )) cmd = scaleVirtualMachine.scaleVirtualMachineCmd() cmd.serviceofferingid = self.big_offering_dynamic_scaling_disabled.id cmd.id = self.virtual_machine_with_service_offering_dynamic_scaling_disabled.id try: self.apiclient.scaleVirtualMachine(cmd) except Exception as e: if "LicenceRestriction" in str(e): self.skipTest("Your XenServer License does not allow scaling") else: pass else: self.fail("Expected an exception to be thrown, failing") self.debug("Scaling VM-ID: %s to service offering: %s for which dynamic scaling is enabled and VM state %s" % ( self.virtual_machine_with_service_offering_dynamic_scaling_disabled.id, self.big_offering.id, self.virtual_machine.state )) cmd = scaleVirtualMachine.scaleVirtualMachineCmd() cmd.serviceofferingid = self.big_offering.id cmd.id = self.virtual_machine_with_service_offering_dynamic_scaling_disabled.id try: self.apiclient.scaleVirtualMachine(cmd) except Exception as e: if "LicenceRestriction" in str(e): self.skipTest("Your XenServer License does not allow scaling") else: pass else: self.fail("Expected an exception to be thrown, failing") return @attr(tags=["advanced", "basic"], required_hardware="false") def test_03_scale_vm(self): """Test scale virtual machine which is not dynamically scalable to a service offering. Scaling operation should fail. """ # Validate the following # Scale up the vm which is not dynamically scalable and see if scaling operation fails # VirtualMachine should be updated to tell cloudstack # it has PV tools # available and successfully scaled. We will only mock # that behaviour # here but it is not expected in production since the VM # scaling is not # guaranteed until tools are installed, vm rebooted # If hypervisor is Vmware, then check if # the vmware tools are installed and the process is running # Vmware tools are necessary for scale VM operation if self.hypervisor.lower() == "vmware": sshClient = self.virtual_machine_not_dynamically_scalable.get_ssh_client() result = str( sshClient.execute("service vmware-tools status")).lower() self.debug("and result is: %s" % result) if not "running" in result: self.skipTest("Skipping scale VM operation because\ VMware tools are not installed on the VM") list_vm_response = VirtualMachine.list( self.apiclient, id=self.virtual_machine_not_dynamically_scalable.id ) self.assertEqual( isinstance(list_vm_response, list), True, "Check list response returns a valid list" ) self.assertNotEqual( list_vm_response, None, "Check virtual machine is in listVirtualMachines" ) vm_response = list_vm_response[0] self.assertEqual( vm_response.id, self.virtual_machine_not_dynamically_scalable.id, "Check virtual machine ID of scaled VM" ) self.assertEqual( vm_response.isdynamicallyscalable, False, "Check if VM is not dynamically scalable" ) self.debug("Scaling VM-ID: %s to service offering: %s for which dynamic scaling is enabled and VM state %s" % ( self.virtual_machine_not_dynamically_scalable.id, self.big_offering.id, self.virtual_machine.state )) cmd = scaleVirtualMachine.scaleVirtualMachineCmd() cmd.serviceofferingid = self.big_offering.id cmd.id = self.virtual_machine_not_dynamically_scalable.id try: self.apiclient.scaleVirtualMachine(cmd) except Exception as e: if "LicenceRestriction" in str(e): self.skipTest("Your XenServer License does not allow scaling") else: pass else: self.fail("Expected an exception to be thrown, failing") return