mirror of
https://github.com/apache/cloudstack.git
synced 2025-10-26 08:42:29 +01:00
CLOUDSTACK-6402: Fix StopCommand so that VMs are not removed accidentally as part of vmsync
Added a new flag 'checkBeforeCleanup' to StopCommand based on which check is done to see if VM is running in HV host. If VM is running then in this case it is not stopped and the operation bails out. Also modified the MS code to call the StopCommand with appropriate value for the flag based on the context. Currently it is only set to 'true' when called from the new vmsync logic based on powerstate of VM. For rest it is set to 'false' meaning no change in behaviour.
This commit is contained in:
parent
2a0236931a
commit
5e90b75c98
@ -25,26 +25,30 @@ public class StopCommand extends RebootCommand {
|
|||||||
private String publicConsoleProxyIpAddress = null;
|
private String publicConsoleProxyIpAddress = null;
|
||||||
boolean executeInSequence = false;
|
boolean executeInSequence = false;
|
||||||
private GPUDeviceTO gpuDevice;
|
private GPUDeviceTO gpuDevice;
|
||||||
|
boolean checkBeforeCleanup = false;
|
||||||
|
|
||||||
protected StopCommand() {
|
protected StopCommand() {
|
||||||
}
|
}
|
||||||
|
|
||||||
public StopCommand(VirtualMachine vm, boolean isProxy, String urlPort, String publicConsoleProxyIpAddress, boolean executeInSequence) {
|
public StopCommand(VirtualMachine vm, boolean isProxy, String urlPort, String publicConsoleProxyIpAddress, boolean executeInSequence, boolean checkBeforeCleanup) {
|
||||||
super(vm);
|
super(vm);
|
||||||
this.isProxy = isProxy;
|
this.isProxy = isProxy;
|
||||||
this.urlPort = urlPort;
|
this.urlPort = urlPort;
|
||||||
this.publicConsoleProxyIpAddress = publicConsoleProxyIpAddress;
|
this.publicConsoleProxyIpAddress = publicConsoleProxyIpAddress;
|
||||||
this.executeInSequence = executeInSequence;
|
this.executeInSequence = executeInSequence;
|
||||||
|
this.checkBeforeCleanup = checkBeforeCleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
public StopCommand(VirtualMachine vm, boolean executeInSequence) {
|
public StopCommand(VirtualMachine vm, boolean executeInSequence, boolean checkBeforeCleanup) {
|
||||||
super(vm);
|
super(vm);
|
||||||
this.executeInSequence = executeInSequence;
|
this.executeInSequence = executeInSequence;
|
||||||
|
this.checkBeforeCleanup = checkBeforeCleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
public StopCommand(String vmName, boolean executeInSequence) {
|
public StopCommand(String vmName, boolean executeInSequence, boolean checkBeforeCleanup) {
|
||||||
super(vmName);
|
super(vmName);
|
||||||
this.executeInSequence = executeInSequence;
|
this.executeInSequence = executeInSequence;
|
||||||
|
this.checkBeforeCleanup = checkBeforeCleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -71,4 +75,8 @@ public class StopCommand extends RebootCommand {
|
|||||||
public void setGpuDevice(GPUDeviceTO gpuDevice) {
|
public void setGpuDevice(GPUDeviceTO gpuDevice) {
|
||||||
this.gpuDevice = gpuDevice;
|
this.gpuDevice = gpuDevice;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean checkBeforeCleanup() {
|
||||||
|
return this.checkBeforeCleanup;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1047,7 +1047,7 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
|||||||
s_logger.info("The guru did not like the answers so stopping " + vm);
|
s_logger.info("The guru did not like the answers so stopping " + vm);
|
||||||
}
|
}
|
||||||
|
|
||||||
StopCommand cmd = new StopCommand(vm, getExecuteInSequence(vm.getHypervisorType()));
|
StopCommand cmd = new StopCommand(vm, getExecuteInSequence(vm.getHypervisorType()), false);
|
||||||
StopAnswer answer = (StopAnswer) _agentMgr.easySend(destHostId, cmd);
|
StopAnswer answer = (StopAnswer) _agentMgr.easySend(destHostId, cmd);
|
||||||
if ( answer != null ) {
|
if ( answer != null ) {
|
||||||
|
|
||||||
@ -1225,9 +1225,9 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected boolean sendStop(VirtualMachineGuru guru, VirtualMachineProfile profile, boolean force) {
|
protected boolean sendStop(VirtualMachineGuru guru, VirtualMachineProfile profile, boolean force, boolean checkBeforeCleanup) {
|
||||||
VirtualMachine vm = profile.getVirtualMachine();
|
VirtualMachine vm = profile.getVirtualMachine();
|
||||||
StopCommand stop = new StopCommand(vm, getExecuteInSequence(vm.getHypervisorType()));
|
StopCommand stop = new StopCommand(vm, getExecuteInSequence(vm.getHypervisorType()), checkBeforeCleanup);
|
||||||
try {
|
try {
|
||||||
StopAnswer answer = (StopAnswer)_agentMgr.send(vm.getHostId(), stop);
|
StopAnswer answer = (StopAnswer)_agentMgr.send(vm.getHostId(), stop);
|
||||||
if (answer != null) {
|
if (answer != null) {
|
||||||
@ -1282,7 +1282,7 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
|||||||
|
|
||||||
if (step == Step.Started || step == Step.Starting || step == Step.Release) {
|
if (step == Step.Started || step == Step.Starting || step == Step.Release) {
|
||||||
if (vm.getHostId() != null) {
|
if (vm.getHostId() != null) {
|
||||||
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop)) {
|
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop, false)) {
|
||||||
s_logger.warn("Failed to stop vm " + vm + " in " + State.Starting + " state as a part of cleanup process");
|
s_logger.warn("Failed to stop vm " + vm + " in " + State.Starting + " state as a part of cleanup process");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1295,26 +1295,26 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
|||||||
}
|
}
|
||||||
} else if (state == State.Stopping) {
|
} else if (state == State.Stopping) {
|
||||||
if (vm.getHostId() != null) {
|
if (vm.getHostId() != null) {
|
||||||
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop)) {
|
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop, false)) {
|
||||||
s_logger.warn("Failed to stop vm " + vm + " in " + State.Stopping + " state as a part of cleanup process");
|
s_logger.warn("Failed to stop vm " + vm + " in " + State.Stopping + " state as a part of cleanup process");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (state == State.Migrating) {
|
} else if (state == State.Migrating) {
|
||||||
if (vm.getHostId() != null) {
|
if (vm.getHostId() != null) {
|
||||||
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop)) {
|
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop, false)) {
|
||||||
s_logger.warn("Failed to stop vm " + vm + " in " + State.Migrating + " state as a part of cleanup process");
|
s_logger.warn("Failed to stop vm " + vm + " in " + State.Migrating + " state as a part of cleanup process");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (vm.getLastHostId() != null) {
|
if (vm.getLastHostId() != null) {
|
||||||
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop)) {
|
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop, false)) {
|
||||||
s_logger.warn("Failed to stop vm " + vm + " in " + State.Migrating + " state as a part of cleanup process");
|
s_logger.warn("Failed to stop vm " + vm + " in " + State.Migrating + " state as a part of cleanup process");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (state == State.Running) {
|
} else if (state == State.Running) {
|
||||||
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop)) {
|
if (!sendStop(guru, profile, cleanUpEvenIfUnableToStop, false)) {
|
||||||
s_logger.warn("Failed to stop vm " + vm + " in " + State.Running + " state as a part of cleanup process");
|
s_logger.warn("Failed to stop vm " + vm + " in " + State.Running + " state as a part of cleanup process");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1488,7 +1488,7 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
|||||||
|
|
||||||
vmGuru.prepareStop(profile);
|
vmGuru.prepareStop(profile);
|
||||||
|
|
||||||
StopCommand stop = new StopCommand(vm, getExecuteInSequence(vm.getHypervisorType()));
|
StopCommand stop = new StopCommand(vm, getExecuteInSequence(vm.getHypervisorType()), false);
|
||||||
|
|
||||||
boolean stopped = false;
|
boolean stopped = false;
|
||||||
StopAnswer answer = null;
|
StopAnswer answer = null;
|
||||||
@ -2464,11 +2464,11 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
|||||||
}
|
}
|
||||||
|
|
||||||
public Command cleanup(VirtualMachine vm) {
|
public Command cleanup(VirtualMachine vm) {
|
||||||
return new StopCommand(vm, getExecuteInSequence(vm.getHypervisorType()));
|
return new StopCommand(vm, getExecuteInSequence(vm.getHypervisorType()), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Command cleanup(String vmName) {
|
public Command cleanup(String vmName) {
|
||||||
return new StopCommand(vmName, getExecuteInSequence(null));
|
return new StopCommand(vmName, getExecuteInSequence(null), false);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4257,7 +4257,7 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac
|
|||||||
|
|
||||||
VirtualMachineGuru vmGuru = getVmGuru(vm);
|
VirtualMachineGuru vmGuru = getVmGuru(vm);
|
||||||
VirtualMachineProfile profile = new VirtualMachineProfileImpl(vm);
|
VirtualMachineProfile profile = new VirtualMachineProfileImpl(vm);
|
||||||
sendStop(vmGuru, profile, true);
|
sendStop(vmGuru, profile, true, true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
stateTransitTo(vm, VirtualMachine.Event.FollowAgentPowerOffReport, null);
|
stateTransitTo(vm, VirtualMachine.Event.FollowAgentPowerOffReport, null);
|
||||||
|
|||||||
@ -466,12 +466,12 @@ public class VirtualMachineManagerImplTest {
|
|||||||
VirtualMachineGuru guru = mock(VirtualMachineGuru.class);
|
VirtualMachineGuru guru = mock(VirtualMachineGuru.class);
|
||||||
VirtualMachine vm = mock(VirtualMachine.class);
|
VirtualMachine vm = mock(VirtualMachine.class);
|
||||||
VirtualMachineProfile profile = mock(VirtualMachineProfile.class);
|
VirtualMachineProfile profile = mock(VirtualMachineProfile.class);
|
||||||
StopAnswer answer = new StopAnswer(new StopCommand(vm, false), "ok", true);
|
StopAnswer answer = new StopAnswer(new StopCommand(vm, false, false), "ok", true);
|
||||||
when(profile.getVirtualMachine()).thenReturn(vm);
|
when(profile.getVirtualMachine()).thenReturn(vm);
|
||||||
when(vm.getHostId()).thenReturn(1L);
|
when(vm.getHostId()).thenReturn(1L);
|
||||||
when(_agentMgr.send(anyLong(), (Command)any())).thenReturn(answer);
|
when(_agentMgr.send(anyLong(), (Command)any())).thenReturn(answer);
|
||||||
|
|
||||||
boolean actual = _vmMgr.sendStop(guru, profile, false);
|
boolean actual = _vmMgr.sendStop(guru, profile, false, false);
|
||||||
|
|
||||||
Assert.assertTrue(actual);
|
Assert.assertTrue(actual);
|
||||||
}
|
}
|
||||||
@ -481,12 +481,12 @@ public class VirtualMachineManagerImplTest {
|
|||||||
VirtualMachineGuru guru = mock(VirtualMachineGuru.class);
|
VirtualMachineGuru guru = mock(VirtualMachineGuru.class);
|
||||||
VirtualMachine vm = mock(VirtualMachine.class);
|
VirtualMachine vm = mock(VirtualMachine.class);
|
||||||
VirtualMachineProfile profile = mock(VirtualMachineProfile.class);
|
VirtualMachineProfile profile = mock(VirtualMachineProfile.class);
|
||||||
StopAnswer answer = new StopAnswer(new StopCommand(vm, false), "fail", false);
|
StopAnswer answer = new StopAnswer(new StopCommand(vm, false, false), "fail", false);
|
||||||
when(profile.getVirtualMachine()).thenReturn(vm);
|
when(profile.getVirtualMachine()).thenReturn(vm);
|
||||||
when(vm.getHostId()).thenReturn(1L);
|
when(vm.getHostId()).thenReturn(1L);
|
||||||
when(_agentMgr.send(anyLong(), (Command)any())).thenReturn(answer);
|
when(_agentMgr.send(anyLong(), (Command)any())).thenReturn(answer);
|
||||||
|
|
||||||
boolean actual = _vmMgr.sendStop(guru, profile, false);
|
boolean actual = _vmMgr.sendStop(guru, profile, false, false);
|
||||||
|
|
||||||
Assert.assertFalse(actual);
|
Assert.assertFalse(actual);
|
||||||
}
|
}
|
||||||
@ -500,7 +500,7 @@ public class VirtualMachineManagerImplTest {
|
|||||||
when(vm.getHostId()).thenReturn(1L);
|
when(vm.getHostId()).thenReturn(1L);
|
||||||
when(_agentMgr.send(anyLong(), (Command)any())).thenReturn(null);
|
when(_agentMgr.send(anyLong(), (Command)any())).thenReturn(null);
|
||||||
|
|
||||||
boolean actual = _vmMgr.sendStop(guru, profile, false);
|
boolean actual = _vmMgr.sendStop(guru, profile, false, false);
|
||||||
|
|
||||||
Assert.assertFalse(actual);
|
Assert.assertFalse(actual);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1135,7 +1135,18 @@ namespace HypervResource
|
|||||||
logger.Info(CloudStackTypes.StopCommand + Utils.CleanString(cmd.ToString()));
|
logger.Info(CloudStackTypes.StopCommand + Utils.CleanString(cmd.ToString()));
|
||||||
string details = null;
|
string details = null;
|
||||||
bool result = false;
|
bool result = false;
|
||||||
|
bool checkBeforeCleanup = cmd.checkBeforeCleanup;
|
||||||
|
String vmName = cmd.vmName;
|
||||||
|
|
||||||
|
if (checkBeforeCleanup == true)
|
||||||
|
{
|
||||||
|
ComputerSystem vm = wmiCallsV2.GetComputerSystem(vmName);
|
||||||
|
if (vm == null || vm.EnabledState == 2)
|
||||||
|
{
|
||||||
|
// VM is not available or vm in running state
|
||||||
|
return ReturnCloudStackTypedJArray(new { result = false, details = "VM is running on host, bailing out", vm = vmName, contextMap = contextMap }, CloudStackTypes.StopAnswer);
|
||||||
|
}
|
||||||
|
}
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
wmiCallsV2.DestroyVm(cmd);
|
wmiCallsV2.DestroyVm(cmd);
|
||||||
|
|||||||
@ -281,7 +281,7 @@ namespace ServerResource.Tests
|
|||||||
HypervResourceController rsrcServer = new HypervResourceController();
|
HypervResourceController rsrcServer = new HypervResourceController();
|
||||||
HypervResourceController.wmiCallsV2 = wmiCallsV2;
|
HypervResourceController.wmiCallsV2 = wmiCallsV2;
|
||||||
|
|
||||||
String sampleStop = "{\"isProxy\":false,\"vmName\":\"i-2-17-VM\",\"contextMap\":{},\"wait\":0}";
|
String sampleStop = "{\"isProxy\":false,\"vmName\":\"i-2-17-VM\",\"contextMap\":{},\"checkBeforeCleanup\":false,\"wait\":0}";
|
||||||
dynamic jsonStopCmd = JsonConvert.DeserializeObject(sampleStop);
|
dynamic jsonStopCmd = JsonConvert.DeserializeObject(sampleStop);
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
|
|||||||
@ -3485,6 +3485,18 @@ public class LibvirtComputingResource extends ServerResourceBase implements Serv
|
|||||||
protected Answer execute(StopCommand cmd) {
|
protected Answer execute(StopCommand cmd) {
|
||||||
final String vmName = cmd.getVmName();
|
final String vmName = cmd.getVmName();
|
||||||
|
|
||||||
|
if (cmd.checkBeforeCleanup()) {
|
||||||
|
try {
|
||||||
|
Connect conn = LibvirtConnection.getConnectionByVmName(vmName);
|
||||||
|
Domain vm = conn.domainLookupByName(cmd.getVmName());
|
||||||
|
if (vm != null && vm.getInfo().state == DomainInfo.DomainState.VIR_DOMAIN_RUNNING) {
|
||||||
|
return new StopAnswer(cmd, "vm is still running on host", false);
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
s_logger.debug("Failed to get vm status in case of checkboforecleanup is true", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
State state = null;
|
State state = null;
|
||||||
synchronized (_vms) {
|
synchronized (_vms) {
|
||||||
state = _vms.get(vmName);
|
state = _vms.get(vmName);
|
||||||
|
|||||||
@ -2740,6 +2740,18 @@ public class VmwareResource implements StoragePoolResource, ServerResource, Vmwa
|
|||||||
try {
|
try {
|
||||||
VirtualMachineMO vmMo = hyperHost.findVmOnHyperHost(cmd.getVmName());
|
VirtualMachineMO vmMo = hyperHost.findVmOnHyperHost(cmd.getVmName());
|
||||||
if (vmMo != null) {
|
if (vmMo != null) {
|
||||||
|
if (cmd.checkBeforeCleanup()) {
|
||||||
|
if (getVmPowerState(vmMo) != PowerState.PowerOff) {
|
||||||
|
String msg = "StopCommand is sent for cleanup and VM " + cmd.getVmName() + " is current running. ignore it.";
|
||||||
|
s_logger.warn(msg);
|
||||||
|
return new StopAnswer(cmd, msg, false);
|
||||||
|
} else {
|
||||||
|
String msg = "StopCommand is sent for cleanup and VM " + cmd.getVmName() + " is indeed stopped already.";
|
||||||
|
s_logger.info(msg);
|
||||||
|
return new StopAnswer(cmd, msg, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
State state = null;
|
State state = null;
|
||||||
synchronized (_vms) {
|
synchronized (_vms) {
|
||||||
state = _vms.get(cmd.getVmName());
|
state = _vms.get(cmd.getVmName());
|
||||||
|
|||||||
@ -3628,6 +3628,12 @@ public abstract class CitrixResourceBase implements ServerResource, HypervisorRe
|
|||||||
return new StopAnswer(cmd, msg, platformstring, false);
|
return new StopAnswer(cmd, msg, platformstring, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (cmd.checkBeforeCleanup() && vmr.powerState == VmPowerState.RUNNING) {
|
||||||
|
String msg = "Vm " + vmName + " is running on host and checkBeforeCleanup flag is set, so bailing out";
|
||||||
|
s_logger.debug(msg);
|
||||||
|
return new StopAnswer(cmd, msg, false);
|
||||||
|
}
|
||||||
|
|
||||||
State state = s_vms.getState(_cluster, vmName);
|
State state = s_vms.getState(_cluster, vmName);
|
||||||
|
|
||||||
synchronized (_cluster.intern()) {
|
synchronized (_cluster.intern()) {
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user