# Installation source spoke classes # # Copyright (C) 2011, 2012 Red Hat, Inc. # # This copyrighted material is made available to anyone wishing to use, # modify, copy, or redistribute it subject to the terms and conditions of # the GNU General Public License v.2, or (at your option) any later version. # This program is distributed in the hope that it will be useful, but WITHOUT # ANY WARRANTY expressed or implied, including the implied warranties 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. Any Red Hat trademarks that are incorporated in the # source code or documentation are not subject to the GNU General Public # License and may only be used or replicated with the express permission of # Red Hat, Inc. # # Red Hat Author(s): Chris Lumens # Martin Sivak # import time import logging log = logging.getLogger("anaconda") import os, signal, re from gi.repository import GLib, Gtk from pyanaconda.flags import flags from pyanaconda.i18n import _, N_, CN_ from pyanaconda.image import opticalInstallMedia, potentialHdisoSources from pyanaconda.ui.communication import hubQ from pyanaconda.ui.helpers import InputCheck, InputCheckHandler from pyanaconda.ui.gui import GUIObject from pyanaconda.ui.gui.helpers import GUIDialogInputCheckHandler, GUISpokeInputCheckHandler from pyanaconda.ui.gui.spokes import NormalSpoke from pyanaconda.ui.categories.software import SoftwareCategory from pyanaconda.ui.gui.utils import blockedHandler, fire_gtk_action from pyanaconda.iutil import ProxyString, ProxyStringError, cmp_obj_attrs from pyanaconda.ui.gui.utils import gtk_call_once, really_hide, really_show, fancy_set_sensitive from pyanaconda.threads import threadMgr, AnacondaThread from pyanaconda.packaging import PackagePayload, payloadMgr from pyanaconda.regexes import REPO_NAME_VALID, URL_PARSE, HOSTNAME_PATTERN_WITHOUT_ANCHORS from pyanaconda import constants from blivet.util import get_mount_device, get_mount_paths __all__ = ["SourceSpoke"] BASEREPO_SETUP_MESSAGE = N_("Setting up installation source...") METADATA_DOWNLOAD_MESSAGE = N_("Downloading package metadata...") # These need to match the IDs in protocolComboBox and repoProtocolComboBox in source.glade. PROTOCOL_HTTP = 'http' PROTOCOL_HTTPS = 'https' PROTOCOL_FTP = 'ftp' PROTOCOL_NFS = 'nfs' PROTOCOL_MIRROR = 'Closest mirror' # Repo Store Columns REPO_ENABLED_COL = 0 REPO_NAME_COL = 1 REPO_OBJ = 2 REPO_PROTO = {PROTOCOL_HTTP: "http://", PROTOCOL_HTTPS: "https://", PROTOCOL_FTP: "ftp://", PROTOCOL_NFS: "nfs://" } def _validateProxy(proxy_string, username_set, password_set): """Validate a proxy string and return an input code usable by InputCheck :param str proxy_string: the proxy URL string :param bool username_set: Whether a username has been specified external to the URL :param bool password_set: Whether a password has been speicifed external to the URL """ proxy_match = URL_PARSE.match(proxy_string) if not proxy_match: return _("Invalid proxy URL") # Ensure the protocol is something that makes sense protocol = proxy_match.group("protocol") if protocol and protocol not in ('http://', 'https://', 'ftp://'): return _("Invalid proxy protocol: %s") % protocol # Path and anything after makes no sense for a proxy URL # Allow '/' as a path so you can use http://proxy.example.com:8080/ if (proxy_match.group("path") and proxy_match.group("path") != "/") \ or proxy_match.group("query") or proxy_match.group("fragment"): return _("Extra characters in proxy URL") # Check if if authentication data is both in the URL and specified externally if (proxy_match.group("username") or proxy_match.group("password")) and (username_set or password_set): return _("Proxy authentication data duplicated") return InputCheck.CHECK_OK class ProxyDialog(GUIObject, GUIDialogInputCheckHandler): builderObjects = ["proxyDialog"] mainWidgetName = "proxyDialog" uiFile = "spokes/source.glade" def __init__(self, data, proxy_url): GUIObject.__init__(self, data) GUIDialogInputCheckHandler.__init__(self) self.proxyUrl = proxy_url self._proxyCheck = self.builder.get_object("enableProxyCheck") self._proxyInfoBox = self.builder.get_object("proxyInfoBox") self._authCheck = self.builder.get_object("enableAuthCheck") self._proxyAuthBox = self.builder.get_object("proxyAuthBox") self._proxyURLEntry = self.builder.get_object("proxyURLEntry") self._proxyUsernameEntry = self.builder.get_object("proxyUsernameEntry") self._proxyPasswordEntry = self.builder.get_object("proxyPasswordEntry") self._proxyOkButton = self.builder.get_object("proxyOkButton") self._proxyValidate = self.add_check(self._proxyURLEntry, self._checkProxyURL) self._proxyValidate.update_check_status() def _checkProxyURL(self, inputcheck): proxy_string = self.get_input(inputcheck.input_obj) # Don't set an error icon on empty input, but keep the add button insensitive. if not proxy_string: return InputCheck.CHECK_SILENT username_set = self._proxyUsernameEntry.is_sensitive() and self._proxyUsernameEntry.get_text() password_set = self._proxyPasswordEntry.is_sensitive() and self._proxyPasswordEntry.get_text() return _validateProxy(proxy_string, username_set, password_set) def set_status(self, inputcheck): # Use the superclass set_status to set the error message GUIDialogInputCheckHandler.set_status(self, inputcheck) # Change the sensitivity of the Add button self._proxyOkButton.set_sensitive(inputcheck.check_status == InputCheck.CHECK_OK) # Update the proxy validation check on username and password changes to catch # changes in duplicated authentication data def on_proxyUsernameEntry_changed(self, entry, user_data=None): self._proxyValidate.update_check_status() def on_proxyPasswordEntry_changed(self, entry, user_data=None): self._proxyValidate.update_check_status() def on_proxy_cancel_clicked(self, *args): self.window.destroy() def on_proxy_ok_clicked(self, *args): if self._proxyCheck.get_active(): url = self._proxyURLEntry.get_text() if self._authCheck.get_active(): username = self._proxyUsernameEntry.get_text() password = self._proxyPasswordEntry.get_text() else: username = None password = None proxy = ProxyString(url=url, username=username, password=password) self.proxyUrl = proxy.url else: self.proxyUrl = "" self.window.destroy() def on_proxy_enable_toggled(self, button, *args): self._proxyInfoBox.set_sensitive(button.get_active()) if button.get_active(): self.set_status(self._proxyValidate) else: self._proxyOkButton.set_sensitive(True) def on_proxy_auth_toggled(self, button, *args): self._proxyAuthBox.set_sensitive(button.get_active()) self._proxyValidate.update_check_status() def refresh(self): GUIObject.refresh(self) if not self.proxyUrl: self._proxyCheck.set_active(False) self.on_proxy_enable_toggled(self._proxyCheck) self._authCheck.set_active(False) self.on_proxy_auth_toggled(self._authCheck) return try: proxy = ProxyString(self.proxyUrl) if proxy.username: self._proxyUsernameEntry.set_text(proxy.username) if proxy.password: self._proxyPasswordEntry.set_text(proxy.password) self._proxyURLEntry.set_text(proxy.noauth_url) except ProxyStringError as e: log.error("Failed to parse proxy for ProxyDialog.refresh %s: %s", self.proxyUrl, e) return self._proxyCheck.set_active(True) self._authCheck.set_active(bool(proxy.username or proxy.password)) self.on_proxy_enable_toggled(self._proxyCheck) self.on_proxy_auth_toggled(self._authCheck) def run(self): self.window.run() class MediaCheckDialog(GUIObject): builderObjects = ["mediaCheckDialog"] mainWidgetName = "mediaCheckDialog" uiFile = "spokes/source.glade" def __init__(self, data): GUIObject.__init__(self, data) self.progressBar = self.builder.get_object("mediaCheck-progressBar") self._pid = None def _checkisoEndsCB(self, pid, status): doneButton = self.builder.get_object("doneButton") verifyLabel = self.builder.get_object("verifyLabel") if os.WIFSIGNALED(status): pass elif status == 0: verifyLabel.set_text(_("This media is good to install from.")) else: verifyLabel.set_text(_("This media is not good to install from.")) self.progressBar.set_fraction(1.0) doneButton.set_sensitive(True) GLib.spawn_close_pid(pid) self._pid = None def _checkisoStdoutWatcher(self, fd, condition): if condition == GLib.IOCondition.HUP: return False channel = GLib.IOChannel(fd) line = channel.readline().strip() if not line.isdigit(): return True pct = float(line)/100 if pct > 1.0: pct = 1.0 self.progressBar.set_fraction(pct) return True def run(self, devicePath): (retval, self._pid, _stdin, stdout, _stderr) = \ GLib.spawn_async_with_pipes(None, ["checkisomd5", "--gauge", devicePath], [], GLib.SpawnFlags.DO_NOT_REAP_CHILD|GLib.SpawnFlags.SEARCH_PATH, None, None) if not retval: return # This function waits for checkisomd5 to end and then cleans up after it. GLib.child_watch_add(self._pid, self._checkisoEndsCB) # This function watches the process's stdout. GLib.io_add_watch(stdout, GLib.IOCondition.IN|GLib.IOCondition.HUP, self._checkisoStdoutWatcher) self.window.run() def on_close(self, *args): if self._pid: os.kill(self._pid, signal.SIGKILL) self.window.destroy() def on_done_clicked(self, *args): self.window.destroy() # This class is responsible for popping up the dialog that allows the user to # choose the ISO image they want to use. We can get away with this instead of # selecting a directory because we no longer support split media. # # Two assumptions about the use of this class: # (1) This class is responsible for mounting and unmounting the partition # containing the ISO images. # (2) When you call refresh() with a currentFile argument or when you get a # result from run(), the file path you use is relative to the root of the # mounted partition. In other words, it will not contain the # "/mnt/isodir/install" part. This is consistent with the rest of anaconda. class IsoChooser(GUIObject): builderObjects = ["isoChooserDialog", "isoFilter"] mainWidgetName = "isoChooserDialog" uiFile = "spokes/source.glade" def __init__(self, data): GUIObject.__init__(self, data) self._chooser = self.builder.get_object("isoChooserDialog") # pylint: disable=arguments-differ def refresh(self, currentFile=""): GUIObject.refresh(self) self._chooser.connect("current-folder-changed", self.on_folder_changed) self._chooser.set_filename(constants.ISO_DIR + "/" + currentFile) def run(self, dev): retval = None unmount = not dev.format.status mounts = get_mount_paths(dev.path) # We have to check both ISO_DIR and the DRACUT_ISODIR because we # still reference both, even though /mnt/install is a symlink to # /run/install. Finding mount points doesn't handle the symlink if constants.ISO_DIR not in mounts and constants.DRACUT_ISODIR not in mounts: # We're not mounted to either location, so do the mount dev.format.mount(mountpoint=constants.ISO_DIR) # If any directory was chosen, return that. Otherwise, return None. rc = self.window.run() if rc == 1: f = self._chooser.get_filename() if f: retval = f.replace(constants.ISO_DIR, "") if unmount: dev.format.unmount() self.window.destroy() return retval # There doesn't appear to be any way to restrict a GtkFileChooser to a # given directory (see https://bugzilla.gnome.org/show_bug.cgi?id=155729) # so we'll just have to fake it by setting you back to inside the directory # should you change out of it. def on_folder_changed(self, chooser): d = chooser.get_current_folder() if not d: return if not d.startswith(constants.ISO_DIR): chooser.set_current_folder(constants.ISO_DIR) class SourceSpoke(NormalSpoke, GUISpokeInputCheckHandler): builderObjects = ["isoChooser", "isoFilter", "partitionStore", "sourceWindow", "dirImage", "repoStore"] mainWidgetName = "sourceWindow" uiFile = "spokes/source.glade" helpFile = "SourceSpoke.xml" category = SoftwareCategory icon = "media-optical-symbolic" title = CN_("GUI|Spoke", "_INSTALLATION SOURCE") def __init__(self, *args, **kwargs): NormalSpoke.__init__(self, *args, **kwargs) GUISpokeInputCheckHandler.__init__(self) self._currentIsoFile = None self._ready = False self._error = False self._proxyUrl = "" self._proxyChange = False self._cdrom = None def apply(self): # If askmethod was provided on the command line, entering the source # spoke wipes that out. if flags.askmethod: flags.askmethod = False payloadMgr.restartThread(self.storage, self.data, self.payload, self.instclass, checkmount=False) self.clear_info() def _method_changed(self): """ Check to see if the install method has changed. :returns: True if it changed, False if not :rtype: bool """ import copy old_source = copy.deepcopy(self.data.method) if self._autodetectButton.get_active(): if not self._cdrom: return False self.data.method.method = "cdrom" self.payload.install_device = self._cdrom if old_source.method == "cdrom": # XXX maybe we should always redo it for cdrom in case they # switched disks return False elif self._isoButton.get_active(): # If the user didn't select a partition (not sure how that would # happen) or didn't choose a directory (more likely), then return # as if they never did anything. part = self._get_selected_partition() if not part or not self._currentIsoFile: return False self.data.method.method = "harddrive" self.data.method.partition = part.name # The / gets stripped off by payload.ISOImage self.data.method.dir = "/" + self._currentIsoFile if (old_source.method == "harddrive" and self.storage.devicetree.resolveDevice(old_source.partition) == part and old_source.dir in [self._currentIsoFile, "/" + self._currentIsoFile]): return False # Make sure anaconda doesn't touch this device. part.protected = True self.storage.config.protectedDevSpecs.append(part.name) elif self._mirror_active(): # this preserves the url for later editing self.data.method.method = None self.data.method.proxy = self._proxyUrl if not old_source.method and self.payload.baseRepo and \ not self._proxyChange: return False elif self._http_active() or self._ftp_active(): url = self._urlEntry.get_text().strip() mirrorlist = False # If the user didn't fill in the URL entry, just return as if they # selected nothing. if url == "": return False # Make sure the URL starts with the protocol. yum will want that # to know how to fetch, and the refresh method needs that to know # which element of the combo to default to should this spoke be # revisited. if self._ftp_active() and not url.startswith("ftp://"): url = "ftp://" + url elif self._protocolComboBox.get_active_id() == PROTOCOL_HTTP and not url.startswith("http://"): url = "http://" + url mirrorlist = self._mirrorlistCheckbox.get_active() elif self._protocolComboBox.get_active_id() == PROTOCOL_HTTPS and not url.startswith("https://"): url = "https://" + url mirrorlist = self._mirrorlistCheckbox.get_active() if old_source.method == "url" and not self._proxyChange and \ ((not mirrorlist and old_source.url == url) or \ (mirrorlist and old_source.mirrorlist == url)): return False self.data.method.method = "url" self.data.method.proxy = self._proxyUrl if mirrorlist: self.data.method.mirrorlist = url self.data.method.url = "" else: self.data.method.url = url self.data.method.mirrorlist = "" elif self._nfs_active(): url = self._urlEntry.get_text().strip() if url == "": return False self.data.method.method = "nfs" try: (self.data.method.server, self.data.method.dir) = url.split(":", 2) except ValueError as e: log.error("ValueError: %s", e) gtk_call_once(self.set_warning, _("Failed to set up installation source; check the repo url")) self._error = True return self.data.method.opts = self.builder.get_object("nfsOptsEntry").get_text() or "" if (old_source.method == "nfs" and old_source.server == self.data.method.server and old_source.dir == self.data.method.dir and old_source.opts == self.data.method.opts): return False # If the user moved from an HDISO method to some other, we need to # clear the protected bit on that device. if old_source.method == "harddrive" and old_source.partition: self._currentIsoFile = None self._isoChooserButton.set_label(self._origIsoChooserButton) self._isoChooserButton.set_use_underline(True) if old_source.partition in self.storage.config.protectedDevSpecs: self.storage.config.protectedDevSpecs.remove(old_source.partition) dev = self.storage.devicetree.getDeviceByName(old_source.partition) if dev: dev.protected = False self._proxyChange = False return True @property def changed(self): method_changed = self._method_changed() update_payload_repos = self._update_payload_repos() return method_changed or update_payload_repos or self._error @property def completed(self): """ WARNING: This can be called before _initialize is done, make sure that it doesn't access things that are not setup (eg. payload.*) until it is ready """ if flags.automatedInstall and self.ready and not self.payload.baseRepo: return False else: return not self._error and self.ready and (self.data.method.method or self.payload.baseRepo) @property def mandatory(self): return True @property def ready(self): return (self._ready and not threadMgr.get(constants.THREAD_PAYLOAD) and not threadMgr.get(constants.THREAD_SOFTWARE_WATCHER) and not threadMgr.get(constants.THREAD_CHECK_SOFTWARE)) @property def status(self): if threadMgr.get(constants.THREAD_CHECK_SOFTWARE): return _("Checking software dependencies...") elif not self.ready: return _(BASEREPO_SETUP_MESSAGE) elif not self.payload.baseRepo: return _("Error setting up base repository") elif self._error: return _("Error setting up software source") elif self.data.method.method == "url": return self.data.method.url or self.data.method.mirrorlist elif self.data.method.method == "nfs": return _("NFS server %s") % self.data.method.server elif self.data.method.method == "cdrom": return _("Local media") elif self.data.method.method == "harddrive": if not self._currentIsoFile: return _("Error setting up ISO file") return os.path.basename(self._currentIsoFile) elif self.payload.baseRepo: return _("Closest mirror") else: return _("Nothing selected") def _grabObjects(self): self._autodetectButton = self.builder.get_object("autodetectRadioButton") self._autodetectBox = self.builder.get_object("autodetectBox") self._autodetectDeviceLabel = self.builder.get_object("autodetectDeviceLabel") self._autodetectLabel = self.builder.get_object("autodetectLabel") self._isoButton = self.builder.get_object("isoRadioButton") self._isoBox = self.builder.get_object("isoBox") self._networkButton = self.builder.get_object("networkRadioButton") self._networkBox = self.builder.get_object("networkBox") self._urlEntry = self.builder.get_object("urlEntry") self._protocolComboBox = self.builder.get_object("protocolComboBox") self._isoChooserButton = self.builder.get_object("isoChooserButton") self._origIsoChooserButton = self._isoChooserButton.get_label() # Attach a validator to the URL entry. Start it as disabled, and it will be # enabled/disabled as entry sensitivity is enabled/disabled. self._urlCheck = self.add_check(self._urlEntry, self._checkURLEntry) self._urlCheck.enabled = False self._mirrorlistCheckbox = self.builder.get_object("mirrorlistCheckbox") self._noUpdatesCheckbox = self.builder.get_object("noUpdatesCheckbox") self._verifyIsoButton = self.builder.get_object("verifyIsoButton") # addon repo objects self._repoEntryBox = self.builder.get_object("repoEntryBox") self._repoStore = self.builder.get_object("repoStore") self._repoSelection = self.builder.get_object("repoSelection") self._repoNameEntry = self.builder.get_object("repoNameEntry") self._repoProtocolComboBox = self.builder.get_object("repoProtocolComboBox") self._repoUrlEntry = self.builder.get_object("repoUrlEntry") self._repoMirrorlistCheckbox = self.builder.get_object("repoMirrorlistCheckbox") self._repoProxyUrlEntry = self.builder.get_object("repoProxyUrlEntry") self._repoProxyUsernameEntry = self.builder.get_object("repoProxyUsernameEntry") self._repoProxyPasswordEntry = self.builder.get_object("repoProxyPasswordEntry") self._repoView = self.builder.get_object("repoTreeView") # Create a check for duplicate repo ids # Call InputCheckHandler directly since this check operates on rows of a TreeModel # instead of GtkEntry inputs. Updating the check is handled by the signal handlers # connected to repoStore. self._duplicateRepoCheck = InputCheckHandler.add_check(self, self._repoStore, self._checkDuplicateRepos) # Create dictionaries for the checks on fields in individual repos # These checks will be added and removed as repos are added and removed from repoStore self._repoNameChecks = {} self._repoURLChecks = {} self._repoProxyChecks = {} # updates option container self._updatesBox = self.builder.get_object("updatesBox") self._proxyButton = self.builder.get_object("proxyButton") self._nfsOptsBox = self.builder.get_object("nfsOptsBox") # Connect scroll events on the viewport with focus events on the box mainViewport = self.builder.get_object("mainViewport") mainBox = self.builder.get_object("mainBox") mainBox.set_focus_vadjustment(mainViewport.get_vadjustment()) def initialize(self): NormalSpoke.initialize(self) self._grabObjects() # I shouldn't have to do this outside GtkBuilder, but it really doesn't # want to let me pass in user data. # See also: https://bugzilla.gnome.org/show_bug.cgi?id=727919 self._autodetectButton.connect("toggled", self.on_source_toggled, self._autodetectBox) self._isoButton.connect("toggled", self.on_source_toggled, self._isoBox) self._networkButton.connect("toggled", self.on_source_toggled, self._networkBox) self._networkButton.connect("toggled", self._updateURLEntryCheck) # Show or hide the updates option based on the installclass if self.instclass.installUpdates: really_show(self._updatesBox) else: really_hide(self._updatesBox) self._repoNameWarningBox = self.builder.get_object("repoNameWarningBox") self._repoNameWarningLabel = self.builder.get_object("repoNameWarningLabel") threadMgr.add(AnacondaThread(name=constants.THREAD_SOURCE_WATCHER, target=self._initialize)) # Register listeners for payload events payloadMgr.addListener(payloadMgr.STATE_START, self._payload_refresh) payloadMgr.addListener(payloadMgr.STATE_STORAGE, self._probing_storage) payloadMgr.addListener(payloadMgr.STATE_GROUP_MD, self._downloading_package_md) payloadMgr.addListener(payloadMgr.STATE_FINISHED, self._payload_finished) payloadMgr.addListener(payloadMgr.STATE_ERROR, self._payload_error) def _payload_refresh(self): hubQ.send_not_ready("SoftwareSelectionSpoke") hubQ.send_not_ready(self.__class__.__name__) hubQ.send_message(self.__class__.__name__, _(BASEREPO_SETUP_MESSAGE)) # this sleep is lame, but without it the message above doesn't seem # to get processed by the hub in time, and is never shown. # FIXME this should get removed when we figure out how to ensure # that the message takes effect on the hub before we try to mount # a bad NFS server. time.sleep(1) def _probing_storage(self): hubQ.send_message(self.__class__.__name__, _("Probing storage...")) def _downloading_package_md(self): # Reset the error state from previous payloads self._error = False hubQ.send_message(self.__class__.__name__, _(METADATA_DOWNLOAD_MESSAGE)) def _payload_finished(self): hubQ.send_ready("SoftwareSelectionSpoke", False) hubQ.send_ready(self.__class__.__name__, False) def _payload_error(self): self._error = True hubQ.send_message(self.__class__.__name__, payloadMgr.error) if not (hasattr(self.data.method, "proxy") and self.data.method.proxy): gtk_call_once(self.set_warning, _("Failed to set up installation source; check the repo url")) else: gtk_call_once(self.set_warning, _("Failed to set up installation source; check the repo url and proxy settings")) hubQ.send_ready(self.__class__.__name__, False) def _initialize(self): threadMgr.wait(constants.THREAD_PAYLOAD) added = False # If there's no fallback mirror to use, we should just disable that option # in the UI. if not self.payload.mirrorEnabled: model = self._protocolComboBox.get_model() itr = model.get_iter_first() while itr and model[itr][self._protocolComboBox.get_id_column()] != PROTOCOL_MIRROR: itr = model.iter_next(itr) if itr: model.remove(itr) # If we've previously set up to use a CD/DVD method, the media has # already been mounted by payload.setup. We can't try to mount it # again. So just use what we already know to create the selector. # Otherwise, check to see if there's anything available. if self.data.method.method == "cdrom": self._cdrom = self.payload.install_device elif not flags.automatedInstall: self._cdrom = opticalInstallMedia(self.storage.devicetree) if self._cdrom: fire_gtk_action(self._autodetectDeviceLabel.set_text, _("Device: %s") % self._cdrom.name) fire_gtk_action(self._autodetectLabel.set_text, _("Label: %s") % (getattr(self._cdrom.format, "label", "") or "")) added = True if self.data.method.method == "harddrive": self._currentIsoFile = self.payload.ISOImage # These UI elements default to not being showable. If optical install # media were found, mark them to be shown. if added: gtk_call_once(self._autodetectBox.set_no_show_all, False) gtk_call_once(self._autodetectButton.set_no_show_all, False) # Add the mirror manager URL in as the default for HTTP and HTTPS. # We'll override this later in the refresh() method, if they've already # provided a URL. # FIXME self._reset_repoStore() self._ready = True # Wait to make sure the other threads are done before sending ready, otherwise # the spoke may not be set sensitive by _handleCompleteness in the hub. while not self.ready: time.sleep(1) hubQ.send_ready(self.__class__.__name__, False) def refresh(self): NormalSpoke.refresh(self) # Find all hard drive partitions that could hold an ISO and add each # to the partitionStore. This has to be done here because if the user # has done partitioning first, they may have blown away partitions # found during _initialize on the partitioning spoke. store = self.builder.get_object("partitionStore") store.clear() added = False active = 0 idx = 0 if self.data.method.method == "harddrive": methodDev = self.storage.devicetree.resolveDevice(self.data.method.partition) for dev in potentialHdisoSources(self.storage.devicetree): # path model size format type uuid of format dev_info = { "model" : self._sanitize_model(dev.disk.model), "path" : dev.path, "size" : dev.size, "format": dev.format.name or "", "label" : dev.format.label or dev.format.uuid or "" } # With the label in here, the combo box can appear really long thus pushing the "pick an image" # and the "verify" buttons off the screen. if dev_info["label"] != "": dev_info["label"] = "\n" + dev_info["label"] store.append([dev, "%(model)s %(path)s (%(size)s) %(format)s %(label)s" % dev_info]) if self.data.method.method == "harddrive" and dev == methodDev: active = idx added = True idx += 1 # Again, only display these widgets if an HDISO source was found. self._isoBox.set_no_show_all(not added) self._isoBox.set_visible(added) self._isoButton.set_no_show_all(not added) self._isoButton.set_visible(added) if added: combo = self.builder.get_object("isoPartitionCombo") combo.set_active(active) # We default to the mirror list, and then if the method tells us # something different later, we can change it. self._protocolComboBox.set_active_id(PROTOCOL_MIRROR) self._urlEntry.set_sensitive(False) self._updateURLEntryCheck() # Set up the default state of UI elements. if self.data.method.method == "url": self._networkButton.set_active(True) proto = self.data.method.url or self.data.method.mirrorlist if proto.startswith("http:"): self._protocolComboBox.set_active_id(PROTOCOL_HTTP) l = 7 elif proto.startswith("https:"): self._protocolComboBox.set_active_id(PROTOCOL_HTTPS) l = 8 elif proto.startswith("ftp:"): self._protocolComboBox.set_active_id(PROTOCOL_FTP) l = 6 else: self._protocolComboBox.set_active_id(PROTOCOL_HTTP) l = 0 self._urlEntry.set_sensitive(True) self._urlEntry.set_text(proto[l:]) self._updateURLEntryCheck() self._mirrorlistCheckbox.set_active(bool(self.data.method.mirrorlist)) self._proxyUrl = self.data.method.proxy elif self.data.method.method == "nfs": self._networkButton.set_active(True) self._protocolComboBox.set_active_id(PROTOCOL_NFS) self._urlEntry.set_text("%s:%s" % (self.data.method.server, self.data.method.dir)) self._urlEntry.set_sensitive(True) self._updateURLEntryCheck() self.builder.get_object("nfsOptsEntry").set_text(self.data.method.opts or "") elif self.data.method.method == "harddrive": self._isoButton.set_active(True) self._isoBox.set_sensitive(True) self._verifyIsoButton.set_sensitive(True) if self._currentIsoFile: self._isoChooserButton.set_label(os.path.basename(self._currentIsoFile)) else: self._isoChooserButton.set_label("") self._isoChooserButton.set_use_underline(False) else: # No method was given in advance, so now we need to make a sensible # guess. Go with autodetected media if that was provided, and then # fall back to closest mirror. if not self._autodetectButton.get_no_show_all(): self._autodetectButton.set_active(True) self.data.method.method = "cdrom" else: self._networkButton.set_active(True) self.data.method.method = None self._proxyUrl = self.data.method.proxy self._setup_no_updates() # Setup the addon repos self._reset_repoStore() if self.data.method.method == "harddrive" and \ get_mount_device(constants.DRACUT_ISODIR) == get_mount_device(constants.DRACUT_REPODIR): # If the stage2 image is mounted from an HDISO source, there's really # no way we can tear down that source to allow the user to change it. # Thus, this portion of the spoke should be insensitive. for widget in [self._autodetectButton, self._autodetectBox, self._isoButton, self._isoBox, self._networkButton, self._networkBox]: widget.set_sensitive(False) widget.set_tooltip_text(_("The installation source is in use by the installer and cannot be changed.")) else: # Then, some widgets get enabled/disabled/greyed out depending on # how others are set up. We can use the signal handlers to handle # that condition here too. self.on_protocol_changed(self._protocolComboBox) def _setup_no_updates(self): """ Setup the state of the No Updates checkbox. If closest mirror is not selected, check it. If closest mirror is selected, and "updates" repo is enabled, uncheck it. """ self._updatesBox.set_sensitive(self._mirror_active()) active = not self._mirror_active() or not self.payload.isRepoEnabled("updates") self._noUpdatesCheckbox.set_active(active) @property def showable(self): return isinstance(self.payload, PackagePayload) def _mirror_active(self): return self._protocolComboBox.get_active_id() == PROTOCOL_MIRROR def _http_active(self): return self._protocolComboBox.get_active_id() in [PROTOCOL_HTTP, PROTOCOL_HTTPS] def _ftp_active(self): return self._protocolComboBox.get_active_id() == PROTOCOL_FTP def _nfs_active(self): return self._protocolComboBox.get_active_id() == PROTOCOL_NFS def _get_selected_partition(self): store = self.builder.get_object("partitionStore") combo = self.builder.get_object("isoPartitionCombo") selected = combo.get_active() if selected == -1: return None else: return store[selected][0] def _sanitize_model(self, model): return model.replace("_", " ") # Input checks # This method is shared by the checks on urlEntry and repoUrlEntry def _checkURL(self, inputcheck, combo): url_string = self.get_input(inputcheck.input_obj).strip() # If this is HTTP/HTTPS/FTP, use the URL_PARSE regex combo_protocol = combo.get_active_id() if combo_protocol in (PROTOCOL_HTTP, PROTOCOL_HTTPS, PROTOCOL_FTP): if not url_string: return _("URL is empty") m = URL_PARSE.match(url_string) if not m: return _("Invalid URL") # If there is a protocol in the URL, and the protocol matches the # combo box, just remove it. This makes it more convenient to paste # in URLs. It'll probably freak out people who are typing out http:// # in the box themselves, but why would you do that? Don't do that. # If the protocols don't match, complain. url_protocol = m.group('protocol') if url_protocol: if (url_protocol == 'http://' and combo_protocol == PROTOCOL_HTTP) or \ (url_protocol == 'https://' and combo_protocol == PROTOCOL_HTTPS) or \ (url_protocol == 'ftp://' and combo_protocol == PROTOCOL_FTP): # Disable the check to block a recursive check call inputcheck.enabled = False inputcheck.input_obj.set_text(url_string[len(url_protocol):]) inputcheck.enabled = True else: return _("Protocol in URL does not match selected protocol") elif combo_protocol == PROTOCOL_NFS: if not url_string: return _("NFS server is empty") # Make sure the part before the colon looks like a hostname, # and that the path is not empty host, _colon, path = url_string.partition(':') if not re.match('^' + HOSTNAME_PATTERN_WITHOUT_ANCHORS + '$', host): return _("Invalid host name") if not path: return _("Remote directory is required") return InputCheck.CHECK_OK def _checkURLEntry(self, inputcheck): return self._checkURL(inputcheck, self._protocolComboBox) def _checkRepoURL(self, inputcheck): return self._checkURL(inputcheck, self._repoProtocolComboBox) # Update the check on urlEntry when the sensitity or selected protocol changes def _updateURLEntryCheck(self, *args): self._urlCheck.enabled = self._urlEntry.is_sensitive() self._urlCheck.update_check_status() # Force a status update to clear any disabled errors self.set_status(self._urlCheck) def _checkDuplicateRepos(self, inputcheck): repo_names = [r[REPO_OBJ].name for r in inputcheck.input_obj] if len(repo_names) != len(frozenset(repo_names)): return _("Duplicate repository names.") return InputCheck.CHECK_OK def _checkRepoName(self, inputcheck): repo_name = self.get_input(inputcheck.input_obj).strip() if not repo_name: return _("Empty repository name") if not REPO_NAME_VALID.match(repo_name): return _("Invalid repository name") cnames = [constants.BASE_REPO_NAME] + \ self.payload.DEFAULT_REPOS + \ [r for r in self.payload.repos if r not in self.payload.addOns] if repo_name in cnames: return _("Repository name conflicts with internal repository name.") return InputCheck.CHECK_OK def _checkRepoProxy(self, inputcheck): # If nfs is selected as the protocol, skip the proxy check if self._repoProtocolComboBox.get_active_id() == PROTOCOL_NFS: return InputCheck.CHECK_OK # Empty proxies are OK, as long as the username and password are empty too proxy_string = self.get_input(inputcheck.input_obj).strip() username_set = self._repoProxyUsernameEntry.is_sensitive() and self._repoProxyUsernameEntry.get_text().strip() password_set = self._repoProxyPasswordEntry.is_sensitive() and self._repoProxyPasswordEntry.get_text().strip() if not (proxy_string or username_set or password_set): return InputCheck.CHECK_OK return _validateProxy(proxy_string, username_set, password_set) # Signal handlers. def on_source_toggled(self, button, relatedBox): # When a radio button is clicked, this handler gets called for both # the newly enabled button as well as the previously enabled (now # disabled) button. enabled = button.get_active() relatedBox.set_sensitive(enabled) self._setup_no_updates() def on_back_clicked(self, button): """If any input validation checks failed, keep the user on the screen. Otherwise, do the usual thing.""" failed_check = next(self.failed_checks, None) # If the failed check is the duplicate repo check, focus the repo TreeView if failed_check == self._duplicateRepoCheck: self._repoView.grab_focus() return # If the failed check is on one of the repo fields, select the repo in the # TreeView and focus the field elif failed_check in self._repoNameChecks.values(): self._repoSelection.select_path(failed_check.data.get_path()) self._repoNameEntry.grab_focus() return elif failed_check in self._repoURLChecks.values(): self._repoSelection.select_path(failed_check.data.get_path()) self._repoUrlEntry.grab_focus() return elif failed_check in self._repoProxyChecks.values(): self._repoSelection.select_path(failed_check.data.get_path()) self._repoProxyUrlEntry.grab_focus() return # Otherwise let GUISpokeInputCheckHandler figure out what to focus elif not GUISpokeInputCheckHandler.on_back_clicked(self, button): return NormalSpoke.on_back_clicked(self, button) def on_chooser_clicked(self, button): dialog = IsoChooser(self.data) # If the chooser has been run once before, we should make it default to # the previously selected file. if self._currentIsoFile: dialog.refresh(currentFile=self._currentIsoFile) else: dialog.refresh() with self.main_window.enlightbox(dialog.window): f = dialog.run(self._get_selected_partition()) if f and f.endswith(".iso"): self._currentIsoFile = f button.set_label(os.path.basename(f)) button.set_use_underline(False) self._verifyIsoButton.set_sensitive(True) def on_proxy_clicked(self, button): dialog = ProxyDialog(self.data, self._proxyUrl) with self.main_window.enlightbox(dialog.window): dialog.refresh() dialog.run() if self._proxyUrl != dialog.proxyUrl: self._proxyChange = True self._proxyUrl = dialog.proxyUrl def on_verify_iso_clicked(self, button): p = self._get_selected_partition() f = self._currentIsoFile if not p or not f: return dialog = MediaCheckDialog(self.data) with self.main_window.enlightbox(dialog.window): unmount = not p.format.status mounts = get_mount_paths(p.path) # We have to check both ISO_DIR and the DRACUT_ISODIR because we # still reference both, even though /mnt/install is a symlink to # /run/install. Finding mount points doesn't handle the symlink if constants.ISO_DIR not in mounts and constants.DRACUT_ISODIR not in mounts: # We're not mounted to either location, so do the mount p.format.mount(mountpoint=constants.ISO_DIR) dialog.run(constants.ISO_DIR + "/" + f) if unmount: p.format.unmount() def on_verify_media_clicked(self, button): if not self._cdrom: return dialog = MediaCheckDialog(self.data) with self.main_window.enlightbox(dialog.window): dialog.run("/dev/" + self._cdrom.name) def on_protocol_changed(self, combo): # Only allow the URL entry to be used if we're using an HTTP/FTP # method that's not the mirror list, or an NFS method. self._urlEntry.set_sensitive(self._http_active() or self._ftp_active() or self._nfs_active()) # Only allow thse widgets to be shown if it makes sense for the # the currently selected protocol. self._proxyButton.set_sensitive(self._http_active() or self._mirror_active()) self._nfsOptsBox.set_visible(self._nfs_active()) self._mirrorlistCheckbox.set_visible(self._http_active()) self._setup_no_updates() # Any changes to the protocol combo box also need to update the check to see # if the protocol now matches (e.g., user puts in a ftp:// URL with http selected # in the combo box, then switches the combo box to ftp). self._updateURLEntryCheck() def _update_payload_repos(self): """ Change the packaging repos to match the new edits This will add new repos to the addon repo list, remove ones that were removed and update any changes made to existing ones. :returns: True if any repo was changed, added or removed :rtype: bool """ REPO_ATTRS=("name", "baseurl", "mirrorlist", "proxy", "enabled") changed = False ui_orig_names = [r[REPO_OBJ].orig_name for r in self._repoStore] # Remove repos from payload that were removed in the UI for repo_name in [r for r in self.payload.addOns if r not in ui_orig_names]: repo = self.payload.getAddOnRepo(repo_name) # TODO: Need an API to do this w/o touching yum (not addRepo) self.payload.data.repo.dataList().remove(repo) changed = True for repo, orig_repo in [(r[REPO_OBJ],self.payload.getAddOnRepo(r[REPO_OBJ].orig_name)) for r in self._repoStore]: if not orig_repo: # TODO: Need an API to do this w/o touching yum (not addRepo) self.payload.data.repo.dataList().append(repo) changed = True elif not cmp_obj_attrs(orig_repo, repo, REPO_ATTRS): for attr in REPO_ATTRS: setattr(orig_repo, attr, getattr(repo, attr)) changed = True return changed def _reset_repoStore(self): """ Reset the list of repos. Populate the list with all the addon repos from payload.addOns. If the list has no element, clear the repo entry fields. """ # Remove the repo checks for check in self._repoNameChecks.values() + self._repoURLChecks.values() + self._repoProxyChecks.values(): self.remove_check(check) self._repoNameChecks = {} self._repoURLChecks = {} self._repoProxyChecks = {} self._repoStore.clear() repos = self.payload.addOns log.debug("Setting up repos: %s", repos) for name in repos: repo = self.payload.getAddOnRepo(name) ks_repo = self.data.RepoData(name=repo.name, baseurl=repo.baseurl, mirrorlist=repo.mirrorlist, proxy=repo.proxy, enabled=repo.enabled) # Track the original name, user may change .name ks_repo.orig_name = name self._repoStore.append([self.payload.isRepoEnabled(name), ks_repo.name, ks_repo]) if len(self._repoStore) > 0: self._repoSelection.select_path(0) else: self._clear_repo_info() self._repoEntryBox.set_sensitive(False) def _unique_repo_name(self, name): """ Return a unique variation of the name if it already exists in the repo store. :param str name: Name to check :returns: name or name with _%d appended The returned name will be 1 greater than any other entry in the store with a _%d at the end of it. """ # Does this name exist in the store? If not, return it. if not any(r[REPO_NAME_COL] == name for r in self._repoStore): return name # If the name already ends with a _\d+ it needs to be stripped. match = re.match(r"(.*)_\d+$", name) if match: name = match.group(1) # Find all of the names with _\d+ at the end name_re = re.compile(r"("+re.escape(name)+r")_(\d+)") matches = (name_re.match(r[REPO_NAME_COL]) for r in self._repoStore) matches = [int(m.group(2)) for m in matches if m is not None] # Get the highest number, add 1, append to name highest_index = max(matches) if matches else 0 return name + ("_%d" % (highest_index + 1)) def on_repoSelection_changed(self, *args): """ Called when the selection changed. Update the repo text boxes with the current information """ itr = self._repoSelection.get_selected()[1] if not itr: return self._update_repo_info(self._repoStore[itr][REPO_OBJ]) def on_repoEnable_toggled(self, renderer, path): """ Called when the repo Enable checkbox is clicked """ enabled = not self._repoStore[path][REPO_ENABLED_COL] self._repoStore[path][REPO_ENABLED_COL] = enabled self._repoStore[path][REPO_OBJ].enabled = enabled def _clear_repo_info(self): """ Clear the text from the repo entry fields and reset the checkbox and combobox. """ self._repoNameEntry.set_text("") with blockedHandler(self._repoMirrorlistCheckbox, self.on_repoMirrorlistCheckbox_toggled): self._repoMirrorlistCheckbox.set_active(False) self._repoUrlEntry.set_text("") self._repoProtocolComboBox.set_active(0) self._repoProxyUrlEntry.set_text("") self._repoProxyUsernameEntry.set_text("") self._repoProxyPasswordEntry.set_text("") def _update_repo_info(self, repo): """ Update the text boxes with data from repo :param repo: kickstart repository object :type repo: RepoData """ self._repoNameEntry.set_text(repo.name) with blockedHandler(self._repoMirrorlistCheckbox, self.on_repoMirrorlistCheckbox_toggled): if repo.mirrorlist: url = repo.mirrorlist self._repoMirrorlistCheckbox.set_active(True) else: url = repo.baseurl self._repoMirrorlistCheckbox.set_active(False) if url: for idx, proto in REPO_PROTO.items(): if url.startswith(proto): self._repoProtocolComboBox.set_active_id(idx) self._repoUrlEntry.set_text(url[len(proto):]) break else: # Unknown protocol, just set the url then self._repoUrlEntry.set_text(url) else: self._repoUrlEntry.set_text("") if not repo.proxy: self._repoProxyUrlEntry.set_text("") self._repoProxyUsernameEntry.set_text("") self._repoProxyPasswordEntry.set_text("") else: try: proxy = ProxyString(repo.proxy) if proxy.username: self._repoProxyUsernameEntry.set_text(proxy.username) if proxy.password: self._repoProxyPasswordEntry.set_text(proxy.password) self._repoProxyUrlEntry.set_text(proxy.noauth_url) except ProxyStringError as e: log.error("Failed to parse proxy for repo %s: %s", repo.name, e) return def on_noUpdatesCheckbox_toggled(self, *args): """ Toggle the enable state of the updates repo Before final release this will also toggle the updates-testing repo """ if self._noUpdatesCheckbox.get_active(): self.payload.disableRepo("updates") if not constants.isFinal: self.payload.disableRepo("updates-testing") else: self.payload.enableRepo("updates") if not constants.isFinal: self.payload.enableRepo("updates-testing") def on_addRepo_clicked(self, button): """ Add a new repository """ name = self._unique_repo_name("New_Repository") repo = self.data.RepoData(name=name) repo.ks_repo = True repo.orig_name = "" itr = self._repoStore.append([True, repo.name, repo]) self._repoSelection.select_iter(itr) self._repoEntryBox.set_sensitive(True) def on_removeRepo_clicked(self, button): """ Remove the selected repository """ itr = self._repoSelection.get_selected()[1] if not itr: return # Remove the input validation checks for this repo repo = self._repoStore[itr][REPO_OBJ] self.remove_check(self._repoNameChecks[repo]) self.remove_check(self._repoURLChecks[repo]) self.remove_check(self._repoProxyChecks[repo]) del self._repoNameChecks[repo] del self._repoURLChecks[repo] del self._repoProxyChecks[repo] self._repoStore.remove(itr) if len(self._repoStore) == 0: self._clear_repo_info() self._repoEntryBox.set_sensitive(False) def on_resetRepos_clicked(self, button): """ Revert to the default list of repositories """ self._reset_repoStore() def on_repoNameEntry_changed(self, entry): """ repo name changed """ itr = self._repoSelection.get_selected()[1] if not itr: return repo = self._repoStore[itr][REPO_OBJ] name = self._repoNameEntry.get_text().strip() repo.name = name self._repoStore.set_value(itr, REPO_NAME_COL, name) self._repoNameChecks[repo].update_check_status() def on_repoUrl_changed(self, *args): """ proxy url or protocol changed """ itr = self._repoSelection.get_selected()[1] if not itr: return repo = self._repoStore[itr][REPO_OBJ] idx = self._repoProtocolComboBox.get_active_id() proto = REPO_PROTO[idx] url = self._repoUrlEntry.get_text().strip() if self._repoMirrorlistCheckbox.get_active(): repo.mirorlist = proto + url else: repo.baseurl = proto + url self._repoURLChecks[repo].update_check_status() def on_repoMirrorlistCheckbox_toggled(self, *args): """ mirror state changed """ itr = self._repoSelection.get_selected()[1] if not itr: return repo = self._repoStore[itr][REPO_OBJ] # This is called by set_active so only swap if there is something # in the variable. if self._repoMirrorlistCheckbox.get_active() and repo.baseurl: repo.mirrorlist = repo.baseurl repo.baseurl = "" elif repo.mirrorlist: repo.baseurl = repo.mirrorlist repo.mirrorlist = "" def on_repoProxy_changed(self, *args): """ Update the selected repo's proxy settings """ itr = self._repoSelection.get_selected()[1] if not itr: return repo = self._repoStore[itr][REPO_OBJ] url = self._repoProxyUrlEntry.get_text().strip() username = self._repoProxyUsernameEntry.get_text().strip() or None password = self._repoProxyPasswordEntry.get_text().strip() or None self._repoProxyChecks[repo].update_check_status() try: proxy = ProxyString(url=url, username=username, password=password) repo.proxy = proxy.url except ProxyStringError as e: log.error("Failed to parse proxy - %s:%s@%s: %s", username, password, url, e) def on_repoStore_row_changed(self, model, path, itr, user_data=None): self._duplicateRepoCheck.update_check_status() def on_repoStore_row_deleted(self, model, path, user_data=None): self._duplicateRepoCheck.update_check_status() def on_repoStore_row_inserted(self, model, path, itr, user_data=None): self._duplicateRepoCheck.update_check_status() repo = model[itr][REPO_OBJ] # Add checks for the repo fields # Use InputCheckHandler.add_check instead of GUISpokeInputCheckHandler.add_check since # the input fields are used by every repo, so the changed signal handler is shared by # more than one check and needs to update only the active one. # It would be nice if we could store itr as the means of accessing this row later, # and GtkListStore sets GTK_TREE_MODEL_ITERS_PERSIST which is supposed to let us # do something like that, but as part of a grand practical joke the iter passed in # to this method is different from the iter used everywhere else, and is useless # once this method returns. Instead, create a TreeRowReference and work backwards # from that using paths any time we need to reference the store. self._repoNameChecks[repo] = InputCheckHandler.add_check(self, self._repoNameEntry, self._checkRepoName, Gtk.TreeRowReference.new(model, path)) self._repoURLChecks[repo] = InputCheckHandler.add_check(self, self._repoUrlEntry, self._checkRepoURL, Gtk.TreeRowReference.new(model, path)) self._repoProxyChecks[repo] = InputCheckHandler.add_check(self, self._repoProxyUrlEntry, self._checkRepoProxy, Gtk.TreeRowReference.new(model, path)) def on_repoProtocolComboBox_changed(self, combobox, user_data=None): # Set the mirrorlist and proxy fields sensitivity depending on whether NFS was selected sensitive = not(self._repoProtocolComboBox.get_active_id() == PROTOCOL_NFS) fancy_set_sensitive(self._repoMirrorlistCheckbox, sensitive) fancy_set_sensitive(self._repoProxyUrlEntry, sensitive) fancy_set_sensitive(self._repoProxyUsernameEntry, sensitive) fancy_set_sensitive(self._repoProxyPasswordEntry, sensitive) # Re-run the proxy check itr = self._repoSelection.get_selected()[1] if itr: repo = self._repoStore[itr][REPO_OBJ] self._repoProxyChecks[repo].update_check_status()