2014-01-29 01:53:11 +04:00
# Copyright (C) 2014 Red Hat, Inc.
#
2018-04-04 16:35:41 +03:00
# This work is licensed under the GNU GPLv2 or later.
2018-03-20 22:00:02 +03:00
# See the COPYING file in the top-level directory.
2014-01-29 01:53:11 +04:00
import logging
import os
from gi . repository import Gtk
import virtinst
2014-09-13 00:10:45 +04:00
from . import uiutil
from . baseclass import vmmGObjectUI
2014-01-29 01:53:11 +04:00
class vmmAddStorage ( vmmGObjectUI ) :
__gsignals__ = {
2018-03-15 15:10:09 +03:00
" browse-clicked " : ( vmmGObjectUI . RUN_FIRST , None , [ object ] ) ,
" storage-toggled " : ( vmmGObjectUI . RUN_FIRST , None , [ object ] )
2014-01-29 01:53:11 +04:00
}
def __init__ ( self , conn , builder , topwin ) :
vmmGObjectUI . __init__ ( self , " addstorage.ui " , None ,
builder = builder , topwin = topwin )
self . conn = conn
self . builder . connect_signals ( {
2015-09-14 01:01:39 +03:00
" on_storage_browse_clicked " : self . _browse_storage ,
" on_storage_select_toggled " : self . _toggle_storage_select ,
2014-01-29 01:53:11 +04:00
} )
2015-09-14 01:01:39 +03:00
self . top_box = self . widget ( " storage-box " )
2014-01-29 01:53:11 +04:00
def _cleanup ( self ) :
self . conn = None
##########################
# Initialization methods #
##########################
def _get_default_dir ( self ) :
2015-05-03 02:33:16 +03:00
return virtinst . StoragePool . get_default_dir ( self . conn . get_backend ( ) )
2014-01-29 01:53:11 +04:00
def _get_ideal_path_info ( self , name ) :
path = self . _get_default_dir ( )
2014-02-05 02:30:24 +04:00
fmt = self . conn . get_default_storage_format ( )
suffix = virtinst . StorageVolume . get_file_extension_for_format ( fmt )
return ( path , name , suffix or " .img " )
2014-01-29 01:53:11 +04:00
def _get_ideal_path ( self , name ) :
target , name , suffix = self . _get_ideal_path_info ( name )
return os . path . join ( target , name ) + suffix
def _host_disk_space ( self ) :
pool = self . conn . get_default_pool ( )
path = self . _get_default_dir ( )
avail = 0
if pool and pool . is_active ( ) :
2014-09-12 20:08:44 +04:00
# Rate limit this, since it can be spammed at dialog startup time
2015-04-09 21:42:25 +03:00
if pool . secs_since_last_refresh ( ) > 10 :
2014-09-12 20:08:44 +04:00
pool . refresh ( )
2014-01-29 01:53:11 +04:00
avail = int ( pool . get_available ( ) )
elif not self . conn . is_remote ( ) and os . path . exists ( path ) :
vfs = os . statvfs ( os . path . dirname ( path ) )
2017-10-11 14:35:47 +03:00
avail = vfs . f_frsize * vfs . f_bavail
2014-01-29 01:53:11 +04:00
return float ( avail / 1024.0 / 1024.0 / 1024.0 )
def _update_host_space ( self ) :
widget = self . widget ( " phys-hd-label " )
try :
max_storage = self . _host_disk_space ( )
2017-07-24 11:26:48 +03:00
except Exception :
2014-01-29 01:53:11 +04:00
logging . exception ( " Error determining host disk space " )
widget . set_markup ( " " )
return
def pretty_storage ( size ) :
2014-06-16 07:56:02 +04:00
return " %.1f GiB " % float ( size )
2014-01-29 01:53:11 +04:00
2015-08-09 21:01:01 +03:00
hd_label = ( _ ( " %s available in the default location " ) %
2014-01-29 01:53:11 +04:00
pretty_storage ( max_storage ) )
hd_label = ( " <span color= ' #484848 ' > %s </span> " % hd_label )
widget . set_markup ( hd_label )
def _check_default_pool_active ( self ) :
default_pool = self . conn . get_default_pool ( )
if default_pool and not default_pool . is_active ( ) :
res = self . err . yes_no ( _ ( " Default pool is not active. " ) ,
_ ( " Storage pool ' %s ' is not active. "
" Would you like to start the pool "
" now? " ) % default_pool . get_name ( ) )
if not res :
return False
# Try to start the pool
try :
default_pool . start ( )
logging . info ( " Started pool ' %s ' " , default_pool . get_name ( ) )
2017-05-05 19:47:21 +03:00
except Exception as e :
2014-01-29 01:53:11 +04:00
return self . err . show_err ( _ ( " Could not start storage_pool "
" ' %s ' : %s " ) %
( default_pool . get_name ( ) , str ( e ) ) )
return True
##############
# Public API #
##############
@staticmethod
2014-02-11 03:08:59 +04:00
def check_path_search ( src , conn , path ) :
2014-01-29 01:53:11 +04:00
skip_paths = src . config . get_perms_fix_ignore ( )
2018-10-12 01:52:45 +03:00
searchdata = virtinst . DeviceDisk . check_path_search (
2014-02-11 03:08:59 +04:00
conn . get_backend ( ) , path )
2018-10-12 01:52:45 +03:00
broken_paths = searchdata . fixlist [ : ]
2014-02-11 03:08:59 +04:00
for p in broken_paths [ : ] :
2014-01-29 01:53:11 +04:00
if p in skip_paths :
broken_paths . remove ( p )
if not broken_paths :
return
logging . debug ( " No search access for dirs: %s " , broken_paths )
resp , chkres = src . err . warn_chkbox (
_ ( " The emulator may not have search permissions "
" for the path ' %s ' . " ) % path ,
_ ( " Do you want to correct this now? " ) ,
_ ( " Don ' t ask about these directories again. " ) ,
buttons = Gtk . ButtonsType . YES_NO )
if chkres :
src . config . add_perms_fix_ignore ( broken_paths )
if not resp :
return
logging . debug ( " Attempting to correct permission issues. " )
2018-10-12 01:52:45 +03:00
errors = virtinst . DeviceDisk . fix_path_search (
conn . get_backend ( ) , searchdata )
2014-01-29 01:53:11 +04:00
if not errors :
return
errmsg = _ ( " Errors were encountered changing permissions for the "
" following directories: " )
details = " "
2017-06-15 15:18:26 +03:00
for p , error in errors . items ( ) :
if p not in broken_paths :
2014-01-29 01:53:11 +04:00
continue
2017-06-15 15:18:26 +03:00
details + = " %s : %s \n " % ( p , error )
2015-10-02 19:37:51 +03:00
details + = " \n It is very likely the VM will fail to start up. "
2014-01-29 01:53:11 +04:00
logging . debug ( " Permission errors: \n %s " , details )
ignore , chkres = src . err . err_chkbox ( errmsg , details ,
_ ( " Don ' t ask about these directories again. " ) )
if chkres :
2017-10-11 14:35:46 +03:00
src . config . add_perms_fix_ignore ( list ( errors . keys ( ) ) )
2014-01-29 01:53:11 +04:00
def reset_state ( self ) :
self . _update_host_space ( )
2015-09-14 01:01:39 +03:00
self . widget ( " storage-create " ) . set_active ( True )
2015-09-14 02:30:10 +03:00
self . widget ( " storage-size " ) . set_value ( 20 )
2015-09-14 01:01:39 +03:00
self . widget ( " storage-entry " ) . set_text ( " " )
self . widget ( " storage-create-box " ) . set_sensitive ( True )
2014-01-29 01:53:11 +04:00
storage_tooltip = None
can_storage = ( not self . conn . is_remote ( ) or
self . conn . is_storage_capable ( ) )
2015-09-14 01:01:39 +03:00
use_storage = self . widget ( " storage-select " )
storage_area = self . widget ( " storage-box " )
2014-01-29 01:53:11 +04:00
storage_area . set_sensitive ( can_storage )
if not can_storage :
storage_tooltip = _ ( " Connection does not support storage "
" management. " )
use_storage . set_sensitive ( True )
storage_area . set_tooltip_text ( storage_tooltip or " " )
def get_default_path ( self , name , collidelist = None ) :
collidelist = collidelist or [ ]
pool = self . conn . get_default_pool ( )
default_dir = self . _get_default_dir ( )
def path_exists ( p ) :
return os . path . exists ( p ) or p in collidelist
if not pool :
# Use old generating method
origf = os . path . join ( default_dir , name + " .img " )
f = origf
n = 1
while path_exists ( f ) and n < 100 :
f = os . path . join ( default_dir , name +
" - " + str ( n ) + " .img " )
n + = 1
if path_exists ( f ) :
f = origf
path = f
else :
target , ignore , suffix = self . _get_ideal_path_info ( name )
# Sanitize collidelist to work with the collision checker
newcollidelist = [ ]
for c in collidelist :
if c and os . path . dirname ( c ) == pool . get_target_path ( ) :
newcollidelist . append ( os . path . basename ( c ) )
path = virtinst . StorageVolume . find_free_name (
pool . get_backend ( ) , name ,
suffix = suffix , collidelist = newcollidelist )
path = os . path . join ( target , path )
return path
def is_default_storage ( self ) :
2015-09-14 01:01:39 +03:00
return self . widget ( " storage-create " ) . get_active ( )
2014-01-29 01:53:11 +04:00
2015-09-14 02:05:02 +03:00
def validate_storage ( self , vmname ,
path = None , device = " disk " , collidelist = None ) :
2015-09-14 01:42:10 +03:00
if self . is_default_storage ( ) :
# Make sure default pool is running
2014-01-29 01:53:11 +04:00
ret = self . _check_default_pool_active ( )
if not ret :
return False
2015-09-13 23:02:35 +03:00
if path is None :
2015-09-14 01:42:10 +03:00
if self . is_default_storage ( ) :
path = self . get_default_path ( vmname , collidelist or [ ] )
2015-09-13 23:02:35 +03:00
else :
2015-09-14 01:01:39 +03:00
path = self . widget ( " storage-entry " ) . get_text ( ) . strip ( )
2015-09-13 23:02:35 +03:00
if not path and device in [ " disk " , " lun " ] :
return self . err . val_err ( _ ( " A storage path must be specified. " ) )
2018-03-20 19:18:35 +03:00
disk = virtinst . DeviceDisk ( self . conn . get_backend ( ) )
2015-09-13 23:02:35 +03:00
disk . path = path or None
disk . device = device
if disk . wants_storage_creation ( ) :
pool = disk . get_parent_pool ( )
2015-09-14 01:42:10 +03:00
size = uiutil . spin_get_helper ( self . widget ( " storage-size " ) )
sparse = False
2018-03-20 19:18:35 +03:00
vol_install = virtinst . DeviceDisk . build_vol_install (
2015-09-13 23:02:35 +03:00
disk . conn , os . path . basename ( disk . path ) , pool ,
2015-09-14 02:05:02 +03:00
size , sparse )
2015-09-13 23:02:35 +03:00
disk . set_vol_install ( vol_install )
2015-09-14 02:05:02 +03:00
fmt = self . conn . get_default_storage_format ( )
2018-09-06 03:13:56 +03:00
if disk . get_vol_install ( ) . supports_property ( " format " ) :
2015-09-14 02:05:02 +03:00
logging . debug ( " Using default prefs format= %s for path= %s " ,
fmt , disk . path )
disk . get_vol_install ( ) . format = fmt
else :
logging . debug ( " path= %s can not use default prefs format= %s , "
" not setting it " , disk . path , fmt )
2015-09-13 23:02:35 +03:00
disk . validate ( )
2014-01-29 01:53:11 +04:00
return disk
def validate_disk_object ( self , disk ) :
isfatal , errmsg = disk . is_size_conflict ( )
if not isfatal and errmsg :
# Fatal errors are reported when setting 'size'
res = self . err . ok_cancel ( _ ( " Not Enough Free Space " ) , errmsg )
if not res :
return False
# Disk collision
names = disk . is_conflict_disk ( )
if names :
res = self . err . yes_no (
_ ( ' Disk " %s " is already in use by other guests %s ' ) %
( disk . path , names ) ,
_ ( " Do you really want to use the disk? " ) )
if not res :
return False
2014-02-11 03:08:59 +04:00
self . check_path_search ( self , self . conn , disk . path )
2014-01-29 01:53:11 +04:00
#############
# Listeners #
#############
def _browse_storage ( self , ignore ) :
2015-09-14 01:01:39 +03:00
self . emit ( " browse-clicked " , self . widget ( " storage-entry " ) )
2014-01-29 01:53:11 +04:00
def _toggle_storage_select ( self , src ) :
act = src . get_active ( )
2015-09-14 01:01:39 +03:00
self . widget ( " storage-browse-box " ) . set_sensitive ( act )
2014-01-29 01:53:11 +04:00
self . emit ( " storage-toggled " , src )