mirror of
https://github.com/apache/cloudstack.git
synced 2025-10-26 08:42:29 +01:00
CLOUDSTACK-1638: Introduce NetworkMigrationResponder
The location of the virtual machine is provided by DeployDestination, which will be passed in NetworkGuru#reserve and NetworkElement#prepare. During the virtual machine migration, it actually changes DeployDestination and it looks like that it will tell that event to network components as it has NetworkManager#prepareNicForMigration. The problem is that althogh the interface has that method, NetworkManagerImpl does not tell the DeployDestination changes to network components. So IMHO, we need to add calls of NetworkGuru#reserve and NetworkElement#prepare in NetworkManagerImpl#prepareNicForMigration . And then, we also need to add calls NetworkGuru#release and NetworkElement#release after the migration, otherwise the network resources that plugin reserved will be kept even when the vm leaves off. (Sheng Yang: rebase code, add license header) Signed-off-by: Sheng Yang <sheng.yang@citrix.com>
This commit is contained in:
parent
cf6045f1aa
commit
7260e8d83f
70
api/src/com/cloud/network/NetworkMigrationResponder.java
Normal file
70
api/src/com/cloud/network/NetworkMigrationResponder.java
Normal file
@ -0,0 +1,70 @@
|
||||
// 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;
|
||||
|
||||
import com.cloud.deploy.DeployDestination;
|
||||
import com.cloud.vm.NicProfile;
|
||||
import com.cloud.vm.ReservationContext;
|
||||
import com.cloud.vm.VirtualMachine;
|
||||
import com.cloud.vm.VirtualMachineProfile;
|
||||
|
||||
/**
|
||||
* NetworkGuru and NetworkElements that implement this interface
|
||||
* will be called during Virtual Machine migration.
|
||||
*/
|
||||
public interface NetworkMigrationResponder {
|
||||
/**
|
||||
* Prepare for migration.
|
||||
*
|
||||
* This method will be called per nic before the vm migration.
|
||||
* @param nic
|
||||
* @param network
|
||||
* @param vm
|
||||
* @param dest
|
||||
* @param context
|
||||
* @return true when operation was successful.
|
||||
*/
|
||||
public boolean prepareMigration(NicProfile nic, Network network, VirtualMachineProfile<? extends VirtualMachine> vm, DeployDestination dest, ReservationContext context);
|
||||
|
||||
/**
|
||||
* Cancel for migration preparation.
|
||||
*
|
||||
* This method will be called per nic when the entire vm migration
|
||||
* process failed and need to release the resouces that was
|
||||
* allocated at the migration preparation.
|
||||
* @param nic destination nic
|
||||
* @param network destination network
|
||||
* @param vm destination vm profile
|
||||
* @param src The context nic migrates from.
|
||||
* @param dst The context nic migrates to.
|
||||
*/
|
||||
public void rollbackMigration(NicProfile nic, Network network, VirtualMachineProfile<? extends VirtualMachine> vm, ReservationContext src, ReservationContext dst);
|
||||
|
||||
/**
|
||||
* Commit the migration resource.
|
||||
*
|
||||
* This method will be called per nic when the entire vm migration
|
||||
* process was successful. This is useful to release the resource of
|
||||
* source deployment where vm has left.
|
||||
* @param nic source nic
|
||||
* @param network source network
|
||||
* @param vm source vm profile
|
||||
* @param src the context nic migrates from.
|
||||
* @param dst the context nic migrates to.
|
||||
*/
|
||||
public void commitMigration(NicProfile nic, Network network, VirtualMachineProfile<? extends VirtualMachine> vm, ReservationContext src, ReservationContext dst);
|
||||
}
|
||||
@ -123,7 +123,34 @@ public interface NetworkManager {
|
||||
Pair<NetworkGuru, NetworkVO> implementNetwork(long networkId, DeployDestination dest, ReservationContext context) throws ConcurrentOperationException, ResourceUnavailableException,
|
||||
InsufficientCapacityException;
|
||||
|
||||
<T extends VMInstanceVO> void prepareNicForMigration(VirtualMachineProfile<T> vm, DeployDestination dest);
|
||||
/**
|
||||
* prepares vm nic change for migration
|
||||
*
|
||||
* This method will be called in migration transaction before the vm migration.
|
||||
* @param vm
|
||||
* @param dest
|
||||
*/
|
||||
void prepareNicForMigration(VirtualMachineProfile<? extends VMInstanceVO> vm, DeployDestination dest);
|
||||
|
||||
/**
|
||||
* commit vm nic change for migration
|
||||
*
|
||||
* This method will be called in migration transaction after the successful
|
||||
* vm migration.
|
||||
* @param src
|
||||
* @param dst
|
||||
*/
|
||||
void commitNicForMigration(VirtualMachineProfile<? extends VMInstanceVO> src, VirtualMachineProfile<? extends VMInstanceVO> dst);
|
||||
|
||||
/**
|
||||
* rollback vm nic change for migration
|
||||
*
|
||||
* This method will be called in migaration transaction after vm migration
|
||||
* failure.
|
||||
* @param src
|
||||
* @param dst
|
||||
*/
|
||||
void rollbackNicForMigration(VirtualMachineProfile<? extends VMInstanceVO> src, VirtualMachineProfile<? extends VMInstanceVO> dst);
|
||||
|
||||
boolean shutdownNetwork(long networkId, ReservationContext context, boolean cleanupElements);
|
||||
|
||||
|
||||
@ -2000,8 +2000,9 @@ public class NetworkManagerImpl extends ManagerBase implements NetworkManager, L
|
||||
}
|
||||
|
||||
@Override
|
||||
public <T extends VMInstanceVO> void prepareNicForMigration(VirtualMachineProfile<T> vm, DeployDestination dest) {
|
||||
public void prepareNicForMigration(VirtualMachineProfile<? extends VMInstanceVO> vm, DeployDestination dest) {
|
||||
List<NicVO> nics = _nicDao.listByVmId(vm.getId());
|
||||
ReservationContext context = new ReservationContextImpl(UUID.randomUUID().toString(), null, null);
|
||||
for (NicVO nic : nics) {
|
||||
NetworkVO network = _networksDao.findById(nic.getNetworkId());
|
||||
Integer networkRate = _networkModel.getNetworkRate(network.getId(), vm.getId());
|
||||
@ -2009,11 +2010,80 @@ public class NetworkManagerImpl extends ManagerBase implements NetworkManager, L
|
||||
NetworkGuru guru = AdapterBase.getAdapterByName(_networkGurus, network.getGuruName());
|
||||
NicProfile profile = new NicProfile(nic, network, nic.getBroadcastUri(), nic.getIsolationUri(), networkRate,
|
||||
_networkModel.isSecurityGroupSupportedInNetwork(network), _networkModel.getNetworkTag(vm.getHypervisorType(), network));
|
||||
if(guru instanceof NetworkMigrationResponder){
|
||||
if(!((NetworkMigrationResponder) guru).prepareMigration(profile, network, vm, dest, context)){
|
||||
s_logger.error("NetworkGuru "+guru+" prepareForMigration failed."); // XXX: Transaction error
|
||||
}
|
||||
}
|
||||
for (NetworkElement element : _networkElements) {
|
||||
if(element instanceof NetworkMigrationResponder){
|
||||
if(!((NetworkMigrationResponder) element).prepareMigration(profile, network, vm, dest, context)){
|
||||
s_logger.error("NetworkElement "+element+" prepareForMigration failed."); // XXX: Transaction error
|
||||
}
|
||||
}
|
||||
}
|
||||
guru.updateNicProfile(profile, network);
|
||||
vm.addNic(profile);
|
||||
}
|
||||
}
|
||||
|
||||
private NicProfile findNicProfileById(VirtualMachineProfile<? extends VMInstanceVO> vm, long id){
|
||||
for(NicProfile nic: vm.getNics()){
|
||||
if(nic.getId() == id){
|
||||
return nic;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void commitNicForMigration(
|
||||
VirtualMachineProfile<? extends VMInstanceVO> src,
|
||||
VirtualMachineProfile<? extends VMInstanceVO> dst) {
|
||||
for(NicProfile nicSrc: src.getNics()){
|
||||
NetworkVO network = _networksDao.findById(nicSrc.getNetworkId());
|
||||
NetworkGuru guru = AdapterBase.getAdapterByName(_networkGurus, network.getGuruName());
|
||||
NicProfile nicDst = findNicProfileById(dst, nicSrc.getId());
|
||||
ReservationContext src_context = new ReservationContextImpl(nicSrc.getReservationId(), null, null);
|
||||
ReservationContext dst_context = new ReservationContextImpl(nicDst.getReservationId(), null, null);
|
||||
|
||||
if(guru instanceof NetworkMigrationResponder){
|
||||
((NetworkMigrationResponder) guru).commitMigration(nicSrc, network, src, src_context, dst_context);
|
||||
}
|
||||
for (NetworkElement element : _networkElements) {
|
||||
if(element instanceof NetworkMigrationResponder){
|
||||
((NetworkMigrationResponder) element).commitMigration(nicSrc, network, src, src_context, dst_context);
|
||||
}
|
||||
}
|
||||
// update the reservation id
|
||||
NicVO nicVo = _nicDao.findById(nicDst.getId());
|
||||
nicVo.setReservationId(nicDst.getReservationId());
|
||||
_nicDao.persist(nicVo);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void rollbackNicForMigration(
|
||||
VirtualMachineProfile<? extends VMInstanceVO> src,
|
||||
VirtualMachineProfile<? extends VMInstanceVO> dst) {
|
||||
for(NicProfile nicDst: dst.getNics()){
|
||||
NetworkVO network = _networksDao.findById(nicDst.getNetworkId());
|
||||
NetworkGuru guru = AdapterBase.getAdapterByName(_networkGurus, network.getGuruName());
|
||||
NicProfile nicSrc = findNicProfileById(src, nicDst.getId());
|
||||
ReservationContext src_context = new ReservationContextImpl(nicSrc.getReservationId(), null, null);
|
||||
ReservationContext dst_context = new ReservationContextImpl(nicDst.getReservationId(), null, null);
|
||||
|
||||
if(guru instanceof NetworkMigrationResponder){
|
||||
((NetworkMigrationResponder) guru).rollbackMigration(nicDst, network, dst, src_context, dst_context);
|
||||
}
|
||||
for (NetworkElement element : _networkElements) {
|
||||
if(element instanceof NetworkMigrationResponder){
|
||||
((NetworkMigrationResponder) element).rollbackMigration(nicDst, network, dst, src_context, dst_context);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
@DB
|
||||
public void release(VirtualMachineProfile<? extends VMInstanceVO> vmProfile, boolean forced) throws
|
||||
|
||||
@ -1414,6 +1414,11 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
||||
alertType = AlertManager.ALERT_TYPE_CONSOLE_PROXY_MIGRATE;
|
||||
}
|
||||
|
||||
VirtualMachineProfile<VMInstanceVO> vmSrc = new VirtualMachineProfileImpl<VMInstanceVO>(vm);
|
||||
for(NicProfile nic: _networkMgr.getNicProfiles(vm)){
|
||||
vmSrc.addNic(nic);
|
||||
}
|
||||
|
||||
VirtualMachineProfile<VMInstanceVO> profile = new VirtualMachineProfileImpl<VMInstanceVO>(vm);
|
||||
_networkMgr.prepareNicForMigration(profile, dest);
|
||||
this.volumeMgr.prepareForMigration(profile, dest);
|
||||
@ -1439,6 +1444,7 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
||||
throw new AgentUnavailableException("Operation timed out", dstHostId);
|
||||
} finally {
|
||||
if (pfma == null) {
|
||||
_networkMgr.rollbackNicForMigration(vmSrc, profile);
|
||||
work.setStep(Step.Done);
|
||||
_workDao.update(work.getId(), work);
|
||||
}
|
||||
@ -1447,10 +1453,12 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
||||
vm.setLastHostId(srcHostId);
|
||||
try {
|
||||
if (vm == null || vm.getHostId() == null || vm.getHostId() != srcHostId || !changeState(vm, Event.MigrationRequested, dstHostId, work, Step.Migrating)) {
|
||||
_networkMgr.rollbackNicForMigration(vmSrc, profile);
|
||||
s_logger.info("Migration cancelled because state has changed: " + vm);
|
||||
throw new ConcurrentOperationException("Migration cancelled because state has changed: " + vm);
|
||||
}
|
||||
} catch (NoTransitionException e1) {
|
||||
_networkMgr.rollbackNicForMigration(vmSrc, profile);
|
||||
s_logger.info("Migration cancelled because " + e1.getMessage());
|
||||
throw new ConcurrentOperationException("Migration cancelled because " + e1.getMessage());
|
||||
}
|
||||
@ -1502,6 +1510,7 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
||||
} finally {
|
||||
if (!migrated) {
|
||||
s_logger.info("Migration was unsuccessful. Cleaning up: " + vm);
|
||||
_networkMgr.rollbackNicForMigration(vmSrc, profile);
|
||||
|
||||
_alertMgr.sendAlert(alertType, fromHost.getDataCenterId(), fromHost.getPodId(), "Unable to migrate vm " + vm.getInstanceName() + " from host " + fromHost.getName() + " in zone "
|
||||
+ dest.getDataCenter().getName() + " and pod " + dest.getPod().getName(), "Migrate Command failed. Please check logs.");
|
||||
@ -1516,6 +1525,8 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
||||
} catch (NoTransitionException e) {
|
||||
s_logger.warn(e.getMessage());
|
||||
}
|
||||
}else{
|
||||
_networkMgr.commitNicForMigration(vmSrc, profile);
|
||||
}
|
||||
|
||||
work.setStep(Step.Done);
|
||||
|
||||
@ -277,12 +277,6 @@ public class MockNetworkManagerImpl extends ManagerBase implements NetworkManage
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public <T extends VMInstanceVO> void prepareNicForMigration(VirtualMachineProfile<T> vm, DeployDestination dest) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean destroyNetwork(long networkId, ReservationContext context) {
|
||||
// TODO Auto-generated method stub
|
||||
@ -966,4 +960,27 @@ public class MockNetworkManagerImpl extends ManagerBase implements NetworkManage
|
||||
public PublicIp assignPublicIpAddressFromVlans(long dcId, Long podId, Account owner, VlanType type, List<Long> vlanDbIds, Long networkId, String requestedIp, boolean isSystem) throws InsufficientAddressCapacityException {
|
||||
return null; //To change body of implemented methods use File | Settings | File Templates.
|
||||
}
|
||||
|
||||
@Override
|
||||
public void prepareNicForMigration(
|
||||
VirtualMachineProfile<? extends VMInstanceVO> vm,
|
||||
DeployDestination dest) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public void commitNicForMigration(
|
||||
VirtualMachineProfile<? extends VMInstanceVO> src,
|
||||
VirtualMachineProfile<? extends VMInstanceVO> dst) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public void rollbackNicForMigration(
|
||||
VirtualMachineProfile<? extends VMInstanceVO> src,
|
||||
VirtualMachineProfile<? extends VMInstanceVO> dst) {
|
||||
// TODO Auto-generated method stub
|
||||
}
|
||||
}
|
||||
|
||||
@ -845,18 +845,6 @@ public class MockNetworkManagerImpl extends ManagerBase implements NetworkManage
|
||||
|
||||
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see com.cloud.network.NetworkManager#prepareNicForMigration(com.cloud.vm.VirtualMachineProfile, com.cloud.deploy.DeployDestination)
|
||||
*/
|
||||
@Override
|
||||
public <T extends VMInstanceVO> void prepareNicForMigration(VirtualMachineProfile<T> vm, DeployDestination dest) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see com.cloud.network.NetworkManager#shutdownNetwork(long, com.cloud.vm.ReservationContext, boolean)
|
||||
@ -1471,4 +1459,40 @@ public class MockNetworkManagerImpl extends ManagerBase implements NetworkManage
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
@Override
|
||||
public void prepareNicForMigration(
|
||||
VirtualMachineProfile<? extends VMInstanceVO> vm,
|
||||
DeployDestination dest) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
@Override
|
||||
public void commitNicForMigration(
|
||||
VirtualMachineProfile<? extends VMInstanceVO> src,
|
||||
VirtualMachineProfile<? extends VMInstanceVO> dst) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
@Override
|
||||
public void rollbackNicForMigration(
|
||||
VirtualMachineProfile<? extends VMInstanceVO> src,
|
||||
VirtualMachineProfile<? extends VMInstanceVO> dst) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user