diff --git a/icons/appsprefs.png b/icons/appsprefs.png new file mode 100644 index 0000000..71324c7 Binary files /dev/null and b/icons/appsprefs.png differ diff --git a/icons/newfirewall.png b/icons/newfirewall.png new file mode 100644 index 0000000..85c3602 Binary files /dev/null and b/icons/newfirewall.png differ diff --git a/icons/on.png b/icons/on.png new file mode 100644 index 0000000..8bb39f8 Binary files /dev/null and b/icons/on.png differ diff --git a/qubesmanager/restore.py b/qubesmanager/restore.py new file mode 100644 index 0000000..247f470 --- /dev/null +++ b/qubesmanager/restore.py @@ -0,0 +1,129 @@ +#!/usr/bin/python2.6 +# +# The Qubes OS Project, http://www.qubes-os.org +# +# Copyright (C) 2012 Agnieszka Kostrzewa +# Copyright (C) 2012 Marek Marczykowski +# +# 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 General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. +# +# + +import sys +import os +from PyQt4.QtCore import * +from PyQt4.QtGui import * + +from qubes.qubes import QubesVmCollection +from qubes.qubes import QubesException +from qubes.qubes import QubesDaemonPidfile +from qubes.qubes import QubesHost + +import qubesmanager.resources_rc + +from pyinotify import WatchManager, Notifier, ThreadedNotifier, EventsCodes, ProcessEvent + +import subprocess +import time +import threading +from operator import itemgetter + +from ui_restoredlg import * +from multiselectwidget import * + + + +class RestoreVMsWindow(Ui_Restore, QWizard): + + def __init__(self, parent=None): + super(RestoreVMsWindow, self).__init__(parent) + + self.setupUi(self) + + self.selectVMsWidget = MultiSelectWidget(self) + self.verticalLayout.insertWidget(1, self.selectVMsWidget) + + self.selectVMsWidget.available_list.addItem("netVM1") + self.selectVMsWidget.available_list.addItem("appVM1") + self.selectVMsWidget.available_list.addItem("appVM2") + self.selectVMsWidget.available_list.addItem("templateVM1") + + + def reject(self): + self.done(0) + + def save_and_apply(self): + pass + + @pyqtSlot(name='on_selectPathButton_clicked') + def selectPathButton_clicked(self): + self.path = self.pathLineEdit.text() + newPath = QFileDialog.getExistingDirectory(self, 'Select backup directory.') + if newPath: + self.pathLineEdit.setText(newPath) + self.path = newPath + +# Bases on the original code by: +# Copyright (c) 2002-2007 Pascal Varet + +def handle_exception( exc_type, exc_value, exc_traceback ): + import sys + import os.path + import traceback + + filename, line, dummy, dummy = traceback.extract_tb( exc_traceback ).pop() + filename = os.path.basename( filename ) + error = "%s: %s" % ( exc_type.__name__, exc_value ) + + QMessageBox.critical(None, "Houston, we have a problem...", + "Whoops. A critical error has occured. This is most likely a bug " + "in Qubes Restore VMs application.

" + "%s" % error + + "at line %d of file %s.

" + % ( line, filename )) + + + + +def main(): + + global qubes_host + qubes_host = QubesHost() + + global app + app = QApplication(sys.argv) + app.setOrganizationName("The Qubes Project") + app.setOrganizationDomain("http://qubes-os.org") + app.setApplicationName("Qubes Restore VMs") + + sys.excepthook = handle_exception + + qvm_collection = QubesVmCollection() + qvm_collection.lock_db_for_reading() + qvm_collection.load() + qvm_collection.unlock_db() + + global restore_window + restore_window = RestoreVMsWindow() + + restore_window.show() + + app.exec_() + app.exit() + + + +if __name__ == "__main__": + main() diff --git a/qubesmanager/settings.py b/qubesmanager/settings.py new file mode 100644 index 0000000..a213af7 --- /dev/null +++ b/qubesmanager/settings.py @@ -0,0 +1,149 @@ +#!/usr/bin/python2.6 +# +# The Qubes OS Project, http://www.qubes-os.org +# +# Copyright (C) 2012 Agnieszka Kostrzewa +# Copyright (C) 2012 Marek Marczykowski +# +# 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 General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. +# +# + +import sys +import os +from PyQt4.QtCore import * +from PyQt4.QtGui import * + +from qubes.qubes import QubesVmCollection +from qubes.qubes import QubesException +from qubes.qubes import qubes_appmenu_create_cmd +from qubes.qubes import qubes_appmenu_remove_cmd +from qubes.qubes import QubesDaemonPidfile +from qubes.qubes import QubesHost +from qubes.qubes import qrexec_client_path + +import qubesmanager.resources_rc + +from pyinotify import WatchManager, Notifier, ThreadedNotifier, EventsCodes, ProcessEvent + +import subprocess +import time +import threading +from operator import itemgetter + +from ui_settingsdlg import * +from multiselectwidget import * + + + +class VMSettingsWindow(Ui_SettingsDialog, QDialog): + + def __init__(self, vm, parent=None): + super(VMSettingsWindow, self).__init__(parent) + + self.setupUi(self) + + self.connect(self.buttonBox, SIGNAL("accepted()"), self.save_and_apply) + self.connect(self.buttonBox, SIGNAL("rejected()"), self.reject) + + self.app_list = MultiSelectWidget(self) + self.dev_list = MultiSelectWidget(self) + + self.apps_layout.addWidget(self.app_list) + self.devices_layout.addWidget(self.dev_list) + + self.vm = vm + if self.vm.template_vm: + self.source_vm = self.vm.template_vm + else: + self.source_vm = self.vm + + + #self.fill_apps_list() + #self.fill_devices_list() + + def reject(self): + self.done(0) + + def save_and_apply(self): + pass + +# Bases on the original code by: +# Copyright (c) 2002-2007 Pascal Varet + +def handle_exception( exc_type, exc_value, exc_traceback ): + import sys + import os.path + import traceback + + filename, line, dummy, dummy = traceback.extract_tb( exc_traceback ).pop() + filename = os.path.basename( filename ) + error = "%s: %s" % ( exc_type.__name__, exc_value ) + + QMessageBox.critical(None, "Houston, we have a problem...", + "Whoops. A critical error has occured. This is most likely a bug " + "in Qubes VM Settings application.

