2018-02-12 12:54:27 +01:00
/ * Copyright ( c ) Citrix Systems , Inc .
* All rights reserved .
*
* Redistribution and use in source and binary forms ,
* with or without modification , are permitted provided
* that the following conditions are met :
*
* * Redistributions of source code must retain the above
* copyright notice , this list of conditions and the
* following disclaimer .
* * Redistributions in binary form must reproduce the above
* copyright notice , this list of conditions and the
* following disclaimer in the documentation and / or other
* materials provided with the distribution .
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
* CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES ,
* INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL ,
* SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING ,
* BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS
* INTERRUPTION ) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY ,
* WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT ( INCLUDING
* NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE .
* /
using System ;
using System.Collections.Generic ;
using System.IO ;
using System.Linq ;
using System.Reflection ;
using XenAdmin.Network ;
using XenAdmin.Core ;
using XenAPI ;
namespace XenAdmin.Actions
{
2018-06-27 19:05:00 +02:00
public class UploadSupplementalPackAction : AsyncAction , IByteProgressAction
2018-02-12 12:54:27 +01:00
{
private static readonly log4net . ILog log = log4net . LogManager . GetLogger ( System . Reflection . MethodBase . GetCurrentMethod ( ) . DeclaringType ) ;
private readonly string suppPackFilePath ;
2019-07-19 14:59:49 +02:00
private long _totalUpdateSize ;
2018-02-12 12:54:27 +01:00
private readonly List < Host > servers ;
2019-01-10 14:24:42 +01:00
private long totalCount ;
private long totalUploaded ;
2018-06-27 19:05:00 +02:00
private readonly string _updateName ;
2018-03-14 13:19:14 +01:00
2019-01-10 14:24:42 +01:00
public Dictionary < Host , SR > SrsWithUploadedUpdatesPerHost = new Dictionary < Host , SR > ( ) ;
public readonly Dictionary < Host , XenRef < VDI > > VdiRefsPerHost = new Dictionary < Host , XenRef < VDI > > ( ) ;
2018-03-14 13:19:14 +01:00
2019-01-10 14:24:42 +01:00
public string ByteProgressDescription { get ; set ; }
private Pool_update poolUpdate ;
2018-02-12 12:54:27 +01:00
public Pool_update PoolUpdate
{
get { return poolUpdate ; }
}
public UploadSupplementalPackAction ( IXenConnection connection , List < Host > selectedServers , string path , bool suppressHistory )
: base ( connection , null , Messages . SUPP_PACK_UPLOADING , suppressHistory )
{
2019-01-10 14:24:42 +01:00
Host = Helpers . GetMaster ( connection ) ? ? throw new NullReferenceException ( ) ;
2018-02-12 12:54:27 +01:00
ApiMethodsToRoleCheck . Add ( "VDI.create" ) ;
ApiMethodsToRoleCheck . Add ( "VDI.destroy" ) ;
ApiMethodsToRoleCheck . Add ( "VDI.set_other_config" ) ;
ApiMethodsToRoleCheck . Add ( "http/put_import_raw_vdi" ) ;
suppPackFilePath = path ;
2018-06-27 19:05:00 +02:00
_updateName = Path . GetFileNameWithoutExtension ( suppPackFilePath ) ;
2018-02-12 12:54:27 +01:00
servers = selectedServers ;
}
protected override void Run ( )
{
SafeToExit = false ;
2019-07-19 14:59:49 +02:00
_totalUpdateSize = new FileInfo ( suppPackFilePath ) . Length ;
2018-07-20 20:08:08 +02:00
var srList = SelectTargetSr ( ) ;
2018-02-12 12:54:27 +01:00
if ( srList . Count = = 0 )
2018-07-20 18:36:25 +02:00
throw new Exception ( Messages . HOTFIX_APPLY_ERROR_NO_SR ) ;
2018-02-12 12:54:27 +01:00
totalCount = srList . Count ;
foreach ( var sr in srList )
{
Result = UploadSupplementalPack ( sr ) ;
}
}
public override void RecomputeCanCancel ( )
{
CanCancel = ! Cancelling ;
}
private string UploadSupplementalPack ( SR sr )
{
2018-06-27 19:05:00 +02:00
this . Description = String . Format ( Messages . SUPP_PACK_UPLOADING_TO , _updateName , sr . Name ( ) ) ;
log . DebugFormat ( "Creating vdi of size {0} bytes on SR '{1}'" , _totalUpdateSize , sr . Name ( ) ) ;
2018-02-12 12:54:27 +01:00
VDI vdi = NewVDI ( sr ) ;
2018-07-20 18:36:25 +02:00
var vdiRef = VDI . create ( Session , vdi ) ;
2018-02-12 12:54:27 +01:00
Host localStorageHost = sr . GetStorageHost ( ) ;
string hostUrl ;
if ( localStorageHost = = null )
{
Uri uri = new Uri ( Session . Url ) ;
hostUrl = uri . Host ;
}
else
{
log . DebugFormat ( "SR is not shared -- redirecting to {0}" , localStorageHost . address ) ;
hostUrl = localStorageHost . address ;
}
log . DebugFormat ( "Using {0} for import" , hostUrl ) ;
2018-07-20 18:36:25 +02:00
string result ;
2018-02-12 12:54:27 +01:00
try
{
2018-07-20 18:36:25 +02:00
log . DebugFormat ( "Uploading file '{0}' to VDI '{1}' on SR '{2}'" , suppPackFilePath , vdi . Name ( ) , sr . Name ( ) ) ;
2018-02-12 12:54:27 +01:00
HTTP . UpdateProgressDelegate progressDelegate = delegate ( int percent )
{
2018-06-27 19:05:00 +02:00
var sr1 = sr ;
var descr = string . Format ( Messages . UPLOAD_PATCH_UPLOADING_TO_SR_PROGRESS_DESCRIPTION , _updateName , sr1 . Name ( ) ,
2018-06-28 17:28:57 +02:00
Util . DiskSizeString ( percent * _totalUpdateSize / 100 , "F1" ) , Util . DiskSizeString ( _totalUpdateSize ) ) ;
2018-06-27 19:05:00 +02:00
var actionPercent = ( int ) ( ( totalUploaded * 100 + percent ) / totalCount ) ;
ByteProgressDescription = descr ;
Tick ( actionPercent , descr ) ;
2018-02-12 12:54:27 +01:00
} ;
Session session = NewSession ( ) ;
2020-12-24 00:07:03 +01:00
RelatedTask = Task . create ( Session , "put_import_raw_vdi_task" , hostUrl ) ;
log . DebugFormat ( "HTTP PUTTING file from {0} to {1}" , suppPackFilePath , hostUrl ) ;
HTTP_actions . put_import_raw_vdi ( progressDelegate ,
( ) = > XenAdminConfigManager . Provider . ForcedExiting | | GetCancelling ( ) ,
XenAdminConfigManager . Provider . GetProxyTimeout ( true ) ,
hostUrl ,
XenAdminConfigManager . Provider . GetProxyFromSettings ( Connection ) ,
suppPackFilePath , RelatedTask . opaque_ref , session . opaque_ref , vdiRef . opaque_ref ) ;
PollToCompletion ( ) ;
result = Result ;
2018-02-12 12:54:27 +01:00
}
catch ( Exception ex )
{
2020-12-24 00:07:03 +01:00
PollToCompletion ( suppressFailures : true ) ;
2018-02-12 12:54:27 +01:00
if ( vdiRef ! = null )
{
try
{
2020-12-24 00:07:03 +01:00
log . ErrorFormat ( "Failed to import a virtual disk over HTTP. Deleting VDI '{0}' on a best effort basis." , vdiRef . opaque_ref ) ;
2018-07-20 18:36:25 +02:00
VDI . destroy ( Session , vdiRef ) ;
2018-02-12 12:54:27 +01:00
}
catch ( Exception removeEx )
{
2018-07-20 18:36:25 +02:00
log . Error ( "Failed to remove VDI." , removeEx ) ;
2018-02-12 12:54:27 +01:00
}
2017-03-16 12:41:14 +01:00
}
2020-12-24 00:07:03 +01:00
if ( ex is CancelledException | | ex is HTTP . CancelledException | | ex . InnerException is CancelledException )
throw new CancelledException ( ) ;
log . Error ( "Failed to import a virtual disk over HTTP" , ex ) ;
2017-03-16 12:41:14 +01:00
//after having tried to remove the VDI, the original exception is thrown for the UI
if ( ex is TargetInvocationException & & ex . InnerException ! = null )
throw ex . InnerException ;
2018-06-27 19:05:00 +02:00
else
throw ;
2018-02-12 12:54:27 +01:00
}
//introduce ISO for Ely and higher
if ( Helpers . ElyOrGreater ( Connection ) )
{
try
{
var poolUpdateRef = Pool_update . introduce ( Connection . Session , vdiRef ) ;
poolUpdate = Connection . WaitForCache ( poolUpdateRef ) ;
if ( poolUpdate = = null )
throw new Exception ( Messages . UPDATE_ERROR_INTRODUCE ) ; // This should not happen, because such case will result in a XAPI Failure. But this code has to be protected at this point.
}
2018-07-24 12:50:36 +02:00
catch ( Exception ex )
2018-02-12 12:54:27 +01:00
{
2018-07-24 12:50:36 +02:00
//clean-up the VDI we've just created
try
{
log . ErrorFormat ( "Deleting VDI '{0}' on a best effor basis." , vdiRef ) ;
VDI . destroy ( Session , vdiRef ) ;
}
catch ( Exception removeEx )
2018-02-12 12:54:27 +01:00
{
2018-07-24 12:50:36 +02:00
log . Error ( "Failed to remove VDI" , removeEx ) ;
}
var failure = ex as Failure ;
if ( failure ! = null & & failure . ErrorDescription ! = null & & failure . ErrorDescription . Count > 1 & & failure . ErrorDescription [ 0 ] = = Failure . UPDATE_ALREADY_EXISTS )
{
string uuidFound = failure . ErrorDescription [ 1 ] ;
poolUpdate = Connection . Cache . Pool_updates . FirstOrDefault ( pu = > string . Equals ( pu . uuid , uuidFound , StringComparison . InvariantCultureIgnoreCase ) ) ;
2018-02-12 12:54:27 +01:00
}
else
{
2018-07-24 12:50:36 +02:00
log . Error ( "Failed to introduce the update" , ex ) ;
poolUpdate = null ;
2018-02-12 12:54:27 +01:00
throw ;
}
}
}
else
{
poolUpdate = null ;
}
2019-01-10 14:24:42 +01:00
if ( localStorageHost ! = null )
VdiRefsPerHost . Add ( localStorageHost , vdiRef ) ;
else // shared SR
foreach ( var server in servers )
VdiRefsPerHost . Add ( server , vdiRef ) ;
2018-02-12 12:54:27 +01:00
totalUploaded + + ;
2019-01-10 14:24:42 +01:00
Description = string . Format ( Messages . SUPP_PACK_UPLOADED , sr . Name ( ) ) ;
2018-03-14 13:19:14 +01:00
foreach ( Host host in servers )
2019-01-10 14:24:42 +01:00
SrsWithUploadedUpdatesPerHost [ host ] = sr ;
2018-03-14 13:19:14 +01:00
2018-02-12 12:54:27 +01:00
return result ;
}
private VDI NewVDI ( SR sr )
{
VDI vdi = new VDI ( ) ;
vdi . Connection = Connection ;
vdi . read_only = false ;
vdi . SR = new XenRef < SR > ( sr ) ;
2018-06-27 19:05:00 +02:00
vdi . virtual_size = _totalUpdateSize ;
2018-02-12 12:54:27 +01:00
vdi . name_label = new FileInfo ( suppPackFilePath ) . Name ;
vdi . name_description = Helpers . ElyOrGreater ( Connection ) ? Messages . UPDATE_TEMP_VDI_DESCRIPTION : Messages . SUPP_PACK_TEMP_VDI_DESCRIPTION ;
vdi . sharable = false ;
vdi . type = vdi_type . user ;
vdi . SetVmHint ( "" ) ;
//mark the vdi as being a temporary supp pack iso
vdi . other_config = new Dictionary < string , string > { { "supp_pack_iso" , "true" } } ;
return vdi ;
}
2018-07-20 20:08:08 +02:00
private List < SR > SelectTargetSr ( )
2018-02-12 12:54:27 +01:00
{
2018-07-20 20:08:08 +02:00
/ * For ely or greater ( update ISOs ) we need an SR that can be seen from master ;
* that would be a shared SR or the master ' s local SR .
*
* For earlier ( supplemental packs ) we need an SR that can be seen from all hosts ;
* that would be a shared SR , otherwise we have to upload to each hosts ' s local SR
*
* The selection priority is default SRs over non - default and shared over local .
* /
2018-02-12 12:54:27 +01:00
SR defaultSr = Pool ! = null ? Pool . Connection . Resolve ( Pool . default_SR ) : null ;
2018-07-20 20:08:08 +02:00
var serversToConsider = Helpers . ElyOrGreater ( Connection )
? new List < Host > { Helpers . GetMaster ( Connection ) }
: new List < Host > ( servers ) ;
2018-02-12 12:54:27 +01:00
2018-07-20 20:08:08 +02:00
var srList = new List < SR > ( ) ;
foreach ( var host in serversToConsider )
2018-02-12 12:54:27 +01:00
{
2018-07-20 20:08:08 +02:00
var visibleSRs = Connection . Cache . SRs . Where ( sr = > CanStoreUpdateForHost ( sr , host ) ) . ToList ( ) ;
if ( defaultSr ! = null & & visibleSRs . Contains ( defaultSr ) & & ! srList . Contains ( defaultSr ) )
2018-02-12 12:54:27 +01:00
{
srList . Add ( defaultSr ) ;
2018-07-20 20:08:08 +02:00
continue ;
2018-02-12 12:54:27 +01:00
}
2018-07-20 20:08:08 +02:00
var sharedSr = visibleSRs . FirstOrDefault ( sr = > sr . shared ) ;
if ( sharedSr ! = null & & ! srList . Contains ( sharedSr ) )
2018-02-12 12:54:27 +01:00
{
2018-07-20 20:08:08 +02:00
srList . Add ( sharedSr ) ;
continue ;
2018-02-12 12:54:27 +01:00
}
2018-07-20 20:08:08 +02:00
if ( visibleSRs . Count > 0 )
srList . Add ( visibleSRs [ 0 ] ) ;
2018-02-12 12:54:27 +01:00
}
2018-07-20 20:08:08 +02:00
return srList ;
2018-02-12 12:54:27 +01:00
}
2018-07-20 20:08:08 +02:00
private bool CanStoreUpdateForHost ( SR sr , Host host )
2018-02-12 12:54:27 +01:00
{
2018-07-20 20:08:08 +02:00
if ( ! sr . SupportsVdiCreate ( ) )
return false ;
if ( sr . FreeSpace ( ) < _totalUpdateSize )
return false ;
var canBeSeen = false ;
foreach ( var pbdRef in sr . PBDs )
{
var pbd = Connection . Resolve ( pbdRef ) ;
if ( pbd ! = null & & pbd . currently_attached & & pbd . host . opaque_ref = = host . opaque_ref )
{
canBeSeen = true ;
break ;
}
}
return canBeSeen ;
2018-02-12 12:54:27 +01:00
}
2018-07-20 20:08:08 +02:00
}
}