mirror of
				https://github.com/apache/cloudstack.git
				synced 2025-10-26 08:42:29 +01:00 
			
		
		
		
	GPU enabled hosts from non-GPU VM deployment. Cluster reordering is based on the number of unique host tags in a cluster, cluster with most number of unique host tags will put at the end of list. Hosts with GPU capability will get tagged with implicit tags defined by global config param 'implicit.host.tags' at the time os host discovery. Also added FirstFitPlannerTest unit test file. (cherry picked from commit 39fe766c2b6fb6edd4c1bf828625b29d9bb87719)
		
			
				
	
	
		
			414 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			414 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| // 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.vm;
 | |
| 
 | |
| import static org.junit.Assert.assertNull;
 | |
| 
 | |
| import java.io.IOException;
 | |
| import java.util.ArrayList;
 | |
| import java.util.List;
 | |
| 
 | |
| import javax.inject.Inject;
 | |
| import javax.naming.ConfigurationException;
 | |
| 
 | |
| import org.junit.Before;
 | |
| import org.junit.BeforeClass;
 | |
| import org.junit.Test;
 | |
| import org.junit.runner.RunWith;
 | |
| import org.mockito.Matchers;
 | |
| import org.mockito.Mockito;
 | |
| import org.springframework.context.annotation.Bean;
 | |
| import org.springframework.context.annotation.ComponentScan;
 | |
| import org.springframework.context.annotation.ComponentScan.Filter;
 | |
| import org.springframework.context.annotation.Configuration;
 | |
| import org.springframework.context.annotation.FilterType;
 | |
| import org.springframework.core.type.classreading.MetadataReader;
 | |
| import org.springframework.core.type.classreading.MetadataReaderFactory;
 | |
| import org.springframework.core.type.filter.TypeFilter;
 | |
| import org.springframework.test.context.ContextConfiguration;
 | |
| import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
 | |
| import org.springframework.test.context.support.AnnotationConfigContextLoader;
 | |
| 
 | |
| import org.apache.cloudstack.affinity.AffinityGroupProcessor;
 | |
| import org.apache.cloudstack.affinity.AffinityGroupService;
 | |
| import org.apache.cloudstack.affinity.dao.AffinityGroupDao;
 | |
| import org.apache.cloudstack.affinity.dao.AffinityGroupVMMapDao;
 | |
| import org.apache.cloudstack.engine.cloud.entity.api.db.dao.VMReservationDao;
 | |
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreManager;
 | |
| import org.apache.cloudstack.framework.config.dao.ConfigurationDao;
 | |
| import org.apache.cloudstack.framework.messagebus.MessageBus;
 | |
| import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao;
 | |
| import org.apache.cloudstack.test.utils.SpringUtils;
 | |
| 
 | |
| import com.cloud.agent.AgentManager;
 | |
| import com.cloud.capacity.CapacityManager;
 | |
| import com.cloud.capacity.dao.CapacityDao;
 | |
| import com.cloud.dc.ClusterDetailsDao;
 | |
| import com.cloud.dc.ClusterVO;
 | |
| import com.cloud.dc.DataCenterVO;
 | |
| import com.cloud.dc.dao.ClusterDao;
 | |
| import com.cloud.dc.dao.DataCenterDao;
 | |
| import com.cloud.dc.dao.DedicatedResourceDao;
 | |
| import com.cloud.dc.dao.HostPodDao;
 | |
| import com.cloud.deploy.DataCenterDeployment;
 | |
| import com.cloud.deploy.DeployDestination;
 | |
| import com.cloud.deploy.DeploymentClusterPlanner;
 | |
| import com.cloud.deploy.DeploymentPlanner;
 | |
| import com.cloud.deploy.DeploymentPlanner.ExcludeList;
 | |
| import com.cloud.deploy.DeploymentPlanner.PlannerResourceUsage;
 | |
| import com.cloud.deploy.DeploymentPlanningManagerImpl;
 | |
| import com.cloud.deploy.FirstFitPlanner;
 | |
| import com.cloud.deploy.PlannerHostReservationVO;
 | |
| import com.cloud.deploy.dao.PlannerHostReservationDao;
 | |
| import com.cloud.exception.AffinityConflictException;
 | |
| import com.cloud.exception.InsufficientServerCapacityException;
 | |
| import com.cloud.gpu.dao.HostGpuGroupsDao;
 | |
