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());
|
|
}
|
|
}
|