/**
* Copyright (C) 2010 Cloud.com, Inc. All rights reserved.
*
* This software is licensed under the GNU General Public License v3 or later.
*
* It is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*
*/
package com.cloud.api.commands;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.apache.log4j.Logger;
import com.cloud.api.BaseCmd;
import com.cloud.api.ServerApiException;
import com.cloud.dc.VlanVO;
import com.cloud.dc.Vlan.VlanType;
import com.cloud.exception.InsufficientAddressCapacityException;
import com.cloud.exception.InternalErrorException;
import com.cloud.exception.InvalidParameterValueException;
import com.cloud.exception.ResourceAllocationException;
import com.cloud.network.IPAddressVO;
import com.cloud.user.Account;
import com.cloud.utils.Pair;
public class AssociateIPAddrCmd extends BaseCmd {
public static final Logger s_logger = Logger.getLogger(AssociateIPAddrCmd.class.getName());
private static final String s_name = "associateipaddressresponse";
private static final List> s_properties = new ArrayList>();
static {
s_properties.add(new Pair(BaseCmd.Properties.ACCOUNT, Boolean.FALSE));
s_properties.add(new Pair(BaseCmd.Properties.DOMAIN_ID, Boolean.FALSE));
s_properties.add(new Pair(BaseCmd.Properties.ACCOUNT_OBJ, Boolean.FALSE));
s_properties.add(new Pair(BaseCmd.Properties.USER_ID, Boolean.FALSE));
s_properties.add(new Pair(BaseCmd.Properties.ZONE_ID, Boolean.TRUE));
s_properties.add(new Pair(BaseCmd.Properties.ONE_TO_ONE_NAT, Boolean.FALSE));
}
public String getName() {
return s_name;
}
public static String getResultObjectName() {
return "addressinfo";
}
public List> getProperties() {
return s_properties;
}
@Override
public List> execute(Map params) {
Long zoneId = (Long)params.get(BaseCmd.Properties.ZONE_ID.getName());
Account account = (Account)params.get(BaseCmd.Properties.ACCOUNT_OBJ.getName());
Long userId = (Long)params.get(BaseCmd.Properties.USER_ID.getName());
String accountName = (String)params.get(BaseCmd.Properties.ACCOUNT.getName());
Long domainId = (Long)params.get(BaseCmd.Properties.DOMAIN_ID.getName());
String newIpAddr = null;
String errorDesc = null;
Long accountId = null;
boolean isAdmin = false;
if ((account == null) || isAdmin(account.getType())) {
isAdmin = true;
if (domainId != null) {
if ((account != null) && !getManagementServer().isChildDomain(account.getDomainId(), domainId)) {
throw new ServerApiException(BaseCmd.PARAM_ERROR, "Invalid domain id (" + domainId + ") given, unable to associate IP address.");
}
if (accountName != null) {
Account userAccount = getManagementServer().findAccountByName(accountName, domainId);
if (userAccount != null) {
accountId = userAccount.getId();
} else {
throw new ServerApiException(BaseCmd.ACCOUNT_ERROR, "Unable to find account " + accountName + " in domain " + domainId);
}
}
} else if (account != null) {
// the admin is acquiring an IP address
accountId = account.getId();
domainId = account.getDomainId();
} else {
throw new ServerApiException(BaseCmd.PARAM_ERROR, "Account information is not specified.");
}
} else {
accountId = account.getId();
domainId = account.getDomainId();
}
if (userId == null) {
userId = Long.valueOf(1);
}
try {
newIpAddr = getManagementServer().associateIpAddress(userId.longValue(), accountId.longValue(), domainId.longValue(), zoneId.longValue());
} catch (ResourceAllocationException rae) {
if (rae.getResourceType().equals("vm")) throw new ServerApiException (BaseCmd.VM_ALLOCATION_ERROR, rae.getMessage());
else if (rae.getResourceType().equals("ip")) throw new ServerApiException (BaseCmd.IP_ALLOCATION_ERROR, rae.getMessage());
} catch (InvalidParameterValueException ex1) {
s_logger.error("error associated IP Address with userId: " + userId, ex1);
throw new ServerApiException (BaseCmd.NET_INVALID_PARAM_ERROR, ex1.getMessage());
} catch (InsufficientAddressCapacityException ex2) {
throw new ServerApiException (BaseCmd.NET_IP_ASSOC_ERROR, ex2.getMessage());
} catch (InternalErrorException ex3){
throw new ServerApiException (BaseCmd.NET_IP_ASSOC_ERROR, ex3.getMessage());
} catch (Exception ex4) {
throw new ServerApiException (BaseCmd.NET_IP_ASSOC_ERROR, "Unable to associate IP address");
}
if (newIpAddr == null) {
s_logger.warn("unable to associate IP address for user " + ((errorDesc != null) ? (", reason: " + errorDesc) : null));
throw new ServerApiException(BaseCmd.NET_IP_ASSOC_ERROR, "unable to associate IP address for user " + ((errorDesc != null) ? (", reason: " + errorDesc) : null));
}
List> embeddedObject = new ArrayList>();
List> returnValues = new ArrayList>();
try {
returnValues.add(new Pair(BaseCmd.Properties.IP_ADDRESS.getName(), newIpAddr));
List ipAddresses = getManagementServer().listPublicIpAddressesBy(accountId.longValue(), true, null, null);
IPAddressVO ipAddress = null;
for (Iterator iter = ipAddresses.iterator(); iter.hasNext();) {
IPAddressVO current = iter.next();
if (current.getAddress().equals(newIpAddr)) {
ipAddress = current;
break;
}
}
if (ipAddress == null) {
return returnValues;
}
if (ipAddress.getAllocated() != null) {
returnValues.add(new Pair(BaseCmd.Properties.ALLOCATED.getName(), getDateString(ipAddress.getAllocated())));
}
returnValues.add(new Pair(BaseCmd.Properties.ZONE_ID.getName(), Long.valueOf(ipAddress.getDataCenterId()).toString()));
returnValues.add(new Pair(BaseCmd.Properties.ZONE_NAME.getName(), getManagementServer().findDataCenterById(ipAddress.getDataCenterId()).getName()));
returnValues.add(new Pair(BaseCmd.Properties.IS_SOURCE_NAT.getName(), Boolean.valueOf(ipAddress.isSourceNat()).toString()));
//get account information
Account accountTemp = getManagementServer().findAccountById(ipAddress.getAccountId());
returnValues.add(new Pair(BaseCmd.Properties.ACCOUNT.getName(), accountTemp.getAccountName()));
returnValues.add(new Pair(BaseCmd.Properties.DOMAIN_ID.getName(), accountTemp.getDomainId()));
returnValues.add(new Pair(BaseCmd.Properties.DOMAIN.getName(), getManagementServer().findDomainIdById(accountTemp.getDomainId()).getName()));
VlanVO vlan = getManagementServer().findVlanById(ipAddress.getVlanDbId());
boolean forVirtualNetworks = vlan.getVlanType().equals(VlanType.VirtualNetwork);
returnValues.add(new Pair(BaseCmd.Properties.FOR_VIRTUAL_NETWORK.getName(), forVirtualNetworks));
//show this info to admin only
if (isAdmin == true) {
returnValues.add(new Pair(BaseCmd.Properties.VLAN_DB_ID.getName(), Long.valueOf(ipAddress.getVlanDbId()).toString()));
returnValues.add(new Pair(BaseCmd.Properties.VLAN_ID.getName(), vlan.getVlanId()));
}
embeddedObject.add(new Pair("publicipaddress", new Object[] { returnValues } ));
} catch (Exception ex) {
s_logger.error("error!", ex);
throw new ServerApiException(BaseCmd.INTERNAL_ERROR, "Internal Error encountered while assigning IP address " + newIpAddr + " to account " + accountId);
}
return embeddedObject;
}
}