| # 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 |