| import com.cloud.host.dao.HostDao;
 | |
| import com.cloud.host.dao.HostTagsDao;
 | |
| import com.cloud.hypervisor.Hypervisor.HypervisorType;
 | |
| import com.cloud.resource.ResourceManager;
 | |
| import com.cloud.service.ServiceOfferingVO;
 | |
| import com.cloud.service.dao.ServiceOfferingDetailsDao;
 | |
| import com.cloud.storage.Storage.ProvisioningType;
 | |
| import com.cloud.storage.StorageManager;
 | |
| import com.cloud.storage.dao.DiskOfferingDao;
 | |
| import com.cloud.storage.dao.GuestOSCategoryDao;
 | |
| import com.cloud.storage.dao.GuestOSDao;
 | |
| import com.cloud.storage.dao.StoragePoolHostDao;
 | |
| import com.cloud.storage.dao.VolumeDao;
 | |
| import com.cloud.user.AccountManager;
 | |
| import com.cloud.utils.component.ComponentContext;
 | |
| import com.cloud.vm.dao.UserVmDao;
 | |
| import com.cloud.vm.dao.VMInstanceDao;
 | |
| 
 | |
| @RunWith(SpringJUnit4ClassRunner.class)
 | |
| @ContextConfiguration(loader = AnnotationConfigContextLoader.class)
 | |
