Package com.cloud.baremetal

Source Code of com.cloud.baremetal.ExternalDhcpManagerImpl

// 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.baremetal;

import java.util.HashMap;
import java.util.List;
import java.util.Map;

import javax.ejb.Local;
import javax.naming.ConfigurationException;

import org.apache.log4j.Logger;

import com.cloud.agent.AgentManager;
import com.cloud.agent.api.Answer;
import com.cloud.agent.api.StartupCommand;
import com.cloud.agent.api.StartupExternalDhcpCommand;
import com.cloud.agent.api.routing.DhcpEntryCommand;
import com.cloud.dc.DataCenter;
import com.cloud.dc.DataCenterVO;
import com.cloud.dc.HostPodVO;
import com.cloud.dc.dao.DataCenterDao;
import com.cloud.dc.dao.HostPodDao;
import com.cloud.deploy.DeployDestination;
import com.cloud.exception.InvalidParameterValueException;
import com.cloud.exception.ResourceUnavailableException;
import com.cloud.host.Host;
import com.cloud.host.Host.Type;
import com.cloud.host.HostVO;
import com.cloud.host.dao.HostDao;
import com.cloud.hypervisor.Hypervisor.HypervisorType;
import com.cloud.network.Network;
import com.cloud.resource.ResourceManager;
import com.cloud.resource.ResourceStateAdapter;
import com.cloud.resource.ServerResource;
import com.cloud.resource.UnableDeleteHostException;
import com.cloud.utils.component.Inject;
import com.cloud.utils.db.DB;
import com.cloud.utils.db.Transaction;
import com.cloud.utils.exception.CloudRuntimeException;
import com.cloud.vm.NicProfile;
import com.cloud.vm.ReservationContext;
import com.cloud.vm.UserVmVO;
import com.cloud.vm.VirtualMachine;
import com.cloud.vm.VirtualMachineProfile;
import com.cloud.vm.dao.NicDao;
import com.cloud.vm.dao.UserVmDao;

@Local(value = {ExternalDhcpManager.class})
public class ExternalDhcpManagerImpl implements ExternalDhcpManager, ResourceStateAdapter {
  private static final org.apache.log4j.Logger s_logger = Logger.getLogger(ExternalDhcpManagerImpl.class);
  protected String _name;
  @Inject DataCenterDao _dcDao;
  @Inject HostDao _hostDao;
  @Inject AgentManager _agentMgr;
  @Inject HostPodDao _podDao;
  @Inject UserVmDao _userVmDao;
  @Inject ResourceManager _resourceMgr;
  @Inject NicDao _nicDao;
 
  @Override
  public boolean configure(String name, Map<String, Object> params) throws ConfigurationException {
    _resourceMgr.registerResourceStateAdapter(this.getClass().getSimpleName(), this);
    return true;
  }

  @Override
  public boolean start() {
    return true;
  }

  @Override
  public boolean stop() {
    _resourceMgr.unregisterResourceStateAdapter(this.getClass().getSimpleName());
    return true;
  }

  @Override
  public String getName() {
    return _name;
  }

  protected String getDhcpServerGuid(String zoneId, String name, String ip) {
    return zoneId + "-" + name + "-" + ip;
  }
 
 
  @Override @DB
  public Host addDhcpServer(Long zoneId, Long podId, String type, String url, String username, String password) { 
    DataCenterVO zone = _dcDao.findById(zoneId);
    if (zone == null) {
      throw new InvalidParameterValueException("Could not find zone with ID: " + zoneId);
    }
   
    HostPodVO pod = _podDao.findById(podId);
    if (pod == null) {
      throw new InvalidParameterValueException("Could not find pod with ID: " + podId);
    }
   
    List<HostVO> dhcps = _resourceMgr.listAllUpAndEnabledHosts(Host.Type.ExternalDhcp, null, podId, zoneId);
    if (dhcps.size() != 0) {
      throw new InvalidParameterValueException("Already had a DHCP server in Pod: " + podId + " zone: " + zoneId);
    }
   
   
    String ipAddress = url;
    String guid = getDhcpServerGuid(Long.toString(zoneId) + "-" + Long.toString(podId), "ExternalDhcp", ipAddress);
    Map params = new HashMap<String, String>();
    params.put("type", type);
    params.put("zone", Long.toString(zoneId));
    params.put("pod", podId.toString());
    params.put("ip", ipAddress);
    params.put("username", username);
    params.put("password", password);
    params.put("guid", guid);
    params.put("pod", Long.toString(podId));
    params.put("gateway", pod.getGateway());
    String dns = zone.getDns1();
    if (dns == null) {
      dns = zone.getDns2();
    }
    params.put("dns", dns);
   
    ServerResource resource = null;
    try {
      if (type.equalsIgnoreCase(DhcpServerType.Dnsmasq.getName())) {
        resource = new DnsmasqResource();
        resource.configure("Dnsmasq resource", params);
      } else if (type.equalsIgnoreCase(DhcpServerType.Dhcpd.getName())) {
        resource = new DhcpdResource();
        resource.configure("Dhcpd resource", params);
      } else {
        throw new CloudRuntimeException("Unsupport DHCP server " + type);
      }
    } catch (Exception e) {
      s_logger.debug(e);
      throw new CloudRuntimeException(e.getMessage());
    }
   
    Host dhcpServer = _resourceMgr.addHost(zoneId, resource, Host.Type.ExternalDhcp, params);
    if (dhcpServer == null) {
      throw new CloudRuntimeException("Cannot add external Dhcp server as a host");
    }
   
    Transaction txn = Transaction.currentTxn();
        txn.start();
        pod.setExternalDhcp(true);
        _podDao.update(pod.getId(), pod);
        txn.commit();
    return dhcpServer;
  }
 
