2013-01-23 17:28:19 +00:00
|
|
|
# The base classes for Anaconda TUI Spokes
|
|
|
|
#
|
|
|
|
# Copyright (C) (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.
|
|
|
|
#
|
2014-04-07 12:38:09 +00:00
|
|
|
from pyanaconda.ui.tui import simpleline as tui
|
|
|
|
from pyanaconda.ui.tui.tuiobject import TUIObject, YesNoDialog
|
2015-03-23 11:36:12 +00:00
|
|
|
from pyanaconda.ui.common import Spoke, StandaloneSpoke, NormalSpoke
|
2014-04-07 12:38:09 +00:00
|
|
|
from pyanaconda.users import validatePassword, cryptPassword
|
|
|
|
import re
|
|
|
|
from collections import namedtuple
|
|
|
|
from pyanaconda.iutil import setdeepattr, getdeepattr
|
2015-03-23 11:36:12 +00:00
|
|
|
from pyanaconda.i18n import N_, _
|
|
|
|
from pyanaconda.constants import PASSWORD_CONFIRM_ERROR_TUI, PW_ASCII_CHARS
|
2015-05-30 11:20:59 +00:00
|
|
|
from pyanaconda.constants import PASSWORD_WEAK, PASSWORD_WEAK_WITH_ERROR
|
2013-01-23 17:28:19 +00:00
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
__all__ = ["TUISpoke", "EditTUISpoke", "EditTUIDialog", "EditTUISpokeEntry",
|
|
|
|
"StandaloneSpoke", "NormalTUISpoke"]
|
2013-01-23 17:28:19 +00:00
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
# Inherit abstract methods from Spoke
|
|
|
|
# pylint: disable=abstract-method
|
2013-01-23 17:28:19 +00:00
|
|
|
class TUISpoke(TUIObject, tui.Widget, Spoke):
|
|
|
|
"""Base TUI Spoke class implementing the pyanaconda.ui.common.Spoke API.
|
2016-04-10 04:00:00 +00:00
|
|
|
It also acts as a Widget so we can easily add it to Hub, where is shows
|
|
|
|
as a summary box with title, description and completed checkbox.
|
2013-01-23 17:28:19 +00:00
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
:param title: title of this spoke
|
|
|
|
:type title: str
|
2013-01-23 17:28:19 +00:00
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
:param category: category this spoke belongs to
|
|
|
|
:type category: string
|
|
|
|
|
|
|
|
.. inheritance-diagram:: TUISpoke
|
|
|
|
:parts: 3
|
2013-01-23 17:28:19 +00:00
|
|
|
"""
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
title = N_("Default spoke title")
|
2013-01-23 17:28:19 +00:00
|
|
|
|
|
|
|
def __init__(self, app, data, storage, payload, instclass):
|
2015-03-23 11:36:12 +00:00
|
|
|
if self.__class__ is TUISpoke:
|
|
|
|
raise TypeError("TUISpoke is an abstract class")
|
|
|
|
|
2013-01-23 17:28:19 +00:00
|
|
|
TUIObject.__init__(self, app, data)
|
|
|
|
tui.Widget.__init__(self)
|
2015-03-23 11:36:12 +00:00
|
|
|
Spoke.__init__(self, storage, payload, instclass)
|
2013-01-23 17:28:19 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def status(self):
|
|
|
|
return _("testing status...")
|
|
|
|
|
|
|
|
@property
|
|
|
|
def completed(self):
|
|
|
|
return True
|
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
def refresh(self, args=None):
|
2013-01-23 17:28:19 +00:00
|
|
|
TUIObject.refresh(self, args)
|
|
|
|
return True
|
|
|
|
|
|
|
|
def input(self, args, key):
|
|
|
|
"""Handle the input, the base class just forwards it to the App level."""
|
|
|
|
return key
|
|
|
|
|
|
|
|
def render(self, width):
|
|
|
|
"""Render the summary representation for Hub to internal buffer."""
|
|
|
|
tui.Widget.render(self, width)
|
2014-04-07 12:38:09 +00:00
|
|
|
|
|
|
|
if self.mandatory and not self.completed:
|
|
|
|
key = "!"
|
2015-03-23 11:36:12 +00:00
|
|
|
elif self.completed:
|
2014-04-07 12:38:09 +00:00
|
|
|
key = "x"
|
2015-03-23 11:36:12 +00:00
|
|
|
else:
|
|
|
|
key = " "
|
2014-04-07 12:38:09 +00:00
|
|
|
|
|
|
|
# always set completed = True here; otherwise key value won't be
|
|
|
|
# displayed if completed (spoke value from above) is False
|
2016-04-10 04:00:00 +00:00
|
|
|
c = tui.CheckboxWidget(key=key, completed=True,
|
|
|
|
title=_(self.title), text=self.status)
|
2013-01-23 17:28:19 +00:00
|
|
|
c.render(width)
|
|
|
|
self.draw(c)
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
class NormalTUISpoke(TUISpoke, NormalSpoke):
|
2016-04-10 04:00:00 +00:00
|
|
|
"""
|
|
|
|
.. inheritance-diagram:: NormalTUISpoke
|
|
|
|
:parts: 3
|
|
|
|
"""
|
2013-01-23 17:28:19 +00:00
|
|
|
pass
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
EditTUISpokeEntry = namedtuple("EditTUISpokeEntry", ["title", "attribute", "aux", "visible"])
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
# Inherit abstract methods from NormalTUISpoke
|
|
|
|
# pylint: disable=abstract-method
|
2014-04-07 12:38:09 +00:00
|
|
|
class EditTUIDialog(NormalTUISpoke):
|
2016-04-10 04:00:00 +00:00
|
|
|
"""Spoke/dialog used to read new value of textual or password data
|
|
|
|
|
|
|
|
.. inheritance-diagram:: EditTUIDialog
|
|
|
|
:parts: 3
|
|
|
|
|
|
|
|
To override the wrong input message set the wrong_input_message attribute
|
|
|
|
to a translated string.
|
|
|
|
"""
|
2015-03-23 11:36:12 +00:00
|
|
|
title = N_("New value")
|
2014-04-07 12:38:09 +00:00
|
|
|
PASSWORD = re.compile(".*")
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
def __init__(self, app, data, storage, payload, instclass, policy_name=""):
|
2015-03-23 11:36:12 +00:00
|
|
|
if self.__class__ is EditTUIDialog:
|
|
|
|
raise TypeError("EditTUIDialog is an abstract class")
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
NormalTUISpoke.__init__(self, app, data, storage, payload, instclass)
|
|
|
|
self.value = None
|
2016-04-10 04:00:00 +00:00
|
|
|
self.policy = None
|
|
|
|
self.wrong_input_message = None
|
2014-04-07 12:38:09 +00:00
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
# Configure the password policy, if available. Otherwise use defaults.
|
|
|
|
self.policy = self.data.anaconda.pwpolicy.get_policy(policy_name)
|
|
|
|
if not self.policy:
|
|
|
|
self.policy = self.data.anaconda.PwPolicyData()
|
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
def refresh(self, args=None):
|
2014-04-07 12:38:09 +00:00
|
|
|
self._window = []
|
|
|
|
self.value = None
|
|
|
|
return True
|
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
def prompt(self, entry=None):
|
2014-04-07 12:38:09 +00:00
|
|
|
if not entry:
|
|
|
|
return None
|
|
|
|
|
|
|
|
if entry.aux == self.PASSWORD:
|
|
|
|
pw = self._app.raw_input(_("%s: ") % entry.title, hidden=True)
|
|
|
|
confirm = self._app.raw_input(_("%s (confirm): ") % entry.title, hidden=True)
|
|
|
|
|
|
|
|
if (pw and not confirm) or (confirm and not pw):
|
|
|
|
print(_("You must enter your root password and confirm it by typing"
|
|
|
|
" it a second time to continue."))
|
|
|
|
return None
|
|
|
|
if (pw != confirm):
|
|
|
|
print(_(PASSWORD_CONFIRM_ERROR_TUI))
|
|
|
|
return None
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
# If an empty password was provided, unset the value
|
|
|
|
if not pw:
|
|
|
|
self.value = ""
|
|
|
|
return None
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
valid, strength, message = validatePassword(pw, user=None, minlen=self.policy.minlen)
|
2014-04-07 12:38:09 +00:00
|
|
|
|
|
|
|
if not valid:
|
|
|
|
print(message)
|
|
|
|
return None
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
if strength < self.policy.minquality:
|
|
|
|
if self.policy.strict:
|
|
|
|
done_msg = ""
|
|
|
|
else:
|
|
|
|
done_msg = _("\nWould you like to use it anyway?")
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
if message:
|
2016-04-10 04:00:00 +00:00
|
|
|
error = _(PASSWORD_WEAK_WITH_ERROR) % message + " " + done_msg
|
2014-04-07 12:38:09 +00:00
|
|
|
else:
|
2015-05-30 11:20:59 +00:00
|
|
|
error = _(PASSWORD_WEAK) % done_msg
|
|
|
|
|
|
|
|
if not self.policy.strict:
|
|
|
|
question_window = YesNoDialog(self._app, error)
|
|
|
|
self._app.switch_screen_modal(question_window)
|
|
|
|
if not question_window.answer:
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
print(error)
|
2014-04-07 12:38:09 +00:00
|
|
|
return None
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
if any(char not in PW_ASCII_CHARS for char in pw):
|
|
|
|
print(_("You have provided a password containing non-ASCII characters.\n"
|
|
|
|
"You may not be able to switch between keyboard layouts to login.\n"))
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
self.value = cryptPassword(pw)
|
|
|
|
return None
|
|
|
|
else:
|
2017-01-09 02:09:07 +00:00
|
|
|
return _("Enter a new value for '%s' and press [Enter]\n") % entry.title
|
2014-04-07 12:38:09 +00:00
|
|
|
|
|
|
|
def input(self, entry, key):
|
2017-01-09 02:09:07 +00:00
|
|
|
if callable(entry.aux):
|
|
|
|
valid, err_msg = entry.aux(key)
|
|
|
|
if not valid:
|
|
|
|
if err_msg is not None:
|
|
|
|
self.wrong_input_message = err_msg
|
|
|
|
else:
|
|
|
|
valid = entry.aux.match(key)
|
|
|
|
|
|
|
|
if valid:
|
2014-04-07 12:38:09 +00:00
|
|
|
self.value = key
|
|
|
|
self.close()
|
|
|
|
return True
|
|
|
|
else:
|
2016-04-10 04:00:00 +00:00
|
|
|
if self.wrong_input_message:
|
|
|
|
print(self.wrong_input_message)
|
|
|
|
else:
|
|
|
|
print(_("You have provided an invalid value\n"))
|
2014-04-07 12:38:09 +00:00
|
|
|
return NormalTUISpoke.input(self, entry, key)
|
|
|
|
|
2017-01-09 02:09:07 +00:00
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
class OneShotEditTUIDialog(EditTUIDialog):
|
|
|
|
"""The same as EditTUIDialog, but closes automatically after
|
|
|
|
the value is read
|
|
|
|
"""
|
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
def prompt(self, entry=None):
|
2014-04-07 12:38:09 +00:00
|
|
|
ret = None
|
|
|
|
|
|
|
|
if entry:
|
|
|
|
ret = EditTUIDialog.prompt(self, entry)
|
|
|
|
if ret is None:
|
|
|
|
self.close()
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
# Inherit abstract methods from NormalTUISpoke
|
|
|
|
# pylint: disable=abstract-method
|
2014-04-07 12:38:09 +00:00
|
|
|
class EditTUISpoke(NormalTUISpoke):
|
|
|
|
"""Spoke with declarative semantics, it contains
|
|
|
|
a list of titles, attribute names and regexps
|
|
|
|
that specify the fields of an object the user
|
|
|
|
allowed to edit.
|
2016-04-10 04:00:00 +00:00
|
|
|
|
|
|
|
.. inheritance-diagram:: EditTUISpoke
|
|
|
|
:parts: 3
|
2014-04-07 12:38:09 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
# self.data's subattribute name
|
|
|
|
# empty string means __init__ will provide
|
|
|
|
# something else
|
|
|
|
edit_data = ""
|
|
|
|
|
|
|
|
# constants to be used in the aux field
|
|
|
|
# and mark the entry as a password or checkbox field
|
|
|
|
PASSWORD = EditTUIDialog.PASSWORD
|
|
|
|
CHECK = "check"
|
|
|
|
|
|
|
|
# list of fields in the format of named tuples like:
|
|
|
|
# EditTUISpokeEntry(title, attribute, aux, visible)
|
|
|
|
# title - Nontranslated title of the entry
|
|
|
|
# attribute - The edited object's attribute name
|
2017-01-09 02:09:07 +00:00
|
|
|
# aux - Compiled regular expression or
|
|
|
|
# a callable taking the value and
|
|
|
|
# returning (valid:bool, err_msg:str) tuple,
|
|
|
|
# or one of the two constants from above.
|
2014-04-07 12:38:09 +00:00
|
|
|
# It will be used to check the value typed
|
|
|
|
# by user and to show the proper entry
|
|
|
|
# for password, text or checkbox.
|
|
|
|
# visible - True, False or a function that accepts
|
|
|
|
# two arguments - self and the edited object
|
|
|
|
# It is evaluated and used to display or
|
|
|
|
# hide this attribute's entry
|
|
|
|
edit_fields = [
|
|
|
|
]
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
def __init__(self, app, data, storage, payload, instclass, policy_name=""):
|
2015-03-23 11:36:12 +00:00
|
|
|
if self.__class__ is EditTUISpoke:
|
|
|
|
raise TypeError("EditTUISpoke is an abstract class")
|
|
|
|
|
2014-04-07 12:38:09 +00:00
|
|
|
NormalTUISpoke.__init__(self, app, data, storage, payload, instclass)
|
2015-05-30 11:20:59 +00:00
|
|
|
self.dialog = OneShotEditTUIDialog(app, data, storage, payload, instclass, policy_name=policy_name)
|
2014-04-07 12:38:09 +00:00
|
|
|
|
|
|
|
# self.args should hold the object this Spoke is supposed
|
|
|
|
# to edit
|
|
|
|
self.args = None
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
@property
|
|
|
|
def visible_fields(self):
|
|
|
|
"""Get the list of currently visible entries"""
|
|
|
|
|
|
|
|
# it would be nice to have this a static list, but visibility of the
|
|
|
|
# entries often depends on the current state of the spoke and thus
|
|
|
|
# changes dynamically
|
|
|
|
ret = []
|
|
|
|
for entry in self.edit_fields:
|
|
|
|
if callable(entry.visible) and entry.visible(self, self.args):
|
|
|
|
ret.append(entry)
|
|
|
|
elif not callable(entry.visible) and entry.visible:
|
|
|
|
ret.append(entry)
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
2016-04-10 04:00:00 +00:00
|
|
|
def refresh(self, args=None):
|
2014-04-07 12:38:09 +00:00
|
|
|
NormalTUISpoke.refresh(self, args)
|
|
|
|
|
|
|
|
if args:
|
|
|
|
self.args = args
|
|
|
|
elif self.edit_data:
|
|
|
|
self.args = self.data
|
|
|
|
for key in self.edit_data.split("."):
|
|
|
|
self.args = getattr(self.args, key)
|
|
|
|
|
|
|
|
def _prep_text(i, entry):
|
|
|
|
number = tui.TextWidget("%2d)" % i)
|
|
|
|
title = tui.TextWidget(_(entry.title))
|
|
|
|
value = getdeepattr(self.args, entry.attribute)
|
|
|
|
value = tui.TextWidget(value)
|
|
|
|
|
|
|
|
return tui.ColumnWidget([(3, [number]), (None, [title, value])], 1)
|
|
|
|
|
|
|
|
def _prep_check(i, entry):
|
|
|
|
number = tui.TextWidget("%2d)" % i)
|
|
|
|
value = getdeepattr(self.args, entry.attribute)
|
|
|
|
ch = tui.CheckboxWidget(title=_(entry.title), completed=bool(value))
|
|
|
|
|
|
|
|
return tui.ColumnWidget([(3, [number]), (None, [ch])], 1)
|
|
|
|
|
|
|
|
def _prep_password(i, entry):
|
|
|
|
number = tui.TextWidget("%2d)" % i)
|
|
|
|
title = tui.TextWidget(_(entry.title))
|
|
|
|
value = ""
|
|
|
|
if len(getdeepattr(self.args, entry.attribute)) > 0:
|
|
|
|
value = _("Password set.")
|
|
|
|
value = tui.TextWidget(value)
|
|
|
|
|
|
|
|
return tui.ColumnWidget([(3, [number]), (None, [title, value])], 1)
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
for idx, entry in enumerate(self.visible_fields):
|
2014-04-07 12:38:09 +00:00
|
|
|
entry_type = entry.aux
|
|
|
|
if entry_type == self.PASSWORD:
|
|
|
|
w = _prep_password(idx+1, entry)
|
|
|
|
elif entry_type == self.CHECK:
|
|
|
|
w = _prep_check(idx+1, entry)
|
|
|
|
else:
|
|
|
|
w = _prep_text(idx+1, entry)
|
|
|
|
|
|
|
|
self._window.append(w)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
def input(self, args, key):
|
|
|
|
try:
|
|
|
|
idx = int(key) - 1
|
2015-03-23 11:36:12 +00:00
|
|
|
if idx >= 0 and idx < len(self.visible_fields):
|
2015-05-30 11:20:59 +00:00
|
|
|
if self.visible_fields[idx].aux == self.CHECK:
|
2015-03-23 11:36:12 +00:00
|
|
|
setdeepattr(self.args, self.visible_fields[idx].attribute,
|
2015-05-30 11:20:59 +00:00
|
|
|
not getdeepattr(self.args, self.visible_fields[idx][1]))
|
2014-04-07 12:38:09 +00:00
|
|
|
self.app.redraw()
|
|
|
|
self.apply()
|
|
|
|
else:
|
2015-03-23 11:36:12 +00:00
|
|
|
self.app.switch_screen_modal(self.dialog, self.visible_fields[idx])
|
2014-04-07 12:38:09 +00:00
|
|
|
if self.dialog.value is not None:
|
2015-03-23 11:36:12 +00:00
|
|
|
setdeepattr(self.args, self.visible_fields[idx].attribute,
|
2014-04-07 12:38:09 +00:00
|
|
|
self.dialog.value)
|
|
|
|
self.apply()
|
|
|
|
return True
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
return NormalTUISpoke.input(self, args, key)
|
|
|
|
|
|
|
|
class StandaloneTUISpoke(TUISpoke, StandaloneSpoke):
|
2016-04-10 04:00:00 +00:00
|
|
|
"""
|
|
|
|
.. inheritance-diagram:: StandaloneTUISpoke
|
|
|
|
:parts: 3
|
|
|
|
"""
|
2013-01-23 17:28:19 +00:00
|
|
|
pass
|