Я создал скрипт Python, который приостановил бы все работающие виртуальные машины VirtualBox, а затем настроил систему для запуска сценария при выходе из системы в качестве запланированной задачи.
Я не знаю, насколько надежен этот метод. Как уже отмечали другие, существуют ограничения на то, как долго система будет ждать завершения задачи Winlogon 7002. Но лично у меня не было никаких проблем с предоставлением пригодных для сохранения состояний даже с несколькими запущенными виртуальными машинами на 4+ ГБ общей виртуальной памяти.
Вот шаги, чтобы настроить это:
- Загрузите и установите Python 2.7.x с python.org
- Создайте файл скрипта Python где-нибудь в вашей системе, используя Блокнот или любой другой текстовый редактор (см. Ниже)
- Открыть планировщик задач
- Выберите «Действие» -> «Создать базовую задачу» и с помощью мастера создайте задачу со следующими настройками.
- Имя по вашему выбору
- Запустите задачу, когда определенное событие зарегистрировано
- Журнал: Система
- Источник: Winlogon
- Код события: 7002
- Начать программу
- Рядом с Program / Script , введите полный путь к вашему
python.exe
, напримерc:\Python27\python.exe
- Рядом с полем Добавить аргументы введите полный путь к файлу скрипта Python, например, я помещаю свой в подпапку папки с документами, так что это
C:\Users\rakslice\Documents\vboxsuspend\vboxsuspend.py
- Выберите Готово.
Теперь виртуальные машины VirtualBox должны быть приостановлены при выходе из системы / перезагрузке / завершении работы.
Скрипт python для выключения находится ниже:
# 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()