2016-02-18 02:53:35 +03:00
# Copyright (C) 2015-2016 Red Hat, Inc. All rights reserved.
#
# 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.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
import subprocess
from . import cfg
2021-06-11 18:35:31 +03:00
from . cmdhandler import options_to_cli_args , LvmExecutionMeta , call_lvm
2016-02-18 02:53:35 +03:00
import dbus
2017-03-15 18:58:41 +03:00
from . utils import pv_range_append , pv_dest_ranges , log_error , log_debug , \
2021-06-11 18:35:31 +03:00
mt_async_call
from . request import RequestEntry
2016-11-11 18:46:12 +03:00
import threading
2017-06-02 20:25:01 +03:00
import time
2021-06-11 18:35:31 +03:00
import traceback
2016-02-18 02:53:35 +03:00
def pv_move_lv_cmd ( move_options , lv_full_name ,
pv_source , pv_source_range , pv_dest_range_list ) :
cmd = [ ' pvmove ' , ' -i ' , ' 1 ' ]
cmd . extend ( options_to_cli_args ( move_options ) )
if lv_full_name :
cmd . extend ( [ ' -n ' , lv_full_name ] )
pv_range_append ( cmd , pv_source , * pv_source_range )
pv_dest_ranges ( cmd , pv_dest_range_list )
return cmd
def lv_merge_cmd ( merge_options , lv_full_name ) :
cmd = [ ' lvconvert ' , ' --merge ' , ' -i ' , ' 1 ' ]
cmd . extend ( options_to_cli_args ( merge_options ) )
cmd . append ( lv_full_name )
return cmd
2021-06-11 18:35:31 +03:00
def _load_wrapper ( ignored ) :
cfg . load ( )
2017-03-15 18:58:41 +03:00
2017-06-02 20:25:01 +03:00
2021-06-11 18:35:31 +03:00
def _move_callback ( job_state , line_str ) :
try :
if line_str . count ( ' : ' ) == 2 :
( device , ignore , percentage ) = line_str . split ( ' : ' )
2017-06-02 20:25:01 +03:00
2021-06-11 18:35:31 +03:00
job_state . Percent = int ( round (
float ( percentage . strip ( ) [ : - 1 ] ) , 1 ) )
2016-11-04 02:25:12 +03:00
2021-06-11 18:35:31 +03:00
# While the move is in progress we need to periodically update
# the state to reflect where everything is at. we will do this
# by scheduling the load to occur in the main work queue.
r = RequestEntry (
- 1 , _load_wrapper , ( " _move_callback: load " , ) , None , None , False )
cfg . worker_q . put ( r )
except ValueError :
log_error ( " Trying to parse percentage which failed for %s " % line_str )
2016-11-04 02:25:12 +03:00
2021-06-11 18:35:31 +03:00
def _move_merge ( interface_name , command , job_state ) :
# We need to execute these command stand alone by forking & exec'ing
# the command always as we will be getting periodic output from them on
# the status of the long running operation.
2017-03-15 22:19:55 +03:00
2022-08-10 01:43:00 +03:00
meta = LvmExecutionMeta ( time . time ( ) , 0 , command )
2022-08-10 01:55:27 +03:00
cfg . flightrecorder . add ( meta )
2016-11-04 02:25:12 +03:00
2021-06-11 18:35:31 +03:00
ec , stdout , stderr = call_lvm ( command , line_cb = _move_callback ,
cb_data = job_state )
2022-08-10 01:43:00 +03:00
ended = time . time ( )
meta . completed ( ended , ec , stdout , stderr )
2017-06-02 20:25:01 +03:00
2021-06-11 18:35:31 +03:00
if ec == 0 :
2016-11-04 02:25:12 +03:00
job_state . Percent = 100
else :
2016-06-28 20:07:21 +03:00
raise dbus . exceptions . DBusException (
interface_name ,
2021-06-11 18:35:31 +03:00
' Exit code %s , stderr = %s ' % ( str ( ec ) , stderr ) )
2016-02-18 02:53:35 +03:00
2016-06-28 20:07:21 +03:00
cfg . load ( )
return ' / '
2016-02-18 02:53:35 +03:00
def move ( interface_name , lv_name , pv_src_obj , pv_source_range ,
2016-06-28 20:07:21 +03:00
pv_dests_and_ranges , move_options , job_state ) :
2016-02-18 02:53:35 +03:00
"""
Common code for the pvmove handling .
: param interface_name : What dbus interface we are providing for
: param lv_name : Optional ( None or name of LV to move )
: param pv_src_obj : dbus object patch for source PV
: param pv_source_range : ( 0 , 0 to ignore , else start , end segments )
: param pv_dests_and_ranges : Array of PV object paths and start / end segs
: param move_options : Hash with optional arguments
2016-06-28 20:07:21 +03:00
: param job_state : Used to convey information about jobs between processes
: return : ' / ' When complete , the empty object path
2016-02-18 02:53:35 +03:00
"""
pv_dests = [ ]
pv_src = cfg . om . get_object_by_path ( pv_src_obj )
if pv_src :
# Check to see if we are handling a move to a specific
# destination(s)
if len ( pv_dests_and_ranges ) :
for pr in pv_dests_and_ranges :
pv_dbus_obj = cfg . om . get_object_by_path ( pr [ 0 ] )
if not pv_dbus_obj :
raise dbus . exceptions . DBusException (
interface_name ,
' PV Destination ( %s ) not found ' % pr [ 0 ] )
pv_dests . append ( ( pv_dbus_obj . lvm_id , pr [ 1 ] , pr [ 2 ] ) )
cmd = pv_move_lv_cmd ( move_options ,
lv_name ,
pv_src . lvm_id ,
pv_source_range ,
pv_dests )
2016-06-28 20:07:21 +03:00
return _move_merge ( interface_name , cmd , job_state )
2016-02-18 02:53:35 +03:00
else :
raise dbus . exceptions . DBusException (
interface_name , ' pv_src_obj ( %s ) not found ' % pv_src_obj )
2016-06-28 20:07:21 +03:00
def merge ( interface_name , lv_uuid , lv_name , merge_options , job_state ) :
2016-02-18 02:53:35 +03:00
# Make sure we have a dbus object representing it
dbo = cfg . om . get_object_by_uuid_lvm_id ( lv_uuid , lv_name )
if dbo :
cmd = lv_merge_cmd ( merge_options , dbo . lvm_id )
2016-06-28 20:07:21 +03:00
return _move_merge ( interface_name , cmd , job_state )
2016-02-18 02:53:35 +03:00
else :
raise dbus . exceptions . DBusException (
interface_name ,
' LV with uuid %s and name %s not present! ' % ( lv_uuid , lv_name ) )
2016-11-11 18:46:12 +03:00
def _run_cmd ( req ) :
log_debug (
" _run_cmd: Running method: %s with args %s " %
( str ( req . method ) , str ( req . arguments ) ) )
req . run_cmd ( )
log_debug ( " _run_cmd: complete! " )
def cmd_runner ( request ) :
2017-09-25 23:18:43 +03:00
t = threading . Thread ( target = _run_cmd , args = ( request , ) ,
name = " cmd_runner %s " % str ( request . method ) )
2016-11-11 18:46:12 +03:00
t . start ( )