2013-01-23 17:28:19 +00:00
|
|
|
# Base classes for all user interfaces.
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
#
|
|
|
|
|
|
|
|
__all__ = ["UserInterface"]
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
import copy
|
|
|
|
from pyanaconda.ui.common import collect
|
|
|
|
|
|
|
|
class PathDict(dict):
|
|
|
|
"""Dictionary class supporting + operator"""
|
|
|
|
def __add__(self, ext):
|
|
|
|
new_dict = copy.copy(self)
|
2015-05-30 11:20:59 +00:00
|
|
|
for key, value in ext.items():
|
2015-03-23 11:36:12 +00:00
|
|
|
try:
|
|
|
|
new_dict[key].extend(value)
|
|
|
|
except KeyError:
|
|
|
|
new_dict[key] = value[:]
|
|
|
|
|
|
|
|
return new_dict
|
2013-01-23 17:28:19 +00:00
|
|
|
|
|
|
|
class UserInterface(object):
|
|
|
|
"""This is the base class for all kinds of install UIs. It primarily
|
|
|
|
defines what kinds of dialogs and entry widgets every interface must
|
|
|
|
provide that the rest of anaconda may rely upon.
|
|
|
|
"""
|
|
|
|
def __init__(self, storage, payload, instclass):
|
|
|
|
"""Create a new UserInterface instance.
|
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
The arguments this base class accepts defines the API that interfaces
|
|
|
|
have to work with. A UserInterface does not get free reign over
|
|
|
|
everything in the anaconda class, as that would be a big mess.
|
|
|
|
Instead, a UserInterface may count on the following:
|
|
|
|
|
|
|
|
storage -- An instance of storage.Storage. This is useful for
|
|
|
|
determining what storage devices are present and how
|
|
|
|
they are configured.
|
|
|
|
payload -- An instance of a packaging.Payload subclass. This
|
|
|
|
is useful for displaying and selecting packages to
|
|
|
|
install, and in carrying out the actual installation.
|
|
|
|
instclass -- An instance of a BaseInstallClass subclass. This
|
|
|
|
is useful for determining distribution-specific
|
|
|
|
installation information like default package
|
|
|
|
selections and default partitioning.
|
2013-01-23 17:28:19 +00:00
|
|
|
"""
|
|
|
|
if self.__class__ is UserInterface:
|
|
|
|
raise TypeError("UserInterface is an abstract class.")
|
|
|
|
|
|
|
|
self.storage = storage
|
|
|
|
self.payload = payload
|
|
|
|
self.instclass = instclass
|
|
|
|
|
|
|
|
# Register this interface with the top-level ErrorHandler.
|
|
|
|
from pyanaconda.errors import errorHandler
|
|
|
|
errorHandler.ui = self
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
paths = PathDict({})
|
|
|
|
|
|
|
|
@property
|
|
|
|
def tty_num(self):
|
|
|
|
"""Returns the number of tty the UserInterface is running on."""
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def update_paths(cls, pathdict):
|
2017-01-09 02:09:07 +00:00
|
|
|
"""Receives pathdict and appends it's contents to the current class defined search path dictionary."""
|
2016-04-10 04:00:00 +00:00
|
|
|
for k, v in pathdict.items():
|
2014-04-07 12:38:09 +00:00
|
|
|
cls.paths.setdefault(k, [])
|
|
|
|
cls.paths[k].extend(v)
|
|
|
|
|
2013-01-23 17:28:19 +00:00
|
|
|
def setup(self, data):
|
|
|
|
"""Construct all the objects required to implement this interface.
|
2017-01-09 02:09:07 +00:00
|
|
|
|
|
|
|
This method must be provided by all subclasses.
|
2013-01-23 17:28:19 +00:00
|
|
|
"""
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def run(self):
|
2017-01-09 02:09:07 +00:00
|
|
|
"""Run the interface.
|
|
|
|
|
|
|
|
This should do little more than just pass through to something else's run method,
|
|
|
|
but is provided here in case more is needed. This method must be provided by all subclasses.
|
2013-01-23 17:28:19 +00:00
|
|
|
"""
|
|
|
|
raise NotImplementedError
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
@property
|
|
|
|
def meh_interface(self):
|
2017-01-09 02:09:07 +00:00
|
|
|
"""Returns an interface for exception handling (defined by python-meh's AbstractIntf class)."""
|
2014-04-07 12:38:09 +00:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2013-01-23 17:28:19 +00:00
|
|
|
###
|
|
|
|
### MESSAGE HANDLING METHODS
|
|
|
|
###
|
|
|
|
def showError(self, message):
|
2017-01-09 02:09:07 +00:00
|
|
|
"""Display an error dialog with the given message.
|
|
|
|
|
|
|
|
There is no return value. This method must be implemented by all UserInterface
|
|
|
|
subclasses.
|
2013-01-23 17:28:19 +00:00
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
In the code, this method should be used sparingly and only for
|
|
|
|
critical errors that anaconda cannot figure out how to recover from.
|
2013-01-23 17:28:19 +00:00
|
|
|
"""
|
|
|
|
raise NotImplementedError
|
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
def showDetailedError(self, message, details, buttons=None):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
2013-01-23 17:28:19 +00:00
|
|
|
def showYesNoQuestion(self, message):
|
2017-01-09 02:09:07 +00:00
|
|
|
"""Display a dialog with the given message that presents the user a yes or no choice.
|
|
|
|
|
|
|
|
This method returns True if the yes choice is selected,
|
|
|
|
and False if the no choice is selected.
|
|
|
|
From here, anaconda can figure out what to do next.
|
|
|
|
This method must be implemented by all UserInterface subclasses.
|
|
|
|
|
|
|
|
In the code, this method should be used sparingly and only for those
|
|
|
|
times where anaconda cannot make a reasonable decision. We don't
|
|
|
|
want to overwhelm the user with choices.
|
2013-01-23 17:28:19 +00:00
|
|
|
"""
|
|
|
|
raise NotImplementedError
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
def _collectActionClasses(self, module_pattern_w_path, standalone_class):
|
2017-01-09 02:09:07 +00:00
|
|
|
"""Collect all the Hub and Spoke classes which should be enqueued for processing.
|
2013-01-23 17:28:19 +00:00
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
:param module_pattern_w_path: the full name patterns (pyanaconda.ui.gui.spokes.%s)
|
|
|
|
and directory paths to modules we are about to import
|
|
|
|
:type module_pattern_w_path: list of (string, string)
|
2014-04-07 12:38:09 +00:00
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
:param standalone_class: the parent type of Spokes we want to pick up
|
|
|
|
:type standalone_class: common.StandaloneSpoke based types
|
2014-04-07 12:38:09 +00:00
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
:return: list of Spoke classes with standalone_class as a parent
|
|
|
|
:rtype: list of Spoke classes
|
2013-01-23 17:28:19 +00:00
|
|
|
"""
|
2014-04-07 12:38:09 +00:00
|
|
|
standalones = []
|
|
|
|
|
|
|
|
for module_pattern, path in module_pattern_w_path:
|
|
|
|
standalones.extend(collect(module_pattern, path, lambda obj: issubclass(obj, standalone_class) and \
|
|
|
|
getattr(obj, "preForHub", False) or getattr(obj, "postForHub", False)))
|
|
|
|
|
|
|
|
return standalones
|
2013-01-23 17:28:19 +00:00
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
def _orderActionClasses(self, spokes, hubs):
|
2017-01-09 02:09:07 +00:00
|
|
|
"""Order all the Hub and Spoke classes which should be enqueued for processing according to their pre/post dependencies.
|
2013-01-23 17:28:19 +00:00
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
:param spokes: the classes we are to about order according
|
|
|
|
to the hub dependencies
|
|
|
|
:type spokes: list of Spoke instances
|
2014-04-07 12:38:09 +00:00
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
:param hubs: the list of Hub classes we check to be in pre/postForHub
|
|
|
|
attribute of Spokes to pick up
|
|
|
|
:type hubs: common.Hub based types
|
2014-04-07 12:38:09 +00:00
|
|
|
"""
|
2013-01-23 17:28:19 +00:00
|
|
|
|
|
|
|
actionClasses = []
|
|
|
|
for hub in hubs:
|
2015-05-30 11:20:59 +00:00
|
|
|
actionClasses.extend(sorted(filter(lambda obj, h=hub: getattr(obj, "preForHub", None) == h, spokes),
|
2013-01-23 17:28:19 +00:00
|
|
|
key=lambda obj: obj.priority))
|
|
|
|
actionClasses.append(hub)
|
2015-05-30 11:20:59 +00:00
|
|
|
actionClasses.extend(sorted(filter(lambda obj, h=hub: getattr(obj, "postForHub", None) == h, spokes),
|
2013-01-23 17:28:19 +00:00
|
|
|
key=lambda obj: obj.priority))
|
|
|
|
|
|
|
|
return actionClasses
|