mirror of
				https://github.com/apache/cloudstack.git
				synced 2025-10-26 08:42:29 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			333 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Java
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			333 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Java
		
	
	
		
			Executable File
		
	
	
	
	
| // 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.network.guru;
 | |
| 
 | |
| import java.util.List;
 | |
| 
 | |
| import javax.ejb.Local;
 | |
| import javax.inject.Inject;
 | |
| 
 | |
| import org.apache.log4j.Logger;
 | |
| 
 | |
| import org.apache.cloudstack.engine.orchestration.service.NetworkOrchestrationService;
 | |
| 
 | |
| import com.cloud.dc.DataCenter;
 | |
| import com.cloud.dc.DataCenter.NetworkType;
 | |
| import com.cloud.dc.dao.DataCenterDao;
 | |
| import com.cloud.dc.dao.VlanDao;
 | |
| import com.cloud.deploy.DeployDestination;
 | |
| import com.cloud.deploy.DeploymentPlan;
 | |
| import com.cloud.exception.ConcurrentOperationException;
 | |
| import com.cloud.exception.InsufficientAddressCapacityException;
 | |
| import com.cloud.exception.InsufficientVirtualNetworkCapcityException;
 | |
| import com.cloud.exception.InvalidParameterValueException;
 | |
| import com.cloud.network.IpAddressManager;
 | |
| import com.cloud.network.Ipv6AddressManager;
 | |
| import com.cloud.network.Network;
 | |
| import com.cloud.network.Network.GuestType;
 | |
| import com.cloud.network.Network.Service;
 | |
| import com.cloud.network.Network.State;
 | |
| import com.cloud.network.NetworkModel;
 | |
| import com.cloud.network.NetworkProfile;
 | |
| import com.cloud.network.Networks.BroadcastDomainType;
 | |
| import com.cloud.network.Networks.Mode;
 | |
| import com.cloud.network.Networks.TrafficType;
 | |
| import com.cloud.network.dao.IPAddressDao;
 | |
| import com.cloud.network.dao.IPAddressVO;
 | |
| import com.cloud.network.dao.NetworkVO;
 | |
| import com.cloud.network.dao.UserIpv6AddressDao;
 | |
| import com.cloud.offering.NetworkOffering;
 | |
| import com.cloud.offerings.dao.NetworkOfferingDao;
 | |
| import com.cloud.user.Account;
 | |
| import com.cloud.utils.component.AdapterBase;
 | |
| import com.cloud.utils.db.DB;
 | |
| import com.cloud.utils.db.Transaction;
 | |
| import com.cloud.vm.Nic;
 | |
| import com.cloud.vm.Nic.ReservationStrategy;
 | |
| import com.cloud.vm.NicProfile;
 | |
| import com.cloud.vm.NicVO;
 | |
| import com.cloud.vm.ReservationContext;
 | |
| import com.cloud.vm.VirtualMachine;
 | |
| import com.cloud.vm.VirtualMachineProfile;
 | |
| import com.cloud.vm.dao.NicDao;
 | |
| import com.cloud.vm.dao.NicSecondaryIpDao;
 | |
| 
 | |
| @Local(value = { NetworkGuru.class })
 | |
