mirror of
https://github.com/SabreTools/SabreTools.Serialization.git
synced 2026-04-05 22:01:33 +00:00
This change looks dramatic, but it's just separating out the already-split namespaces into separate top-level folders. In theory, every single one could be built into their own Nuget package. `SabreTools.Serialization` still builds the normal Nuget package that is used by all other projects and includes all namespaces.
177 lines
6.5 KiB
C#
177 lines
6.5 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Text.RegularExpressions;
|
|
#if NET462_OR_GREATER || NETCOREAPP || NETSTANDARD2_0_OR_GREATER
|
|
using SharpCompress.Archives;
|
|
using SharpCompress.Archives.Zip;
|
|
using SharpCompress.Readers;
|
|
#endif
|
|
|
|
#pragma warning disable SYSLIB1045 // Convert to 'GeneratedRegexAttribute'
|
|
namespace SabreTools.Wrappers
|
|
{
|
|
public partial class PKZIP : IExtractable
|
|
{
|
|
/// <inheritdoc/>
|
|
public bool Extract(string outputDirectory, bool includeDebug)
|
|
=> Extract(outputDirectory, lookForHeader: false, includeDebug);
|
|
|
|
/// <inheritdoc cref="Extract(string, bool)"/>
|
|
public bool Extract(string outputDirectory, bool lookForHeader, bool includeDebug)
|
|
{
|
|
if (_dataSource is null || !_dataSource.CanRead)
|
|
return false;
|
|
|
|
#if NET462_OR_GREATER || NETCOREAPP || NETSTANDARD2_0_OR_GREATER
|
|
try
|
|
{
|
|
var readerOptions = new ReaderOptions() { LookForHeader = lookForHeader };
|
|
var zipFile = (ZipArchive)ZipArchive.OpenArchive(_dataSource, readerOptions);
|
|
|
|
// If the file exists
|
|
if (!string.IsNullOrEmpty(Filename) && File.Exists(Filename!))
|
|
{
|
|
// Find all file parts
|
|
FileInfo[] parts = [.. ArchiveFactory.GetFileParts(new FileInfo(Filename))];
|
|
|
|
// If there are multiple parts
|
|
if (parts.Length > 1)
|
|
zipFile = (ZipArchive)ZipArchive.OpenArchive(parts, readerOptions);
|
|
|
|
// Try to read the file path if no entries are found
|
|
else if (zipFile.Entries.Count == 0)
|
|
zipFile = (ZipArchive)ZipArchive.OpenArchive(parts, readerOptions);
|
|
|
|
// If the archive is somehow null
|
|
if (zipFile is null)
|
|
return false;
|
|
}
|
|
|
|
foreach (var entry in zipFile.Entries)
|
|
{
|
|
try
|
|
{
|
|
// If the entry is a directory
|
|
if (entry.IsDirectory)
|
|
continue;
|
|
|
|
// If the entry has an invalid key
|
|
if (entry.Key is null)
|
|
continue;
|
|
|
|
// If the entry is partial due to an incomplete multi-part archive, skip it
|
|
if (!entry.IsComplete)
|
|
continue;
|
|
|
|
// If the entry is password-protected, skip it
|
|
if (entry.IsEncrypted)
|
|
{
|
|
if (includeDebug) Console.WriteLine($"File {entry.Key} in zip is password-protected!");
|
|
|
|
continue;
|
|
}
|
|
|
|
// Ensure directory separators are consistent
|
|
string filename = entry.Key;
|
|
if (Path.DirectorySeparatorChar == '\\')
|
|
filename = filename.Replace('/', '\\');
|
|
else if (Path.DirectorySeparatorChar == '/')
|
|
filename = filename.Replace('\\', '/');
|
|
|
|
// Ensure the full output directory exists
|
|
filename = Path.Combine(outputDirectory, filename);
|
|
var directoryName = Path.GetDirectoryName(filename);
|
|
if (directoryName is not null && !Directory.Exists(directoryName))
|
|
Directory.CreateDirectory(directoryName);
|
|
|
|
entry.WriteToFile(filename);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
if (includeDebug) Console.Error.WriteLine(ex);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
if (includeDebug) Console.Error.WriteLine(ex);
|
|
return false;
|
|
}
|
|
#else
|
|
Console.WriteLine("Extraction is not supported for this framework!");
|
|
Console.WriteLine();
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
/// <summary>
|
|
/// Try to find all parts of the archive, if possible
|
|
/// </summary>
|
|
/// <param name="firstPart">Path of the first archive part</param>
|
|
/// <returns>List of all found parts, if possible</returns>
|
|
public static List<string> FindParts(string firstPart)
|
|
{
|
|
// Define the regex patterns
|
|
const string zipPattern = @"^(.*\.)(zipx?|zx?[0-9]+)$";
|
|
const string genericPattern = @"^(.*\.)([0-9]+)$";
|
|
|
|
// Ensure the full path is available
|
|
firstPart = Path.GetFullPath(firstPart);
|
|
string filename = Path.GetFileName(firstPart);
|
|
string? directory = Path.GetDirectoryName(firstPart);
|
|
|
|
// Make the output list
|
|
List<string> parts = [];
|
|
|
|
// Determine which pattern is being used
|
|
Match match;
|
|
Func<int, string> nextPartFunc;
|
|
if (Regex.IsMatch(filename, zipPattern, RegexOptions.IgnoreCase))
|
|
{
|
|
match = Regex.Match(filename, zipPattern, RegexOptions.IgnoreCase);
|
|
nextPartFunc = (i) =>
|
|
{
|
|
return string.Concat(
|
|
match.Groups[1].Value,
|
|
Regex.Replace(match.Groups[2].Value, @"[^xz]", ""),
|
|
$"{i:D2}");
|
|
};
|
|
}
|
|
else if (Regex.IsMatch(filename, genericPattern, RegexOptions.IgnoreCase))
|
|
{
|
|
match = Regex.Match(filename, genericPattern, RegexOptions.IgnoreCase);
|
|
nextPartFunc = (i) =>
|
|
{
|
|
return string.Concat(
|
|
match.Groups[1].Value,
|
|
$"{i + 1}".PadLeft(match.Groups[2].Value.Length, '0')
|
|
);
|
|
};
|
|
}
|
|
else
|
|
{
|
|
return [firstPart];
|
|
}
|
|
|
|
// Loop and add the files
|
|
parts.Add(firstPart);
|
|
for (int i = 1; ; i++)
|
|
{
|
|
string nextPart = nextPartFunc(i);
|
|
if (directory is not null)
|
|
nextPart = Path.Combine(directory, nextPart);
|
|
|
|
if (!File.Exists(nextPart))
|
|
break;
|
|
|
|
parts.Add(nextPart);
|
|
}
|
|
|
|
return parts;
|
|
}
|
|
}
|
|
}
|