mirror of
https://github.com/xcp-ng/xenadmin.git
synced 2025-01-22 00:00:40 +01:00
979f88a90c
If the tar contains huge files, this causes memory spikes, even OutOfMemoryException. Instead, copy the file from the input to the output stream in chunks. For this purpose the sanitisation code was moved to XenCenterLib to avoid exposing the tar internals to the UI project. Also, made some minor improvements in error handling, clean up, and logging. Signed-off-by: Konstantina Chremmou <konstantina.chremmou@citrix.com>
228 lines
8.9 KiB
C#
228 lines
8.9 KiB
C#
/* 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 XenCenterLib.Archive;
|
|
|
|
|
|
namespace XenAdmin.Actions
|
|
{
|
|
public class ZipStatusReportAction : AsyncAction
|
|
{
|
|
private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
|
|
|
|
/// <summary>
|
|
/// The folder containing the raw files as downloaded from the server
|
|
/// </summary>
|
|
private readonly string _inputTempFolder;
|
|
|
|
/// <summary>
|
|
/// Temporary folder in which we assemble the log files from the server
|
|
/// before repackaging them in a single zip file.
|
|
/// </summary>
|
|
private string _extractTempDir;
|
|
|
|
/// <summary>
|
|
/// The destination zip file for the repackaged server log files
|
|
/// </summary>
|
|
private readonly string _destFile;
|
|
|
|
/// <summary>
|
|
/// Creates a new instance of the action for zipping downloaded server log files
|
|
/// </summary>
|
|
/// <param name="tempFolder">Temporary folder to store the downloaded logs</param>
|
|
/// <param name="destFile">The target file to store the compressed result</param>
|
|
/// <param name="suppressHistory">Whether to suppress history in the Events TabPage</param>
|
|
public ZipStatusReportAction(string tempFolder, string destFile, bool suppressHistory=true)
|
|
: base(null, Messages.BUGTOOL_SAVING, Messages.BUGTOOL_SAVING, suppressHistory)
|
|
{
|
|
_inputTempFolder = tempFolder;
|
|
_destFile = destFile;
|
|
}
|
|
|
|
protected override void Run()
|
|
{
|
|
do
|
|
{
|
|
_extractTempDir = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
|
|
} while (Directory.Exists(_extractTempDir));
|
|
|
|
Directory.CreateDirectory(_extractTempDir);
|
|
|
|
try
|
|
{
|
|
// Calculate total bytes to save
|
|
long bytesToCompress = 0, bytesToExtract = 0, bytesExtracted = 0;
|
|
|
|
var files = Directory.GetFiles(_inputTempFolder);
|
|
|
|
foreach (string inputFile in files)
|
|
bytesToExtract += new FileInfo(inputFile).Length;
|
|
|
|
foreach (string inputFile in files)
|
|
{
|
|
if (inputFile.ToLowerInvariant().EndsWith(".tar"))
|
|
{
|
|
// Sanitize and un-tar each of the raw server tars to the temp extraction directory
|
|
|
|
string outFilename = inputFile.Substring(0, inputFile.Length - 4);
|
|
if (outFilename.Length == 0)
|
|
outFilename = Path.GetRandomFileName();
|
|
string outputDir = Path.Combine(_extractTempDir, Path.GetFileName(outFilename));
|
|
|
|
string sanitizedTar = Path.GetTempFileName();
|
|
TarSanitization.SanitizeTarForWindows(inputFile, sanitizedTar, CheckCancellation);
|
|
|
|
using (FileStream fs = File.OpenRead(sanitizedTar))
|
|
using (ArchiveIterator tarIterator = ArchiveFactory.Reader(ArchiveFactory.Type.Tar, fs))
|
|
{
|
|
Directory.CreateDirectory(outputDir);
|
|
tarIterator.ExtractAllContents(outputDir);
|
|
bytesToCompress += Core.Helpers.GetDirSize(new DirectoryInfo(outputDir));
|
|
}
|
|
|
|
try
|
|
{
|
|
File.Delete(sanitizedTar);
|
|
}
|
|
catch
|
|
{
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Just copy vanilla input files unmodified to the temp directory
|
|
string outputFile = Path.Combine(_extractTempDir, Path.GetFileName(inputFile));
|
|
File.Copy(inputFile, outputFile);
|
|
bytesToCompress += new FileInfo(outputFile).Length;
|
|
}
|
|
|
|
bytesExtracted += new FileInfo(inputFile).Length;
|
|
File.Delete(inputFile);
|
|
PercentComplete = (int)(50.0 * bytesExtracted / bytesToExtract);
|
|
CheckCancellation();
|
|
}
|
|
|
|
// Now zip up all the temporarily extracted files into a single zip file for the user
|
|
log.DebugFormat("Packing {0} of bug report files into zip file {1}",
|
|
Util.DiskSizeString(bytesToCompress), _destFile);
|
|
|
|
ZipToOutputFile(_extractTempDir, CheckCancellation, p => PercentComplete = 50 + p / 2);
|
|
CleanupFiles();
|
|
PercentComplete = 100;
|
|
}
|
|
catch (CancelledException)
|
|
{
|
|
CleanupFiles(true);
|
|
throw;
|
|
}
|
|
catch (Exception exn)
|
|
{
|
|
try
|
|
{
|
|
log.Error("Failed to package sanitized server status report: ", exn);
|
|
log.Debug("Attempting to package raw downloaded server files.");
|
|
ZipToOutputFile(_inputTempFolder, CheckCancellation);
|
|
}
|
|
catch(CancelledException)
|
|
{
|
|
CleanupFiles(true);
|
|
throw;
|
|
}
|
|
catch
|
|
{
|
|
log.Debug("Failed to package raw downloaded server files.");
|
|
}
|
|
|
|
throw new Exception(Messages.STATUS_REPORT_ZIP_FAILED);
|
|
}
|
|
}
|
|
|
|
private void CheckCancellation()
|
|
{
|
|
if (Cancelling)
|
|
throw new CancelledException();
|
|
}
|
|
|
|
private void ZipToOutputFile(string folderToZip, Action cancellingDelegate = null, Action<int> progressDelegate = null)
|
|
{
|
|
using (ArchiveWriter zip = ArchiveFactory.Writer(ArchiveFactory.Type.Zip, File.OpenWrite(_destFile)))
|
|
zip.CreateArchive(folderToZip, cancellingDelegate, progressDelegate);
|
|
}
|
|
|
|
private void CleanupFiles(bool deleteDestFile = false)
|
|
{
|
|
try
|
|
{
|
|
log.Debug("Deleting temporary directory with raw downloaded server files");
|
|
Directory.Delete(_inputTempFolder, true);
|
|
}
|
|
catch (Exception exn)
|
|
{
|
|
log.Warn("Could not delete temporary directory with raw downloaded server files", exn);
|
|
}
|
|
|
|
try
|
|
{
|
|
log.Debug("Deleting directory with temporarily extracted files");
|
|
Directory.Delete(_extractTempDir, true);
|
|
}
|
|
catch (Exception exn)
|
|
{
|
|
log.Warn("Could not delete directory with temporarily extracted files", exn);
|
|
}
|
|
|
|
try
|
|
{
|
|
if (deleteDestFile)
|
|
{
|
|
log.Debug("Deleting destination zip file");
|
|
File.Delete(_destFile);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Warn("Could not delete destination zip file", ex);
|
|
}
|
|
}
|
|
|
|
public override void RecomputeCanCancel()
|
|
{
|
|
CanCancel = !Cancelling && !IsCompleted;
|
|
}
|
|
|
|
protected override void CancelRelatedTask()
|
|
{
|
|
}
|
|
}
|
|
}
|