2025-09-08 01:14:58 +01:00
|
|
|
using System;
|
2025-09-08 02:05:07 +01:00
|
|
|
using System.IO;
|
2025-09-08 01:14:08 +01:00
|
|
|
using Aaru.CommonTypes.Enums;
|
2025-09-08 02:05:07 +01:00
|
|
|
using Aaru.CommonTypes.Interfaces;
|
2025-09-08 01:54:06 +01:00
|
|
|
using Aaru.CommonTypes.Structs;
|
2025-09-08 02:05:07 +01:00
|
|
|
using Aaru.Compression;
|
|
|
|
|
using Aaru.Filters;
|
|
|
|
|
using Aaru.Helpers.IO;
|
2025-09-08 01:54:06 +01:00
|
|
|
using FileAttributes = System.IO.FileAttributes;
|
2025-09-08 01:14:08 +01:00
|
|
|
|
|
|
|
|
namespace Aaru.Archives;
|
|
|
|
|
|
|
|
|
|
public sealed partial class Ha
|
|
|
|
|
{
|
|
|
|
|
#region IArchive Members
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public ErrorNumber GetFilename(int entryNumber, out string fileName)
|
|
|
|
|
{
|
|
|
|
|
fileName = null;
|
|
|
|
|
|
|
|
|
|
if(!Opened) return ErrorNumber.NotOpened;
|
|
|
|
|
|
|
|
|
|
if(entryNumber < 0 || entryNumber >= _entries.Count) return ErrorNumber.OutOfRange;
|
|
|
|
|
|
|
|
|
|
fileName = _entries[entryNumber].Filename;
|
|
|
|
|
|
|
|
|
|
return ErrorNumber.NoError;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 01:14:58 +01:00
|
|
|
/// <inheritdoc />
|
|
|
|
|
public ErrorNumber GetEntryNumber(string fileName, bool caseInsensitiveMatch, out int entryNumber)
|
|
|
|
|
{
|
|
|
|
|
entryNumber = -1;
|
|
|
|
|
|
|
|
|
|
if(!Opened) return ErrorNumber.NotOpened;
|
|
|
|
|
|
|
|
|
|
if(entryNumber < 0 || entryNumber >= _entries.Count) return ErrorNumber.OutOfRange;
|
|
|
|
|
|
|
|
|
|
StringComparison comparison = caseInsensitiveMatch
|
|
|
|
|
? StringComparison.CurrentCultureIgnoreCase
|
|
|
|
|
: StringComparison.CurrentCulture;
|
|
|
|
|
|
|
|
|
|
for(int i = 0, count = _entries.Count; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
if(!_entries[i].Filename.Equals(fileName, comparison)) continue;
|
|
|
|
|
|
|
|
|
|
entryNumber = i;
|
|
|
|
|
|
|
|
|
|
return ErrorNumber.NoError;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ErrorNumber.NoSuchFile;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 01:23:08 +01:00
|
|
|
/// <inheritdoc />
|
|
|
|
|
public ErrorNumber GetCompressedSize(int entryNumber, out long length)
|
|
|
|
|
{
|
|
|
|
|
length = -1;
|
|
|
|
|
|
|
|
|
|
if(!Opened) return ErrorNumber.NotOpened;
|
|
|
|
|
|
|
|
|
|
if(entryNumber < 0 || entryNumber >= _entries.Count) return ErrorNumber.OutOfRange;
|
|
|
|
|
|
|
|
|
|
length = _entries[entryNumber].Compressed;
|
|
|
|
|
|
|
|
|
|
return ErrorNumber.NoError;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public ErrorNumber GetUncompressedSize(int entryNumber, out long length)
|
|
|
|
|
{
|
|
|
|
|
length = -1;
|
|
|
|
|
|
|
|
|
|
if(!Opened) return ErrorNumber.NotOpened;
|
|
|
|
|
|
|
|
|
|
if(entryNumber < 0 || entryNumber >= _entries.Count) return ErrorNumber.OutOfRange;
|
|
|
|
|
|
|
|
|
|
length = _entries[entryNumber].Uncompressed;
|
|
|
|
|
|
|
|
|
|
return ErrorNumber.NoError;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 01:25:49 +01:00
|
|
|
/// <inheritdoc />
|
|
|
|
|
public ErrorNumber GetAttributes(int entryNumber, out FileAttributes attributes)
|
|
|
|
|
{
|
|
|
|
|
attributes = FileAttributes.None;
|
|
|
|
|
|
|
|
|
|
if(!Opened) return ErrorNumber.NotOpened;
|
|
|
|
|
|
|
|
|
|
if(entryNumber < 0 || entryNumber >= _entries.Count) return ErrorNumber.OutOfRange;
|
|
|
|
|
|
|
|
|
|
attributes = _entries[entryNumber].Attributes;
|
|
|
|
|
|
|
|
|
|
return ErrorNumber.NoError;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 01:54:06 +01:00
|
|
|
/// <inheritdoc />
|
|
|
|
|
public ErrorNumber Stat(int entryNumber, out FileEntryInfo stat)
|
|
|
|
|
{
|
|
|
|
|
stat = null;
|
|
|
|
|
|
|
|
|
|
if(!Opened) return ErrorNumber.NotOpened;
|
|
|
|
|
|
|
|
|
|
if(entryNumber < 0 || entryNumber >= _entries.Count) return ErrorNumber.OutOfRange;
|
|
|
|
|
|
|
|
|
|
Entry entry = _entries[entryNumber];
|
|
|
|
|
|
|
|
|
|
stat = new FileEntryInfo
|
|
|
|
|
{
|
|
|
|
|
Attributes = CommonTypes.Structs.FileAttributes.None,
|
|
|
|
|
Blocks = entry.Uncompressed / 512,
|
|
|
|
|
BlockSize = 512,
|
|
|
|
|
Length = entry.Uncompressed,
|
|
|
|
|
LastWriteTime = entry.LastWrite,
|
|
|
|
|
LastWriteTimeUtc = entry.LastWrite
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if(entry.Uncompressed % 512 != 0) stat.Blocks++;
|
|
|
|
|
|
|
|
|
|
if(entry.Attributes.HasFlag(FileAttributes.Directory))
|
|
|
|
|
stat.Attributes |= CommonTypes.Structs.FileAttributes.Directory;
|
|
|
|
|
else
|
|
|
|
|
stat.Attributes |= CommonTypes.Structs.FileAttributes.File;
|
|
|
|
|
|
|
|
|
|
if(entry.Attributes.HasFlag(FileAttributes.Archive))
|
|
|
|
|
stat.Attributes |= CommonTypes.Structs.FileAttributes.Archive;
|
|
|
|
|
|
|
|
|
|
if(entry.Attributes.HasFlag(FileAttributes.Hidden))
|
|
|
|
|
stat.Attributes |= CommonTypes.Structs.FileAttributes.Hidden;
|
|
|
|
|
|
|
|
|
|
if(entry.Attributes.HasFlag(FileAttributes.ReadOnly))
|
|
|
|
|
stat.Attributes |= CommonTypes.Structs.FileAttributes.ReadOnly;
|
|
|
|
|
|
|
|
|
|
if(entry.Attributes.HasFlag(FileAttributes.System))
|
|
|
|
|
stat.Attributes |= CommonTypes.Structs.FileAttributes.System;
|
|
|
|
|
|
|
|
|
|
stat.Mode = entry.Mode;
|
|
|
|
|
stat.UID = entry.Uid;
|
|
|
|
|
stat.GID = entry.Gid;
|
|
|
|
|
|
|
|
|
|
return ErrorNumber.NoError;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 02:05:07 +01:00
|
|
|
/// <inheritdoc />
|
|
|
|
|
public ErrorNumber GetEntry(int entryNumber, out IFilter filter)
|
|
|
|
|
{
|
|
|
|
|
filter = null;
|
|
|
|
|
|
|
|
|
|
if(!Opened) return ErrorNumber.NotOpened;
|
|
|
|
|
|
|
|
|
|
if(entryNumber < 0 || entryNumber >= _entries.Count) return ErrorNumber.OutOfRange;
|
|
|
|
|
|
|
|
|
|
switch(_entries[entryNumber].Method)
|
|
|
|
|
{
|
|
|
|
|
case Method.Directory:
|
|
|
|
|
return ErrorNumber.IsDirectory;
|
|
|
|
|
case >= Method.Special:
|
|
|
|
|
return ErrorNumber.InvalidArgument;
|
|
|
|
|
case > Method.HSC:
|
|
|
|
|
return ErrorNumber.NotSupported;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Stream stream = new OffsetStream(new NonClosableStream(_stream),
|
|
|
|
|
_entries[entryNumber].DataOffset,
|
|
|
|
|
_entries[entryNumber].DataOffset + _entries[entryNumber].Compressed);
|
|
|
|
|
|
|
|
|
|
if(_entries[entryNumber].Uncompressed == 0) stream = new MemoryStream([]);
|
|
|
|
|
|
|
|
|
|
if(_entries[entryNumber].Method == Method.ASC)
|
|
|
|
|
stream = new HaStream(stream, _entries[entryNumber].Uncompressed, HaStream.HaMethod.ASC);
|
|
|
|
|
|
|
|
|
|
if(_entries[entryNumber].Method == Method.HSC)
|
|
|
|
|
stream = new HaStream(stream, _entries[entryNumber].Uncompressed, HaStream.HaMethod.HSC);
|
|
|
|
|
|
|
|
|
|
filter = new ZZZNoFilter();
|
|
|
|
|
ErrorNumber errno = filter.Open(stream);
|
|
|
|
|
|
|
|
|
|
if(errno == ErrorNumber.NoError) return ErrorNumber.NoError;
|
|
|
|
|
|
|
|
|
|
stream.Close();
|
|
|
|
|
|
|
|
|
|
return errno;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-08 01:14:08 +01:00
|
|
|
#endregion
|
|
|
|
|
}
|