  @Override
  public DhcpServerResponse getApiResponse(Host dhcpServer) {
    DhcpServerResponse response = new DhcpServerResponse();
    response.setId(dhcpServer.getId());
    return response;
  }

  private void prepareBareMetalDhcpEntry(NicProfile nic, DhcpEntryCommand cmd) {
    Long vmId = nic.getVmId();
    UserVmVO vm = _userVmDao.findById(vmId);
    if (vm == null || vm.getHypervisorType() != HypervisorType.BareMetal) {
      s_logger.debug("VM " + vmId + " is not baremetal machine, skip preparing baremetal DHCP entry");
      return;
    }
   
    List<HostVO> servers = _resourceMgr.listAllUpAndEnabledHosts(Host.Type.PxeServer, null, vm.getPodIdToDeployIn(), vm.getDataCenterIdToDeployIn());
    if (servers.size() != 1) {
      throw new CloudRuntimeException("Wrong number of PXE server found in zone " + vm.getDataCenterIdToDeployIn()
          + " Pod " + vm.getPodIdToDeployIn() + ", number is " + servers.size());
    }
    HostVO pxeServer = servers.get(0);
    cmd.setNextServer(pxeServer.getPrivateIpAddress());
    s_logger.debug("Set next-server to " + pxeServer.getPrivateIpAddress() + " for VM " + vm.getId());
  }
 
  @Override
  public boolean addVirtualMachineIntoNetwork(Network network, NicProfile nic, VirtualMachineProfile<? extends VirtualMachine> profile, DeployDestination dest,
      ReservationContext context) throws ResourceUnavailableException {
    Long zoneId = profile.getVirtualMachine().getDataCenterIdToDeployIn();
    Long podId = profile.getVirtualMachine().getPodIdToDeployIn();
    List<HostVO> hosts = _resourceMgr.listAllUpAndEnabledHosts(Type.ExternalDhcp, null, podId, zoneId);
    if (hosts.size() == 0) {
      throw new CloudRuntimeException("No external Dhcp found in zone " + zoneId + " pod " + podId);
    }
   
    if (hosts.size() > 1) {
      throw new CloudRuntimeException("Something wrong, more than 1 external Dhcp found in zone " + zoneId + " pod " + podId);
    }
   
    HostVO h = hosts.get(0);
    String dns = nic.getDns1();
    if (dns == null) {
      dns = nic.getDns2();
    }
    DhcpEntryCommand dhcpCommand = new DhcpEntryCommand(nic.getMacAddress(), nic.getIp4Address(), profile.getVirtualMachine().getHostName(), dns, nic.getGateway());
    String errMsg = String.format("Set dhcp entry on external DHCP %1$s failed(ip=%2$s, mac=%3$s, vmname=%4$s)",
        h.getPrivateIpAddress(), nic.getIp4Address(), nic.getMacAddress(), profile.getVirtualMachine().getHostName());
    //prepareBareMetalDhcpEntry(nic, dhcpCommand);
    try {
      Answer ans = _agentMgr.send(h.getId(), dhcpCommand);
      if (ans.getResult()) {
        s_logger.debug(String.format("Set dhcp entry on external DHCP %1$s successfully(ip=%2$s, mac=%3$s, vmname=%4$s)",
            h.getPrivateIpAddress(), nic.getIp4Address(), nic.getMacAddress(), profile.getVirtualMachine().getHostName()));
        return true;
      } else {
        s_logger.debug(errMsg + " " + ans.getDetails());
        throw new ResourceUnavailableException(errMsg, DataCenter.class, zoneId);
      }
    } catch (Exception e) {
      s_logger.debug(errMsg, e);
      throw new ResourceUnavailableException(errMsg + e.getMessage(), DataCenter.class, zoneId);
    }
  }

  @Override
    public HostVO createHostVOForConnectedAgent(HostVO host, StartupCommand[] cmd) {
      // TODO Auto-generated method stub
      return null;
    }

  @Override
    public HostVO createHostVOForDirectConnectAgent(HostVO host, StartupCommand[] startup, ServerResource resource, Map<String, String> details,
            List<String> hostTags) {
        if (!(startup[0] instanceof StartupExternalDhcpCommand)) {
            return null;
        }
       
        host.setType(Host.Type.ExternalDhcp);
        return host;
    }

  @Override
    public DeleteHostAnswer deleteHost(HostVO host, boolean isForced, boolean isForceDeleteStorage) throws UnableDeleteHostException {
      // TODO Auto-generated method stub
      return null;
    }
}
TOP

Related Classes of com.cloud.baremetal.ExternalDhcpManagerImpl

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.