Eu criei um script python que suspendia todas as VMs do VirtualBox em execução e depois configurava o sistema para executar o script no logout como uma tarefa agendada.
Eu não sei quão confiável este método é exatamente. Como outros já observaram , há limites para quanto tempo o sistema aguardará a conclusão de uma tarefa do Winlogon 7002. Mas, pessoalmente, não tive nenhum problema em fornecer estados de economia utilizáveis, mesmo com várias VMs em execução em mais de 4 GB de RAM geral da VM.
Aqui estão os passos para configurá-lo:
- Faça o download e instale o Python 2.7.x em python.org
- Crie o arquivo de script python em algum lugar do sistema usando o Bloco de Notas ou qualquer outro editor de texto simples (veja abaixo)
- Abra o Agendador de Tarefas
- Escolher ação - > Crie uma Tarefa Básica ... e use o assistente para criar uma tarefa com as seguintes configurações
- Um nome de sua escolha
- Iniciar a tarefa quando um evento específico é registrado
- Log: sistema
- Fonte: Winlogon
- ID do evento: 7002
- Iniciar um programa
- Ao lado de Programa / Script , insira o caminho completo para seu
python.exe
, por exemplo c:\Python27\python.exe
- Ao lado de Adicionar argumentos , insira o caminho completo onde você colocou o arquivo de script python, por exemplo, coloquei o meu em uma subpasta da pasta my documents, portanto, isso é
C:\Users\rakslice\Documents\vboxsuspend\vboxsuspend.py
- Escolha Concluir.
Agora as VMs do VirtualBox devem ser suspensas no logout / restart / shutdown.
O script python para fazer o desligamento está abaixo:
# A script to suspend all running VirtualBox VMs
import os
import subprocess
import sys
class VM(object):
def __init__(self, name, uuid):
self.name = name
self.uuid = uuid
def __repr__(self):
return "VM(%r,%r)" % (self.name, self.uuid)
class VBoxRunner(object):
def __init__(self):
program_files = os.environ["ProgramW6432"]
vbox_dir = os.path.join(program_files, "Oracle", "VirtualBox")
self.vboxmanage_filename = os.path.join(vbox_dir, "VBoxManage.exe")
def vbox_run(self, *args):
subprocess.check_call([self.vboxmanage_filename] + list(args))
def vbox_run_output(self, *args):
return subprocess.check_output([self.vboxmanage_filename] + list(args))
def list(self, running=True):
if running:
list_cmd = "runningvms"
else:
list_cmd = "vms"
return [self.parse_vm_list_entry(x) for x in self.vbox_run_output("list", list_cmd).strip().split("\n")]
def suspend_all(self):
success = True
stopped_some_vms = False
vms = self.list(running=True)
for vm in vms:
if vm is None:
continue
# noinspection PyBroadException
try:
self.suspend_vm(vm)
except:
success = False
else:
stopped_some_vms = True
if not stopped_some_vms:
self.message("No running vms")
return success
@staticmethod
def parse_vm_list_entry(x):
""":type x: str"""
if not x.startswith('"'):
return None
end_pos = x.find('"', 1)
if end_pos == -1:
return None
name = x[1:end_pos]
assert x[end_pos + 1: end_pos + 3] == " {"
assert x.endswith("}")
uuid = x[end_pos + 2:]
return VM(name, uuid)
@staticmethod
def message(msg):
print >>sys.stderr, msg
def suspend_vm(self, vm):
assert isinstance(vm, VM)
self.vbox_run("controlvm", vm.uuid, "savestate")
def main():
vr = VBoxRunner()
success = vr.suspend_all()
if not success:
sys.exit(1)
if __name__ == "__main__":
main()