manager/qubesmanager/create_new_vm.py

228 lines
7.9 KiB
Python
Raw Normal View History

#!/usr/bin/python2
#
# The Qubes OS Project, http://www.qubes-os.org
#
# Copyright (C) 2012 Agnieszka Kostrzewa <agnieszka.kostrzewa@gmail.com>
# Copyright (C) 2012 Marek Marczykowski <marmarek@mimuw.edu.pl>
2017-07-12 14:10:15 +02:00
# Copyright (C) 2017 Wojtek Porczyk <woju@invisiblethingslab.com>
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License along
# with this program; if not, see <http://www.gnu.org/licenses/>.
#
#
2017-07-12 14:10:15 +02:00
import sys
import threading
import time
import subprocess
2017-07-12 14:10:15 +02:00
from PyQt4 import QtCore, QtGui # pylint: disable=import-error
2017-07-12 14:10:15 +02:00
import qubesadmin
import qubesadmin.tools
import qubesadmin.exc
2017-07-12 14:10:15 +02:00
from . import utils
from .ui_newappvmdlg import Ui_NewVMDlg # pylint: disable=import-error
2017-07-12 14:10:15 +02:00
from .thread_monitor import ThreadMonitor
class NewVmDlg(QtGui.QDialog, Ui_NewVMDlg):
def __init__(self, qtapp, app, parent=None):
2017-07-12 14:10:15 +02:00
super(NewVmDlg, self).__init__(parent)
self.setupUi(self)
2017-07-12 14:10:15 +02:00
self.qtapp = qtapp
self.app = app
# Theoretically we should be locking for writing here and unlock
# only after the VM creation finished. But the code would be
# more messy...
# Instead we lock for writing in the actual worker thread
2017-07-12 14:10:15 +02:00
self.label_list, self.label_idx = utils.prepare_label_choice(
self.label,
self.app, None,
None,
allow_default=False)
self.template_list, self.template_idx = utils.prepare_vm_choice(
self.template_vm,
self.app, None,
self.app.default_template,
(lambda vm: vm.klass == 'TemplateVM'),
2017-07-12 14:10:15 +02:00
allow_internal=False, allow_default=True, allow_none=False)
self.netvm_list, self.netvm_idx = utils.prepare_vm_choice(
self.netvm,
self.app, None,
self.app.default_netvm,
(lambda vm: vm.provides_network),
allow_internal=False, allow_default=True, allow_none=True)
self.name.setValidator(QtGui.QRegExpValidator(
QtCore.QRegExp("[a-zA-Z0-9-]*", QtCore.Qt.CaseInsensitive), None))
2017-07-12 14:10:15 +02:00
self.name.selectAll()
self.name.setFocus()
if not self.template_list:
QtGui.QMessageBox.warning(None,
2017-07-12 14:10:15 +02:00
self.tr('No template available!'),
self.tr('Cannot create a qube when no template exists.'))
# Order of types is important and used elsewhere; if it's changed
# check for changes needed in self.type_change and TODO
type_list = [self.tr("AppVM"),
self.tr("Standalone qube based on a template"),
self.tr("Standalone qube not based on a template")]
self.vm_type.addItems(type_list)
self.vm_type.currentIndexChanged.connect(self.type_change)
self.launch_settings.stateChanged.connect(self.settings_change)
self.install_system.stateChanged.connect(self.install_change)
def reject(self):
self.done(0)
def accept(self):
vmclass = ('AppVM' if self.vm_type.currentIndex() == 0
else 'StandaloneVM')
2017-07-12 14:10:15 +02:00
name = str(self.name.text())
try:
self.app.domains[name]
except LookupError:
pass
else:
QtGui.QMessageBox.warning(None,
2017-07-12 14:10:15 +02:00
self.tr('Incorrect qube name!'),
self.tr('A qube with the name <b>{}</b> already exists in the '
'system!').format(name))
return
2017-07-12 14:10:15 +02:00
label = self.label_list[self.label.currentIndex()]
if self.template_vm.currentIndex() == -1:
template = None
else:
template = self.template_list[self.template_vm.currentIndex()]
2017-07-12 14:10:15 +02:00
properties = {}
properties['provides_network'] = self.provides_network.isChecked()
properties['virt_mode'] = 'hvm'
2017-07-12 14:10:15 +02:00
properties['netvm'] = self.netvm_list[self.netvm.currentIndex()]
thread_monitor = ThreadMonitor()
2017-07-12 14:10:15 +02:00
thread = threading.Thread(target=self.do_create_vm,
args=(self.app, vmclass, name, label, template, properties,
thread_monitor))
thread.daemon = True
thread.start()
progress = QtGui.QProgressDialog(
2017-07-12 14:10:15 +02:00
self.tr("Creating new qube <b>{}</b>...").format(name), "", 0, 0)
progress.setCancelButton(None)
progress.setModal(True)
progress.show()
while not thread_monitor.is_finished():
2017-07-12 14:10:15 +02:00
self.qtapp.processEvents()
time.sleep(0.1)
progress.hide()
2017-07-12 14:10:15 +02:00
if not thread_monitor.success:
QtGui.QMessageBox.warning(None,
2017-07-12 14:10:15 +02:00
self.tr("Error creating the qube!"),
self.tr("ERROR: {}").format(thread_monitor.error_msg))
self.done(0)
if thread_monitor.success:
if self.launch_settings.isChecked():
subprocess.check_call(['qubes-vm-settings', name])
if self.install_system.isChecked():
subprocess.check_call(
['qubes-vm-boot-from-device', name])
@staticmethod
2017-07-12 14:10:15 +02:00
def do_create_vm(app, vmclass, name, label, template, properties,
thread_monitor):
try:
if vmclass == 'StandaloneVM' and template is not None:
if template is qubesadmin.DEFAULT:
src_vm = app.default_template
else:
src_vm = template
vm = app.clone_vm(src_vm, name, vmclass)
vm.label = label
for k, v in properties.items():
setattr(vm, k, v)
else:
vm = app.add_new_vm(vmclass,
name=name, label=label, template=template)
for k, v in properties.items():
setattr(vm, k, v)
2017-07-12 14:10:15 +02:00
except qubesadmin.exc.QubesException as qex:
thread_monitor.set_error_msg(str(qex))
except Exception as ex: # pylint: disable=broad-except
thread_monitor.set_error_msg(repr(ex))
thread_monitor.set_finished()
def type_change(self):
# AppVM
if self.vm_type.currentIndex() == 0:
self.template_vm.setEnabled(True)
self.template_vm.setCurrentIndex(0)
self.install_system.setEnabled(False)
self.install_system.setChecked(False)
# Standalone - based on a template
if self.vm_type.currentIndex() == 1:
self.template_vm.setEnabled(True)
self.template_vm.setCurrentIndex(0)
self.install_system.setEnabled(False)
self.install_system.setChecked(False)
# Standalone - not based on a template
if self.vm_type.currentIndex() == 2:
self.template_vm.setEnabled(False)
self.template_vm.setCurrentIndex(-1)
self.install_system.setEnabled(True)
self.install_system.setChecked(True)
def install_change(self):
if self.install_system.isChecked():
self.launch_settings.setChecked(False)
def settings_change(self):
if self.launch_settings.isChecked() and self.install_system.isEnabled():
self.install_system.setChecked(False)
2017-07-12 14:10:15 +02:00
parser = qubesadmin.tools.QubesArgumentParser()
def main(args=None):
args = parser.parse_args(args)
qtapp = QtGui.QApplication(sys.argv)
2017-07-12 14:10:15 +02:00
qtapp.setOrganizationName('Invisible Things Lab')
qtapp.setOrganizationDomain('https://www.qubes-os.org/')
qtapp.setApplicationName('Create qube')
2017-07-12 14:10:15 +02:00
dialog = NewVmDlg(qtapp, args.app)
dialog.exec_()