CLOUDSTACK-7828.Avoid marking IPs already in Allocated as Allocated again. Use row lock to ensure that prev state is either Allocating or Free. This will inturn avoid logging duplicate events

This commit is contained in:
Santhosh Edukulla 2014-11-03 12:40:13 +05:30
parent 34b7288ca4
commit 1c01da3e08

View File

@ -807,32 +807,32 @@ public class IpAddressManagerImpl extends ManagerBase implements IpAddressManage
@Override @Override
public void markPublicIpAsAllocated(final IPAddressVO addr) { public void markPublicIpAsAllocated(final IPAddressVO addr) {
assert (addr.getState() == IpAddress.State.Allocating || addr.getState() == IpAddress.State.Free) : "Unable to transition from state " + addr.getState() + " to "
+ IpAddress.State.Allocated;
Transaction.execute(new TransactionCallbackNoReturn() { Transaction.execute(new TransactionCallbackNoReturn() {
@Override @Override
public void doInTransactionWithoutResult(TransactionStatus status) { public void doInTransactionWithoutResult(TransactionStatus status) {
Account owner = _accountMgr.getAccount(addr.getAllocatedToAccountId()); Account owner = _accountMgr.getAccount(addr.getAllocatedToAccountId());
synchronized (this) {
addr.setState(IpAddress.State.Allocated); if (_ipAddressDao.lockRow(addr.getId(),true) != null) {
_ipAddressDao.update(addr.getId(), addr); IPAddressVO userIp = _ipAddressDao.findById(addr.getId());
if (userIp.getState() == IpAddress.State.Allocating || addr.getState() == IpAddress.State.Free) {
// Save usage event addr.setState(IpAddress.State.Allocated);
if (owner.getAccountId() != Account.ACCOUNT_ID_SYSTEM) { _ipAddressDao.update(addr.getId(), addr);
VlanVO vlan = _vlanDao.findById(addr.getVlanId()); // Save usage event
if (owner.getAccountId() != Account.ACCOUNT_ID_SYSTEM) {
String guestType = vlan.getVlanType().toString(); VlanVO vlan = _vlanDao.findById(addr.getVlanId());
String guestType = vlan.getVlanType().toString();
if (!isIpDedicated(addr)) { if (!isIpDedicated(addr)) {
UsageEventUtils.publishUsageEvent(EventTypes.EVENT_NET_IP_ASSIGN, owner.getId(), addr.getDataCenterId(), addr.getId(), addr.getAddress().toString(), UsageEventUtils.publishUsageEvent(EventTypes.EVENT_NET_IP_ASSIGN, owner.getId(), addr.getDataCenterId(), addr.getId(), addr.getAddress().toString(),
addr.isSourceNat(), guestType, addr.getSystem(), addr.getClass().getName(), addr.getUuid()); addr.isSourceNat(), guestType, addr.getSystem(), addr.getClass().getName(), addr.getUuid());
} }
if (updateIpResourceCount(addr)) {
if (updateIpResourceCount(addr)) { _resourceLimitMgr.incrementResourceCount(owner.getId(), ResourceType.public_ip);
_resourceLimitMgr.incrementResourceCount(owner.getId(), ResourceType.public_ip); }
}
}
}
} }
} }
}
}); });
} }
@ -922,7 +922,6 @@ public class IpAddressManagerImpl extends ManagerBase implements IpAddressManage
public boolean applyIpAssociations(Network network, boolean continueOnError) throws ResourceUnavailableException { public boolean applyIpAssociations(Network network, boolean continueOnError) throws ResourceUnavailableException {
List<IPAddressVO> userIps = _ipAddressDao.listByAssociatedNetwork(network.getId(), null); List<IPAddressVO> userIps = _ipAddressDao.listByAssociatedNetwork(network.getId(), null);
boolean success = true; boolean success = true;
// CloudStack will take a lazy approach to associate an acquired public IP to a network service provider as // CloudStack will take a lazy approach to associate an acquired public IP to a network service provider as
// it will not know what service an acquired IP will be used for. An IP is actually associated with a provider when first // it will not know what service an acquired IP will be used for. An IP is actually associated with a provider when first
// rule is applied. Similarly when last rule on the acquired IP is revoked, IP is not associated with any provider // rule is applied. Similarly when last rule on the acquired IP is revoked, IP is not associated with any provider
@ -941,7 +940,6 @@ public class IpAddressManagerImpl extends ManagerBase implements IpAddressManage
} }
} }
} }
return success; return success;
} }