| public class DirectNetworkGuru extends AdapterBase implements NetworkGuru {
 | |
|     private static final Logger s_logger = Logger.getLogger(DirectNetworkGuru.class);
 | |
| 
 | |
|     @Inject
 | |
|     DataCenterDao _dcDao;
 | |
|     @Inject
 | |
|     VlanDao _vlanDao;
 | |
|     @Inject
 | |
|     NetworkModel _networkModel;
 | |
|     @Inject
 | |
|     NetworkOrchestrationService _networkMgr;
 | |
|     @Inject
 | |
|     IPAddressDao _ipAddressDao;
 | |
|     @Inject
 | |
|     NetworkOfferingDao _networkOfferingDao;
 | |
|     @Inject
 | |
|     UserIpv6AddressDao _ipv6Dao;
 | |
|     @Inject
 | |
|     Ipv6AddressManager _ipv6Mgr;
 | |
|     @Inject
 | |
|     NicSecondaryIpDao _nicSecondaryIpDao;
 | |
|     @Inject
 | |
|     NicDao _nicDao;
 | |
|     @Inject
 | |
|     IpAddressManager _ipAddrMgr;
 | |
| 
 | |
|     private static final TrafficType[] _trafficTypes = {TrafficType.Guest};
 | |
|     
 | |
|     @Override
 | |
|     public boolean isMyTrafficType(TrafficType type) {
 | |
|     	for (TrafficType t : _trafficTypes) {
 | |
|     		if (t == type) {
 | |
|     			return true;
 | |
|     		}
 | |
|     	}
 | |
|     	return false;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public TrafficType[] getSupportedTrafficType() {
 | |
|     	return _trafficTypes;
 | |
|     }
 | |
|     
 | |
|     protected boolean canHandle(NetworkOffering offering, DataCenter dc) {
 | |
|         // this guru handles only Guest networks in Advance zone with source nat service disabled
 | |
|         if (dc.getNetworkType() == NetworkType.Advanced && isMyTrafficType(offering.getTrafficType()) && offering.getGuestType() == GuestType.Shared) {
 | |
|             return true;
 | |
|         } else {
 | |
|             s_logger.trace("We only take care of Guest networks of type " + GuestType.Shared);
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public Network design(NetworkOffering offering, DeploymentPlan plan, Network userSpecified, Account owner) {
 | |
|         DataCenter dc = _dcDao.findById(plan.getDataCenterId());
 | |
| 
 | |
|         if (!canHandle(offering, dc)) {
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         State state = State.Allocated;
 | |
|         if (dc.getNetworkType() == NetworkType.Basic) {
 | |
|             state = State.Setup;
 | |
|         }
 | |
| 
 | |
|         NetworkVO config = new NetworkVO(offering.getTrafficType(), Mode.Dhcp, BroadcastDomainType.Vlan, offering.getId(), state, plan.getDataCenterId(), plan.getPhysicalNetworkId());
 | |
| 
 | |
|         if (userSpecified != null) {
 | |
|             if ((userSpecified.getCidr() == null && userSpecified.getGateway() != null) || (userSpecified.getCidr() != null && userSpecified.getGateway() == null)) {
 | |
|                 throw new InvalidParameterValueException("cidr and gateway must be specified together.");
 | |
|             }
 | |
| 
 | |
|             if ((userSpecified.getIp6Cidr() == null && userSpecified.getIp6Gateway() != null) || (userSpecified.getIp6Cidr() != null && userSpecified.getIp6Gateway() == null)) {
 | |
|                 throw new InvalidParameterValueException("cidrv6 and gatewayv6 must be specified together.");
 | |
|             }
 | |
| 
 | |
|             if (userSpecified.getCidr() != null) {
 | |
|                 config.setCidr(userSpecified.getCidr());
 | |
|                 config.setGateway(userSpecified.getGateway());
 | |
|             }
 | |
| 
 | |
|             if (userSpecified.getIp6Cidr() != null) {
 | |
|                 config.setIp6Cidr(userSpecified.getIp6Cidr());
 | |
|                 config.setIp6Gateway(userSpecified.getIp6Gateway());
 | |
|             }
 | |
| 
 | |
|             if (userSpecified.getBroadcastUri() != null) {
 | |
|                 config.setBroadcastUri(userSpecified.getBroadcastUri());
 | |
|                 config.setState(State.Setup);
 | |
|             }
 | |
| 
 | |
|             if (userSpecified.getBroadcastDomainType() != null) {
 | |
|                 config.setBroadcastDomainType(userSpecified.getBroadcastDomainType());
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         boolean isSecurityGroupEnabled = _networkModel.areServicesSupportedByNetworkOffering(offering.getId(), Service.SecurityGroup);
 | |
|         if (isSecurityGroupEnabled) {
 | |
|         	if (userSpecified.getIp6Cidr() != null) {
 | |
|                 throw new InvalidParameterValueException("Didn't support security group with IPv6");
 | |
|         	}
 | |
|             config.setName("SecurityGroupEnabledNetwork");
 | |
|             config.setDisplayText("SecurityGroupEnabledNetwork");
 | |
|         }
 | |
| 
 | |
|         return config;
 | |
|     }
 | |
| 
 | |
|     protected DirectNetworkGuru() {
 | |
|         super();
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void updateNicProfile(NicProfile profile, Network network) {
 | |
|         DataCenter dc = _dcDao.findById(network.getDataCenterId());
 | |
|         if (profile != null) {
 | |
|             profile.setDns1(dc.getDns1());
 | |
|             profile.setDns2(dc.getDns2());
 | |
|             profile.setIp6Dns1(dc.getIp6Dns1());
 | |
|             profile.setIp6Dns2(dc.getIp6Dns2());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public NicProfile allocate(Network network, NicProfile nic, VirtualMachineProfile vm) throws InsufficientVirtualNetworkCapcityException,
 | |
|             InsufficientAddressCapacityException, ConcurrentOperationException {
 | |
| 
 | |
|         DataCenter dc = _dcDao.findById(network.getDataCenterId());
 | |
| 
 | |
|         if (nic == null) {
 | |
|             nic = new NicProfile(ReservationStrategy.Create, null, null, null, null);
 | |
|         } else if (nic.getIp4Address() == null && nic.getIp6Address() == null) {
 | |
|             nic.setStrategy(ReservationStrategy.Start);
 | |
|         } else {
 | |
|             nic.setStrategy(ReservationStrategy.Create);
 | |
|         }
 | |
| 
 | |
|         allocateDirectIp(nic, network, vm, dc, nic.getRequestedIpv4(), nic.getRequestedIpv6());
 | |
|         nic.setStrategy(ReservationStrategy.Create);
 | |
| 
 | |
|         if (nic.getMacAddress() == null) {
 | |
|             nic.setMacAddress(_networkModel.getNextAvailableMacAddressInNetwork(network.getId()));
 | |
|             if (nic.getMacAddress() == null) {
 | |
|                 throw new InsufficientAddressCapacityException("Unable to allocate more mac addresses", Network.class, network.getId());
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return nic;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void reserve(NicProfile nic, Network network, VirtualMachineProfile vm, DeployDestination dest, ReservationContext context)
 | |
|             throws InsufficientVirtualNetworkCapcityException, InsufficientAddressCapacityException, ConcurrentOperationException {
 | |
|         if (nic.getIp4Address() == null && nic.getIp6Address() == null) {
 | |
|             allocateDirectIp(nic, network, vm, dest.getDataCenter(), null, null);
 | |
|             nic.setStrategy(ReservationStrategy.Create);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @DB
 | |
|     protected void allocateDirectIp(NicProfile nic, Network network, VirtualMachineProfile vm, DataCenter dc, String requestedIp4Addr, String requestedIp6Addr)
 | |
|             throws InsufficientVirtualNetworkCapcityException,
 | |
|             InsufficientAddressCapacityException {
 | |
|         
 | |
|         Transaction txn = Transaction.currentTxn();
 | |
|         txn.start();
 | |
|         _ipAddrMgr.allocateDirectIp(nic, dc, vm, network, requestedIp4Addr, requestedIp6Addr);
 | |
|         //save the placeholder nic if the vm is the Virtual router
 | |
|         if (vm.getType() == VirtualMachine.Type.DomainRouter) {
 | |
|             Nic placeholderNic = _networkModel.getPlaceholderNicForRouter(network, null);
 | |
|             if (placeholderNic == null) {
 | |
|                 s_logger.debug("Saving placeholder nic with ip4 address " + nic.getIp4Address() + " and ipv6 address " + nic.getIp6Address() + " for the network " + network);
 | |
|                 _networkMgr.savePlaceholderNic(network, nic.getIp4Address(), nic.getIp6Address(), VirtualMachine.Type.DomainRouter);
 | |
|             }
 | |
|         }
 | |
|         txn.commit();
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public boolean release(NicProfile nic, VirtualMachineProfile vm, String reservationId) {
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public Network implement(Network network, NetworkOffering offering, DeployDestination destination, ReservationContext context) throws InsufficientVirtualNetworkCapcityException {
 | |
|         return network;
 | |
|     }
 | |
| 
 | |
|     @Override @DB
 | |
|     public void deallocate(Network network, NicProfile nic, VirtualMachineProfile vm) {
 | |
|     	if (s_logger.isDebugEnabled()) {
 | |
|             s_logger.debug("Deallocate network: networkId: " + nic.getNetworkId() + ", ip: " + nic.getIp4Address());
 | |
|         }
 | |
|     	
 | |
|     	if (nic.getIp4Address() != null) {
 | |
|             IPAddressVO ip = _ipAddressDao.findByIpAndSourceNetworkId(nic.getNetworkId(), nic.getIp4Address());
 | |
|             if (ip != null) {
 | |
|                 Transaction txn = Transaction.currentTxn();
 | |
|                 txn.start();
 | |
|                 
 | |
|                 // if the ip address a part of placeholder, don't release it
 | |
|                 Nic placeholderNic = _networkModel.getPlaceholderNicForRouter(network, null);
 | |
|                 if (placeholderNic != null && placeholderNic.getIp4Address().equalsIgnoreCase(ip.getAddress().addr())) {
 | |
|                     s_logger.debug("Not releasing direct ip " + ip.getId() +" yet as its ip is saved in the placeholder");
 | |
|                 } else {
 | |
|                     _ipAddrMgr.markIpAsUnavailable(ip.getId());
 | |
|                     _ipAddressDao.unassignIpAddress(ip.getId());
 | |
|                 }
 | |
|                
 | |
|                 //unassign nic secondary ip address
 | |
|                 s_logger.debug("remove nic " + nic.getId() + " secondary ip ");
 | |
|                 List<String> nicSecIps = null;
 | |
|                 nicSecIps = _nicSecondaryIpDao.getSecondaryIpAddressesForNic(nic.getId());
 | |
|                 for (String secIp: nicSecIps) {
 | |
|                     IPAddressVO pubIp = _ipAddressDao.findByIpAndSourceNetworkId(nic.getNetworkId(), secIp);
 | |
|                     _ipAddrMgr.markIpAsUnavailable(pubIp.getId());
 | |
|                     _ipAddressDao.unassignIpAddress(pubIp.getId());
 | |
|                 }
 | |
|     
 | |
|                 txn.commit();
 | |
|             }
 | |
|     	}
 | |
|     	
 | |
|     	if (nic.getIp6Address() != null) {
 | |
|     		_ipv6Mgr.revokeDirectIpv6Address(nic.getNetworkId(), nic.getIp6Address());
 | |
|     	}
 | |
|         nic.deallocate();
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void shutdown(NetworkProfile network, NetworkOffering offering) {
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     @DB
 | |
|     public boolean trash(Network network, NetworkOffering offering) {
 | |
|         //Have to remove all placeholder nics
 | |
|         List<NicVO> nics = _nicDao.listPlaceholderNicsByNetworkId(network.getId());
 | |
|         Transaction txn = Transaction.currentTxn();
 | |
|         txn.start();
 | |
|         for (Nic nic : nics) {
 | |
|             if (nic.getIp4Address() != null) {
 | |
|                 s_logger.debug("Releasing ip " + nic.getIp4Address() + " of placeholder nic " + nic);
 | |
|                 IPAddressVO ip = _ipAddressDao.findByIpAndSourceNetworkId(nic.getNetworkId(), nic.getIp4Address());
 | |
|                 _ipAddrMgr.markIpAsUnavailable(ip.getId());
 | |
|                 _ipAddressDao.unassignIpAddress(ip.getId());
 | |
|                 s_logger.debug("Removing placeholder nic " + nic);
 | |
|                 _nicDao.remove(nic.getId());
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         txn.commit();
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void updateNetworkProfile(NetworkProfile networkProfile) {
 | |
|         DataCenter dc = _dcDao.findById(networkProfile.getDataCenterId());
 | |
|         networkProfile.setDns1(dc.getDns1());
 | |
|         networkProfile.setDns2(dc.getDns2());
 | |
|     }
 | |
| }
 |