ホストコンピューターのシャットダウンまたは再起動時にVirtualBox Windowsで実行されているゲストVMを正常にシャットダウンするソリューションを見つけようとしています。
最も安全なオプションは、ホストがシャットダウンを開始したときに「状態保存」コマンドをトリガーすることですが、VMが状態の保存と電源オフを完了するまでホストが十分に待機するかどうかは不明です。
誰もがこの(一見基本的な)問題に対して堅牢なソリューションを持っていますか?
ホストコンピューターのシャットダウンまたは再起動時にVirtualBox Windowsで実行されているゲストVMを正常にシャットダウンするソリューションを見つけようとしています。
最も安全なオプションは、ホストがシャットダウンを開始したときに「状態保存」コマンドをトリガーすることですが、VMが状態の保存と電源オフを完了するまでホストが十分に待機するかどうかは不明です。
誰もがこの(一見基本的な)問題に対して堅牢なソリューションを持っていますか?
回答:
同様の問題があり、VirtualBoxをサービスとして実行して解決しました。
http://vboxvmservice.sourceforge.net/
VBoxVMServiceを使用すると、マシンのシャットダウン(状態の保存、電源オフ)および起動の方法を選択できます。サービスとして実行されているため、Windowsはシステムのシャットダウンプロセス中に自動的にシャットダウンするまで待機します。
残念ながら、VirtualBox GUIを介して起動されたVMではこれは不可能のようです。GUIがホストのシャットダウンイベントをキャッチして反応する場合でも、VirtualBoxサービスは終了します:https : //forums.virtualbox.org/viewtopic.php? p=278668#p278668
グラフィカルコンソールが必要ない場合は、VBoxHeadlessTrayまたはVBoxVMServiceが最適です。どちらも、Windowsホストのシャットダウンと再起動での自動保存と再開をサポートしています。
VirtualBox 5.0では、「取り外し可能なUI」スタートモードが導入されています。このモードは、独立したUIプロセスでヘッドレスVMを起動します。ただし、グラフィックパフォーマンスは低下し、3Dアクセラレーションはまだサポートされていません。ただし、将来的にはこれをVBoxHeadlessTrayと組み合わせることができるかもしれません(VBoxHeadlessTrayはまだ5.0をサポートしていません。)VBoxHeadlessTray GitHubリポジトリおよびVirtualBox 5サポートを追加するための対応するGitHubプルリクエストへのリンク。
編集:VBoxVmService は、バージョン5.0の新しいデタッチャブルモードもサポートしていません。これまでのところ唯一のヘッドレス。そのための機能リクエストを追加しました。
スタートメニューの電源ボタンの代わりに使用する3つのバッチスクリプトがあります。
do_shutdown.bat(10秒の待機時間でPCをシャットダウンします。VMの状態を保存するために10秒の時間を与えるのではなく、10秒以内にシャットダウンをキャンセルできるようにします。カウントダウンは、VMのシャットダウン後に開始します)
"C:\VirtualBox\VBoxManage.exe" controlvm "Ubuntu Server" savestate
"C:\VirtualBox\VBoxManage.exe" controlvm "Ubuntu Minimal" savestate
shutdown /s /t 10
do_reboot.bat(vmがシャットダウンした直後にリブートします)
"C:\VirtualBox\VBoxManage.exe" controlvm "Ubuntu Server" savestate
"C:\VirtualBox\VBoxManage.exe" controlvm "Ubuntu Minimal" savestate
shutdown /r /t 0
do_cancel.bat(10秒の待機時間内にpc-shutdownをキャンセルできます。do_shutdown.batでシャットダウンしたため、vmを再度再起動します)
shutdown /a
C:\VirtualBox\VBoxManage.exe startvm "Ubuntu Minimal" --type headless
C:\VirtualBox\VBoxManage.exe startvm "Ubuntu Server" --type headless
代わりにsavestate
、次のいずれかを使用することもできます
poweroff - pulls the plug
(probably not a good idea...)
acpipowerbutton - presses the power off button for a clean shutdown
("The system is going down for power off NOW!" to all consoles)
acpisleepbutton - tells the os to go to sleep
(probably just as bad as poweroff)
パーティーに遅れてすみません。これには正確な答えがありますが、コマンドラインfooが必要です。詳細については、次のスレッドの投稿を参照してください:https : //forums.virtualbox.org/viewtopic.php?f=6&t=53684#p285540
探しているコマンドは次のとおりです。
「C:\ Program Files \ Oracle \ VirtualBox \ VBoxManage.exe」setextradata「VM NAME」GUI / DefaultCloseActionシャットダウン
これは複数のVMで使用しているもので、ウィンドウを閉じると自動的に安全なシャットダウンが開始されます。Windowsをシャットダウンし、すべてを閉じようとすると、これらのプロセスが完了するまで待機します。
同様の質問があり、このページを見つけました。テスト用のVMがたくさんあるため、VirtualBoxをサービスとして実行したくありません。通常、VirtualBox UIで実行する別のVMを選択します。コンピューターをシャットダウンすると、各VMの状態を手動で保存するのが面倒です。この場合、スクリプトを使用して実行中のすべてのVMを保存することが実用的なソリューションのようです。Daniel Fの答えをより一般的にするために、実行中のすべてのVMの状態を明示的に名前を付けずに自動的に保存するこれらのスクリプトを書きました。
Windows用のsaveRunningVMs.bat:
set VBoxManageEXE="%ProgramFiles%\Oracle\VirtualBox\VBoxManage.exe"
set ListRunningVMS=%VboxManageEXE% list runningvms
for /f tokens^=2^,4^ delims^=^" %%p in ('%ListRunningVMS%') do %VBoxManageEXE% controlvm %%p savestate
echo all vms saved, you can shutdown now.
rem shutdown /s /t 10
Linux用のsaveRunningVMs.sh:
#!/bin/bash
vboxmanage list runningvms | while read line; do
#echo "VBoxManage controlvm $uuid savestate;"
echo $line
if [[ $line =~ \{(.*)\} ]]
then
vboxmanage controlvm ${BASH_REMATCH[1]} savestate
fi
done
実行中のすべてのVirtualBox VMを一時停止するPythonスクリプトを作成してから、スケジュールされたタスクとしてログアウト時にスクリプトを実行するようにシステムを設定しました。
この方法が正確にどれだけ信頼できるかわかりません。他の人が指摘したように、Winlogon 7002タスクが完了するまでシステムが待機する時間には制限があります。しかし、個人的には、4 GBを超えるVM RAM全体で複数の実行中のVMを使用する場合でも、使用可能な保存状態を提供することに何の問題もありません。
設定手順は次のとおりです。
python.exe
、たとえばへのフルパスを入力しますc:\Python27\python.exe
C:\Users\rakslice\Documents\vboxsuspend\vboxsuspend.py
これで、ログアウト/再起動/シャットダウン時にVirtualBox VMを一時停止する必要があります。
シャットダウンを行う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()