/* 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.IO; using System.Reflection; using XenAdmin.Network; using XenAdmin.Core; using XenAPI; namespace XenAdmin.Actions { public class UploadPatchAction : AsyncAction, IByteProgressAction { private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); private Pool_patch _patch; private readonly string retailPatchPath; private readonly string _patchName; private readonly bool deleteFileOnCancel; private readonly long _totalPatchSize; public UploadPatchAction(IXenConnection connection, string path, bool suppressHistory, bool deleteFileOnCancel) : base(connection, null, Messages.UPLOADING_PATCH, suppressHistory) { Host coordinator = Helpers.GetCoordinator(connection); this.deleteFileOnCancel = deleteFileOnCancel; if (coordinator == null) throw new NullReferenceException(); ApiMethodsToRoleCheck.Add("pool.sync_database"); ApiMethodsToRoleCheck.Add("http/put_pool_patch_upload"); retailPatchPath = path; _patchName = Path.GetFileNameWithoutExtension(retailPatchPath); _totalPatchSize = (new FileInfo(path)).Length; Host = coordinator; } public string ByteProgressDescription { get; set; } public Pool_patch Patch { get { return _patch; } } protected override void Run() { SafeToExit = false; try { _patch = UploadRetailPatch(); log.InfoFormat("File '{0}' uploaded to server '{1}'", _patchName, Host.Name()); Tick(100, string.Format(Messages.UPLOAD_PATCH_UPLOADED_DESCRIPTION, _patchName)); } catch (Failure f) { // Need to check if the patch already exists. // If it does then we use it and ignore the PATCH_ALREADY_EXISTS error (CA-110209). if (f.ErrorDescription != null && f.ErrorDescription.Count > 1 && f.ErrorDescription[0] == Failure.PATCH_ALREADY_EXISTS) { string uuid = f.ErrorDescription[1]; Session session = Host.Connection.DuplicateSession(); _patch = Connection.Resolve(Pool_patch.get_by_uuid(session, uuid)); } else throw; } } public override void RecomputeCanCancel() { CanCancel = !Cancelling; } private Pool_patch UploadRetailPatch() { Session session = NewSession(); Host coordinator = Helpers.GetCoordinator(Connection); log.InfoFormat("Uploading file '{0}' to server '{1}'", _patchName, coordinator.Name()); this.Description = string.Format(Messages.UPLOAD_PATCH_UPLOADING_DESCRIPTION, _patchName); try { RelatedTask = Task.create(session, "put_pool_patch_upload_task", coordinator.address); log.DebugFormat("HTTP PUTTING file from {0} to {1}", retailPatchPath, coordinator.address); HTTP_actions.put_pool_patch_upload(UpdateProgress, () => XenAdminConfigManager.Provider.ForcedExiting || GetCancelling(), XenAdminConfigManager.Provider.GetProxyTimeout(true), coordinator.address, XenAdminConfigManager.Provider.GetProxyFromSettings(Connection), retailPatchPath, RelatedTask.opaque_ref, session.opaque_ref); PollToCompletion(); return Connection.WaitForCache(new XenRef(new XenRef(Result))); } catch (Exception e) { PollToCompletion(suppressFailures: true); if (e is TargetInvocationException ex) { if (ex.InnerException == null) throw; else throw ex.InnerException; } if (e is CancelledException || e is HTTP.CancelledException || e.InnerException is CancelledException) { if (deleteFileOnCancel && File.Exists(retailPatchPath)) File.Delete(retailPatchPath); throw new CancelledException(); } throw; } } private void UpdateProgress(int percent) { var descr = string.Format(Messages.UPLOAD_PATCH_UPLOADING_PROGRESS_DESCRIPTION, _patchName, Util.DiskSizeString(percent * _totalPatchSize / 100, "F1"), Util.DiskSizeString(_totalPatchSize)); ByteProgressDescription = descr; Tick(percent, descr); } } public interface IByteProgressAction { string ByteProgressDescription { get; set; } } }