mirror of
				https://github.com/apache/cloudstack.git
				synced 2025-11-04 00:02:37 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			200 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
# 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.
 | 
						|
 | 
						|
#Test from the Marvin - Testing in Python wiki
 | 
						|
 | 
						|
import time
 | 
						|
 | 
						|
#All tests inherit from cloudstackTestCase
 | 
						|
from marvin.cloudstackTestCase import cloudstackTestCase
 | 
						|
 | 
						|
#Import Integration Libraries
 | 
						|
 | 
						|
#base - contains all resources as entities and defines create, delete, list operations on them
 | 
						|
from marvin.lib.base import Account, VirtualMachine, Cluster, Host, ServiceOffering, Configurations, SimulatorMock
 | 
						|
 | 
						|
#utils - utility classes for common cleanup, external library wrappers etc
 | 
						|
from marvin.lib.utils import cleanup_resources
 | 
						|
 | 
						|
#common - commonly used methods for all tests are listed here
 | 
						|
from marvin.lib.common import get_zone, get_domain, get_template
 | 
						|
 | 
						|
from nose.plugins.attrib import attr
 | 
						|
 | 
						|
class TestDeployVMHA(cloudstackTestCase):
 | 
						|
    """Test VM HA
 | 
						|
    """
 | 
						|
 | 
						|
    def setUp(self):
 | 
						|
        self.testdata = self.testClient.getParsedTestDataConfig()
 | 
						|
        self.apiclient = self.testClient.getApiClient()
 | 
						|
 | 
						|
        # Get Zone, Domain and Default Built-in template
 | 
						|
        self.domain = get_domain(self.apiclient)
 | 
						|
        self.zone = get_zone(self.apiclient, self.testClient.getZoneForTests())
 | 
						|
 | 
						|
        self.testdata["mode"] = self.zone.networktype
 | 
						|
        self.template = get_template(self.apiclient, self.zone.id, self.testdata["ostype"])
 | 
						|
 | 
						|
        self.hosts = []
 | 
						|
        suitablecluster = None
 | 
						|
        clusters = Cluster.list(self.apiclient)
 | 
						|
        self.assertTrue(isinstance(clusters, list) and len(clusters) > 0, msg = "No clusters found")
 | 
						|
        for cluster in clusters:
 | 
						|
            self.hosts = Host.list(self.apiclient, clusterid=cluster.id, type='Routing')
 | 
						|
            if isinstance(self.hosts, list) and len(self.hosts) >= 2:
 | 
						|
                suitablecluster = cluster
 | 
						|
                break
 | 
						|
        self.assertTrue(isinstance(self.hosts, list) and len(self.hosts) >= 2, msg = "Atleast 2 hosts required in cluster for VM HA test")
 | 
						|
        #update host tags
 | 
						|
        for host in self.hosts:
 | 
						|
            Host.update(self.apiclient, id=host.id, hosttags=self.testdata["service_offerings"]["hasmall"]["hosttags"])
 | 
						|
 | 
						|
        #create a user account
 | 
						|
        self.account = Account.create(
 | 
						|
            self.apiclient,
 | 
						|
            self.testdata["account"],
 | 
						|
            domainid=self.domain.id
 | 
						|
        )
 | 
						|
        #create a service offering
 | 
						|
        self.service_offering = ServiceOffering.create(
 | 
						|
            self.apiclient,
 | 
						|
            self.testdata["service_offerings"]["hasmall"]
 | 
						|
        )
 | 
						|
        #deploy ha vm
 | 
						|
        self.virtual_machine = VirtualMachine.create(
 | 
						|
            self.apiclient,
 | 
						|
            self.testdata["virtual_machine"],
 | 
						|
            accountid=self.account.name,
 | 
						|
            zoneid=self.zone.id,
 | 
						|
            domainid=self.account.domainid,
 | 
						|
            serviceofferingid=self.service_offering.id,
 | 
						|
            templateid=self.template.id
 | 
						|
        )
 | 
						|
        list_vms = VirtualMachine.list(self.apiclient, id=self.virtual_machine.id)
 | 
						|
        self.debug(
 | 
						|
            "Verify listVirtualMachines response for virtual machine: %s"\
 | 
						|
            % self.virtual_machine.id
 | 
						|
        )
 | 
						|
        self.assertTrue(isinstance(list_vms, list) and len(list_vms) == 1, msg = "List VM response was empty")
 | 
						|
        self.virtual_machine = list_vms[0]
 | 
						|
 | 
						|
        self.mock_checkhealth = SimulatorMock.create(
 | 
						|
            apiclient=self.apiclient,
 | 
						|
            command="CheckHealthCommand",
 | 
						|
            zoneid=suitablecluster.zoneid,
 | 
						|
            podid=suitablecluster.podid,
 | 
						|
            clusterid=suitablecluster.id,
 | 
						|
            hostid=self.virtual_machine.hostid,
 | 
						|
            value="result:fail")
 | 
						|
        self.mock_ping = SimulatorMock.create(
 | 
						|
            apiclient=self.apiclient,
 | 
						|
            command="PingCommand",
 | 
						|
            zoneid=suitablecluster.zoneid,
 | 
						|
            podid=suitablecluster.podid,
 | 
						|
            clusterid=suitablecluster.id,
 | 
						|
            hostid=self.virtual_machine.hostid,
 | 
						|
            value="result:fail")
 | 
						|
        self.mock_checkvirtualmachine = SimulatorMock.create(
 | 
						|
            apiclient=self.apiclient,
 | 
						|
            command="CheckVirtualMachineCommand",
 | 
						|
            zoneid=suitablecluster.zoneid,
 | 
						|
            podid=suitablecluster.podid,
 | 
						|
            clusterid=suitablecluster.id,
 | 
						|
            hostid=self.virtual_machine.hostid,
 | 
						|
            value="result:fail")
 | 
						|
        self.mock_pingtest = SimulatorMock.create(
 | 
						|
            apiclient=self.apiclient,
 | 
						|
            command="PingTestCommand",
 | 
						|
            zoneid=suitablecluster.zoneid,
 | 
						|
            podid=suitablecluster.podid,
 | 
						|
            value="result:fail")
 | 
						|
        self.mock_checkonhost_list = []
 | 
						|
        for host in self.hosts:
 | 
						|
            if host.id != self.virtual_machine.hostid:
 | 
						|
                self.mock_checkonhost_list.append(SimulatorMock.create(
 | 
						|
                    apiclient=self.apiclient,
 | 
						|
                    command="CheckOnHostCommand",
 | 
						|
                    zoneid=suitablecluster.zoneid,
 | 
						|
                    podid=suitablecluster.podid,
 | 
						|
                    clusterid=suitablecluster.id,
 | 
						|
                    hostid=host.id,
 | 
						|
                    value="result:fail"))
 | 
						|
        #build cleanup list
 | 
						|
        self.cleanup = [
 | 
						|
            self.service_offering,
 | 
						|
            self.account,
 | 
						|
            self.mock_checkhealth,
 | 
						|
            self.mock_ping,
 | 
						|
            self.mock_checkvirtualmachine,
 | 
						|
            self.mock_pingtest
 | 
						|
        ]
 | 
						|
        self.cleanup = self.cleanup + self.mock_checkonhost_list
 | 
						|
 | 
						|
    @attr(tags = ['advanced'], required_hardware="false")
 | 
						|
    def test_vm_ha(self):
 | 
						|
        """Test VM HA
 | 
						|
 | 
						|
        # Validate the following:
 | 
						|
        # VM started on other host in cluster
 | 
						|
        """
 | 
						|
 | 
						|
        #wait for VM to HA
 | 
						|
        ping_timeout = Configurations.list(self.apiclient, name="ping.timeout")
 | 
						|
        ping_interval = Configurations.list(self.apiclient, name="ping.interval")
 | 
						|
        total_duration = int(float(ping_timeout[0].value) * float(ping_interval[0].value))
 | 
						|
        time.sleep(total_duration)
 | 
						|
 | 
						|
        duration = 0
 | 
						|
        vm = None
 | 
						|
        while duration < total_duration:
 | 
						|
            list_vms = VirtualMachine.list(self.apiclient, id=self.virtual_machine.id)
 | 
						|
            self.assertTrue(isinstance(list_vms, list) and len(list_vms) == 1, msg = "List VM response was empty")
 | 
						|
            vm = list_vms[0]
 | 
						|
            if vm.hostid != self.virtual_machine.hostid and vm.state == "Running":
 | 
						|
                break
 | 
						|
            else:
 | 
						|
                time.sleep(10)
 | 
						|
                duration = duration + 10
 | 
						|
 | 
						|
        self.assertEqual(
 | 
						|
            vm.id,
 | 
						|
            self.virtual_machine.id,
 | 
						|
            "VM ids do not match")
 | 
						|
        self.assertEqual(
 | 
						|
            vm.name,
 | 
						|
            self.virtual_machine.name,
 | 
						|
            "VM names do not match")
 | 
						|
        self.assertEqual(
 | 
						|
            vm.state,
 | 
						|
            "Running",
 | 
						|
            msg="VM is not in Running state")
 | 
						|
        self.assertNotEqual(
 | 
						|
            vm.hostid,
 | 
						|
            self.virtual_machine.hostid,
 | 
						|
            msg="VM is not started on another host as part of HA")
 | 
						|
 | 
						|
    def tearDown(self):
 | 
						|
        try:
 | 
						|
            for host in self.hosts:
 | 
						|
                Host.update(self.apiclient, id=host.id, hosttags="")
 | 
						|
 | 
						|
            cleanup_resources(self.apiclient, self.cleanup)
 | 
						|
        except Exception as e:
 | 
						|
            self.debug("Warning! Exception in tearDown: %s" % e)
 | 
						|
 |