Files
sharpcompress/tests/SharpCompress.Test/Zip/ZipArchiveTests.cs
2021-02-14 13:52:55 +00:00

574 lines
21 KiB
C#

using System;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using SharpCompress.Archives;
using SharpCompress.Archives.Zip;
using SharpCompress.Common;
using SharpCompress.Readers;
using SharpCompress.Writers;
using SharpCompress.Writers.Zip;
using Xunit;
namespace SharpCompress.Test.Zip
{
public class ZipArchiveTests : ArchiveTests
{
public ZipArchiveTests()
{
UseExtensionInsteadOfNameToVerify = true;
}
[Fact]
public async ValueTask Zip_ZipX_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.zipx");
}
[Fact]
public async ValueTask Zip_BZip2_Streamed_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.bzip2.dd.zip");
}
[Fact]
public async ValueTask Zip_BZip2_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.bzip2.zip");
}
[Fact]
public async ValueTask Zip_Deflate_Streamed2_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.deflate.dd-.zip");
}
[Fact]
public async ValueTask Zip_Deflate_Streamed_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.deflate.dd.zip");
}
[Fact]
public async ValueTask Zip_Deflate_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.deflate.zip");
}
[Fact]
public async ValueTask Zip_Deflate64_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.deflate64.zip");
}
[Fact]
public async ValueTask Zip_LZMA_Streamed_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.lzma.dd.zip");
}
[Fact]
public async ValueTask Zip_LZMA_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.lzma.zip");
}
[Fact]
public async ValueTask Zip_PPMd_Streamed_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.ppmd.dd.zip");
}
[Fact]
public async ValueTask Zip_PPMd_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.ppmd.zip");
}
[Fact]
public async ValueTask Zip_None_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.none.zip");
}
[Fact]
public async ValueTask Zip_BZip2_Streamed_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.bzip2.dd.zip");
}
[Fact]
public async ValueTask Zip_BZip2_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.bzip2.zip");
}
[Fact]
public async ValueTask Zip_Deflate_Streamed2_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.deflate.dd-.zip");
}
[Fact]
public async ValueTask Zip_Deflate_Streamed_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.deflate.dd.zip");
}
[Fact]
public async ValueTask Zip_Deflate_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.deflate.zip");
}
[Fact]
public async ValueTask Zip_Deflate64_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.deflate64.zip");
}
[Fact]
public async ValueTask Zip_LZMA_Streamed_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.lzma.dd.zip");
}
[Fact]
public async ValueTask Zip_LZMA_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.lzma.zip");
}
[Fact]
public async ValueTask Zip_PPMd_Streamed_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.ppmd.dd.zip");
}
[Fact]
public async ValueTask Zip_PPMd_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.ppmd.zip");
}
[Fact]
public async ValueTask Zip_None_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.none.zip");
}
[Fact]
public async ValueTask Zip_Zip64_ArchiveStreamRead()
{
await ArchiveStreamReadAsync("Zip.zip64.zip");
}
[Fact]
public async ValueTask Zip_Zip64_ArchiveFileRead()
{
await ArchiveStreamReadAsync("Zip.zip64.zip");
}
[Fact]
public async ValueTask Zip_Random_Write_Remove()
{
string scratchPath = Path.Combine(SCRATCH_FILES_PATH, "Zip.deflate.mod.zip");
string unmodified = Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.noEmptyDirs.zip");
string modified = Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.mod.zip");
await using (var archive = ZipArchive.Open(unmodified))
{
var entry = await archive.Entries.SingleAsync(x => x.Key.EndsWith("jpg"));
await archive.RemoveEntryAsync(entry);
WriterOptions writerOptions = new ZipWriterOptions(CompressionType.Deflate);
writerOptions.ArchiveEncoding.Default = Encoding.GetEncoding(866);
await archive.SaveToAsync(scratchPath, writerOptions);
}
await CompareArchivesByPathAsync(modified, scratchPath, Encoding.GetEncoding(866));
}
[Fact]
public async Task Zip_Random_Write_Add()
{
string jpg = Path.Combine(ORIGINAL_FILES_PATH, "jpg", "test.jpg");
string scratchPath = Path.Combine(SCRATCH_FILES_PATH, "Zip.deflate.mod.zip");
string unmodified = Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.mod.zip");
string modified = Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.mod2.zip");
await using (var archive = ZipArchive.Open(unmodified))
{
await archive.AddEntryAsync("jpg\\test.jpg", jpg);
WriterOptions writerOptions = new ZipWriterOptions(CompressionType.Deflate);
writerOptions.ArchiveEncoding.Default = Encoding.GetEncoding(866);
await archive.SaveToAsync(scratchPath, writerOptions);
}
await CompareArchivesByPathAsync(modified, scratchPath, Encoding.GetEncoding(866));
}
[Fact]
public async Task Zip_Save_Twice()
{
string scratchPath1 = Path.Combine(SCRATCH_FILES_PATH, "a.zip");
string scratchPath2 = Path.Combine(SCRATCH_FILES_PATH, "b.zip");
await using (var arc = ZipArchive.Create())
{
string str = "test.txt";
var source = new MemoryStream(Encoding.UTF8.GetBytes(str));
await arc.AddEntryAsync("test.txt", source, true, source.Length);
await arc.SaveToAsync(scratchPath1, CompressionType.Deflate);
await arc.SaveToAsync(scratchPath2, CompressionType.Deflate);
}
Assert.Equal(new FileInfo(scratchPath1).Length, new FileInfo(scratchPath2).Length);
}
[Fact]
public async ValueTask Zip_Removal_Poly()
{
string scratchPath = Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.noEmptyDirs.zip");
await using ZipArchive vfs = (ZipArchive)await ArchiveFactory.OpenAsync(scratchPath);
var e = await vfs.Entries.FirstAsync(v => v.Key.EndsWith("jpg"));
await vfs.RemoveEntryAsync(e);
Assert.Null(await vfs.Entries.FirstOrDefaultAsync(v => v.Key.EndsWith("jpg")));
Assert.Null(await ((IArchive)vfs).Entries.FirstOrDefaultAsync(v => v.Key.EndsWith("jpg")));
}
[Fact]
public async ValueTask Zip_Create_NoDups()
{
await using (var arc = ZipArchive.Create())
{
await arc.AddEntryAsync("1.txt", new MemoryStream());
await Assert.ThrowsAsync<ArchiveException>(async () => await arc.AddEntryAsync("\\1.txt", new MemoryStream()));
}
}
[Fact]
public async Task Zip_Create_Same_Stream()
{
string scratchPath1 = Path.Combine(SCRATCH_FILES_PATH, "a.zip");
string scratchPath2 = Path.Combine(SCRATCH_FILES_PATH, "b.zip");
await using (var arc = ZipArchive.Create())
{
await using (var stream = new MemoryStream(Encoding.UTF8.GetBytes("qwert")))
{
await arc.AddEntryAsync("1.txt", stream, false, stream.Length);
await arc.AddEntryAsync("2.txt", stream, false, stream.Length);
await arc.SaveToAsync(scratchPath1, CompressionType.Deflate);
await arc.SaveToAsync(scratchPath2, CompressionType.Deflate);
}
}
Assert.Equal(new FileInfo(scratchPath1).Length, new FileInfo(scratchPath2).Length);
}
[Fact]
public async Task Zip_Create_New()
{
foreach (var file in Directory.EnumerateFiles(ORIGINAL_FILES_PATH, "*.*", SearchOption.AllDirectories))
{
var newFileName = file.Substring(ORIGINAL_FILES_PATH.Length);
if (newFileName.StartsWith(Path.DirectorySeparatorChar.ToString()))
{
newFileName = newFileName.Substring(1);
}
newFileName = Path.Combine(SCRATCH_FILES_PATH, newFileName);
var newDir = Path.GetDirectoryName(newFileName);
if (!Directory.Exists(newDir))
{
Directory.CreateDirectory(newDir);
}
File.Copy(file, newFileName);
}
string scratchPath = Path.Combine(SCRATCH2_FILES_PATH, "Zip.deflate.noEmptyDirs.zip");
string unmodified = Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.noEmptyDirs.zip");
await using (var archive = ZipArchive.Create())
{
await archive.AddAllFromDirectoryAsync(SCRATCH_FILES_PATH);
WriterOptions writerOptions = new ZipWriterOptions(CompressionType.Deflate);
writerOptions.ArchiveEncoding.Default = Encoding.GetEncoding(866);
await archive.SaveToAsync(scratchPath, writerOptions);
}
await CompareArchivesByPathAsync(unmodified, scratchPath, Encoding.GetEncoding(866));
Directory.Delete(SCRATCH_FILES_PATH, true);
}
[Fact]
public async ValueTask Zip_Create_New_Add_Remove()
{
foreach (var file in Directory.EnumerateFiles(ORIGINAL_FILES_PATH, "*.*", SearchOption.AllDirectories))
{
var newFileName = file.Substring(ORIGINAL_FILES_PATH.Length);
if (newFileName.StartsWith(Path.DirectorySeparatorChar.ToString()))
{
newFileName = newFileName.Substring(1);
}
newFileName = Path.Combine(SCRATCH_FILES_PATH, newFileName);
var newDir = Path.GetDirectoryName(newFileName);
if (!Directory.Exists(newDir))
{
Directory.CreateDirectory(newDir);
}
File.Copy(file, newFileName);
}
string scratchPath = Path.Combine(SCRATCH2_FILES_PATH, "Zip.deflate.noEmptyDirs.zip");
await using (var archive = ZipArchive.Create())
{
await archive.AddAllFromDirectoryAsync(SCRATCH_FILES_PATH);
await archive.RemoveEntryAsync(await archive.Entries.SingleAsync(x => x.Key.EndsWith("jpg", StringComparison.OrdinalIgnoreCase)));
Assert.Null(await archive.Entries.FirstOrDefaultAsync(x => x.Key.EndsWith("jpg")));
}
Directory.Delete(SCRATCH_FILES_PATH, true);
}
[Fact]
public async ValueTask Zip_Deflate_WinzipAES_Read()
{
await using (var reader = ZipArchive.Open(Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.WinzipAES.zip"), new ReaderOptions()
{
Password = "test"
}))
{
await foreach (var entry in reader.Entries.Where(x => !x.IsDirectory))
{
await entry.WriteEntryToDirectoryAsync(SCRATCH_FILES_PATH, new ExtractionOptions()
{
ExtractFullPath = true,
Overwrite = true
});
}
}
VerifyFiles();
}
[Fact]
public async ValueTask Zip_Deflate_WinzipAES_MultiOpenEntryStream()
{
await using (var reader = ZipArchive.Open(Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.WinzipAES2.zip"), new ReaderOptions()
{
Password = "test"
}))
{
await foreach (var entry in reader.Entries.Where(x => !x.IsDirectory))
{
var stream = await entry.OpenEntryStreamAsync();
Assert.NotNull(stream);
var ex = await Record.ExceptionAsync(async () => stream = await entry.OpenEntryStreamAsync());
Assert.Null(ex);
}
}
}
[Fact]
public async ValueTask Zip_Read_Volume_Comment()
{
await using (var reader = ZipArchive.Open(Path.Combine(TEST_ARCHIVES_PATH, "Zip.zip64.zip"), new ReaderOptions()
{
Password = "test"
}))
{
var isComplete = await reader.IsCompleteAsync();
Assert.Equal(1, await reader.Volumes.CountAsync());
string expectedComment = "Encoding:utf-8 || Compression:Deflate levelDefault || Encrypt:None || ZIP64:Always\r\nCreated at 2017-Jan-23 14:10:43 || DotNetZip Tool v1.9.1.8\r\nTest zip64 archive";
Assert.Equal(expectedComment, (await reader.Volumes.FirstAsync()).Comment);
}
}
[Fact]
public async ValueTask Zip_BZip2_Pkware_Read()
{
await using (var reader = ZipArchive.Open(Path.Combine(TEST_ARCHIVES_PATH, "Zip.bzip2.pkware.zip"), new ReaderOptions()
{
Password = "test"
}))
{
await foreach (var entry in reader.Entries.Where(x => !x.IsDirectory))
{
await entry.WriteEntryToDirectoryAsync(SCRATCH_FILES_PATH, new ExtractionOptions()
{
ExtractFullPath = true,
Overwrite = true
});
}
}
VerifyFiles();
}
[Fact]
public async Task Zip_Random_Entry_Access()
{
string unmodified = Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.noEmptyDirs.zip");
ZipArchive a = ZipArchive.Open(unmodified);
int count = 0;
await foreach (var e in a.Entries)
{
count++;
}
//Prints 3
Assert.Equal(3, count);
await a.DisposeAsync();
a = ZipArchive.Open(unmodified);
int count2 = 0;
await foreach (var e in a.Entries)
{
count2++;
//Stop at last file
if (count2 == count)
{
var s = await e.OpenEntryStreamAsync();
byte[] b = new byte[1];
await s.ReadAsync(b, 0, 1); //Actually access stream
await s.DisposeAsync();
break;
}
}
int count3 = 0;
await foreach (var e in a.Entries)
{
count3++;
}
Assert.Equal(3, count3);
}
[Fact]
public async ValueTask Zip_Deflate_PKWear_Multipy_Entry_Access()
{
string zipFile = Path.Combine(TEST_ARCHIVES_PATH, "Zip.deflate.pkware.zip");
await using FileStream fileStream = File.Open(zipFile, FileMode.Open);
await using IArchive archive = await ArchiveFactory.OpenAsync(fileStream, new ReaderOptions { Password = "12345678" });
var entries = archive.Entries.Where(entry => !entry.IsDirectory);
await foreach (IArchiveEntry entry in entries)
{
for (var i = 0; i < 100; i++)
{
await using (var memoryStream = new MemoryStream())
await using (Stream entryStream = await entry.OpenEntryStreamAsync())
{
await entryStream.CopyToAsync(memoryStream);
}
}
}
}
[SkippableFact]
public async ValueTask Zip_Evil_Throws_Exception()
{
//windows only because of the paths
Skip.IfNot(Environment.OSVersion.Platform == PlatformID.Win32NT);
string zipFile = Path.Combine(TEST_ARCHIVES_PATH, "Zip.Evil.zip");
await Assert.ThrowsAnyAsync<Exception>(async () =>
{
await using (var archive = ZipArchive.Open(zipFile))
{
await foreach (var entry in archive.Entries.Where(entry => !entry.IsDirectory))
{
await entry.WriteEntryToDirectoryAsync(SCRATCH_FILES_PATH, new ExtractionOptions()
{
ExtractFullPath = true,
Overwrite = true
});
}
}
});
}
private class NonSeekableMemoryStream : MemoryStream
{
public override bool CanSeek => false;
}
[Fact]
public async Task TestSharpCompressWithEmptyStream()
{
MemoryStream stream = new NonSeekableMemoryStream();
await using (IWriter zipWriter = await WriterFactory.OpenAsync(stream, ArchiveType.Zip, CompressionType.Deflate))
{
await zipWriter.WriteAsync("foo.txt", new MemoryStream(new byte[0]));
await zipWriter.WriteAsync("foo2.txt", new MemoryStream(new byte[10]));
}
stream = new MemoryStream(stream.ToArray());
await File.WriteAllBytesAsync(Path.Combine(SCRATCH_FILES_PATH, "foo.zip"), stream.ToArray());
await using (var zipArchive = ZipArchive.Open(stream))
{
await foreach (var entry in zipArchive.Entries)
{
await using (var entryStream = await entry.OpenEntryStreamAsync())
{
MemoryStream tempStream = new MemoryStream();
const int bufSize = 0x1000;
byte[] buf = new byte[bufSize];
int bytesRead = 0;
while ((bytesRead = await entryStream.ReadAsync(buf, 0, bufSize)) > 0)
{
await tempStream.WriteAsync(buf, 0, bytesRead);
}
}
}
}
}
[Fact]
public async Task Zip_BadLocalExtra_Read()
{
string zipPath = Path.Combine(TEST_ARCHIVES_PATH, "Zip.badlocalextra.zip");
await using (ZipArchive za = ZipArchive.Open(zipPath))
{
var ex = await Record.ExceptionAsync(async () =>
{
var firstEntry = await za.Entries.FirstAsync(x => x.Key == "first.txt");
var buffer = new byte[4096];
await using (var memoryStream = new MemoryStream())
await using (var firstStream = await firstEntry.OpenEntryStreamAsync())
{
await firstStream.CopyToAsync(memoryStream);
Assert.Equal(199, memoryStream.Length);
}
});
Assert.Null(ex);
}
}
[Fact]
public async ValueTask Zip_NoCompression_DataDescriptors_Read()
{
string zipPath = Path.Combine(TEST_ARCHIVES_PATH, "Zip.none.datadescriptors.zip");
await using (ZipArchive za = ZipArchive.Open(zipPath))
{
var firstEntry = await za.Entries.FirstAsync(x => x.Key == "first.txt");
await using (var memoryStream = new MemoryStream())
await using (var firstStream = await firstEntry.OpenEntryStreamAsync())
{
await firstStream.CopyToAsync(memoryStream);
Assert.Equal(199, memoryStream.Length);
}
}
}
[Fact]
public async Task Zip_LongComment_Read()
{
string zipPath = Path.Combine(TEST_ARCHIVES_PATH, "Zip.LongComment.zip");
await using(ZipArchive za = ZipArchive.Open(zipPath))
{
var count = await za.Entries.CountAsync();
Assert.Equal(1, count);
}
}
}
}