| public class DeploymentPlanningManagerImplTest {
 | |
| 
 | |
|     @Inject
 | |
|     DeploymentPlanningManagerImpl _dpm;
 | |
| 
 | |
|     @Inject
 | |
|     PlannerHostReservationDao _plannerHostReserveDao;
 | |
| 
 | |
|     @Inject
 | |
|     VirtualMachineProfileImpl vmProfile;
 | |
| 
 | |
|     @Inject
 | |
|     AffinityGroupVMMapDao _affinityGroupVMMapDao;
 | |
| 
 | |
|     @Inject
 | |
|     ExcludeList avoids;
 | |
| 
 | |
|     @Inject
 | |
|     DataCenterVO dc;
 | |
| 
 | |
|     @Inject
 | |
|     DataCenterDao _dcDao;
 | |
| 
 | |
|     @Inject
 | |
|     FirstFitPlanner _planner;
 | |
| 
 | |
|     @Inject
 | |
|     ClusterDao _clusterDao;
 | |
| 
 | |
|     @Inject
 | |
|     DedicatedResourceDao _dedicatedDao;
 | |
| 
 | |
|     private static long domainId = 5L;
 | |
| 
 | |
|     private static long dataCenterId = 1L;
 | |
| 
 | |
|     @BeforeClass
 | |
|     public static void setUp() throws ConfigurationException {
 | |
|     }
 | |
| 
 | |
|     @Before
 | |
|     public void testSetUp() {
 | |
|         ComponentContext.initComponentsLifeCycle();
 | |
| 
 | |
|         PlannerHostReservationVO reservationVO = new PlannerHostReservationVO(200L, 1L, 2L, 3L, PlannerResourceUsage.Shared);
 | |
|         Mockito.when(_plannerHostReserveDao.persist(Matchers.any(PlannerHostReservationVO.class))).thenReturn(reservationVO);
 | |
|         Mockito.when(_plannerHostReserveDao.findById(Matchers.anyLong())).thenReturn(reservationVO);
 | |
|         Mockito.when(_affinityGroupVMMapDao.countAffinityGroupsForVm(Matchers.anyLong())).thenReturn(0L);
 | |
| 
 | |
|         VMInstanceVO vm = new VMInstanceVO();
 | |
|         Mockito.when(vmProfile.getVirtualMachine()).thenReturn(vm);
 | |
| 
 | |
|         Mockito.when(_dcDao.findById(Matchers.anyLong())).thenReturn(dc);
 | |
|         Mockito.when(dc.getId()).thenReturn(dataCenterId);
 | |
| 
 | |
|         ClusterVO clusterVO = new ClusterVO();
 | |
|         clusterVO.setHypervisorType(HypervisorType.XenServer.toString());
 | |
|         Mockito.when(_clusterDao.findById(Matchers.anyLong())).thenReturn(clusterVO);
 | |
| 
 | |
|         Mockito.when(_planner.getName()).thenReturn("FirstFitPlanner");
 | |
|         List<DeploymentPlanner> planners = new ArrayList<DeploymentPlanner>();
 | |
|         planners.add(_planner);
 | |
|         _dpm.setPlanners(planners);
 | |
| 
 | |
|     }
 | |
| 
 | |
|     @Test
 | |
|     public void dataCenterAvoidTest() throws InsufficientServerCapacityException, AffinityConflictException {
 | |
|         ServiceOfferingVO svcOffering =
 | |
|             new ServiceOfferingVO("testOffering", 1, 512, 500, 1, 1, false, false, false, "test dpm",
 | |
|                     ProvisioningType.THIN, false, false, null, false, VirtualMachine.Type.User, domainId,
 | |
|                 null, "FirstFitPlanner");
 | |
|         Mockito.when(vmProfile.getServiceOffering()).thenReturn(svcOffering);
 | |
| 
 | |
|         DataCenterDeployment plan = new DataCenterDeployment(dataCenterId);
 | |
| 
 | |
|         Mockito.when(avoids.shouldAvoid((DataCenterVO)Matchers.anyObject())).thenReturn(true);
 | |
|         DeployDestination dest = _dpm.planDeployment(vmProfile, plan, avoids, null);
 | |
|         assertNull("DataCenter is in avoid set, destination should be null! ", dest);
 | |
|     }
 | |
| 
 | |
|     @Test
 | |
|     public void plannerCannotHandleTest() throws InsufficientServerCapacityException, AffinityConflictException {
 | |
|         ServiceOfferingVO svcOffering =
 | |
|             new ServiceOfferingVO("testOffering", 1, 512, 500, 1, 1, false, false, false, "test dpm",
 | |
|                     ProvisioningType.THIN, false, false, null, false, VirtualMachine.Type.User, domainId,
 | |
|                 null, "UserDispersingPlanner");
 | |
|         Mockito.when(vmProfile.getServiceOffering()).thenReturn(svcOffering);
 | |
| 
 | |
|         DataCenterDeployment plan = new DataCenterDeployment(dataCenterId);
 | |
|         Mockito.when(avoids.shouldAvoid((DataCenterVO)Matchers.anyObject())).thenReturn(false);
 | |
| 
 | |
|         Mockito.when(_planner.canHandle(vmProfile, plan, avoids)).thenReturn(false);
 | |
|         DeployDestination dest = _dpm.planDeployment(vmProfile, plan, avoids, null);
 | |
|         assertNull("Planner cannot handle, destination should be null! ", dest);
 | |
|     }
 | |
| 
 | |
|     @Test
 | |
|     public void emptyClusterListTest() throws InsufficientServerCapacityException, AffinityConflictException {
 | |
|         ServiceOfferingVO svcOffering =
 | |
|             new ServiceOfferingVO("testOffering", 1, 512, 500, 1, 1, false, false, false, "test dpm",
 | |
|                 ProvisioningType.THIN, false, false, null, false, VirtualMachine.Type.User, domainId,
 | |
|                 null, "FirstFitPlanner");
 | |
|         Mockito.when(vmProfile.getServiceOffering()).thenReturn(svcOffering);
 | |
| 
 | |
|         DataCenterDeployment plan = new DataCenterDeployment(dataCenterId);
 | |
|         Mockito.when(avoids.shouldAvoid((DataCenterVO)Matchers.anyObject())).thenReturn(false);
 | |
|         Mockito.when(_planner.canHandle(vmProfile, plan, avoids)).thenReturn(true);
 | |
| 
 | |
|         Mockito.when(((DeploymentClusterPlanner)_planner).orderClusters(vmProfile, plan, avoids)).thenReturn(null);
 | |
|         DeployDestination dest = _dpm.planDeployment(vmProfile, plan, avoids, null);
 | |
|         assertNull("Planner cannot handle, destination should be null! ", dest);
 | |
|     }
 | |
| 
 | |
|     @Configuration
 | |
|     @ComponentScan(basePackageClasses = {DeploymentPlanningManagerImpl.class}, includeFilters = {@Filter(value = TestConfiguration.Library.class,
 | |
|                                                                                                          type = FilterType.CUSTOM)}, useDefaultFilters = false)
 | |
|     public static class TestConfiguration extends SpringUtils.CloudStackTestConfiguration {
 | |
| 
 | |
|         @Bean
 | |
|         public FirstFitPlanner firstFitPlanner() {
 | |
|             return Mockito.mock(FirstFitPlanner.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public DeploymentPlanner deploymentPlanner() {
 | |
|             return Mockito.mock(DeploymentPlanner.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public DataCenterVO dataCenter() {
 | |
|             return Mockito.mock(DataCenterVO.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public ExcludeList excludeList() {
 | |
|             return Mockito.mock(ExcludeList.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public VirtualMachineProfileImpl virtualMachineProfileImpl() {
 | |
|             return Mockito.mock(VirtualMachineProfileImpl.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public HostTagsDao hostTagsDao() {
 | |
|             return Mockito.mock(HostTagsDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public ClusterDetailsDao clusterDetailsDao() {
 | |
|             return Mockito.mock(ClusterDetailsDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public ResourceManager resourceManager() {
 | |
|             return Mockito.mock(ResourceManager.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public ServiceOfferingDetailsDao serviceOfferingDetailsDao() {
 | |
|             return Mockito.mock(ServiceOfferingDetailsDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public DataStoreManager cataStoreManager() {
 | |
|             return Mockito.mock(DataStoreManager.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public StorageManager storageManager() {
 | |
|             return Mockito.mock(StorageManager.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public HostDao hostDao() {
 | |
|             return Mockito.mock(HostDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public HostPodDao hostPodDao() {
 | |
|             return Mockito.mock(HostPodDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public ClusterDao clusterDao() {
 | |
|             return Mockito.mock(ClusterDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public DedicatedResourceDao dedicatedResourceDao() {
 | |
|             return Mockito.mock(DedicatedResourceDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public GuestOSDao guestOSDao() {
 | |
|             return Mockito.mock(GuestOSDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public GuestOSCategoryDao guestOSCategoryDao() {
 | |
|             return Mockito.mock(GuestOSCategoryDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public CapacityManager capacityManager() {
 | |
|             return Mockito.mock(CapacityManager.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public StoragePoolHostDao storagePoolHostDao() {
 | |
|             return Mockito.mock(StoragePoolHostDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public VolumeDao volumeDao() {
 | |
|             return Mockito.mock(VolumeDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public ConfigurationDao configurationDao() {
 | |
|             return Mockito.mock(ConfigurationDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public DiskOfferingDao diskOfferingDao() {
 | |
|             return Mockito.mock(DiskOfferingDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public PrimaryDataStoreDao primaryDataStoreDao() {
 | |
|             return Mockito.mock(PrimaryDataStoreDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public CapacityDao capacityDao() {
 | |
|             return Mockito.mock(CapacityDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public PlannerHostReservationDao plannerHostReservationDao() {
 | |
|             return Mockito.mock(PlannerHostReservationDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public AffinityGroupProcessor affinityGroupProcessor() {
 | |
|             return Mockito.mock(AffinityGroupProcessor.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public AffinityGroupDao affinityGroupDao() {
 | |
|             return Mockito.mock(AffinityGroupDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public AffinityGroupVMMapDao affinityGroupVMMapDao() {
 | |
|             return Mockito.mock(AffinityGroupVMMapDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public AccountManager accountManager() {
 | |
|             return Mockito.mock(AccountManager.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public AgentManager agentManager() {
 | |
|             return Mockito.mock(AgentManager.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public MessageBus messageBus() {
 | |
|             return Mockito.mock(MessageBus.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public UserVmDao userVMDao() {
 | |
|             return Mockito.mock(UserVmDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public VMInstanceDao vmInstanceDao() {
 | |
|             return Mockito.mock(VMInstanceDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public DataCenterDao dataCenterDao() {
 | |
|             return Mockito.mock(DataCenterDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public VMReservationDao reservationDao() {
 | |
|             return Mockito.mock(VMReservationDao.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public AffinityGroupService affinityGroupService() {
 | |
|             return Mockito.mock(AffinityGroupService.class);
 | |
|         }
 | |
| 
 | |
|         @Bean
 | |
|         public HostGpuGroupsDao hostGpuGroupsDap() {
 | |
|             return Mockito.mock(HostGpuGroupsDao.class);
 | |
|         }
 | |
| 
 | |
|         public static class Library implements TypeFilter {
 | |
| 
 | |
|             @Override
 | |
|             public boolean match(MetadataReader mdr, MetadataReaderFactory arg1) throws IOException {
 | |
|                 ComponentScan cs = TestConfiguration.class.getAnnotation(ComponentScan.class);
 | |
|                 return SpringUtils.includedInBasePackageClasses(mdr.getClassMetadata().getClassName(), cs);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |