2013-01-23 17:28:19 +00:00
# Text storage configuration spoke classes
#
2015-03-23 11:36:12 +00:00
# Copyright (C) 2012-2014 Red Hat, Inc.
2013-01-23 17:28:19 +00:00
#
# 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): Jesse Keating <jkeating@redhat.com>
#
# Some of the code here is copied from pyanaconda/ui/gui/spokes/storage.py
# which has the same license and authored by David Lehman <dlehman@redhat.com>
#
2016-04-10 04:00:00 +00:00
import gi
gi . require_version ( " BlockDev " , " 1.0 " )
from gi . repository import BlockDev as blockdev
from pyanaconda . ui . lib . disks import getDisks , applyDiskSelection , checkDiskSelection
2015-03-23 11:36:12 +00:00
from pyanaconda . ui . categories . system import SystemCategory
2013-01-23 17:28:19 +00:00
from pyanaconda . ui . tui . spokes import NormalTUISpoke
from pyanaconda . ui . tui . simpleline import TextWidget , CheckboxWidget
2015-03-23 11:36:12 +00:00
from pyanaconda . ui . tui . tuiobject import YesNoDialog
from pyanaconda . storage_utils import AUTOPART_CHOICES , sanity_check , SanityError , SanityWarning
2013-01-23 17:28:19 +00:00
2015-03-23 11:36:12 +00:00
from blivet import arch
2014-04-07 12:38:09 +00:00
from blivet . size import Size
2016-04-10 04:00:00 +00:00
from blivet . errors import StorageError
2015-03-23 11:36:12 +00:00
from blivet . devices import DASDDevice , FcoeDiskDevice , iScsiDiskDevice , MultipathDevice , ZFCPDiskDevice
2013-01-23 17:28:19 +00:00
from pyanaconda . flags import flags
2015-03-23 11:36:12 +00:00
from pyanaconda . kickstart import doKickstartStorage , resetCustomStorageData
2013-01-23 17:28:19 +00:00
from pyanaconda . threads import threadMgr , AnacondaThread
2015-03-23 11:36:12 +00:00
from pyanaconda . constants import THREAD_STORAGE , THREAD_STORAGE_WATCHER , THREAD_DASDFMT , DEFAULT_AUTOPART_TYPE
2016-04-10 04:00:00 +00:00
from pyanaconda . constants import PAYLOAD_STATUS_PROBING_STORAGE
2014-04-07 12:38:09 +00:00
from pyanaconda . constants_text import INPUT_PROCESSED
2015-03-23 11:36:12 +00:00
from pyanaconda . i18n import _ , P_ , N_
2014-04-07 12:38:09 +00:00
from pyanaconda . bootloader import BootLoaderError
2013-01-23 17:28:19 +00:00
2015-03-23 11:36:12 +00:00
from pykickstart . constants import CLEARPART_TYPE_ALL , CLEARPART_TYPE_LINUX , CLEARPART_TYPE_NONE , AUTOPART_TYPE_LVM
2014-04-07 12:38:09 +00:00
from pykickstart . errors import KickstartValueError
from collections import OrderedDict
2013-01-23 17:28:19 +00:00
import logging
log = logging . getLogger ( " anaconda " )
__all__ = [ " StorageSpoke " , " AutoPartSpoke " ]
2015-03-23 11:36:12 +00:00
CLEARALL = N_ ( " Use All Space " )
CLEARLINUX = N_ ( " Replace Existing Linux system(s) " )
CLEARNONE = N_ ( " Use Free Space " )
2013-01-23 17:28:19 +00:00
2014-04-07 12:38:09 +00:00
PARTTYPES = { CLEARALL : CLEARPART_TYPE_ALL , CLEARLINUX : CLEARPART_TYPE_LINUX ,
2013-01-23 17:28:19 +00:00
CLEARNONE : CLEARPART_TYPE_NONE }
class StorageSpoke ( NormalTUISpoke ) :
2016-04-10 04:00:00 +00:00
""" Storage spoke where users proceed to customize storage features such
as disk selection , partitioning , and fs type .
. . inheritance - diagram : : StorageSpoke
: parts : 3
2014-04-07 12:38:09 +00:00
"""
2015-03-23 11:36:12 +00:00
title = N_ ( " Installation Destination " )
category = SystemCategory
2013-01-23 17:28:19 +00:00
def __init__ ( self , app , data , storage , payload , instclass ) :
NormalTUISpoke . __init__ ( self , app , data , storage , payload , instclass )
self . _ready = False
2014-04-07 12:38:09 +00:00
self . selected_disks = self . data . ignoredisk . onlyuse [ : ]
2015-03-23 11:36:12 +00:00
self . selection = None
2014-04-07 12:38:09 +00:00
self . autopart = None
self . clearPartType = None
2013-01-23 17:28:19 +00:00
# This list gets set up once in initialize and should not be modified
# except perhaps to add advanced devices. It will remain the full list
# of disks that can be included in the install.
self . disks = [ ]
self . errors = [ ]
2014-04-07 12:38:09 +00:00
self . warnings = [ ]
2015-03-23 11:36:12 +00:00
if self . data . zerombr . zerombr and arch . isS390 ( ) :
# if zerombr is specified in a ks file and there are unformatted
2015-05-30 11:20:59 +00:00
# dasds, automatically format them. pass in storage.devicetree here
# instead of storage.disks since mediaPresent is checked on disks;
# a dasd needing dasdfmt will fail this media check though
2016-04-10 04:00:00 +00:00
to_format = storage . devicetree . make_unformatted_dasd_list ( getDisks ( self . storage . devicetree ) )
2015-03-23 11:36:12 +00:00
if to_format :
self . run_dasdfmt ( to_format )
2014-04-07 12:38:09 +00:00
if not flags . automatedInstall :
# default to using autopart for interactive installs
self . data . autopart . autopart = True
2013-01-23 17:28:19 +00:00
@property
def completed ( self ) :
2014-04-07 12:38:09 +00:00
retval = bool ( self . storage . rootDevice and not self . errors )
return retval
2013-01-23 17:28:19 +00:00
@property
def ready ( self ) :
# By default, the storage spoke is not ready. We have to wait until
# storageInitialize is done.
2015-03-23 11:36:12 +00:00
return self . _ready and not ( threadMgr . get ( THREAD_STORAGE_WATCHER ) or threadMgr . get ( THREAD_DASDFMT ) )
2014-04-07 12:38:09 +00:00
@property
def mandatory ( self ) :
return True
@property
def showable ( self ) :
return not flags . dirInstall
2013-01-23 17:28:19 +00:00
@property
def status ( self ) :
""" A short string describing the current status of storage setup. """
msg = _ ( " No disks selected " )
2014-04-07 12:38:09 +00:00
if flags . automatedInstall and not self . storage . rootDevice :
2015-05-30 11:20:59 +00:00
msg = _ ( " Kickstart insufficient " )
2014-04-07 12:38:09 +00:00
elif self . data . ignoredisk . onlyuse :
2013-01-23 17:28:19 +00:00
msg = P_ ( ( " %d disk selected " ) ,
( " %d disks selected " ) ,
len ( self . data . ignoredisk . onlyuse ) ) % len ( self . data . ignoredisk . onlyuse )
if self . errors :
msg = _ ( " Error checking storage configuration " )
2014-04-07 12:38:09 +00:00
elif self . warnings :
msg = _ ( " Warning checking storage configuration " )
2013-01-23 17:28:19 +00:00
# Maybe show what type of clearpart and which disks selected?
elif self . data . autopart . autopart :
msg = _ ( " Automatic partitioning selected " )
else :
msg = _ ( " Custom partitioning selected " )
return msg
def _update_disk_list ( self , disk ) :
""" Update self.selected_disks based on the selection. """
name = disk . name
# if the disk isn't already selected, select it.
if name not in self . selected_disks :
self . selected_disks . append ( name )
# If the disk is already selected, deselect it.
elif name in self . selected_disks :
self . selected_disks . remove ( name )
def _update_summary ( self ) :
""" Update the summary based on the UI. """
count = 0
capacity = 0
2015-03-23 11:36:12 +00:00
free = Size ( 0 )
2013-01-23 17:28:19 +00:00
# pass in our disk list so hidden disks' free space is available
free_space = self . storage . getFreeSpace ( disks = self . disks )
selected = [ d for d in self . disks if d . name in self . selected_disks ]
for disk in selected :
capacity + = disk . size
free + = free_space [ disk . name ] [ 0 ]
count + = 1
summary = ( P_ ( ( " %d disk selected; %s capacity; %s free ... " ) ,
( " %d disks selected; %s capacity; %s free ... " ) ,
2015-03-23 11:36:12 +00:00
count ) % ( count , str ( Size ( capacity ) ) , free ) )
2013-01-23 17:28:19 +00:00
if len ( self . disks ) == 0 :
summary = _ ( " No disks detected. Please shut down the computer, connect at least one disk, and restart to complete installation. " )
elif count == 0 :
summary = ( _ ( " No disks selected; please select at least one disk to install to. " ) )
# Append storage errors to the summary
if self . errors :
summary = summary + " \n " + " \n " . join ( self . errors )
2014-04-07 12:38:09 +00:00
elif self . warnings :
summary = summary + " \n " + " \n " . join ( self . warnings )
2013-01-23 17:28:19 +00:00
return summary
2016-04-10 04:00:00 +00:00
def refresh ( self , args = None ) :
2013-01-23 17:28:19 +00:00
NormalTUISpoke . refresh ( self , args )
# Join the initialization thread to block on it
2014-04-07 12:38:09 +00:00
# This print is foul. Need a better message display
2016-04-10 04:00:00 +00:00
print ( _ ( PAYLOAD_STATUS_PROBING_STORAGE ) )
2014-04-07 12:38:09 +00:00
threadMgr . wait ( THREAD_STORAGE_WATCHER )
2013-01-23 17:28:19 +00:00
# synchronize our local data store with the global ksdata
# Commment out because there is no way to select a disk right
# now without putting it in ksdata. Seems wrong?
#self.selected_disks = self.data.ignoredisk.onlyuse[:]
self . autopart = self . data . autopart . autopart
message = self . _update_summary ( )
# loop through the disks and present them.
for disk in self . disks :
2015-03-23 11:36:12 +00:00
disk_info = self . _format_disk_info ( disk )
c = CheckboxWidget ( title = " %i ) %s " % ( self . disks . index ( disk ) + 1 , disk_info ) ,
2013-01-23 17:28:19 +00:00
completed = ( disk . name in self . selected_disks ) )
self . _window + = [ c , " " ]
2015-03-23 11:36:12 +00:00
# if we have more than one disk, present an option to just
# select all disks
if len ( self . disks ) > 1 :
c = CheckboxWidget ( title = " %i ) %s " % ( len ( self . disks ) + 1 , _ ( " Select all " ) ) ,
completed = ( self . selection == len ( self . disks ) ) )
self . _window + = [ c , " " ]
2013-01-23 17:28:19 +00:00
self . _window + = [ TextWidget ( message ) , " " ]
return True
2015-03-23 11:36:12 +00:00
def _select_all_disks ( self ) :
""" Mark all disks as selected for use in partitioning. """
for disk in self . disks :
if disk . name not in self . selected_disks :
self . _update_disk_list ( disk )
def _format_disk_info ( self , disk ) :
""" Some specialized disks are difficult to identify in the storage
spoke , so add and return extra identifying information about them .
Since this is going to be ugly to do within the confines of the
CheckboxWidget , pre - format the display string right here .
"""
# show this info for all disks
format_str = " %s : %s ( %s ) " % ( disk . model , disk . size , disk . name )
disk_attrs = [ ]
# now check for/add info about special disks
if ( isinstance ( disk , MultipathDevice ) or isinstance ( disk , iScsiDiskDevice ) or isinstance ( disk , FcoeDiskDevice ) ) :
if hasattr ( disk , " wwid " ) :
disk_attrs . append ( disk . wwid )
elif isinstance ( disk , DASDDevice ) :
if hasattr ( disk , " busid " ) :
disk_attrs . append ( disk . busid )
2016-04-10 04:00:00 +00:00
elif isinstance ( disk , ZFCPDiskDevice ) :
2015-03-23 11:36:12 +00:00
if hasattr ( disk , " fcp_lun " ) :
disk_attrs . append ( disk . fcp_lun )
if hasattr ( disk , " wwpn " ) :
disk_attrs . append ( disk . wwpn )
if hasattr ( disk , " hba_id " ) :
disk_attrs . append ( disk . hba_id )
# now append all additional attributes to our string
for attr in disk_attrs :
format_str + = " , %s " % attr
return format_str
2013-01-23 17:28:19 +00:00
def input ( self , args , key ) :
""" Grab the disk choice and update things """
2016-04-10 04:00:00 +00:00
self . errors = [ ]
2013-01-23 17:28:19 +00:00
try :
2014-04-07 12:38:09 +00:00
keyid = int ( key ) - 1
2016-04-10 04:00:00 +00:00
if keyid < 0 :
return key
2015-03-23 11:36:12 +00:00
self . selection = keyid
if len ( self . disks ) > 1 and keyid == len ( self . disks ) :
self . _select_all_disks ( )
else :
self . _update_disk_list ( self . disks [ keyid ] )
2014-04-07 12:38:09 +00:00
return INPUT_PROCESSED
except ( ValueError , IndexError ) :
if key . lower ( ) == " c " :
if self . selected_disks :
2015-03-23 11:36:12 +00:00
# check selected disks to see if we have any unformatted DASDs
# if we're on s390x, since they need to be formatted before we
# can use them.
if arch . isS390 ( ) :
2016-04-10 04:00:00 +00:00
_disks = [ d for d in self . disks if d . name in self . selected_disks ]
to_format = self . storage . devicetree . make_unformatted_dasd_list ( _disks )
2015-03-23 11:36:12 +00:00
if to_format :
self . run_dasdfmt ( to_format )
return None
2016-04-10 04:00:00 +00:00
# make sure no containers were split up by the user's disk
# selection
self . errors . extend ( checkDiskSelection ( self . storage ,
self . selected_disks ) )
if self . errors :
# The disk selection has to make sense before we can
# proceed.
return None
2014-04-07 12:38:09 +00:00
newspoke = AutoPartSpoke ( self . app , self . data , self . storage ,
self . payload , self . instclass )
self . app . switch_screen_modal ( newspoke )
self . apply ( )
self . execute ( )
self . close ( )
return INPUT_PROCESSED
else :
return key
2013-01-23 17:28:19 +00:00
2015-03-23 11:36:12 +00:00
def run_dasdfmt ( self , to_format ) :
"""
This generates the list of DASDs requiring dasdfmt and runs dasdfmt
against them .
"""
# if the storage thread is running, wait on it to complete before taking
# any further actions on devices; most likely to occur if user has
# zerombr in their ks file
threadMgr . wait ( THREAD_STORAGE )
# ask user to verify they want to format if zerombr not in ks file
if not self . data . zerombr . zerombr :
# prepare our msg strings; copied directly from dasdfmt.glade
2015-05-30 11:20:59 +00:00
summary = _ ( " The following unformatted DASDs have been detected on your system. You can choose to format them now with dasdfmt or cancel to leave them unformatted. Unformatted DASDs cannot be used during installation. \n \n " )
2015-03-23 11:36:12 +00:00
warntext = _ ( " Warning: All storage changes made using the installer will be lost when you choose to format. \n \n Proceed to run dasdfmt? \n " )
2016-04-10 04:00:00 +00:00
displaytext = summary + " \n " . join ( " /dev/ " + d . name for d in to_format ) + " \n " + warntext
2015-03-23 11:36:12 +00:00
# now show actual prompt; note -- in cmdline mode, auto-answer for
# this is 'no', so unformatted DASDs will remain so unless zerombr
# is added to the ks file
question_window = YesNoDialog ( self . _app , displaytext )
self . _app . switch_screen_modal ( question_window )
if not question_window . answer :
# no? well fine then, back to the storage spoke with you;
return None
for disk in to_format :
try :
2016-04-10 04:00:00 +00:00
print ( _ ( " Formatting /dev/ %s . This may take a moment. " ) % disk . name )
blockdev . s390 . dasd_format ( disk . name )
except blockdev . S390Error as err :
2015-03-23 11:36:12 +00:00
# Log errors if formatting fails, but don't halt the installer
log . error ( str ( err ) )
continue
2013-01-23 17:28:19 +00:00
def apply ( self ) :
self . autopart = self . data . autopart . autopart
self . data . ignoredisk . onlyuse = self . selected_disks [ : ]
self . data . clearpart . drives = self . selected_disks [ : ]
2014-04-07 12:38:09 +00:00
if self . data . autopart . type is None :
self . data . autopart . type = AUTOPART_TYPE_LVM
2013-01-23 17:28:19 +00:00
if self . autopart :
self . clearPartType = CLEARPART_TYPE_ALL
else :
self . clearPartType = CLEARPART_TYPE_NONE
for disk in self . disks :
if disk . name not in self . selected_disks and \
disk in self . storage . devices :
self . storage . devicetree . hide ( disk )
elif disk . name in self . selected_disks and \
disk not in self . storage . devices :
self . storage . devicetree . unhide ( disk )
self . data . bootloader . location = " mbr "
2014-04-07 12:38:09 +00:00
if self . data . bootloader . bootDrive and \
self . data . bootloader . bootDrive not in self . selected_disks :
self . data . bootloader . bootDrive = " "
self . storage . bootloader . reset ( )
2013-01-23 17:28:19 +00:00
self . storage . config . update ( self . data )
# If autopart is selected we want to remove whatever has been
# created/scheduled to make room for autopart.
# If custom is selected, we want to leave alone any storage layout the
# user may have set up before now.
self . storage . config . clearNonExistent = self . data . autopart . autopart
def execute ( self ) :
print ( _ ( " Generating updated storage configuration " ) )
try :
doKickstartStorage ( self . storage , self . data , self . instclass )
2014-04-07 12:38:09 +00:00
except ( StorageError , KickstartValueError ) as e :
log . error ( " storage configuration failed: %s " , e )
2015-03-23 11:36:12 +00:00
print ( _ ( " storage configuration failed: %s " ) % e )
2013-01-23 17:28:19 +00:00
self . errors = [ str ( e ) ]
2014-04-07 12:38:09 +00:00
self . data . bootloader . bootDrive = " "
2013-01-23 17:28:19 +00:00
self . data . clearpart . type = CLEARPART_TYPE_ALL
self . data . clearpart . initAll = False
self . storage . config . update ( self . data )
2015-03-23 11:36:12 +00:00
self . storage . autoPartType = self . data . autopart . type
2013-01-23 17:28:19 +00:00
self . storage . reset ( )
2015-03-23 11:36:12 +00:00
# now set ksdata back to the user's specified config
applyDiskSelection ( self . storage , self . data , self . selected_disks )
2014-04-07 12:38:09 +00:00
except BootLoaderError as e :
log . error ( " BootLoader setup failed: %s " , e )
2015-03-23 11:36:12 +00:00
print ( _ ( " storage configuration failed: %s " ) % e )
2014-04-07 12:38:09 +00:00
self . errors = [ str ( e ) ]
self . data . bootloader . bootDrive = " "
2013-01-23 17:28:19 +00:00
else :
print ( _ ( " Checking storage configuration... " ) )
2015-03-23 11:36:12 +00:00
exns = sanity_check ( self . storage )
2016-04-10 04:00:00 +00:00
errors = [ str ( exn ) for exn in exns if isinstance ( exn , SanityError ) ]
warnings = [ str ( exn ) for exn in exns if isinstance ( exn , SanityWarning ) ]
2015-03-23 11:36:12 +00:00
( self . errors , self . warnings ) = ( errors , warnings )
2013-01-23 17:28:19 +00:00
for e in self . errors :
log . error ( e )
2015-03-23 11:36:12 +00:00
print ( e )
2014-04-07 12:38:09 +00:00
for w in self . warnings :
2015-03-23 11:36:12 +00:00
log . warning ( w )
print ( w )
finally :
resetCustomStorageData ( self . data )
self . _ready = True
2013-01-23 17:28:19 +00:00
def initialize ( self ) :
NormalTUISpoke . initialize ( self )
2014-04-07 12:38:09 +00:00
threadMgr . add ( AnacondaThread ( name = THREAD_STORAGE_WATCHER ,
2013-01-23 17:28:19 +00:00
target = self . _initialize ) )
self . selected_disks = self . data . ignoredisk . onlyuse [ : ]
# Probably need something here to track which disks are selected?
def _initialize ( self ) :
2014-04-07 12:38:09 +00:00
"""
Secondary initialize so wait for the storage thread to complete before
populating our disk list
"""
2013-01-23 17:28:19 +00:00
2014-04-07 12:38:09 +00:00
threadMgr . wait ( THREAD_STORAGE )
2013-01-23 17:28:19 +00:00
self . disks = sorted ( getDisks ( self . storage . devicetree ) ,
key = lambda d : d . name )
2014-04-07 12:38:09 +00:00
# if only one disk is available, go ahead and mark it as selected
if len ( self . disks ) == 1 :
self . _update_disk_list ( self . disks [ 0 ] )
2013-01-23 17:28:19 +00:00
self . _update_summary ( )
self . _ready = True
class AutoPartSpoke ( NormalTUISpoke ) :
2016-04-10 04:00:00 +00:00
""" Autopartitioning options are presented here.
. . inheritance - diagram : : AutoPartSpoke
: parts : 3
"""
2015-03-23 11:36:12 +00:00
title = N_ ( " Autopartitioning Options " )
category = SystemCategory
2013-01-23 17:28:19 +00:00
def __init__ ( self , app , data , storage , payload , instclass ) :
NormalTUISpoke . __init__ ( self , app , data , storage , payload , instclass )
self . clearPartType = self . data . clearpart . type
2014-04-07 12:38:09 +00:00
self . parttypelist = sorted ( PARTTYPES . keys ( ) )
2013-01-23 17:28:19 +00:00
@property
def indirect ( self ) :
return True
2016-04-10 04:00:00 +00:00
def refresh ( self , args = None ) :
2013-01-23 17:28:19 +00:00
NormalTUISpoke . refresh ( self , args )
# synchronize our local data store with the global ksdata
self . clearPartType = self . data . clearpart . type
# I dislike "is None", but bool(0) returns false :(
if self . clearPartType is None :
# Default to clearing everything.
self . clearPartType = CLEARPART_TYPE_ALL
2015-03-23 11:36:12 +00:00
for i , parttype in enumerate ( self . parttypelist ) :
c = CheckboxWidget ( title = " %i ) %s " % ( i + 1 , _ ( parttype ) ) ,
2014-04-07 12:38:09 +00:00
completed = ( PARTTYPES [ parttype ] == self . clearPartType ) )
2013-01-23 17:28:19 +00:00
self . _window + = [ c , " " ]
2014-04-07 12:38:09 +00:00
message = _ ( " Installation requires partitioning of your hard drive. Select what space to use for the install target. " )
2013-01-23 17:28:19 +00:00
self . _window + = [ TextWidget ( message ) , " " ]
return True
def apply ( self ) :
2014-04-07 12:38:09 +00:00
# kind of a hack, but if we're actually getting to this spoke, there
# is no doubt that we are doing autopartitioning, so set autopart to
# True. In the case of ks installs which may not have defined any
# partition options, autopart was never set to True, causing some
# issues. (rhbz#1001061)
self . data . autopart . autopart = True
2013-01-23 17:28:19 +00:00
self . data . clearpart . type = self . clearPartType
self . data . clearpart . initAll = True
def input ( self , args , key ) :
""" Grab the choice and update things """
2014-04-07 12:38:09 +00:00
try :
keyid = int ( key ) - 1
except ValueError :
if key . lower ( ) == " c " :
newspoke = PartitionSchemeSpoke ( self . app , self . data , self . storage ,
self . payload , self . instclass )
self . app . switch_screen_modal ( newspoke )
self . apply ( )
self . close ( )
return INPUT_PROCESSED
else :
return key
if 0 < = keyid < len ( self . parttypelist ) :
self . clearPartType = PARTTYPES [ self . parttypelist [ keyid ] ]
2013-01-23 17:28:19 +00:00
self . apply ( )
2014-04-07 12:38:09 +00:00
return INPUT_PROCESSED
class PartitionSchemeSpoke ( NormalTUISpoke ) :
""" Spoke to select what partitioning scheme to use on disk(s). """
2015-03-23 11:36:12 +00:00
title = N_ ( " Partition Scheme Options " )
category = SystemCategory
2014-04-07 12:38:09 +00:00
def __init__ ( self , app , data , storage , payload , instclass ) :
NormalTUISpoke . __init__ ( self , app , data , storage , payload , instclass )
2015-03-23 11:36:12 +00:00
self . partschemes = OrderedDict ( )
pre_select = self . data . autopart . type or DEFAULT_AUTOPART_TYPE
for i , item in enumerate ( AUTOPART_CHOICES ) :
self . partschemes [ item [ 0 ] ] = item [ 1 ]
if item [ 1 ] == pre_select :
self . _selection = i
2014-04-07 12:38:09 +00:00
@property
def indirect ( self ) :
return True
def refresh ( self , args = None ) :
NormalTUISpoke . refresh ( self , args )
schemelist = self . partschemes . keys ( )
2015-03-23 11:36:12 +00:00
for i , sch in enumerate ( schemelist ) :
box = CheckboxWidget ( title = " %i ) %s " % ( i + 1 , _ ( sch ) ) , completed = ( i == self . _selection ) )
2014-04-07 12:38:09 +00:00
self . _window + = [ box , " " ]
message = _ ( " Select a partition scheme configuration. " )
self . _window + = [ TextWidget ( message ) , " " ]
return True
def input ( self , args , key ) :
""" Grab the choice and update things. """
2013-01-23 17:28:19 +00:00
try :
2014-04-07 12:38:09 +00:00
keyid = int ( key ) - 1
except ValueError :
if key . lower ( ) == " c " :
self . apply ( )
self . close ( )
return INPUT_PROCESSED
else :
return key
2013-01-23 17:28:19 +00:00
2014-04-07 12:38:09 +00:00
if 0 < = keyid < len ( self . partschemes ) :
self . _selection = keyid
return INPUT_PROCESSED
def apply ( self ) :
""" Apply our selections. """
2016-04-10 04:00:00 +00:00
schemelist = list ( self . partschemes . values ( ) )
2015-03-23 11:36:12 +00:00
self . data . autopart . type = schemelist [ self . _selection ]