/* Copyright (c) Cloud Software Group, Inc. * * 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 ICSharpCode.SharpZipLib.Tar; using XenCenterLib.Compression; namespace XenCenterLib.Archive { public class TarArchiveIterator : ArchiveIterator { private TarInputStream tarStream; private CompressionStream compressionStream; private TarEntry tarEntry; private bool disposed; /// /// Parameterless constructor needed by tests /// public TarArchiveIterator() { } public TarArchiveIterator(Stream compressedTarFile, CompressionFactory.Type compressionType) { if (compressionType == CompressionFactory.Type.Gz) compressionStream = CompressionFactory.Reader(CompressionFactory.Type.Gz, compressedTarFile); else throw new NotSupportedException($"Type {compressionType} is not supported by ArchiveIterator"); tarStream = new TarInputStream(compressionStream); disposed = false; } public TarArchiveIterator(Stream tarFile) { tarStream = new TarInputStream(tarFile); disposed = false; } public override void SetBaseStream(Stream stream) { tarStream = new TarInputStream(stream); disposed = false; } ~TarArchiveIterator() { Dispose(); } public override bool HasNext() { tarEntry = tarStream.GetNextEntry(); if (tarEntry == null) return false; return true; } public override string CurrentFileName() { if (tarEntry == null) return String.Empty; return tarEntry.Name; } public override long CurrentFileSize() { if (tarEntry == null) return 0; return tarEntry.Size; } public override DateTime CurrentFileModificationTime() { if (tarEntry == null) return new DateTime(); return tarEntry.ModTime; } public override bool IsDirectory() { if (tarEntry == null) return false; return tarEntry.IsDirectory; } public override void ExtractCurrentFile(Stream extractedFileContents, Action cancellingDelegate) { if (IsDirectory()) return; byte[] buffer = new byte[32768]; int count; while ((count = tarStream.Read(buffer, 0, buffer.Length)) > 0) { cancellingDelegate?.Invoke(); extractedFileContents.Write(buffer, 0, count); } } protected override void Dispose(bool disposing) { base.Dispose(disposing); if(disposing) { if(!disposed) { tarStream?.Dispose(); compressionStream?.Dispose(); disposed = true; } } } public override bool VerifyCurrentFileAgainstDigest(string algorithmName, byte[] digest) { return StreamUtilities.VerifyAgainstDigest(tarStream, CurrentFileSize(), algorithmName, digest); } } }