| // 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. |
| package com.cloud.storage; |
| |
| import java.lang.reflect.Field; |
| import java.util.ArrayList; |
| import java.util.Arrays; |
| import java.util.HashMap; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.Optional; |
| |
| import com.cloud.dc.HostPodVO; |
| import com.cloud.dc.dao.HostPodDao; |
| import com.cloud.host.HostVO; |
| import com.cloud.host.dao.HostDao; |
| import com.cloud.resource.ResourceManager; |
| import com.cloud.storage.dao.StoragePoolAndAccessGroupMapDao; |
| import org.apache.cloudstack.api.ApiConstants; |
| import org.apache.cloudstack.api.command.admin.storage.ChangeStoragePoolScopeCmd; |
| import org.apache.cloudstack.api.command.admin.storage.ConfigureStorageAccessCmd; |
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; |
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreLifeCycle; |
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreManager; |
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreProvider; |
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreProviderManager; |
| import org.apache.cloudstack.engine.subsystem.api.storage.PrimaryDataStoreDriver; |
| import org.apache.cloudstack.framework.config.ConfigDepot; |
| import org.apache.cloudstack.framework.config.ConfigKey; |
| import org.apache.cloudstack.framework.config.dao.ConfigurationDao; |
| import org.apache.cloudstack.resourcedetail.dao.DiskOfferingDetailsDao; |
| import org.apache.cloudstack.storage.command.CheckDataStoreStoragePolicyComplainceCommand; |
| import org.apache.cloudstack.storage.datastore.db.ObjectStoreDao; |
| import org.apache.cloudstack.storage.datastore.db.ObjectStoreVO; |
| import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; |
| import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailVO; |
| import org.apache.cloudstack.storage.datastore.db.StoragePoolDetailsDao; |
| import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; |
| import org.apache.cloudstack.storage.object.ObjectStore; |
| import org.apache.commons.collections.MapUtils; |
| import org.junit.Assert; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| import org.mockito.InjectMocks; |
| import org.mockito.Mock; |
| import org.mockito.Mockito; |
| import org.mockito.Spy; |
| import org.mockito.junit.MockitoJUnitRunner; |
| import org.mockito.stubbing.Answer; |
| import org.springframework.test.util.ReflectionTestUtils; |
| |
| import com.cloud.agent.AgentManager; |
| import com.cloud.agent.api.Command; |
| import com.cloud.agent.api.StoragePoolInfo; |
| import com.cloud.capacity.Capacity; |
| import com.cloud.capacity.CapacityManager; |
| import com.cloud.capacity.CapacityVO; |
| import com.cloud.dc.ClusterVO; |
| import com.cloud.dc.DataCenter; |
| import com.cloud.dc.DataCenterVO; |
| import com.cloud.dc.VsphereStoragePolicyVO; |
| import com.cloud.dc.dao.ClusterDao; |
| import com.cloud.dc.dao.DataCenterDao; |
| import com.cloud.dc.dao.VsphereStoragePolicyDao; |
| import com.cloud.exception.AgentUnavailableException; |
| import com.cloud.exception.ConnectionException; |
| import com.cloud.exception.InvalidParameterValueException; |
| import com.cloud.exception.OperationTimedoutException; |
| import com.cloud.exception.PermissionDeniedException; |
| import com.cloud.exception.StorageUnavailableException; |
| import com.cloud.host.Host; |
| import com.cloud.hypervisor.Hypervisor.HypervisorType; |
| import com.cloud.hypervisor.HypervisorGuruManager; |
| import com.cloud.storage.dao.VolumeDao; |
| import com.cloud.user.AccountManagerImpl; |
| import com.cloud.utils.Pair; |
| import com.cloud.utils.exception.CloudRuntimeException; |
| import com.cloud.vm.DiskProfile; |
| import com.cloud.vm.VMInstanceVO; |
| import com.cloud.vm.dao.VMInstanceDao; |
| |
| import static org.junit.Assert.assertEquals; |
| import static org.junit.Assert.assertNotNull; |
| import static org.junit.Assert.assertThrows; |
| import static org.junit.Assert.assertTrue; |
| import static org.mockito.Mockito.doReturn; |
| |
| @RunWith(MockitoJUnitRunner.class) |
| public class StorageManagerImplTest { |
| |
| @Mock |
| VolumeDao _volumeDao; |
| |
| @Mock |
| VMInstanceDao vmInstanceDao; |
| @Mock |
| PrimaryDataStoreDao storagePoolDao; |
| @Mock |
| CapacityManager capacityManager; |
| @Mock |
| DiskOfferingDetailsDao diskOfferingDetailsDao; |
| @Mock |
| VsphereStoragePolicyDao vsphereStoragePolicyDao; |
| @Mock |
| HypervisorGuruManager hvGuruMgr; |
| @Mock |
| AgentManager agentManager; |
| @Mock |
| ConfigDepot configDepot; |
| @Mock |
| ConfigurationDao configurationDao; |
| @Mock |
| DataCenterDao dataCenterDao; |
| @Mock |
| AccountManagerImpl accountMgr; |
| @Mock |
| StoragePoolDetailsDao storagePoolDetailsDao; |
| |
| @Mock |
| ClusterDao clusterDao; |
| |
| @Spy |
| @InjectMocks |
| private StorageManagerImpl storageManagerImpl; |
| |
| @Mock |
| private StoragePoolVO storagePoolVOMock; |
| |
| @Mock |
| private VolumeVO volume1VOMock; |
| |
| @Mock |
| private VolumeVO volume2VOMock; |
| |
| @Mock |
| private VMInstanceVO vmInstanceVOMock; |
| |
| @Mock |
| private HostDao hostDao; |
| @Mock |
| private HostPodDao podDao; |
| |
| @Mock |
| private StoragePoolAndAccessGroupMapDao storagePoolAccessGroupMapDao; |
| |
| @Mock |
| private ResourceManager resourceMgr; |
| |
| |
| @Mock |
| protected ObjectStoreDao objectStoreDao; |
| |
| @Mock |
| DataStoreProviderManager dataStoreProviderMgr; |
| |
| @Mock |
| DataStoreManager dataStoreMgr; |
| |
| @Test |
| public void createLocalStoragePoolName() { |
| String hostMockName = "host1"; |
| executeCreateLocalStoragePoolNameForHostName(hostMockName); |
| } |
| |
| @Test |
| public void createLocalStoragePoolNameUsingHostNameWithSpaces() { |
| String hostMockName = " hostNameWithSpaces "; |
| executeCreateLocalStoragePoolNameForHostName(hostMockName); |
| } |
| |
| private void executeCreateLocalStoragePoolNameForHostName(String hostMockName) { |
| String firstBlockUuid = "dsdsh665"; |
| |
| String expectedLocalStorageName = hostMockName.trim() + "-local-" + firstBlockUuid; |
| |
| Host hostMock = Mockito.mock(Host.class); |
| StoragePoolInfo storagePoolInfoMock = Mockito.mock(StoragePoolInfo.class); |
| |
| Mockito.when(hostMock.getName()).thenReturn(hostMockName); |
| Mockito.when(storagePoolInfoMock.getUuid()).thenReturn(firstBlockUuid + "-213151-df21ef333d-2d33f1"); |
| |
| String localStoragePoolName = storageManagerImpl.createLocalStoragePoolName(hostMock, storagePoolInfoMock); |
| assertEquals(expectedLocalStorageName, localStoragePoolName); |
| } |
| |
| private VolumeVO mockVolumeForIsVolumeSuspectedDestroyDuplicateTest() { |
| VolumeVO volumeVO = new VolumeVO("data", 1L, 1L, 1L, 1L, 1L, "data", "data", Storage.ProvisioningType.THIN, 1, null, null, "data", Volume.Type.DATADISK); |
| volumeVO.setPoolId(1L); |
| return volumeVO; |
| } |
| |
| @Test |
| public void testIsVolumeSuspectedDestroyDuplicateNoPool() { |
| VolumeVO volume = mockVolumeForIsVolumeSuspectedDestroyDuplicateTest(); |
| volume.setPoolId(null); |
| Assert.assertFalse(storageManagerImpl.isVolumeSuspectedDestroyDuplicateOfVmVolume(volume)); |
| } |
| |
| @Test |
| public void testIsVolumeSuspectedDestroyDuplicateNoPath() { |
| VolumeVO volume = mockVolumeForIsVolumeSuspectedDestroyDuplicateTest(); |
| Assert.assertFalse(storageManagerImpl.isVolumeSuspectedDestroyDuplicateOfVmVolume(volume)); |
| } |
| |
| @Test |
| public void testIsVolumeSuspectedDestroyDuplicateNoVmId() { |
| VolumeVO volume = mockVolumeForIsVolumeSuspectedDestroyDuplicateTest(); |
| volume.setInstanceId(null); |
| Assert.assertFalse(storageManagerImpl.isVolumeSuspectedDestroyDuplicateOfVmVolume(volume)); |
| } |
| |
| @Test |
| public void testIsVolumeSuspectedDestroyDuplicateNoVm() { |
| VolumeVO volume = mockVolumeForIsVolumeSuspectedDestroyDuplicateTest(); |
| Assert.assertFalse(storageManagerImpl.isVolumeSuspectedDestroyDuplicateOfVmVolume(volume)); |
| } |
| |
| @Test |
| public void testIsVolumeSuspectedDestroyDuplicateNoVmVolumes() { |
| VolumeVO volume = mockVolumeForIsVolumeSuspectedDestroyDuplicateTest(); |
| Mockito.when(vmInstanceDao.findById(1L)).thenReturn(Mockito.mock(VMInstanceVO.class)); |
| Mockito.when(_volumeDao.findUsableVolumesForInstance(1L)).thenReturn(new ArrayList<>()); |
| Assert.assertFalse(storageManagerImpl.isVolumeSuspectedDestroyDuplicateOfVmVolume(volume)); |
| } |
| |
| @Test |
| public void testIsVolumeSuspectedDestroyDuplicateTrue() { |
| Long poolId = 1L; |
| String path = "data"; |
| VolumeVO volume = mockVolumeForIsVolumeSuspectedDestroyDuplicateTest(); |
| volume.setPoolId(poolId); |
| Mockito.when(vmInstanceDao.findById(1L)).thenReturn(Mockito.mock(VMInstanceVO.class)); |
| VolumeVO volumeVO = Mockito.mock(VolumeVO.class); |
| Mockito.when(volumeVO.getPoolId()).thenReturn(poolId); |
| Mockito.when(volumeVO.getPath()).thenReturn(path); |
| Mockito.when(_volumeDao.findUsableVolumesForInstance(1L)).thenReturn(List.of(volumeVO, Mockito.mock(VolumeVO.class))); |
| assertTrue(storageManagerImpl.isVolumeSuspectedDestroyDuplicateOfVmVolume(volume)); |
| } |
| |
| @Test |
| public void storagePoolCompatibleWithVolumePoolTestVolumeWithPoolIdInAllocatedState() { |
| StoragePoolVO storagePool = new StoragePoolVO(); |
| storagePool.setPoolType(Storage.StoragePoolType.PowerFlex); |
| storagePool.setId(1L); |
| VolumeVO volume = new VolumeVO(); |
| volume.setState(Volume.State.Allocated); |
| volume.setPoolId(1L); |
| PrimaryDataStoreDao storagePoolDao = Mockito.mock(PrimaryDataStoreDao.class); |
| storageManagerImpl._storagePoolDao = storagePoolDao; |
| Mockito.doReturn(storagePool).when(storagePoolDao).findById(volume.getPoolId()); |
| Assert.assertFalse(storageManagerImpl.storagePoolCompatibleWithVolumePool(storagePool, volume)); |
| |
| } |
| |
| @Test |
| public void storagePoolCompatibleWithVolumePoolTestVolumeWithoutPoolIdInAllocatedState() { |
| StoragePoolVO storagePool = new StoragePoolVO(); |
| storagePool.setPoolType(Storage.StoragePoolType.PowerFlex); |
| storagePool.setId(1L); |
| VolumeVO volume = new VolumeVO(); |
| volume.setState(Volume.State.Allocated); |
| PrimaryDataStoreDao storagePoolDao = Mockito.mock(PrimaryDataStoreDao.class); |
| storageManagerImpl._storagePoolDao = storagePoolDao; |
| Assert.assertTrue(storageManagerImpl.storagePoolCompatibleWithVolumePool(storagePool, volume)); |
| |
| } |
| |
| @Test |
| public void testExtractUriParamsAsMapWithSolidFireUrl() { |
| String sfUrl = "MVIP=1.2.3.4;SVIP=6.7.8.9;clusterAdminUsername=admin;" + |
| "clusterAdminPassword=password;clusterDefaultMinIops=1000;" + |
| "clusterDefaultMaxIops=2000;clusterDefaultBurstIopsPercentOfMaxIops=2"; |
| Map<String, String> uriParams = storageManagerImpl.extractUriParamsAsMap(sfUrl); |
| Assert.assertTrue(MapUtils.isEmpty(uriParams)); |
| } |
| |
| @Test |
| public void testExtractUriParamsAsMapWithNFSUrl() { |
| String scheme = "nfs"; |
| String host = "HOST"; |
| String path = "/PATH"; |
| String sfUrl = String.format("%s://%s%s", scheme, host, path); |
| Map<String, String> uriParams = storageManagerImpl.extractUriParamsAsMap(sfUrl); |
| Assert.assertTrue(MapUtils.isNotEmpty(uriParams)); |
| Assert.assertEquals(scheme, uriParams.get("scheme")); |
| Assert.assertEquals(host, uriParams.get("host")); |
| Assert.assertEquals(path, uriParams.get("hostPath")); |
| } |
| |
| @Test(expected = InvalidParameterValueException.class) |
| public void testCreateLocalStorageHostFailure() { |
| Map<String, Object> test = new HashMap<>(); |
| test.put("host", null); |
| try { |
| storageManagerImpl.createLocalStorage(test); |
| } catch (ConnectionException e) { |
| throw new RuntimeException(e); |
| } |
| } |
| |
| @Test(expected = InvalidParameterValueException.class) |
| public void testCreateLocalStoragePathFailure() { |
| Map<String, Object> test = new HashMap<>(); |
| test.put("host", "HOST"); |
| test.put("hostPath", ""); |
| try { |
| storageManagerImpl.createLocalStorage(test); |
| } catch (ConnectionException e) { |
| throw new RuntimeException(e); |
| } |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsNullPoolIops() { |
| StoragePool pool = Mockito.mock(StoragePool.class); |
| Mockito.when(pool.getCapacityIops()).thenReturn(null); |
| List<Pair<Volume, DiskProfile>> list = List.of(new Pair<>(Mockito.mock(Volume.class), Mockito.mock(DiskProfile.class))); |
| assertTrue(storageManagerImpl.storagePoolHasEnoughIops(100L, list, pool, false)); |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsSuccess() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.when(pool.getId()).thenReturn(1L); |
| Mockito.when(pool.getCapacityIops()).thenReturn(1000L); |
| Mockito.when(storagePoolDao.findById(1L)).thenReturn(pool); |
| Mockito.when(capacityManager.getUsedIops(pool)).thenReturn(500L); |
| List<Pair<Volume, DiskProfile>> list = List.of(new Pair<>(Mockito.mock(Volume.class), Mockito.mock(DiskProfile.class))); |
| assertTrue(storageManagerImpl.storagePoolHasEnoughIops(100L, list, pool, true)); |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsNegative() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.when(pool.getId()).thenReturn(1L); |
| Mockito.when(pool.getCapacityIops()).thenReturn(550L); |
| Mockito.when(storagePoolDao.findById(1L)).thenReturn(pool); |
| Mockito.when(capacityManager.getUsedIops(pool)).thenReturn(500L); |
| List<Pair<Volume, DiskProfile>> list = List.of(new Pair<>(Mockito.mock(Volume.class), Mockito.mock(DiskProfile.class))); |
| Assert.assertFalse(storageManagerImpl.storagePoolHasEnoughIops(100L, list, pool, true)); |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsNullPool() { |
| Assert.assertFalse(storageManagerImpl.storagePoolHasEnoughIops(100L, null)); |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsNullRequestedIops() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| List<Long> iopsList = Arrays.asList(null, 0L); |
| for (Long iops : iopsList) { |
| assertTrue(storageManagerImpl.storagePoolHasEnoughIops(iops, pool)); |
| } |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsSuccess1() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.doReturn(true).when(storageManagerImpl).storagePoolHasEnoughIops( |
| Mockito.eq(100L), Mockito.anyList(), Mockito.eq(pool), Mockito.eq(false)); |
| assertTrue(storageManagerImpl.storagePoolHasEnoughIops(100L, pool)); |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsNoVolumesOrPool() { |
| List<Pair<Volume, DiskProfile>> list = new ArrayList<>(); |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Assert.assertFalse(storageManagerImpl.storagePoolHasEnoughIops(list, pool)); |
| list = List.of(new Pair<>(Mockito.mock(Volume.class), Mockito.mock(DiskProfile.class))); |
| Assert.assertFalse(storageManagerImpl.storagePoolHasEnoughIops(list, null)); |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsWithVolPoolNullIops() { |
| List<Pair<Volume, DiskProfile>> list = List.of( |
| new Pair<>(Mockito.mock(Volume.class), Mockito.mock(DiskProfile.class))); |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.when(pool.getCapacityIops()).thenReturn(null); |
| assertTrue(storageManagerImpl.storagePoolHasEnoughIops(list, pool)); |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughIopsWithVolPoolCompare() { |
| Volume volume = Mockito.mock(Volume.class); |
| Mockito.when(volume.getDiskOfferingId()).thenReturn(1L); |
| Mockito.when(volume.getMinIops()).thenReturn(100L); |
| DiskProfile profile = Mockito.mock(DiskProfile.class); |
| Mockito.when(profile.getDiskOfferingId()).thenReturn(1L); |
| List<Pair<Volume, DiskProfile>> list = List.of(new Pair<>(volume, profile)); |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.doReturn(true).when(storageManagerImpl) |
| .storagePoolHasEnoughIops(100L, list, pool, true); |
| assertTrue(storageManagerImpl.storagePoolHasEnoughIops(list, pool)); |
| |
| Mockito.when(profile.getDiskOfferingId()).thenReturn(2L); |
| Mockito.when(profile.getMinIops()).thenReturn(200L); |
| Mockito.doReturn(false).when(storageManagerImpl) |
| .storagePoolHasEnoughIops(200L, list, pool, true); |
| Assert.assertFalse(storageManagerImpl.storagePoolHasEnoughIops(list, pool)); |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughSpaceNullSize() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| List<Long> sizeList = Arrays.asList(null, 0L); |
| for (Long size : sizeList) { |
| assertTrue(storageManagerImpl.storagePoolHasEnoughSpace(size, pool)); |
| } |
| } |
| |
| @Test |
| public void testStoragePoolHasEnoughSpaceCompare() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.when(pool.getId()).thenReturn(1L); |
| Mockito.when(storagePoolDao.findById(1L)).thenReturn(pool); |
| Mockito.when(capacityManager.getAllocatedPoolCapacity(pool, null)).thenReturn(2000L); |
| Mockito.doAnswer((Answer<Boolean>) invocationOnMock -> { |
| long total = invocationOnMock.getArgument(1); |
| long asking = invocationOnMock.getArgument(2); |
| return total > asking; |
| }).when(storageManagerImpl).checkPoolforSpace(Mockito.any(StoragePool.class), |
| Mockito.anyLong(), Mockito.anyLong()); |
| assertTrue(storageManagerImpl.storagePoolHasEnoughSpace(1000L, pool)); |
| Assert.assertFalse(storageManagerImpl.storagePoolHasEnoughSpace(2200L, pool)); |
| } |
| |
| @Test |
| public void testIsStoragePoolCompliantWithStoragePolicy() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.when(diskOfferingDetailsDao.getDetail(1L, ApiConstants.STORAGE_POLICY)) |
| .thenReturn("policy"); |
| try { |
| Mockito.doReturn(null) |
| .when(storageManagerImpl).getCheckDatastorePolicyComplianceAnswer("policy", pool); |
| assertTrue(storageManagerImpl.isStoragePoolCompliantWithStoragePolicy(1L, pool)); |
| } catch (StorageUnavailableException e) { |
| Assert.fail(e.getMessage()); |
| } |
| try { |
| Mockito.doReturn(new com.cloud.agent.api.Answer( |
| Mockito.mock(CheckDataStoreStoragePolicyComplainceCommand.class))) |
| .when(storageManagerImpl).getCheckDatastorePolicyComplianceAnswer("policy", pool); |
| assertTrue(storageManagerImpl.isStoragePoolCompliantWithStoragePolicy(1L, pool)); |
| } catch (StorageUnavailableException e) { |
| Assert.fail(e.getMessage()); |
| } |
| try { |
| com.cloud.agent.api.Answer answer = |
| new com.cloud.agent.api.Answer(Mockito.mock(CheckDataStoreStoragePolicyComplainceCommand.class), |
| false, ""); |
| Mockito.doReturn(answer) |
| .when(storageManagerImpl).getCheckDatastorePolicyComplianceAnswer("policy", pool); |
| Assert.assertFalse(storageManagerImpl.isStoragePoolCompliantWithStoragePolicy(1L, pool)); |
| } catch (StorageUnavailableException e) { |
| Assert.fail(e.getMessage()); |
| } |
| } |
| |
| @Test |
| public void testGetCheckDatastorePolicyComplianceAnswerNullAnswer() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| try { |
| Assert.assertNull(storageManagerImpl.getCheckDatastorePolicyComplianceAnswer(null, pool)); |
| Assert.assertNull(storageManagerImpl.getCheckDatastorePolicyComplianceAnswer("", pool)); |
| } catch (StorageUnavailableException e) { |
| Assert.fail(e.getMessage()); |
| } |
| } |
| |
| @Test(expected = StorageUnavailableException.class) |
| public void testGetCheckDatastorePolicyComplianceAnswerNoHost() throws StorageUnavailableException { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.when(pool.getId()).thenReturn(1L); |
| Mockito.when(vsphereStoragePolicyDao.findById(Mockito.anyLong())) |
| .thenReturn(Mockito.mock(VsphereStoragePolicyVO.class)); |
| Mockito.doReturn(new ArrayList<>()).when(storageManagerImpl).getUpHostsInPool(Mockito.anyLong()); |
| storageManagerImpl.getCheckDatastorePolicyComplianceAnswer("1", pool); |
| } |
| |
| @Test(expected = StorageUnavailableException.class) |
| public void testGetCheckDatastorePolicyComplianceAnswerAgentException() throws StorageUnavailableException { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.when(pool.getId()).thenReturn(1L); |
| VsphereStoragePolicyVO policy = Mockito.mock(VsphereStoragePolicyVO.class); |
| Mockito.when(policy.getPolicyId()).thenReturn("some"); |
| Mockito.when(vsphereStoragePolicyDao.findById(Mockito.anyLong())) |
| .thenReturn(policy); |
| Mockito.doReturn(new ArrayList<>(List.of(1L, 2L))).when(storageManagerImpl).getUpHostsInPool(Mockito.anyLong()); |
| Mockito.when(hvGuruMgr.getGuruProcessedCommandTargetHost(Mockito.anyLong(), |
| Mockito.any(CheckDataStoreStoragePolicyComplainceCommand.class))).thenReturn(1L); |
| try { |
| Mockito.when(agentManager.send(Mockito.anyLong(), Mockito.any(Command.class))) |
| .thenThrow(AgentUnavailableException.class); |
| } catch (AgentUnavailableException | OperationTimedoutException e) { |
| Assert.fail(e.getMessage()); |
| } |
| storageManagerImpl.getCheckDatastorePolicyComplianceAnswer("1", pool); |
| try { |
| Mockito.when(agentManager.send(Mockito.anyLong(), Mockito.any(Command.class))) |
| .thenThrow(OperationTimedoutException.class); |
| } catch (AgentUnavailableException | OperationTimedoutException e) { |
| Assert.fail(e.getMessage()); |
| } |
| storageManagerImpl.getCheckDatastorePolicyComplianceAnswer("1", pool); |
| } |
| |
| @Test |
| public void testGetCheckDatastorePolicyComplianceAnswerSuccess() throws StorageUnavailableException { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Mockito.when(pool.getId()).thenReturn(1L); |
| VsphereStoragePolicyVO policy = Mockito.mock(VsphereStoragePolicyVO.class); |
| Mockito.when(policy.getPolicyId()).thenReturn("some"); |
| Mockito.when(vsphereStoragePolicyDao.findById(Mockito.anyLong())) |
| .thenReturn(policy); |
| Mockito.doReturn(new ArrayList<>(List.of(1L, 2L))).when(storageManagerImpl).getUpHostsInPool(Mockito.anyLong()); |
| Mockito.when(hvGuruMgr.getGuruProcessedCommandTargetHost(Mockito.anyLong(), |
| Mockito.any(CheckDataStoreStoragePolicyComplainceCommand.class))).thenReturn(1L); |
| try { |
| Mockito.when(agentManager.send(Mockito.anyLong(), |
| Mockito.any(CheckDataStoreStoragePolicyComplainceCommand.class))) |
| .thenReturn(new com.cloud.agent.api.Answer( |
| Mockito.mock(CheckDataStoreStoragePolicyComplainceCommand.class))); |
| } catch (AgentUnavailableException | OperationTimedoutException e) { |
| Assert.fail(e.getMessage()); |
| } |
| com.cloud.agent.api.Answer answer = |
| storageManagerImpl.getCheckDatastorePolicyComplianceAnswer("1", pool); |
| assertTrue(answer.getResult()); |
| } |
| |
| @Test |
| public void testEnableDefaultDatastoreDownloadRedirectionForExistingInstallationsNoChange() { |
| Mockito.when(configDepot.isNewConfig(StorageManager.DataStoreDownloadFollowRedirects)) |
| .thenReturn(false); |
| storageManagerImpl.enableDefaultDatastoreDownloadRedirectionForExistingInstallations(); |
| Mockito.verify(configurationDao, Mockito.never()).update(Mockito.anyString(), Mockito.anyString()); |
| } |
| |
| @Test |
| public void testEnableDefaultDatastoreDownloadRedirectionForExistingInstallationsOldInstall() { |
| Mockito.when(configDepot.isNewConfig(StorageManager.DataStoreDownloadFollowRedirects)) |
| .thenReturn(true); |
| Mockito.when(dataCenterDao.listAll(Mockito.any())) |
| .thenReturn(List.of(Mockito.mock(DataCenterVO.class))); |
| Mockito.doReturn(true).when(configurationDao).update(Mockito.anyString(), Mockito.anyString()); |
| storageManagerImpl.enableDefaultDatastoreDownloadRedirectionForExistingInstallations(); |
| Mockito.verify(configurationDao, Mockito.times(1)) |
| .update(StorageManager.DataStoreDownloadFollowRedirects.key(), "true"); |
| } |
| |
| @Test |
| public void testEnableDefaultDatastoreDownloadRedirectionForExistingInstallationsNewInstall() { |
| Mockito.when(configDepot.isNewConfig(StorageManager.DataStoreDownloadFollowRedirects)) |
| .thenReturn(true); |
| Mockito.when(dataCenterDao.listAll(Mockito.any())) |
| .thenReturn(new ArrayList<>()); //new installation |
| storageManagerImpl.enableDefaultDatastoreDownloadRedirectionForExistingInstallations(); |
| Mockito.verify(configurationDao, Mockito.never()) |
| .update(StorageManager.DataStoreDownloadFollowRedirects.key(), StorageManager.DataStoreDownloadFollowRedirects.defaultValue()); |
| } |
| |
| @Test |
| public void getStoragePoolNonDestroyedVolumesLogTestNonDestroyedVolumesReturnLog() { |
| Mockito.doReturn(1L).when(storagePoolVOMock).getId(); |
| Mockito.doReturn(1L).when(volume1VOMock).getInstanceId(); |
| Mockito.doReturn("786633d1-a942-4374-9d56-322dd4b0d202").when(volume1VOMock).getUuid(); |
| Mockito.doReturn(1L).when(volume2VOMock).getInstanceId(); |
| Mockito.doReturn("ffb46333-e983-4c21-b5f0-51c5877a3805").when(volume2VOMock).getUuid(); |
| Mockito.doReturn("58760044-928f-4c4e-9fef-d0e48423595e").when(vmInstanceVOMock).getUuid(); |
| |
| Mockito.when(_volumeDao.findByPoolId(storagePoolVOMock.getId(), null)).thenReturn(List.of(volume1VOMock, volume2VOMock)); |
| Mockito.doReturn(vmInstanceVOMock).when(vmInstanceDao).findById(Mockito.anyLong()); |
| |
| String log = storageManagerImpl.getStoragePoolNonDestroyedVolumesLog(storagePoolVOMock.getId()); |
| String expected = String.format("[Volume [%s] (attached to VM [%s]), Volume [%s] (attached to VM [%s])]", volume1VOMock.getUuid(), vmInstanceVOMock.getUuid(), volume2VOMock.getUuid(), vmInstanceVOMock.getUuid()); |
| |
| Assert.assertEquals(expected, log); |
| } |
| |
| private ChangeStoragePoolScopeCmd mockChangeStoragePooolScopeCmd(String newScope) { |
| ChangeStoragePoolScopeCmd cmd = new ChangeStoragePoolScopeCmd(); |
| ReflectionTestUtils.setField(cmd, "id", 1L); |
| ReflectionTestUtils.setField(cmd, "clusterId", 1L); |
| ReflectionTestUtils.setField(cmd, "scope", newScope); |
| return cmd; |
| } |
| |
| private StoragePoolVO mockStoragePoolVOForChangeStoragePoolScope(ScopeType currentScope, StoragePoolStatus status) { |
| StoragePoolVO primaryStorage = new StoragePoolVO(); |
| primaryStorage.setId(1L); |
| primaryStorage.setDataCenterId(1L); |
| primaryStorage.setClusterId(1L); |
| primaryStorage.setStatus(StoragePoolStatus.Disabled); |
| primaryStorage.setScope(currentScope); |
| primaryStorage.setStatus(status); |
| return primaryStorage; |
| } |
| |
| private void prepareTestChangeStoragePoolScope(ScopeType currentScope, StoragePoolStatus status) { |
| final DataCenterVO zone = new DataCenterVO(1L, null, null, null, null, null, null, null, null, null, DataCenter.NetworkType.Advanced, null, null); |
| StoragePoolVO primaryStorage = mockStoragePoolVOForChangeStoragePoolScope(currentScope, status); |
| |
| Mockito.when(accountMgr.isRootAdmin(Mockito.any())).thenReturn(true); |
| Mockito.when(dataCenterDao.findById(1L)).thenReturn(zone); |
| Mockito.when(storagePoolDao.findById(1L)).thenReturn(primaryStorage); |
| } |
| |
| @Test(expected = InvalidParameterValueException.class) |
| public void testChangeStoragePoolScopeNotDisabledException() { |
| prepareTestChangeStoragePoolScope(ScopeType.CLUSTER, StoragePoolStatus.Initialized); |
| |
| ChangeStoragePoolScopeCmd cmd = mockChangeStoragePooolScopeCmd("ZONE"); |
| storageManagerImpl.changeStoragePoolScope(cmd); |
| } |
| |
| @Test(expected = InvalidParameterValueException.class) |
| public void testChangeStoragePoolScopeToZoneHypervisorNotSupported() { |
| prepareTestChangeStoragePoolScope(ScopeType.CLUSTER, StoragePoolStatus.Disabled); |
| |
| final ClusterVO cluster = new ClusterVO(); |
| cluster.setHypervisorType(String.valueOf(HypervisorType.XenServer)); |
| Mockito.when(clusterDao.findById(1L)).thenReturn(cluster); |
| |
| ChangeStoragePoolScopeCmd cmd = mockChangeStoragePooolScopeCmd("ZONE"); |
| storageManagerImpl.changeStoragePoolScope(cmd); |
| } |
| |
| @Test(expected = CloudRuntimeException.class) |
| public void testChangeStoragePoolScopeToClusterVolumesPresentException() { |
| prepareTestChangeStoragePoolScope(ScopeType.ZONE, StoragePoolStatus.Disabled); |
| |
| final ClusterVO cluster = new ClusterVO(); |
| Mockito.when(clusterDao.findById(1L)).thenReturn(cluster); |
| |
| VMInstanceVO instance = Mockito.mock(VMInstanceVO.class); |
| Pair<List<VMInstanceVO>, Integer> vms = new Pair<>(List.of(instance), 1); |
| Mockito.when(vmInstanceDao.listByVmsNotInClusterUsingPool(1L, 1L)).thenReturn(vms); |
| |
| ChangeStoragePoolScopeCmd cmd = mockChangeStoragePooolScopeCmd("CLUSTER"); |
| storageManagerImpl.changeStoragePoolScope(cmd); |
| } |
| |
| @Test |
| public void testCheckNFSMountOptionsForCreateNoNFSMountOptions() { |
| Map<String, String> details = new HashMap<>(); |
| try { |
| storageManagerImpl.checkNFSMountOptionsForCreate(details, HypervisorType.XenServer, ""); |
| } catch (Exception e) { |
| Assert.fail(); |
| } |
| } |
| |
| @Test |
| public void testCheckNFSMountOptionsForCreateNotKVM() { |
| Map<String, String> details = new HashMap<>(); |
| details.put(ApiConstants.NFS_MOUNT_OPTIONS, "vers=4.1"); |
| InvalidParameterValueException exception = assertThrows(InvalidParameterValueException.class, |
| () -> storageManagerImpl.checkNFSMountOptionsForCreate(details, HypervisorType.XenServer, "")); |
| Assert.assertEquals(exception.getMessage(), "NFS options can not be set for the hypervisor type " + HypervisorType.XenServer); |
| } |
| |
| @Test |
| public void testCheckNFSMountOptionsForCreateNotNFS() { |
| Map<String, String> details = new HashMap<>(); |
| details.put(ApiConstants.NFS_MOUNT_OPTIONS, "vers=4.1"); |
| InvalidParameterValueException exception = assertThrows(InvalidParameterValueException.class, |
| () -> storageManagerImpl.checkNFSMountOptionsForCreate(details, HypervisorType.KVM, "")); |
| Assert.assertEquals(exception.getMessage(), "NFS options can only be set on pool type " + Storage.StoragePoolType.NetworkFilesystem); |
| } |
| |
| @Test |
| public void testCheckNFSMountOptionsForUpdateNoNFSMountOptions() { |
| Map<String, String> details = new HashMap<>(); |
| StoragePoolVO pool = new StoragePoolVO(); |
| Long accountId = 1L; |
| try { |
| storageManagerImpl.checkNFSMountOptionsForUpdate(details, pool, accountId); |
| } catch (Exception e) { |
| Assert.fail(); |
| } |
| } |
| |
| @Test |
| public void testCheckNFSMountOptionsForUpdateNotRootAdmin() { |
| Map<String, String> details = new HashMap<>(); |
| StoragePoolVO pool = new StoragePoolVO(); |
| Long accountId = 1L; |
| details.put(ApiConstants.NFS_MOUNT_OPTIONS, "vers=4.1"); |
| Mockito.when(accountMgr.isRootAdmin(accountId)).thenReturn(false); |
| PermissionDeniedException exception = assertThrows(PermissionDeniedException.class, |
| () -> storageManagerImpl.checkNFSMountOptionsForUpdate(details, pool, accountId)); |
| Assert.assertEquals(exception.getMessage(), "Only root admin can modify nfs options"); |
| } |
| |
| @Test |
| public void testCheckNFSMountOptionsForUpdateNotKVM() { |
| Map<String, String> details = new HashMap<>(); |
| StoragePoolVO pool = new StoragePoolVO(); |
| Long accountId = 1L; |
| details.put(ApiConstants.NFS_MOUNT_OPTIONS, "vers=4.1"); |
| Mockito.when(accountMgr.isRootAdmin(accountId)).thenReturn(true); |
| pool.setHypervisor(HypervisorType.XenServer); |
| InvalidParameterValueException exception = assertThrows(InvalidParameterValueException.class, |
| () -> storageManagerImpl.checkNFSMountOptionsForUpdate(details, pool, accountId)); |
| Assert.assertEquals(exception.getMessage(), "NFS options can only be set for the hypervisor type " + HypervisorType.KVM); |
| } |
| |
| @Test |
| public void testCheckNFSMountOptionsForUpdateNotNFS() { |
| Map<String, String> details = new HashMap<>(); |
| StoragePoolVO pool = new StoragePoolVO(); |
| Long accountId = 1L; |
| details.put(ApiConstants.NFS_MOUNT_OPTIONS, "vers=4.1"); |
| Mockito.when(accountMgr.isRootAdmin(accountId)).thenReturn(true); |
| pool.setHypervisor(HypervisorType.KVM); |
| pool.setPoolType(Storage.StoragePoolType.FiberChannel); |
| InvalidParameterValueException exception = assertThrows(InvalidParameterValueException.class, |
| () -> storageManagerImpl.checkNFSMountOptionsForUpdate(details, pool, accountId)); |
| Assert.assertEquals(exception.getMessage(), "NFS options can only be set on pool type " + Storage.StoragePoolType.NetworkFilesystem); |
| } |
| |
| @Test |
| public void testCheckNFSMountOptionsForUpdateNotMaintenance() { |
| Map<String, String> details = new HashMap<>(); |
| StoragePoolVO pool = new StoragePoolVO(); |
| Long accountId = 1L; |
| details.put(ApiConstants.NFS_MOUNT_OPTIONS, "vers=4.1"); |
| Mockito.when(accountMgr.isRootAdmin(accountId)).thenReturn(true); |
| pool.setHypervisor(HypervisorType.KVM); |
| pool.setPoolType(Storage.StoragePoolType.NetworkFilesystem); |
| pool.setStatus(StoragePoolStatus.Up); |
| InvalidParameterValueException exception = assertThrows(InvalidParameterValueException.class, |
| () -> storageManagerImpl.checkNFSMountOptionsForUpdate(details, pool, accountId)); |
| Assert.assertEquals(exception.getMessage(), "The storage pool should be in maintenance mode to edit nfs options"); |
| } |
| |
| @Test(expected = InvalidParameterValueException.class) |
| public void testDuplicateNFSMountOptions() { |
| String nfsMountOpts = "vers=4.1, nconnect=4,vers=4.2"; |
| Map<String, String> details = new HashMap<>(); |
| details.put(ApiConstants.NFS_MOUNT_OPTIONS, nfsMountOpts); |
| storageManagerImpl.checkNFSMountOptionsForCreate(details, HypervisorType.KVM, "nfs"); |
| } |
| |
| @Test(expected = InvalidParameterValueException.class) |
| public void testInvalidNFSMountOptions() { |
| String nfsMountOpts = "vers=4.1=2,"; |
| Map<String, String> details = new HashMap<>(); |
| details.put(ApiConstants.NFS_MOUNT_OPTIONS, nfsMountOpts); |
| StoragePoolVO pool = new StoragePoolVO(); |
| pool.setHypervisor(HypervisorType.KVM); |
| pool.setPoolType(Storage.StoragePoolType.NetworkFilesystem); |
| pool.setStatus(StoragePoolStatus.Maintenance); |
| Long accountId = 1L; |
| Mockito.when(accountMgr.isRootAdmin(accountId)).thenReturn(true); |
| storageManagerImpl.checkNFSMountOptionsForUpdate(details, pool, accountId); |
| } |
| |
| @Test |
| public void testGetStoragePoolMountOptionsNotNFS() { |
| StoragePoolVO pool = new StoragePoolVO(); |
| |
| pool.setPoolType(Storage.StoragePoolType.FiberChannel); |
| Pair<Map<String, String>, Boolean> details = storageManagerImpl.getStoragePoolNFSMountOpts(pool, null); |
| Assert.assertEquals(details.second(), false); |
| Assert.assertEquals(details.first(), null); |
| } |
| |
| @Test |
| public void testGetStoragePoolMountOptions() { |
| Long poolId = 1L; |
| String key = "nfsmountopts"; |
| String value = "vers=4.1,nconnect=2"; |
| StoragePoolDetailVO nfsMountOpts = new StoragePoolDetailVO(poolId, key, value, true); |
| StoragePoolVO pool = new StoragePoolVO(); |
| pool.setId(poolId); |
| pool.setPoolType(Storage.StoragePoolType.NetworkFilesystem); |
| Mockito.when(storagePoolDetailsDao.findDetail(poolId, ApiConstants.NFS_MOUNT_OPTIONS)).thenReturn(nfsMountOpts); |
| |
| Pair<Map<String, String>, Boolean> details = storageManagerImpl.getStoragePoolNFSMountOpts(pool, null); |
| Assert.assertEquals(details.second(), true); |
| Assert.assertEquals(details.first().get(key), value); |
| } |
| |
| @Test |
| public void testGetStoragePoolMountFailureReason() { |
| String error = "Mount failed on kvm host. An incorrect mount option was specified.\nIncorrect mount option."; |
| String failureReason = storageManagerImpl.getStoragePoolMountFailureReason(error); |
| Assert.assertEquals(failureReason, "An incorrect mount option was specified"); |
| } |
| |
| private void overrideDefaultConfigValue(final ConfigKey configKey, final String name, final Object o) throws IllegalAccessException, NoSuchFieldException { |
| Field f = ConfigKey.class.getDeclaredField(name); |
| f.setAccessible(true); |
| f.set(configKey, o); |
| } |
| |
| private Long testCheckPoolforSpaceForResizeSetup(StoragePoolVO pool, Long allocatedSizeWithTemplate) { |
| Long poolId = 10L; |
| Long zoneId = 2L; |
| |
| Long capacityBytes = (long) (allocatedSizeWithTemplate / Double.valueOf(CapacityManager.StorageAllocatedCapacityDisableThreshold.defaultValue()) |
| / Double.valueOf(CapacityManager.StorageOverprovisioningFactor.defaultValue())); |
| Long maxAllocatedSizeForResize = (long) (capacityBytes * Double.valueOf(CapacityManager.StorageOverprovisioningFactor.defaultValue()) |
| * Double.valueOf(CapacityManager.StorageAllocatedCapacityDisableThresholdForVolumeSize.defaultValue())); |
| |
| System.out.println("maxAllocatedSizeForResize = " + maxAllocatedSizeForResize); |
| System.out.println("allocatedSizeWithTemplate = " + allocatedSizeWithTemplate); |
| |
| Mockito.when(pool.getId()).thenReturn(poolId); |
| Mockito.when(pool.getCapacityBytes()).thenReturn(capacityBytes); |
| Mockito.when(storagePoolDao.findById(poolId)).thenReturn(pool); |
| Mockito.when(pool.getPoolType()).thenReturn(Storage.StoragePoolType.NetworkFilesystem); |
| |
| return maxAllocatedSizeForResize - allocatedSizeWithTemplate; |
| } |
| |
| @Test |
| public void testCheckPoolforSpaceForResize1() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Long allocatedSizeWithTemplate = 100L * 1024 * 1024 * 1024; |
| |
| Long maxAskingSize = testCheckPoolforSpaceForResizeSetup(pool, allocatedSizeWithTemplate); |
| Long totalAskingSize = maxAskingSize / 2; |
| |
| boolean result = storageManagerImpl.checkPoolforSpace(pool, allocatedSizeWithTemplate, totalAskingSize, false); |
| Assert.assertFalse(result); |
| } |
| |
| @Test |
| public void testCheckPoolforSpaceForResize2() { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Long allocatedSizeWithTemplate = 100L * 1024 * 1024 * 1024; |
| |
| Long maxAskingSize = testCheckPoolforSpaceForResizeSetup(pool, allocatedSizeWithTemplate); |
| Long totalAskingSize = maxAskingSize / 2; |
| |
| boolean result = storageManagerImpl.checkPoolforSpace(pool, allocatedSizeWithTemplate, totalAskingSize, true); |
| Assert.assertFalse(result); |
| } |
| |
| @Test |
| public void testCheckPoolforSpaceForResize3() throws NoSuchFieldException, IllegalAccessException { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Long allocatedSizeWithTemplate = 100L * 1024 * 1024 * 1024; |
| |
| Long maxAskingSize = testCheckPoolforSpaceForResizeSetup(pool, allocatedSizeWithTemplate); |
| Long totalAskingSize = maxAskingSize + 1; |
| overrideDefaultConfigValue(StorageManagerImpl.AllowVolumeReSizeBeyondAllocation, "_defaultValue", "true"); |
| |
| boolean result = storageManagerImpl.checkPoolforSpace(pool, allocatedSizeWithTemplate, totalAskingSize, true); |
| Assert.assertFalse(result); |
| } |
| |
| @Test |
| public void testCheckPoolforSpaceForResize4() throws NoSuchFieldException, IllegalAccessException { |
| StoragePoolVO pool = Mockito.mock(StoragePoolVO.class); |
| Long allocatedSizeWithTemplate = 100L * 1024 * 1024 * 1024; |
| |
| Long maxAskingSize = testCheckPoolforSpaceForResizeSetup(pool, allocatedSizeWithTemplate); |
| Long totalAskingSize = maxAskingSize / 2; |
| overrideDefaultConfigValue(StorageManagerImpl.AllowVolumeReSizeBeyondAllocation, "_defaultValue", "true"); |
| |
| boolean result = storageManagerImpl.checkPoolforSpace(pool, allocatedSizeWithTemplate, totalAskingSize, true); |
| assertTrue(result); |
| } |
| |
| @Test |
| public void testGetStorageAccessGroupsOnHostAllSAGsPresent() { |
| long hostId = 1L; |
| |
| HostVO host = Mockito.mock(HostVO.class); |
| ClusterVO cluster = Mockito.mock(ClusterVO.class); |
| HostPodVO pod = Mockito.mock(HostPodVO.class); |
| DataCenterVO zone = Mockito.mock(DataCenterVO.class); |
| |
| Mockito.when(hostDao.findById(hostId)).thenReturn(host); |
| Mockito.when(host.getClusterId()).thenReturn(2L); |
| Mockito.when(clusterDao.findById(2L)).thenReturn(cluster); |
| Mockito.when(cluster.getPodId()).thenReturn(3L); |
| Mockito.when(podDao.findById(3L)).thenReturn(pod); |
| Mockito.when(pod.getDataCenterId()).thenReturn(4L); |
| Mockito.when(dataCenterDao.findById(4L)).thenReturn(zone); |
| |
| Mockito.when(host.getStorageAccessGroups()).thenReturn("sag1"); |
| Mockito.when(cluster.getStorageAccessGroups()).thenReturn("sag2"); |
| Mockito.when(pod.getStorageAccessGroups()).thenReturn("sag3"); |
| Mockito.when(zone.getStorageAccessGroups()).thenReturn("sag4"); |
| |
| String[] sags = storageManagerImpl.getStorageAccessGroups(null, null, null, hostId); |
| |
| assertNotNull(sags); |
| assertEquals(4, sags.length); |
| assertEquals("sag1", sags[0]); |
| assertEquals("sag2", sags[1]); |
| assertEquals("sag3", sags[2]); |
| assertEquals("sag4", sags[3]); |
| } |
| |
| @Test |
| public void testGetSingleStorageAccessGroupOnHost() { |
| long hostId = 1L; |
| |
| HostVO host = Mockito.mock(HostVO.class); |
| ClusterVO cluster = Mockito.mock(ClusterVO.class); |
| HostPodVO pod = Mockito.mock(HostPodVO.class); |
| DataCenterVO zone = Mockito.mock(DataCenterVO.class); |
| |
| Mockito.when(hostDao.findById(hostId)).thenReturn(host); |
| Mockito.when(host.getClusterId()).thenReturn(2L); |
| Mockito.when(clusterDao.findById(2L)).thenReturn(cluster); |
| Mockito.when(cluster.getPodId()).thenReturn(3L); |
| Mockito.when(podDao.findById(3L)).thenReturn(pod); |
| Mockito.when(pod.getDataCenterId()).thenReturn(4L); |
| Mockito.when(dataCenterDao.findById(4L)).thenReturn(zone); |
| |
| Mockito.when(host.getStorageAccessGroups()).thenReturn(""); |
| Mockito.when(cluster.getStorageAccessGroups()).thenReturn("sag2"); |
| Mockito.when(pod.getStorageAccessGroups()).thenReturn(null); |
| |
| String[] sags = storageManagerImpl.getStorageAccessGroups(null, null, null, hostId); |
| |
| assertNotNull(sags); |
| assertEquals(1, sags.length); |
| assertEquals("sag2", sags[0]); |
| } |
| |
| @Test |
| public void testGetStoragePoolIopsStats_ReturnsDriverResultWhenNotNull() { |
| StoragePool pool = Mockito.mock(StoragePool.class); |
| PrimaryDataStoreDriver primaryStoreDriver = Mockito.mock(PrimaryDataStoreDriver.class); |
| Pair<Long, Long> expectedResult = new Pair<>(1000L, 500L); |
| Mockito.when(primaryStoreDriver.getStorageIopsStats(pool)).thenReturn(expectedResult); |
| |
| Pair<Long, Long> result = storageManagerImpl.getStoragePoolIopsStats(primaryStoreDriver, pool); |
| |
| Assert.assertSame("Should return the result from primaryStoreDriver.getStorageIopsStats", expectedResult, result); |
| Mockito.verify(primaryStoreDriver, Mockito.never()).getUsedIops(Mockito.any()); |
| Mockito.verify(pool, Mockito.never()).getCapacityIops(); |
| } |
| |
| @Test |
| public void testGetStoragePoolIopsStats_UsedIopsPositive() { |
| StoragePool pool = Mockito.mock(StoragePool.class); |
| PrimaryDataStoreDriver primaryStoreDriver = Mockito.mock(PrimaryDataStoreDriver.class); |
| Mockito.when(primaryStoreDriver.getStorageIopsStats(pool)).thenReturn(null); |
| Mockito.when(primaryStoreDriver.getUsedIops(pool)).thenReturn(500L); |
| Mockito.when(pool.getCapacityIops()).thenReturn(1000L); |
| |
| Pair<Long, Long> result = storageManagerImpl.getStoragePoolIopsStats(primaryStoreDriver, pool); |
| |
| Assert.assertNotNull(result); |
| Assert.assertEquals("Capacity IOPS should match pool's capacity IOPS", 1000L, result.first().longValue()); |
| Assert.assertEquals("Used IOPS should match the positive value returned", 500L, result.second().longValue()); |
| } |
| |
| @Test |
| public void testGetStoragePoolIopsStats_UsedIopsZero() { |
| StoragePool pool = Mockito.mock(StoragePool.class); |
| PrimaryDataStoreDriver primaryStoreDriver = Mockito.mock(PrimaryDataStoreDriver.class); |
| Mockito.when(primaryStoreDriver.getStorageIopsStats(pool)).thenReturn(null); |
| Mockito.when(primaryStoreDriver.getUsedIops(pool)).thenReturn(0L); |
| Mockito.when(pool.getCapacityIops()).thenReturn(1000L); |
| |
| Pair<Long, Long> result = storageManagerImpl.getStoragePoolIopsStats(primaryStoreDriver, pool); |
| |
| Assert.assertNotNull(result); |
| Assert.assertEquals("Capacity IOPS should match pool's capacity IOPS", 1000L, result.first().longValue()); |
| Assert.assertNull("Used IOPS should be null when usedIops <= 0", result.second()); |
| } |
| |
| @Test |
| public void testGetStoragePoolIopsStats_UsedIopsNegative() { |
| StoragePool pool = Mockito.mock(StoragePool.class); |
| PrimaryDataStoreDriver primaryStoreDriver = Mockito.mock(PrimaryDataStoreDriver.class); |
| Mockito.when(primaryStoreDriver.getStorageIopsStats(pool)).thenReturn(null); |
| Mockito.when(primaryStoreDriver.getUsedIops(pool)).thenReturn(-100L); |
| Mockito.when(pool.getCapacityIops()).thenReturn(1000L); |
| |
| Pair<Long, Long> result = storageManagerImpl.getStoragePoolIopsStats(primaryStoreDriver, pool); |
| |
| Assert.assertNotNull(result); |
| Assert.assertEquals("Capacity IOPS should match pool's capacity IOPS", 1000L, result.first().longValue()); |
| Assert.assertNull("Used IOPS should be null when usedIops <= 0", result.second()); |
| } |
| |
| |
| @Test |
| public void testNoStorageAccessGroupsOnHostAndStoragePool() { |
| HostVO host = Mockito.mock(HostVO.class); |
| StoragePoolVO storagePool = Mockito.mock(StoragePoolVO.class); |
| long hostId = 1L; |
| long poolId = 2L; |
| |
| Mockito.when(host.getId()).thenReturn(hostId); |
| doReturn(new String[0]).when(storageManagerImpl).getStorageAccessGroups(null, null, null, hostId); |
| |
| Mockito.when(storagePool.getId()).thenReturn(poolId); |
| storageManagerImpl._storagePoolAccessGroupMapDao = storagePoolAccessGroupMapDao; |
| Mockito.when(storagePoolAccessGroupMapDao.getStorageAccessGroups(poolId)) |
| .thenReturn(new ArrayList<>()); |
| |
| boolean result = storageManagerImpl.checkIfHostAndStoragePoolHasCommonStorageAccessGroups(host, storagePool); |
| |
| assertTrue("Host without storage access groups should connect to a storage pool without storage access groups.", result); |
| } |
| |
| @Test |
| public void testHostWithStorageAccessGroupsAndStoragePoolWithoutStorageAccessGroups() { |
| HostVO host = Mockito.mock(HostVO.class); |
| StoragePoolVO storagePool = Mockito.mock(StoragePoolVO.class); |
| long hostId = 1L; |
| long poolId = 2L; |
| |
| Mockito.when(host.getId()).thenReturn(hostId); |
| doReturn(new String[]{"StorageAccessGroup1"}).when(storageManagerImpl).getStorageAccessGroups(null, null, null, hostId); |
| |
| Mockito.when(storagePool.getId()).thenReturn(poolId); |
| storageManagerImpl._storagePoolAccessGroupMapDao = storagePoolAccessGroupMapDao; |
| Mockito.when(storagePoolAccessGroupMapDao.getStorageAccessGroups(poolId)) |
| .thenReturn(new ArrayList<>()); |
| |
| boolean result = storageManagerImpl.checkIfHostAndStoragePoolHasCommonStorageAccessGroups(host, storagePool); |
| |
| assertTrue("Host with storage access groups should connect to a storage pool without storage access groups.", result); |
| } |
| |
| @Test |
| public void testHostWithStorageAccessGroupsAndStoragePoolWithDifferentStorageAccessGroups() { |
| HostVO host = Mockito.mock(HostVO.class); |
| StoragePoolVO storagePool = Mockito.mock(StoragePoolVO.class); |
| long hostId = 1L; |
| long poolId = 2L; |
| |
| Mockito.when(host.getId()).thenReturn(hostId); |
| doReturn(new String[]{"StorageAccessGroup1"}).when(storageManagerImpl).getStorageAccessGroups(null, null, null, hostId); |
| |
| Mockito.when(storagePool.getId()).thenReturn(poolId); |
| storageManagerImpl._storagePoolAccessGroupMapDao = storagePoolAccessGroupMapDao; |
| Mockito.when(storagePoolAccessGroupMapDao.getStorageAccessGroups(poolId)) |
| .thenReturn(Arrays.asList("StorageAccessGroup2", "StorageAccessGroup3")); |
| |
| boolean result = storageManagerImpl.checkIfHostAndStoragePoolHasCommonStorageAccessGroups(host, storagePool); |
| |
| Assert.assertFalse("Host with storage access groups should not connect to a storage pool with different storage access groups.", result); |
| } |
| |
| @Test |
| public void testHostWithStorageAccessGroupsAndStoragePoolWithMatchingStorageAccessGroups() { |
| HostVO host = Mockito.mock(HostVO.class); |
| StoragePoolVO storagePool = Mockito.mock(StoragePoolVO.class); |
| long hostId = 1L; |
| long poolId = 2L; |
| |
| Mockito.when(host.getId()).thenReturn(hostId); |
| doReturn(new String[]{"StorageAccessGroup1"}).when(storageManagerImpl).getStorageAccessGroups(null, null, null, hostId); |
| |
| Mockito.when(storagePool.getId()).thenReturn(poolId); |
| storageManagerImpl._storagePoolAccessGroupMapDao = storagePoolAccessGroupMapDao; |
| Mockito.when(storagePoolAccessGroupMapDao.getStorageAccessGroups(poolId)) |
| .thenReturn(Arrays.asList("StorageAccessGroup1", "StorageAccessGroup2")); |
| |
| boolean result = storageManagerImpl.checkIfHostAndStoragePoolHasCommonStorageAccessGroups(host, storagePool); |
| |
| assertTrue("Host with matching storage access groups should connect to a storage pool with matching storage access groups.", result); |
| } |
| |
| @Test |
| public void testHostWithEmptySAGsOnHost() { |
| HostVO host = Mockito.mock(HostVO.class); |
| StoragePoolVO storagePool = Mockito.mock(StoragePoolVO.class); |
| long hostId = 1L; |
| long poolId = 2L; |
| |
| Mockito.when(host.getId()).thenReturn(hostId); |
| doReturn(new String[0]).when(storageManagerImpl).getStorageAccessGroups(null, null, null, hostId); |
| |
| Mockito.when(storagePool.getId()).thenReturn(poolId); |
| storageManagerImpl._storagePoolAccessGroupMapDao = storagePoolAccessGroupMapDao; |
| Mockito.when(storagePoolAccessGroupMapDao.getStorageAccessGroups(poolId)) |
| .thenReturn(Arrays.asList("StorageAccessGroup1", "StorageAccessGroup2")); |
| |
| boolean result = storageManagerImpl.checkIfHostAndStoragePoolHasCommonStorageAccessGroups(host, storagePool); |
| |
| Assert.assertFalse("Host with matching storage access groups should connect to a storage pool with matching storage access groups.", result); |
| } |
| |
| @Test |
| public void testVolumeReadyNoVMOrVMStoppedAndPoolsWithMatchingStorageAccessGroups() { |
| StoragePoolVO destPool = Mockito.mock(StoragePoolVO.class); |
| Volume volume = Mockito.mock(Volume.class); |
| long srcPoolId = 2L; |
| long destPoolId = 3L; |
| |
| Mockito.when(volume.getState()).thenReturn(Volume.State.Ready); |
| Mockito.when(volume.getInstanceId()).thenReturn(null); |
| Mockito.when(volume.getPoolId()).thenReturn(srcPoolId); |
| |
| Mockito.when(destPool.getId()).thenReturn(destPoolId); |
| |
| List<String> srcStorageAccessGroups = Arrays.asList("StorageAccessGroup1", "StorageAccessGroup2"); |
| List<String> destStorageAccessGroups = Arrays.asList("StorageAccessGroup1", "StorageAccessGroup2"); |
| |
| doReturn(srcStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(srcPoolId); |
| doReturn(destStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(destPoolId); |
| |
| Pair<Boolean, String> result = storageManagerImpl.checkIfReadyVolumeFitsInStoragePoolWithStorageAccessGroups(destPool, volume); |
| |
| assertTrue("Volume in Ready state and no VM or VM stopped should migrate if both pools have matching storage access groups.", result.first()); |
| } |
| |
| @Test |
| public void testVolumeReadyNoVMOrVMStoppedAndPoolsWithEmptyStorageAccessGroups() { |
| StoragePoolVO destPool = Mockito.mock(StoragePoolVO.class); |
| Volume volume = Mockito.mock(Volume.class); |
| long srcPoolId = 2L; |
| long destPoolId = 3L; |
| |
| Mockito.when(volume.getState()).thenReturn(Volume.State.Ready); |
| Mockito.when(volume.getInstanceId()).thenReturn(null); |
| Mockito.when(volume.getPoolId()).thenReturn(srcPoolId); |
| |
| Mockito.when(destPool.getId()).thenReturn(destPoolId); |
| |
| List<String> srcStorageAccessGroups = new ArrayList<>(); |
| List<String> destStorageAccessGroups = new ArrayList<>(); |
| |
| doReturn(srcStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(srcPoolId); |
| doReturn(destStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(destPoolId); |
| |
| Pair<Boolean, String> result = storageManagerImpl.checkIfReadyVolumeFitsInStoragePoolWithStorageAccessGroups(destPool, volume); |
| |
| assertTrue("Volume with empty storage access groups should be able to fit in the destination pool.", result.first()); |
| } |
| |
| @Test |
| public void testVolumeReadyVMRunningAndHostHasCommonSAGsForBothPools() { |
| StoragePoolVO destPool = Mockito.mock(StoragePoolVO.class); |
| Volume volume = Mockito.mock(Volume.class); |
| long vmId = 10L; |
| long srcPoolId = 2L; |
| long destPoolId = 3L; |
| |
| Mockito.when(volume.getState()).thenReturn(Volume.State.Ready); |
| Mockito.when(volume.getInstanceId()).thenReturn(vmId); |
| Mockito.when(volume.getPoolId()).thenReturn(srcPoolId); |
| |
| Mockito.when(destPool.getId()).thenReturn(destPoolId); |
| |
| List<String> srcStorageAccessGroups = Arrays.asList("StorageAccessGroup1", "StorageAccessGroup2"); |
| List<String> destStorageAccessGroups = Arrays.asList("StorageAccessGroup2", "StorageAccessGroup3"); |
| |
| doReturn(srcStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(srcPoolId); |
| doReturn(destStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(destPoolId); |
| |
| Pair<Boolean, String> result = storageManagerImpl.checkIfReadyVolumeFitsInStoragePoolWithStorageAccessGroups(destPool, volume); |
| |
| assertTrue("Volume with host having common storage access groups should fit in both source and destination pools.", result.first()); |
| } |
| |
| @Test |
| public void testVolumeReadyVMRunningAndHostHasCommonSAGForSourcePoolButNotDestinationPool() { |
| StoragePoolVO destPool = Mockito.mock(StoragePoolVO.class); |
| Volume volume = Mockito.mock(Volume.class); |
| StoragePoolVO srcPool = Mockito.mock(StoragePoolVO.class); |
| long vmId = 10L; |
| long srcPoolId = 2L; |
| long destPoolId = 3L; |
| |
| Mockito.when(volume.getState()).thenReturn(Volume.State.Ready); |
| Mockito.when(volume.getInstanceId()).thenReturn(vmId); |
| Mockito.when(volume.getPoolId()).thenReturn(srcPoolId); |
| |
| Mockito.when(destPool.getId()).thenReturn(destPoolId); |
| Mockito.when(srcPool.getId()).thenReturn(destPoolId); |
| Mockito.doReturn(srcPool).when(storagePoolDao).findById(srcPoolId); |
| |
| List<String> srcStorageAccessGroups = Arrays.asList("StorageAccessGroup1", "StorageAccessGroup2"); |
| List<String> destStorageAccessGroups = Arrays.asList("StorageAccessGroup3", "StorageAccessGroup4"); |
| |
| doReturn(srcStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(srcPoolId); |
| doReturn(destStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(destPoolId); |
| List<Long> poolIds = new ArrayList<>(); |
| poolIds.add(srcPool.getId()); |
| poolIds.add(destPool.getId()); |
| Mockito.doReturn(null).when(storageManagerImpl).findUpAndEnabledHostWithAccessToStoragePools(poolIds); |
| |
| Pair<Boolean, String> result = storageManagerImpl.checkIfReadyVolumeFitsInStoragePoolWithStorageAccessGroups(destPool, volume); |
| |
| Assert.assertFalse("Volume with host having common storage access group for source pool but not destination pool should not fit.", result.first()); |
| } |
| |
| @Test |
| public void testNoCommonHostConnected() { |
| StoragePoolVO destPool = Mockito.mock(StoragePoolVO.class); |
| StoragePoolVO srcPool = Mockito.mock(StoragePoolVO.class); |
| Volume volume = Mockito.mock(Volume.class); |
| long vmId = 10L; |
| long srcPoolId = 2L; |
| long destPoolId = 3L; |
| |
| Mockito.when(volume.getState()).thenReturn(Volume.State.Ready); |
| Mockito.when(volume.getInstanceId()).thenReturn(vmId); |
| Mockito.when(volume.getPoolId()).thenReturn(srcPoolId); |
| |
| Mockito.when(destPool.getId()).thenReturn(destPoolId); |
| Mockito.when(srcPool.getId()).thenReturn(destPoolId); |
| Mockito.doReturn(srcPool).when(storagePoolDao).findById(srcPoolId); |
| List<String> srcStorageAccessGroups = Arrays.asList("StorageAccessGroup3", "StorageAccessGroup4"); |
| List<String> destStorageAccessGroups = Arrays.asList("StorageAccessGroup1", "StorageAccessGroup2"); |
| |
| Mockito.doReturn(srcStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(srcPoolId); |
| Mockito.doReturn(destStorageAccessGroups).when(storagePoolAccessGroupMapDao).getStorageAccessGroups(destPoolId); |
| List<Long> poolIds = new ArrayList<>(); |
| poolIds.add(srcPool.getId()); |
| poolIds.add(destPool.getId()); |
| Mockito.doReturn(null).when(storageManagerImpl).findUpAndEnabledHostWithAccessToStoragePools(poolIds); |
| Pair<Boolean, String> result = storageManagerImpl.checkIfReadyVolumeFitsInStoragePoolWithStorageAccessGroups(destPool, volume); |
| |
| Assert.assertFalse("Volume with host having common storage access group for destination pool but not source pool should not fit.", result.first()); |
| Assert.assertEquals("No common host connected to source and destination storages", result.second()); |
| } |
| |
| @Test |
| public void testConfigureStorageAccess_SkipUpdateForZone() { |
| Long zoneId = 1L; |
| List<String> storageAccessGroups = Arrays.asList("sag1", "sag2"); |
| |
| ConfigureStorageAccessCmd cmd = Mockito.mock(ConfigureStorageAccessCmd.class); |
| Mockito.when(cmd.getZoneId()).thenReturn(zoneId); |
| Mockito.when(cmd.getPodId()).thenReturn(null); |
| Mockito.when(cmd.getClusterId()).thenReturn(null); |
| Mockito.when(cmd.getHostId()).thenReturn(null); |
| Mockito.when(cmd.getStorageId()).thenReturn(null); |
| Mockito.when(cmd.getStorageAccessGroups()).thenReturn(storageAccessGroups); |
| |
| DataCenterVO zone = Mockito.mock(DataCenterVO.class); |
| Mockito.when(zone.getStorageAccessGroups()).thenReturn("sag2,sag1"); |
| Mockito.when(dataCenterDao.findById(zoneId)).thenReturn(zone); |
| |
| boolean result = storageManagerImpl.configureStorageAccess(cmd); |
| |
| Mockito.verify(resourceMgr, Mockito.never()).updateZoneStorageAccessGroups(Mockito.anyLong(), Mockito.anyList()); |
| Mockito.verify(dataCenterDao, Mockito.never()).update(Mockito.eq(zoneId), Mockito.any(DataCenterVO.class)); |
| |
| assertTrue(result); |
| } |
| |
| @Test |
| public void testConfigureStorageAccess_SkipUpdateForPod() { |
| Long podId = 1L; |
| List<String> storageAccessGroups = Arrays.asList("sag1", "sag2"); |
| |
| ConfigureStorageAccessCmd cmd = Mockito.mock(ConfigureStorageAccessCmd.class); |
| Mockito.when(cmd.getZoneId()).thenReturn(null); |
| Mockito.when(cmd.getPodId()).thenReturn(podId); |
| Mockito.when(cmd.getClusterId()).thenReturn(null); |
| Mockito.when(cmd.getHostId()).thenReturn(null); |
| Mockito.when(cmd.getStorageId()).thenReturn(null); |
| Mockito.when(cmd.getStorageAccessGroups()).thenReturn(storageAccessGroups); |
| |
| HostPodVO pod = Mockito.mock(HostPodVO.class); |
| Mockito.when(pod.getDataCenterId()).thenReturn(1L); |
| Mockito.when(pod.getStorageAccessGroups()).thenReturn("sag1,sag2"); |
| Mockito.when(podDao.findById(podId)).thenReturn(pod); |
| Mockito.doNothing().when(storageManagerImpl).checkIfStorageAccessGroupsExistsOnZone(1L, storageAccessGroups); |
| |
| boolean result = storageManagerImpl.configureStorageAccess(cmd); |
| |
| Mockito.verify(resourceMgr, Mockito.never()).updatePodStorageAccessGroups(Mockito.anyLong(), Mockito.anyList()); |
| Mockito.verify(podDao, Mockito.never()).update(Mockito.eq(podId), Mockito.any(HostPodVO.class)); |
| |
| assertTrue(result); |
| } |
| |
| @Test |
| public void testConfigureStorageAccess_SkipUpdateForCluster() { |
| Long clusterId = 1L; |
| List<String> storageAccessGroups = Arrays.asList("sag1", "sag2"); |
| |
| ConfigureStorageAccessCmd cmd = Mockito.mock(ConfigureStorageAccessCmd.class); |
| Mockito.when(cmd.getZoneId()).thenReturn(null); |
| Mockito.when(cmd.getPodId()).thenReturn(null); |
| Mockito.when(cmd.getClusterId()).thenReturn(clusterId); |
| Mockito.when(cmd.getHostId()).thenReturn(null); |
| Mockito.when(cmd.getStorageId()).thenReturn(null); |
| Mockito.when(cmd.getStorageAccessGroups()).thenReturn(storageAccessGroups); |
| |
| ClusterVO cluster = Mockito.mock(ClusterVO.class); |
| Mockito.when(cluster.getPodId()).thenReturn(1L); |
| Mockito.when(cluster.getStorageAccessGroups()).thenReturn("sag1,sag2"); |
| Mockito.when(clusterDao.findById(clusterId)).thenReturn(cluster); |
| Mockito.doNothing().when(storageManagerImpl).checkIfStorageAccessGroupsExistsOnPod(1L, storageAccessGroups); |
| |
| boolean result = storageManagerImpl.configureStorageAccess(cmd); |
| |
| Mockito.verify(resourceMgr, Mockito.never()).updateClusterStorageAccessGroups(Mockito.anyLong(), Mockito.anyList()); |
| Mockito.verify(clusterDao, Mockito.never()).update(Mockito.eq(clusterId), Mockito.any(ClusterVO.class)); |
| |
| assertTrue(result); |
| } |
| |
| @Test |
| public void testConfigureStorageAccess_SkipUpdateForHost() { |
| Long hostId = 1L; |
| List<String> storageAccessGroups = Arrays.asList("sag1", "sag2"); |
| |
| ConfigureStorageAccessCmd cmd = Mockito.mock(ConfigureStorageAccessCmd.class); |
| Mockito.when(cmd.getZoneId()).thenReturn(null); |
| Mockito.when(cmd.getPodId()).thenReturn(null); |
| Mockito.when(cmd.getClusterId()).thenReturn(null); |
| Mockito.when(cmd.getHostId()).thenReturn(hostId); |
| Mockito.when(cmd.getStorageId()).thenReturn(null); |
| Mockito.when(cmd.getStorageAccessGroups()).thenReturn(storageAccessGroups); |
| |
| HostVO host = Mockito.mock(HostVO.class); |
| Mockito.when(host.getClusterId()).thenReturn(1L); |
| Mockito.when(host.getStorageAccessGroups()).thenReturn("sag1,sag2"); |
| Mockito.when(hostDao.findById(hostId)).thenReturn(host); |
| Mockito.doNothing().when(storageManagerImpl).checkIfStorageAccessGroupsExistsOnCluster(1L, storageAccessGroups); |
| |
| boolean result = storageManagerImpl.configureStorageAccess(cmd); |
| |
| Mockito.verify(resourceMgr, Mockito.never()).updateHostStorageAccessGroups(Mockito.anyLong(), Mockito.anyList()); |
| Mockito.verify(hostDao, Mockito.never()).update(Mockito.eq(hostId), Mockito.any(HostVO.class)); |
| |
| assertTrue(result); |
| } |
| |
| @Test |
| public void testConfigureStorageAccess_InvalidNonNullCount() { |
| ConfigureStorageAccessCmd cmd = Mockito.mock(ConfigureStorageAccessCmd.class); |
| |
| Mockito.when(cmd.getZoneId()).thenReturn(1L); |
| Mockito.when(cmd.getPodId()).thenReturn(1L); |
| Mockito.when(cmd.getClusterId()).thenReturn(null); |
| Mockito.when(cmd.getHostId()).thenReturn(null); |
| Mockito.when(cmd.getStorageId()).thenReturn(null); |
| |
| try { |
| storageManagerImpl.configureStorageAccess(cmd); |
| Assert.fail("Expected IllegalArgumentException to be thrown due to nonNullCount validation"); |
| } catch (IllegalArgumentException e) { |
| assertTrue(e.getMessage().contains("Exactly one of zoneid, podid, clusterid, hostid or storagepoolid is required")); |
| } |
| } |
| |
| @Test |
| public void testConfigureStorageAccess_MissingStorageAccessGroups() { |
| ConfigureStorageAccessCmd cmd = Mockito.mock(ConfigureStorageAccessCmd.class); |
| |
| Mockito.when(cmd.getZoneId()).thenReturn(1L); |
| Mockito.when(cmd.getPodId()).thenReturn(null); |
| Mockito.when(cmd.getClusterId()).thenReturn(null); |
| Mockito.when(cmd.getHostId()).thenReturn(null); |
| Mockito.when(cmd.getStorageId()).thenReturn(null); |
| Mockito.when(cmd.getStorageAccessGroups()).thenReturn(null); |
| |
| try { |
| storageManagerImpl.configureStorageAccess(cmd); |
| Assert.fail("Expected InvalidParameterValueException to be thrown due to missing storageAccessGroups"); |
| } catch (InvalidParameterValueException e) { |
| assertTrue(e.getMessage().contains("storageaccessgroups parameter is required")); |
| } |
| } |
| |
| @Test |
| public void testCheckIfStorageAccessGroupsExistsOnZone_NoException() { |
| long zoneId = 1L; |
| List<String> newStorageAccessGroups = Arrays.asList("group1", "group2"); |
| |
| DataCenterVO zoneVO = Mockito.mock(DataCenterVO.class); |
| Mockito.when(dataCenterDao.findById(zoneId)).thenReturn(zoneVO); |
| Mockito.when(zoneVO.getStorageAccessGroups()).thenReturn("group3,group4"); |
| |
| storageManagerImpl.checkIfStorageAccessGroupsExistsOnZone(zoneId, newStorageAccessGroups); |
| } |
| |
| @Test |
| public void testCheckIfStorageAccessGroupsExistsOnZone_ThrowsException() { |
| long zoneId = 1L; |
| List<String> newStorageAccessGroups = Arrays.asList("group1", "group2", "group3"); |
| |
| DataCenterVO zoneVO = Mockito.mock(DataCenterVO.class); |
| Mockito.when(dataCenterDao.findById(zoneId)).thenReturn(zoneVO); |
| Mockito.when(zoneVO.getStorageAccessGroups()).thenReturn("group3,group4"); |
| |
| CloudRuntimeException thrownException = assertThrows(CloudRuntimeException.class, () -> { |
| storageManagerImpl.checkIfStorageAccessGroupsExistsOnZone(zoneId, newStorageAccessGroups); |
| }); |
| |
| assertTrue(thrownException.getMessage().contains("access groups already exist on the zone: [group3]")); |
| } |
| |
| @Test |
| public void testCheckIfStorageAccessGroupsExistsOnPod_NoException() { |
| long podId = 1L; |
| long zoneId = 2L; |
| List<String> newStorageAccessGroups = Arrays.asList("group1", "group2"); |
| |
| HostPodVO podVO = Mockito.mock(HostPodVO.class); |
| DataCenterVO zoneVO = Mockito.mock(DataCenterVO.class); |
| |
| Mockito.when(podVO.getDataCenterId()).thenReturn(zoneId); |
| Mockito.when(podVO.getStorageAccessGroups()).thenReturn("group3,group4"); |
| |
| Mockito.when(zoneVO.getStorageAccessGroups()).thenReturn("group5,group6"); |
| |
| Mockito.when(podDao.findById(podId)).thenReturn(podVO); |
| Mockito.when(dataCenterDao.findById(zoneId)).thenReturn(zoneVO); |
| |
| storageManagerImpl.checkIfStorageAccessGroupsExistsOnPod(podId, newStorageAccessGroups); |
| } |
| |
| @Test |
| public void testCheckIfStorageAccessGroupsExistsOnPod_ThrowsException() { |
| long podId = 1L; |
| long zoneId = 2L; |
| List<String> newStorageAccessGroups = Arrays.asList("group1", "group2", "group3"); |
| |
| HostPodVO podVO = Mockito.mock(HostPodVO.class); |
| DataCenterVO zoneVO = Mockito.mock(DataCenterVO.class); |
| |
| Mockito.when(podVO.getDataCenterId()).thenReturn(zoneId); |
| Mockito.when(podVO.getStorageAccessGroups()).thenReturn("group3,group4"); |
| |
| Mockito.when(zoneVO.getStorageAccessGroups()).thenReturn("group5,group6"); |
| |
| Mockito.when(podDao.findById(podId)).thenReturn(podVO); |
| Mockito.when(dataCenterDao.findById(zoneId)).thenReturn(zoneVO); |
| |
| CloudRuntimeException thrownException = assertThrows(CloudRuntimeException.class, () -> { |
| storageManagerImpl.checkIfStorageAccessGroupsExistsOnPod(podId, newStorageAccessGroups); |
| }); |
| |
| assertTrue(thrownException.getMessage().contains("access groups already exist on the pod: [group3]")); |
| } |
| |
| @Test |
| public void testCheckIfStorageAccessGroupsExistsOnCluster_NoException() { |
| long clusterId = 1L; |
| long podId = 2L; |
| long zoneId = 3L; |
| List<String> newStorageAccessGroups = Arrays.asList("group1", "group2"); |
| |
| ClusterVO clusterVO = Mockito.mock(ClusterVO.class); |
| HostPodVO podVO = Mockito.mock(HostPodVO.class); |
| DataCenterVO zoneVO = Mockito.mock(DataCenterVO.class); |
| |
| Mockito.when(clusterVO.getPodId()).thenReturn(podId); |
| Mockito.when(clusterVO.getStorageAccessGroups()).thenReturn("group4,group5"); |
| |
| Mockito.when(podVO.getDataCenterId()).thenReturn(zoneId); |
| Mockito.when(podVO.getStorageAccessGroups()).thenReturn("group6,group7"); |
| |
| Mockito.when(zoneVO.getStorageAccessGroups()).thenReturn("group8,group9"); |
| |
| Mockito.when(clusterDao.findById(clusterId)).thenReturn(clusterVO); |
| Mockito.when(podDao.findById(podId)).thenReturn(podVO); |
| Mockito.when(dataCenterDao.findById(zoneId)).thenReturn(zoneVO); |
| |
| storageManagerImpl.checkIfStorageAccessGroupsExistsOnCluster(clusterId, newStorageAccessGroups); |
| } |
| |
| @Test |
| public void testCheckIfStorageAccessGroupsExistsOnCluster_ThrowsException() { |
| long clusterId = 1L; |
| long podId = 2L; |
| long zoneId = 3L; |
| List<String> newStorageAccessGroups = Arrays.asList("group1", "group2", "group4"); |
| |
| ClusterVO clusterVO = Mockito.mock(ClusterVO.class); |
| HostPodVO podVO = Mockito.mock(HostPodVO.class); |
| DataCenterVO zoneVO = Mockito.mock(DataCenterVO.class); |
| |
| Mockito.when(clusterVO.getPodId()).thenReturn(podId); |
| Mockito.when(clusterVO.getStorageAccessGroups()).thenReturn("group4,group5"); |
| |
| Mockito.when(podVO.getDataCenterId()).thenReturn(zoneId); |
| Mockito.when(podVO.getStorageAccessGroups()).thenReturn("group6,group7"); |
| |
| Mockito.when(zoneVO.getStorageAccessGroups()).thenReturn("group8,group9"); |
| |
| Mockito.when(clusterDao.findById(clusterId)).thenReturn(clusterVO); |
| Mockito.when(podDao.findById(podId)).thenReturn(podVO); |
| Mockito.when(dataCenterDao.findById(zoneId)).thenReturn(zoneVO); |
| |
| CloudRuntimeException thrownException = assertThrows(CloudRuntimeException.class, () -> { |
| storageManagerImpl.checkIfStorageAccessGroupsExistsOnCluster(clusterId, newStorageAccessGroups); |
| }); |
| |
| assertTrue(thrownException.getMessage().contains("access groups already exist on the cluster: [group4]")); |
| } |
| |
| @Test |
| public void testGetObjectStorageUsedStats() { |
| Long zoneId = 1L; |
| List<ObjectStoreVO> objectStores = new ArrayList<>(); |
| |
| ObjectStoreVO store1 = new ObjectStoreVO(); |
| store1.setAllocatedSize(1000L); |
| store1.setTotalSize(2000L); |
| objectStores.add(store1); |
| |
| ObjectStoreVO store2 = new ObjectStoreVO(); |
| store2.setAllocatedSize(2000L); |
| store2.setTotalSize(4000L); |
| objectStores.add(store2); |
| |
| ObjectStoreVO store3 = new ObjectStoreVO(); |
| store3.setAllocatedSize(null); |
| store3.setTotalSize(null); |
| objectStores.add(store3); |
| |
| Mockito.when(objectStoreDao.listObjectStores()).thenReturn(objectStores); |
| |
| CapacityVO result = storageManagerImpl.getObjectStorageUsedStats(zoneId); |
| |
| Assert.assertEquals(zoneId, result.getDataCenterId()); |
| Assert.assertEquals(Optional.of(3000L), Optional.of(result.getUsedCapacity())); // 1000 + 2000 |
| Assert.assertEquals(6000L, result.getTotalCapacity()); // 2000 + 4000 |
| Assert.assertEquals(Capacity.CAPACITY_TYPE_OBJECT_STORAGE, result.getCapacityType()); |
| Assert.assertNull(result.getPodId()); |
| Assert.assertNull(result.getClusterId()); |
| } |
| |
| @Test |
| public void testGetObjectStorageUsedStatsWithNullSizes() { |
| Long zoneId = 1L; |
| List<ObjectStoreVO> objectStores = new ArrayList<>(); |
| |
| ObjectStoreVO store1 = new ObjectStoreVO(); |
| store1.setAllocatedSize(null); |
| store1.setTotalSize(null); |
| objectStores.add(store1); |
| |
| ObjectStoreVO store2 = new ObjectStoreVO(); |
| store2.setAllocatedSize(null); |
| store2.setTotalSize(null); |
| objectStores.add(store2); |
| |
| Mockito.when(objectStoreDao.listObjectStores()).thenReturn(objectStores); |
| |
| CapacityVO result = storageManagerImpl.getObjectStorageUsedStats(zoneId); |
| |
| Assert.assertEquals(zoneId, result.getDataCenterId()); |
| Assert.assertEquals(Optional.of(0L), Optional.of(result.getUsedCapacity())); |
| Assert.assertEquals(0L, result.getTotalCapacity()); |
| Assert.assertEquals(Capacity.CAPACITY_TYPE_OBJECT_STORAGE, result.getCapacityType()); |
| Assert.assertNull(result.getPodId()); |
| Assert.assertNull(result.getClusterId()); |
| } |
| |
| @Test |
| public void testDiscoverObjectStore() { |
| Long objectStoreId = 1L; |
| |
| String name = "test-store"; |
| String url = "http://10.1.1.33:80"; |
| Long size = 1000L; |
| String providerName = "test-provider"; |
| Map<String, String> details = new HashMap<>(); |
| details.put("key1", "value1"); |
| |
| ObjectStoreVO objectStoreVO = new ObjectStoreVO(); |
| ReflectionTestUtils.setField(objectStoreVO, "id", objectStoreId); |
| objectStoreVO.setName(name); |
| objectStoreVO.setUrl(url); |
| objectStoreVO.setProviderName(providerName); |
| objectStoreVO.setTotalSize(size); |
| |
| DataStoreProvider storeProvider = Mockito.mock(DataStoreProvider.class); |
| DataStoreLifeCycle lifeCycle = Mockito.mock(DataStoreLifeCycle.class); |
| DataStore store = Mockito.mock(DataStore.class); |
| ObjectStore objectStore = Mockito.mock(ObjectStore.class); |
| |
| Mockito.when(dataStoreProviderMgr.getDataStoreProvider(providerName)).thenReturn(storeProvider); |
| Mockito.when(storeProvider.getDataStoreLifeCycle()).thenReturn(lifeCycle); |
| Mockito.when(lifeCycle.initialize(Mockito.any())).thenReturn(store); |
| Mockito.when(store.getId()).thenReturn(1L); |
| Mockito.when(dataStoreMgr.getDataStore(1L, DataStoreRole.Object)).thenReturn(null); |
| |
| ObjectStore result = storageManagerImpl.discoverObjectStore(name, url, size, providerName, details); |
| |
| Mockito.verify(dataStoreProviderMgr).getDataStoreProvider(providerName); |
| Mockito.verify(lifeCycle).initialize(Mockito.any()); |
| Mockito.verify(dataStoreMgr).getDataStore(1L, DataStoreRole.Object); |
| } |
| |
| @Test(expected = InvalidParameterValueException.class) |
| public void testDiscoverObjectStoreInvalidProvider() { |
| // Setup |
| String name = "test-store"; |
| String url = "http://10.1.1.33:80"; |
| Long size = 1000L; |
| String providerName = "invalid-provider"; |
| Map<String, String> details = new HashMap<>(); |
| |
| Mockito.when(dataStoreProviderMgr.getDataStoreProvider(providerName)).thenReturn(null); |
| |
| storageManagerImpl.discoverObjectStore(name, url, size, providerName, details); |
| } |
| |
| @Test(expected = IllegalArgumentException.class) |
| public void testDiscoverObjectStoreInvalidUrl() { |
| String name = "test-store"; |
| String url = "invalid-url"; |
| Long size = 1000L; |
| String providerName = "test-provider"; |
| Map<String, String> details = new HashMap<>(); |
| |
| DataStoreProvider storeProvider = Mockito.mock(DataStoreProvider.class); |
| Mockito.when(dataStoreProviderMgr.getDataStoreProvider(providerName)).thenReturn(storeProvider); |
| |
| storageManagerImpl.discoverObjectStore(name, url, size, providerName, details); |
| } |
| |
| @Test(expected = InvalidParameterValueException.class) |
| public void testDiscoverObjectStoreDuplicateUrl() { |
| String name = "test-store"; |
| String url = "http://10.1.1.33:80"; |
| Long size = 1000L; |
| String providerName = "test-provider"; |
| Map<String, String> details = new HashMap<>(); |
| |
| DataStoreProvider storeProvider = Mockito.mock(DataStoreProvider.class); |
| ObjectStoreVO existingStore = new ObjectStoreVO(); |
| |
| Mockito.when(dataStoreProviderMgr.getDataStoreProvider(providerName)).thenReturn(storeProvider); |
| Mockito.when(objectStoreDao.findByUrl(url)).thenReturn(existingStore); |
| |
| storageManagerImpl.discoverObjectStore(name, url, size, providerName, details); |
| } |
| |
| @Test(expected = CloudRuntimeException.class) |
| public void testDiscoverObjectStoreInitializationFailure() { |
| String name = "test-store"; |
| String url = "http://10.1.1.33:80"; |
| Long size = 1000L; |
| String providerName = "test-provider"; |
| Map<String, String> details = new HashMap<>(); |
| |
| DataStoreProvider storeProvider = Mockito.mock(DataStoreProvider.class); |
| DataStoreLifeCycle lifeCycle = Mockito.mock(DataStoreLifeCycle.class); |
| |
| Mockito.when(dataStoreProviderMgr.getDataStoreProvider(providerName)).thenReturn(storeProvider); |
| Mockito.when(storeProvider.getDataStoreLifeCycle()).thenReturn(lifeCycle); |
| Mockito.when(lifeCycle.initialize(Mockito.any())).thenThrow(new RuntimeException("Initialization failed")); |
| |
| storageManagerImpl.discoverObjectStore(name, url, size, providerName, details); |
| } |
| } |