mirror of
https://github.com/adamhathcock/sharpcompress.git
synced 2026-02-04 05:25:00 +00:00
127 lines
4.2 KiB
C#
127 lines
4.2 KiB
C#
using System;
|
|
using System.IO;
|
|
using System.Runtime.InteropServices;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace SharpCompress.Common;
|
|
|
|
internal static partial class ExtractionMethods
|
|
{
|
|
/// <summary>
|
|
/// Gets the appropriate StringComparison for path checks based on the file system.
|
|
/// Windows uses case-insensitive file systems, while Unix-like systems use case-sensitive file systems.
|
|
/// </summary>
|
|
private static StringComparison PathComparison =>
|
|
RuntimeInformation.IsOSPlatform(OSPlatform.Windows)
|
|
? StringComparison.OrdinalIgnoreCase
|
|
: StringComparison.Ordinal;
|
|
|
|
/// <summary>
|
|
/// Extract to specific directory, retaining filename
|
|
/// </summary>
|
|
public static void WriteEntryToDirectory(
|
|
IEntry entry,
|
|
string destinationDirectory,
|
|
ExtractionOptions? options,
|
|
Action<string, ExtractionOptions?> write
|
|
)
|
|
{
|
|
string destinationFileName;
|
|
var fullDestinationDirectoryPath = Path.GetFullPath(destinationDirectory);
|
|
|
|
//check for trailing slash.
|
|
if (
|
|
fullDestinationDirectoryPath[fullDestinationDirectoryPath.Length - 1]
|
|
!= Path.DirectorySeparatorChar
|
|
)
|
|
{
|
|
fullDestinationDirectoryPath += Path.DirectorySeparatorChar;
|
|
}
|
|
|
|
if (!Directory.Exists(fullDestinationDirectoryPath))
|
|
{
|
|
throw new ExtractionException(
|
|
$"Directory does not exist to extract to: {fullDestinationDirectoryPath}"
|
|
);
|
|
}
|
|
|
|
options ??= new ExtractionOptions() { Overwrite = true };
|
|
|
|
var file = Path.GetFileName(entry.Key.NotNull("Entry Key is null")).NotNull("File is null");
|
|
file = Utility.ReplaceInvalidFileNameChars(file);
|
|
if (options.ExtractFullPath)
|
|
{
|
|
var folder = Path.GetDirectoryName(entry.Key.NotNull("Entry Key is null"))
|
|
.NotNull("Directory is null");
|
|
var destdir = Path.GetFullPath(Path.Combine(fullDestinationDirectoryPath, folder));
|
|
|
|
if (!Directory.Exists(destdir))
|
|
{
|
|
if (!destdir.StartsWith(fullDestinationDirectoryPath, PathComparison))
|
|
{
|
|
throw new ExtractionException(
|
|
"Entry is trying to create a directory outside of the destination directory."
|
|
);
|
|
}
|
|
|
|
Directory.CreateDirectory(destdir);
|
|
}
|
|
destinationFileName = Path.Combine(destdir, file);
|
|
}
|
|
else
|
|
{
|
|
destinationFileName = Path.Combine(fullDestinationDirectoryPath, file);
|
|
}
|
|
|
|
if (!entry.IsDirectory)
|
|
{
|
|
destinationFileName = Path.GetFullPath(destinationFileName);
|
|
|
|
if (!destinationFileName.StartsWith(fullDestinationDirectoryPath, PathComparison))
|
|
{
|
|
throw new ExtractionException(
|
|
"Entry is trying to write a file outside of the destination directory."
|
|
);
|
|
}
|
|
write(destinationFileName, options);
|
|
}
|
|
else if (options.ExtractFullPath && !Directory.Exists(destinationFileName))
|
|
{
|
|
Directory.CreateDirectory(destinationFileName);
|
|
}
|
|
}
|
|
|
|
public static void WriteEntryToFile(
|
|
IEntry entry,
|
|
string destinationFileName,
|
|
ExtractionOptions? options,
|
|
Action<string, FileMode> openAndWrite
|
|
)
|
|
{
|
|
if (entry.LinkTarget != null)
|
|
{
|
|
if (options?.WriteSymbolicLink is null)
|
|
{
|
|
throw new ExtractionException(
|
|
"Entry is a symbolic link but ExtractionOptions.WriteSymbolicLink delegate is null"
|
|
);
|
|
}
|
|
options.WriteSymbolicLink(destinationFileName, entry.LinkTarget);
|
|
}
|
|
else
|
|
{
|
|
var fm = FileMode.Create;
|
|
options ??= new ExtractionOptions() { Overwrite = true };
|
|
|
|
if (!options.Overwrite)
|
|
{
|
|
fm = FileMode.CreateNew;
|
|
}
|
|
|
|
openAndWrite(destinationFileName, fm);
|
|
entry.PreserveExtractionOptions(destinationFileName, options);
|
|
}
|
|
}
|
|
}
|