// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : GZip.cs // Author(s) : Natalia Portillo // // Component : Filters. // // --[ Description ] ---------------------------------------------------------- // // Allow to open files that are compressed using gzip. // // --[ License ] -------------------------------------------------------------- // // This library is free software; you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as // published by the Free Software Foundation; either version 2.1 of the // License, or (at your option) any later version. // // This library is distributed in the hope that it will be useful, but // WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, see . // // ---------------------------------------------------------------------------- // Copyright © 2011-2025 Natalia Portillo // ****************************************************************************/ using System; using System.IO; using System.IO.Compression; using Aaru.CommonTypes.Enums; using Aaru.CommonTypes.Interfaces; using Aaru.Helpers; using Aaru.Helpers.IO; namespace Aaru.Filters; /// /// Decompress gzip files while reading public sealed class GZip : IFilter { Stream _dataStream; uint _decompressedSize; Stream _zStream; #region IFilter Members /// public string Name => Localization.GZip_Name; /// public Guid Id => new("F4996661-4A29-42C9-A2C7-3904EF40F3B0"); /// public string Author => Authors.NataliaPortillo; /// public void Close() { _dataStream?.Close(); _dataStream = null; BasePath = null; } /// public string BasePath { get; private set; } /// public Stream GetDataForkStream() => _zStream; /// public string Path => BasePath; /// public Stream GetResourceForkStream() => null; /// public bool HasResourceFork => false; /// public bool Identify(byte[] buffer) => buffer[0] == 0x1F && buffer[1] == 0x8B && buffer[2] == 0x08; /// public bool Identify(Stream stream) { var buffer = new byte[3]; stream.Seek(0, SeekOrigin.Begin); stream.EnsureRead(buffer, 0, 3); stream.Seek(0, SeekOrigin.Begin); return buffer[0] == 0x1F && buffer[1] == 0x8B && buffer[2] == 0x08; } /// public bool Identify(string path) { if(!File.Exists(path)) return false; var stream = new FileStream(path, FileMode.Open, FileAccess.Read); var buffer = new byte[3]; stream.Seek(0, SeekOrigin.Begin); stream.EnsureRead(buffer, 0, 3); stream.Seek(0, SeekOrigin.Begin); return buffer[0] == 0x1F && buffer[1] == 0x8B && buffer[2] == 0x08; } /// public ErrorNumber Open(byte[] buffer) { var mtimeB = new byte[4]; var isizeB = new byte[4]; _dataStream = new MemoryStream(buffer); BasePath = null; _dataStream.Seek(4, SeekOrigin.Begin); _dataStream.EnsureRead(mtimeB, 0, 4); _dataStream.Seek(-4, SeekOrigin.End); _dataStream.EnsureRead(isizeB, 0, 4); _dataStream.Seek(0, SeekOrigin.Begin); var mtime = BitConverter.ToUInt32(mtimeB, 0); var isize = BitConverter.ToUInt32(isizeB, 0); _decompressedSize = isize; CreationTime = DateHandlers.UnixUnsignedToDateTime(mtime); LastWriteTime = CreationTime; _zStream = new ForcedSeekStream(_decompressedSize, _dataStream, CompressionMode.Decompress); return ErrorNumber.NoError; } /// public ErrorNumber Open(Stream stream) { var mtimeB = new byte[4]; var isizeB = new byte[4]; _dataStream = stream; BasePath = null; _dataStream.Seek(4, SeekOrigin.Begin); _dataStream.EnsureRead(mtimeB, 0, 4); _dataStream.Seek(-4, SeekOrigin.End); _dataStream.EnsureRead(isizeB, 0, 4); _dataStream.Seek(0, SeekOrigin.Begin); var mtime = BitConverter.ToUInt32(mtimeB, 0); var isize = BitConverter.ToUInt32(isizeB, 0); _decompressedSize = isize; CreationTime = DateHandlers.UnixUnsignedToDateTime(mtime); LastWriteTime = CreationTime; _zStream = new ForcedSeekStream(_decompressedSize, _dataStream, CompressionMode.Decompress); return ErrorNumber.NoError; } /// public ErrorNumber Open(string path) { var mtimeB = new byte[4]; var isizeB = new byte[4]; _dataStream = new FileStream(path, FileMode.Open, FileAccess.Read); BasePath = System.IO.Path.GetFullPath(path); _dataStream.Seek(4, SeekOrigin.Begin); _dataStream.EnsureRead(mtimeB, 0, 4); _dataStream.Seek(-4, SeekOrigin.End); _dataStream.EnsureRead(isizeB, 0, 4); _dataStream.Seek(0, SeekOrigin.Begin); var mtime = BitConverter.ToUInt32(mtimeB, 0); var isize = BitConverter.ToUInt32(isizeB, 0); _decompressedSize = isize; var fi = new FileInfo(path); CreationTime = fi.CreationTimeUtc; LastWriteTime = DateHandlers.UnixUnsignedToDateTime(mtime); _zStream = new ForcedSeekStream(_decompressedSize, _dataStream, CompressionMode.Decompress); return ErrorNumber.NoError; } /// public DateTime CreationTime { get; private set; } /// public long DataForkLength => _decompressedSize; /// public DateTime LastWriteTime { get; private set; } /// public long Length => _decompressedSize; /// public long ResourceForkLength => 0; /// public string Filename { get { if(BasePath?.EndsWith(".gz", StringComparison.InvariantCultureIgnoreCase) == true) return BasePath[..^3]; return BasePath?.EndsWith(".gzip", StringComparison.InvariantCultureIgnoreCase) == true ? BasePath[..^5] : BasePath; } } /// public string ParentFolder => System.IO.Path.GetDirectoryName(BasePath); #endregion }