# 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. """ Tests primary storage limits during upload volume """ import unittest from ddt import ddt, data from marvin.cloudstackTestCase import cloudstackTestCase from marvin.codes import PASS, RESOURCE_PRIMARY_STORAGE, FAIL, USER_ACCOUNT from marvin.lib.base import Domain, Account, VirtualMachine, DiskOffering, ServiceOffering, Volume from marvin.lib.common import get_domain, get_zone, update_resource_limit, uploadVolume, matchResourceCount, get_template from marvin.lib.utils import cleanup_resources, validateList from nose.plugins.attrib import attr @ddt class TestPrimaryResourceLimitsVolume(cloudstackTestCase): @classmethod def setUpClass(cls): cloudstacktestclient = super(TestPrimaryResourceLimitsVolume, cls).getClsTestClient() cls.api_client = cloudstacktestclient.getApiClient() cls.hypervisor = cloudstacktestclient.getHypervisorInfo() # Fill services from the external config file cls.services = cloudstacktestclient.getParsedTestDataConfig() # Get Zone, Domain and templates cls.domain = get_domain(cls.api_client) cls.zone = get_zone(cls.api_client, cloudstacktestclient.getZoneForTests()) cls.services["mode"] = cls.zone.networktype cls._cleanup = [] cls.unsupportedStorageType = False cls.template = get_template(cls.api_client, cls.zone.id, cls.services["ostype"]) cls.services["virtual_machine"]["zoneid"] = cls.zone.id cls.services["virtual_machine"]["template"] = cls.template.id cls.services["volume"]["zoneid"] = cls.zone.id try: cls.service_offering = ServiceOffering.create(cls.api_client, cls.services["service_offering"]) cls.services["disk_offering"]["disksize"] = 2 cls.disk_offering = DiskOffering.create(cls.api_client, cls.services["disk_offering"]) cls._cleanup.append(cls.service_offering) cls._cleanup.append(cls.disk_offering) except Exception as e: cls.tearDownClass() raise unittest.SkipTest("Exception in setUpClass: %s" % e) return @classmethod def tearDownClass(cls): try: # Cleanup resources used cleanup_resources(cls.api_client, cls._cleanup) except Exception as e: raise Exception("Warning: Exception during cleanup : %s" % e) return def setUp(self): self.apiclient = self.testClient.getApiClient() self.dbclient = self.testClient.getDbConnection() self.cleanup = [] return def tearDown(self): try: # Clean up, terminate the created instance, volumes and snapshots cleanup_resources(self.apiclient, self.cleanup) pass except Exception as e: raise Exception("Warning: Exception during cleanup : %s" % e) return def setupNormalAccount(self): """Setup the account required for the test""" try: self.domain = Domain.create(self.apiclient, services=self.services["domain"], parentdomainid=self.domain.id) self.account = Account.create(self.apiclient, self.services["account"], domainid=self.domain.id, admin=False) self.cleanup.append(self.account) self.cleanup.append(self.domain) self.virtualMachine = VirtualMachine.create(self.api_client, self.services["virtual_machine"], accountid=self.account.name, domainid=self.account.domainid, diskofferingid=self.disk_offering.id, serviceofferingid=self.service_offering.id) accounts = Account.list(self.apiclient, id=self.account.id) self.assertEqual(validateList(accounts)[0], PASS, "accounts list validation failed") self.initialResourceCount = int(accounts[0].primarystoragetotal) primarystoragelimit = self.initialResourceCount update_resource_limit(self.api_client, RESOURCE_PRIMARY_STORAGE, account=self.account.name, domainid=self.account.domainid, max=primarystoragelimit) except Exception as e: return [FAIL, e] return [PASS, None] # @data(USER_ACCOUNT) @attr(tags=["advanced","basic"], required_hardware="true") def test_attach_volume_exceeding_primary_limits(self): """ # do # 1. create a normal user account and update primary store limits to the current resource count # 2. Upload a volume of any size # 3. Verify that upload volume succeeds # 4. Verify that primary storage count doesn't change # 6. Try attaching volume to VM and verify that the attach fails (as the resource limits exceed) # 7. Verify that primary storage count doesn't change # done """ # create an account, launch a vm with default template and custom disk offering, update the primary store limits to the current primary store resource count response = self.setupNormalAccount() self.assertEqual(response[0], PASS, response[1]) # upload volume and verify that the volume is uploaded volume = Volume.upload(self.apiclient, self.services["configurableData"]["upload_volume"], zoneid=self.zone.id, account=self.account.name, domainid=self.account.domainid) volume.wait_for_upload(self.apiclient) volumes = Volume.list(self.apiclient, id=volume.id, zoneid=self.zone.id, listall=True) validationresult = validateList(volumes) assert validationresult[0] == PASS, "volumes list validation failed: %s" % validationresult[2] assert str(volumes[0].state).lower() == "uploaded", "Volume state should be 'uploaded' but it is %s" % volumes[0].state # verify that the resource count didnt change due to upload volume response = matchResourceCount( self.apiclient, self.initialResourceCount, RESOURCE_PRIMARY_STORAGE, accountid=self.account.id) self.assertEqual(response[0], PASS, response[1]) # attach the above volume to the vm try: self.virtualMachine.attach_volume(self.apiclient, volume=volume) except Exception as e: if "Maximum number of resources of type \'primary_storage\' for account name="+self.account.name in e.message: self.assertTrue(True, "there should be primary store resource limit reached exception") else: self.fail("only resource limit reached exception is expected. some other exception occurred. Failing the test case.") # resource count should match as the attach should fail due to reaching resource limits response = matchResourceCount( self.apiclient, self.initialResourceCount, RESOURCE_PRIMARY_STORAGE, accountid=self.account.id) self.assertEqual(response[0], PASS, response[1]) return