| # 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 cases for validating the Quota balance of accounts |
| """ |
| |
| from marvin.cloudstackTestCase import * |
| from marvin.lib.utils import * |
| from marvin.lib.base import * |
| from marvin.lib.common import * |
| from nose.plugins.attrib import attr |
| |
| |
| class TestQuotaBalance(cloudstackTestCase): |
| |
| @classmethod |
| def setUpClass(cls): |
| testClient = super(TestQuotaBalance, cls).getClsTestClient() |
| cls.apiclient = testClient.getApiClient() |
| cls.services = testClient.getParsedTestDataConfig() |
| cls.mgtSvrDetails = cls.config.__dict__["mgtSvr"][0].__dict__ |
| |
| # Get Zone, Domain and templates |
| cls.domain = get_domain(cls.apiclient) |
| cls.zone = get_zone(cls.apiclient, cls.testClient.getZoneForTests()) |
| cls.zone |
| |
| # Create Account |
| cls.account = Account.create( |
| cls.apiclient, |
| cls.services["account"], |
| domainid=cls.domain.id |
| ) |
| cls._cleanup = [ |
| cls.account, |
| ] |
| |
| cls.services["account"] = cls.account.name |
| |
| if not is_config_suitable(apiclient=cls.apiclient, name='quota.enable.service', value='true'): |
| cls.debug("Quota service is not enabled, therefore the configuration `quota.enable.service` will be set to `true` and the management server will be restarted.") |
| Configurations.update(cls.apiclient, "quota.enable.service", "true") |
| cls.restartServer() |
| |
| return |
| |
| @classmethod |
| def restartServer(cls): |
| """Restart management server""" |
| |
| cls.debug("Restarting management server") |
| sshClient = SshClient( |
| cls.mgtSvrDetails["mgtSvrIp"], |
| 22, |
| cls.mgtSvrDetails["user"], |
| cls.mgtSvrDetails["passwd"] |
| ) |
| |
| command = "service cloudstack-management restart" |
| sshClient.execute(command) |
| |
| # Waits for management to come up in 5 mins, when it's up it will continue |
| timeout = time.time() + 300 |
| while time.time() < timeout: |
| if cls.isManagementUp() is True: |
| time.sleep(30) |
| return |
| time.sleep(5) |
| return cls.fail("Management server did not come up, failing") |
| |
| @classmethod |
| def isManagementUp(cls): |
| try: |
| cls.apiclient.listInfrastructure(listInfrastructure.listInfrastructureCmd()) |
| return True |
| except Exception: |
| return False |
| |
| @classmethod |
| def tearDownClass(cls): |
| try: |
| # Cleanup resources used |
| cleanup_resources(cls.apiclient, 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 = [] |
| self.tariffs = [] |
| return |
| |
| def tearDown(self): |
| try: |
| cleanup_resources(self.apiclient, self.cleanup) |
| self.delete_tariffs() |
| except Exception as e: |
| raise Exception("Warning: Exception during cleanup : %s" % e) |
| return |
| |
| def delete_tariffs(self): |
| for tariff in self.tariffs: |
| cmd = quotaTariffDelete.quotaTariffDeleteCmd() |
| cmd.id = tariff.uuid |
| self.apiclient.quotaTariffDelete(cmd) |
| |
| @attr(tags=["advanced", "smoke", "quota"], required_hardware="false") |
| def test_quota_balance(self): |
| """ |
| Test Quota balance |
| |
| Validate the following |
| 1. Add credits to an account |
| 2. Create Quota tariff for the usage type 21 (VM_DISK_IO_READ) |
| 3. Simulate quota usage by inserting a row in the `cloud_usage` table |
| 4. Update the balance of the account by calling the API quotaUpdate |
| 5. Verify the balance of the account according to the tariff created |
| """ |
| |
| # Create quota tariff for the usage type 21 (VM_DISK_IO_READ) |
| cmd = quotaTariffCreate.quotaTariffCreateCmd() |
| cmd.name = 'Tariff' |
| cmd.value = '10' |
| cmd.usagetype = '21' |
| self.tariffs.append(self.apiclient.quotaTariffCreate(cmd)) |
| |
| # Add credits to the account |
| cmd = quotaCredits.quotaCreditsCmd() |
| cmd.account = self.account.name |
| cmd.domainid = self.domain.id |
| cmd.value = 100 |
| self.apiclient.quotaCredits(cmd) |
| |
| # Fetch account ID from account_uuid |
| account_id_select = f"SELECT id FROM account WHERE uuid = '{self.account.id}';" |
| self.debug(account_id_select) |
| qresultset = self.dbclient.execute(account_id_select) |
| account_id = qresultset[0][0] |
| |
| # Fetch domain ID from domain_uuid |
| domain_id_select = f"SELECT id FROM `domain` d WHERE uuid = '{self.domain.id}';" |
| self.debug(domain_id_select) |
| qresultset = self.dbclient.execute(domain_id_select) |
| domain_id = qresultset[0][0] |
| |
| # Fetch zone ID from zone_uuid |
| zone_id_select = f"SELECT id from data_center dc where dc.uuid = '{self.zone.id}';" |
| self.debug(zone_id_select) |
| qresultset = self.dbclient.execute(zone_id_select) |
| zone_id = qresultset[0][0] |
| |
| start_date = datetime.datetime.now() + datetime.timedelta(seconds=1) |
| end_date = datetime.datetime.now() + datetime.timedelta(hours=1) |
| |
| # Manually insert a usage regarding the usage type 21 (VM_DISK_IO_READ) |
| sql_query = (f"INSERT INTO cloud_usage.cloud_usage (zone_id,account_id,domain_id,description,usage_display,usage_type,raw_usage,vm_instance_id,vm_name,offering_id,template_id," |
| f"usage_id,`type`,`size`,network_id,start_date,end_date,virtual_size,cpu_speed,cpu_cores,memory,quota_calculated,is_hidden,state)" |
| f" VALUES ('{zone_id}','{account_id}','{domain_id}','Test','1 Hrs',21,1,NULL,NULL,NULL,NULL,NULL,'VirtualMachine',NULL,NULL,'{start_date}','{end_date}',NULL,NULL,NULL,NULL,0,0,NULL);") |
| self.debug(sql_query) |
| self.dbclient.execute(sql_query) |
| |
| # Update quota to calculate the balance of the account |
| cmd = quotaUpdate.quotaUpdateCmd() |
| self.apiclient.quotaUpdate(cmd) |
| |
| # Retrieve the quota balance of the account |
| cmd = quotaBalance.quotaBalanceCmd() |
| cmd.domainid = self.account.domainid |
| cmd.account = self.account.name |
| response = self.apiclient.quotaBalance(cmd) |
| |
| self.debug(f"The quota balance for the account {self.account.name} is {response.balance}.") |
| self.assertEqual(response.balance.startquota, 90, f"The `startQuota` response field is supposed to be 90 but was {response.balance.startquota}.") |
| |
| return |