" + "%s" % error + + "at line %d of file %s.

" + % ( line, filename )) + + + + +def main(): + + global qubes_host + qubes_host = QubesHost() + + global app + app = QApplication(sys.argv) + app.setOrganizationName("The Qubes Project") + app.setOrganizationDomain("http://qubes-os.org") + app.setApplicationName("Qubes VM Settings") + + sys.excepthook = handle_exception + + qvm_collection = QubesVmCollection() + qvm_collection.lock_db_for_reading() + qvm_collection.load() + qvm_collection.unlock_db() + + vm = None + + if len(sys.argv) > 1: + vm = qvm_collection.get_vm_by_name(sys.argv[1]) + if vm is None or vm.qid not in qvm_collection: + QMessageBox.critical(None, "Qubes VM Settings Error", + "A VM with the name '{0}' does not exist in the system.".format(sys.argv[1])) + sys.exit(1) + else: + vms_list = [vm.name for vm in qvm_collection.values() if (vm.is_appvm() or vm.is_template())] + vmname = QInputDialog.getItem(None, "Select VM", "Select VM:", vms_list, editable = False) + if not vmname[1]: + sys.exit(1) + vm = qvm_collection.get_vm_by_name(vmname[0]) + + global settings_window + settings_window = VMSettingsWindow(vm) + + settings_window.show() + + app.exec_() + app.exit() + + + +if __name__ == "__main__": + main() diff --git a/restoredlg.ui b/restoredlg.ui new file mode 100644 index 0000000..a37807d --- /dev/null +++ b/restoredlg.ui @@ -0,0 +1,240 @@ + + + Restore + + + + 0 + 0 + 700 + 399 + + + + Qubes Restore VMs + + + + + + + + + + Device + + + + + + + + 0 + 0 + + + + + dev1 + + + + + longdeviceblablabla + + + + + dev2 + + + + + dev3 + + + + + + + + Backup directory: + + + + + + + + + + ... + + + + + + + + 12 + 75 + false + true + false + + + + Select backup source location: + + + + + + + + + + + + 12 + 75 + false + true + false + + + + Select VMs to restore: + + + + + + + + + + + 12 + 75 + false + true + false + + + + Restore options: + + + + + + + + + Do not restore VMs that have missing templates or netvms. + + + skip broken + + + + + + + Do not restore VMs that are already present on the host. + + + skip conflicting + + + + + + + Ignore dom0 username mismatch while restoring homedir. + + + ignore username mismatch + + + + + + + Ignore missing templates or netvms, restore VMs anyway. + + + ignore missing + + + + + + + Force to run, even with root privileges. + + + force root + + + + + + + + + + + + + + 12 + 75 + false + true + false + + + + You're about to perform the following actions: + + + + + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans Serif'; font-size:9pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">A lot of info<br />A lot of info<br />A lot of info<br />A lot of info<br />A lot of info<br />A lot of info<br />A lot of info A lot of info A lot of info A lot of info A lot of info A lot of info</p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">A lot of info A lot of info A lot of info A lot of info A lot of info A lot of info</p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">A lot of info A lot of info A lot of info A lot of info A lot of info A lot of info</p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">A lot of info A lot of info A lot of info A lot of info A lot of info A lot of info</p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">A lot of info A lot of info A lot of info A lot of info A lot of info A lot of info<br />A lot of info</p></body></html> + + + + + + + + 12 + 75 + false + true + false + + + + To accept press Finish. + + + + + + + + +