// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : XZ.cs // Author(s) : Natalia Portillo // // Component : Filters. // // --[ Description ] ---------------------------------------------------------- // // Allow to open files that are compressed using xz. // // --[ 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 Aaru.CommonTypes.Enums; using Aaru.CommonTypes.Interfaces; using Aaru.Helpers; using Aaru.Helpers.IO; using SharpCompress.Compressors.Xz; namespace Aaru.Filters; /// /// Decompress xz files while reading public sealed class XZ : IFilter { Stream _dataStream; Stream _innerStream; void GuessSize() { DataForkLength = 0; // Seek to footer backwards size field _dataStream.Seek(-8, SeekOrigin.End); var tmp = new byte[4]; _dataStream.EnsureRead(tmp, 0, 4); uint backwardSize = (BitConverter.ToUInt32(tmp, 0) + 1) * 4; // Seek to first indexed record _dataStream.Seek(-12 - (backwardSize - 2), SeekOrigin.End); // Skip compressed size tmp = new byte[backwardSize - 2]; _dataStream.EnsureRead(tmp, 0, tmp.Length); ulong number = 0; int ignore = Decode(tmp, tmp.Length, ref number); // Get compressed size _dataStream.Seek(-12 - (backwardSize - 2 - ignore), SeekOrigin.End); tmp = new byte[backwardSize - 2 - ignore]; _dataStream.EnsureRead(tmp, 0, tmp.Length); Decode(tmp, tmp.Length, ref number); DataForkLength = (long)number; _dataStream.Seek(0, SeekOrigin.Begin); } static int Decode(byte[] buf, int sizeMax, ref ulong num) { switch(sizeMax) { case 0: return 0; case > 9: sizeMax = 9; break; } num = (ulong)(buf[0] & 0x7F); var i = 0; while((buf[i++] & 0x80) == 0x80) { if(i >= sizeMax || buf[i] == 0x00) return 0; num |= (ulong)(buf[i] & 0x7F) << i * 7; } return i; } #region IFilter Members /// public string Name => Localization.XZ_Name; /// public Guid Id => new("666A8617-0444-4C05-9F4F-DF0FD758D0D2"); /// public string Author => Authors.NataliaPortillo; /// public void Close() { _dataStream?.Close(); _dataStream = null; BasePath = null; } /// public string BasePath { get; private set; } /// public Stream GetDataForkStream() => _innerStream; /// public string Path => BasePath; /// public Stream GetResourceForkStream() => null; /// public bool HasResourceFork => false; /// public bool Identify(byte[] buffer) => buffer[0] == 0xFD && buffer[1] == 0x37 && buffer[2] == 0x7A && buffer[3] == 0x58 && buffer[4] == 0x5A && buffer[5] == 0x00 && buffer[^2] == 0x59 && buffer[^1] == 0x5A; /// public bool Identify(Stream stream) { var buffer = new byte[6]; var footer = new byte[2]; if(stream.Length < 8) return false; stream.Seek(0, SeekOrigin.Begin); stream.EnsureRead(buffer, 0, 6); stream.Seek(-2, SeekOrigin.End); stream.EnsureRead(footer, 0, 2); stream.Seek(0, SeekOrigin.Begin); return buffer[0] == 0xFD && buffer[1] == 0x37 && buffer[2] == 0x7A && buffer[3] == 0x58 && buffer[4] == 0x5A && buffer[5] == 0x00 && footer[0] == 0x59 && footer[1] == 0x5A; } /// public bool Identify(string path) { if(!File.Exists(path)) return false; var stream = new FileStream(path, FileMode.Open, FileAccess.Read); var buffer = new byte[6]; var footer = new byte[2]; if(stream.Length < 8) return false; stream.Seek(0, SeekOrigin.Begin); stream.EnsureRead(buffer, 0, 6); stream.Seek(-2, SeekOrigin.End); stream.EnsureRead(footer, 0, 2); stream.Seek(0, SeekOrigin.Begin); return buffer[0] == 0xFD && buffer[1] == 0x37 && buffer[2] == 0x7A && buffer[3] == 0x58 && buffer[4] == 0x5A && buffer[5] == 0x00 && footer[0] == 0x59 && footer[1] == 0x5A; } /// public ErrorNumber Open(byte[] buffer) { _dataStream = new MemoryStream(buffer); BasePath = null; CreationTime = DateTime.UtcNow; LastWriteTime = CreationTime; GuessSize(); _innerStream = new ForcedSeekStream(DataForkLength, _dataStream); return ErrorNumber.NoError; } /// public ErrorNumber Open(Stream stream) { _dataStream = stream; BasePath = null; CreationTime = DateTime.UtcNow; LastWriteTime = CreationTime; GuessSize(); _innerStream = new ForcedSeekStream(DataForkLength, _dataStream); return ErrorNumber.NoError; } /// public ErrorNumber Open(string path) { _dataStream = new FileStream(path, FileMode.Open, FileAccess.Read); BasePath = System.IO.Path.GetFullPath(path); var fi = new FileInfo(path); CreationTime = fi.CreationTimeUtc; LastWriteTime = fi.LastWriteTimeUtc; GuessSize(); _innerStream = new ForcedSeekStream(DataForkLength, _dataStream); return ErrorNumber.NoError; } /// public DateTime CreationTime { get; private set; } /// public long DataForkLength { get; private set; } /// public DateTime LastWriteTime { get; private set; } /// public long Length => DataForkLength; /// public long ResourceForkLength => 0; /// public string Filename { get { if(BasePath?.EndsWith(".xz", StringComparison.InvariantCultureIgnoreCase) == true) return BasePath[..^3]; return BasePath?.EndsWith(".xzip", StringComparison.InvariantCultureIgnoreCase) == true ? BasePath[..^5] : BasePath; } } /// public string ParentFolder => System.IO.Path.GetDirectoryName(BasePath); #endregion }