/*
Copyright (C) 2005-2014 Sergey A. Tachenov
This file is part of QuaZIP test suite.
QuaZIP 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.
QuaZIP 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 QuaZIP. If not, see .
See COPYING file for the full LGPL text.
Original ZIP package is copyrighted by Gilles Vollant and contributors,
see quazip/(un)zip.h files for details. Basically it's the zlib license.
*/
#include "testquazipfile.h"
#include "qztest.h"
#include
#include
#include
#include
#include
#include
#include
void TestQuaZipFile::zipUnzip_data()
{
QTest::addColumn("zipName");
QTest::addColumn("fileNames");
QTest::addColumn("fileNameCodec");
QTest::addColumn("password");
QTest::addColumn("zip64");
QTest::newRow("simple") << "simple.zip" << (
QStringList() << "test0.txt" << "testdir1/test1.txt"
<< "testdir2/test2.txt" << "testdir2/subdir/test2sub.txt")
<< QByteArray() << QByteArray() << false;
QTest::newRow("Cyrillic") << "cyrillic.zip" << (
QStringList()
<< QString::fromUtf8("русское имя файла с пробелами.txt"))
<< QByteArray("IBM866") << QByteArray() << false;
QTest::newRow("password") << "password.zip" << (
QStringList() << "test.txt")
<< QByteArray() << QByteArray("PassPass") << false;
QTest::newRow("zip64") << "zip64.zip" << (
QStringList() << "test64.txt")
<< QByteArray() << QByteArray() << true;
}
void TestQuaZipFile::zipUnzip()
{
QFETCH(QString, zipName);
QFETCH(QStringList, fileNames);
QFETCH(QByteArray, fileNameCodec);
QFETCH(QByteArray, password);
QFETCH(bool, zip64);
QFile testFile(zipName);
if (testFile.exists()) {
if (!testFile.remove()) {
QFAIL("Couldn't remove existing archive to create a new one");
}
}
if (!createTestFiles(fileNames)) {
QFAIL("Couldn't create test files for zipping");
}
QuaZip testZip(&testFile);
testZip.setZip64Enabled(zip64);
if (!fileNameCodec.isEmpty())
testZip.setFileNameCodec(fileNameCodec);
QVERIFY(testZip.open(QuaZip::mdCreate));
QString comment = "Test comment";
testZip.setComment(comment);
foreach (QString fileName, fileNames) {
QFile inFile("tmp/" + fileName);
if (!inFile.open(QIODevice::ReadOnly)) {
qDebug("File name: %s", fileName.toUtf8().constData());
QFAIL("Couldn't open input file");
}
QuaZipFile outFile(&testZip);
QVERIFY(outFile.open(QIODevice::WriteOnly, QuaZipNewInfo(fileName,
inFile.fileName()),
password.isEmpty() ? NULL : password.constData()));
for (qint64 pos = 0, len = inFile.size(); pos < len; ) {
char buf[4096];
qint64 readSize = qMin(static_cast(4096), len - pos);
qint64 l;
if ((l = inFile.read(buf, readSize)) != readSize) {
qDebug("Reading %ld bytes from %s at %ld returned %ld",
static_cast(readSize),
fileName.toUtf8().constData(),
static_cast(pos),
static_cast(l));
QFAIL("Read failure");
}
QVERIFY(outFile.write(buf, readSize));
pos += readSize;
}
inFile.close();
outFile.close();
QCOMPARE(outFile.getZipError(), ZIP_OK);
}
testZip.close();
QCOMPARE(testZip.getZipError(), ZIP_OK);
// now test unzip
QuaZip testUnzip(&testFile);
if (!fileNameCodec.isEmpty())
testUnzip.setFileNameCodec(fileNameCodec);
QVERIFY(testUnzip.open(QuaZip::mdUnzip));
QCOMPARE(testUnzip.getComment(), comment);
QVERIFY(testUnzip.goToFirstFile());
foreach (QString fileName, fileNames) {
QuaZipFileInfo64 info;
QVERIFY(testUnzip.getCurrentFileInfo(&info));
QCOMPARE(info.name, fileName);
QCOMPARE(info.isEncrypted(), !password.isEmpty());
QFile original("tmp/" + fileName);
QVERIFY(original.open(QIODevice::ReadOnly));
QuaZipFile archived(&testUnzip);
QVERIFY(archived.open(QIODevice::ReadOnly,
password.isEmpty() ? NULL : password.constData()));
QByteArray originalData = original.readAll();
QByteArray archivedData = archived.readAll();
QCOMPARE(archivedData, originalData);
testUnzip.goToNextFile();
}
if (!password.isEmpty()) {
QVERIFY(testUnzip.goToFirstFile());
QuaZipFileInfo64 info;
QVERIFY(testUnzip.getCurrentFileInfo(&info));
QFile original("tmp/" + info.name);
QVERIFY(original.open(QIODevice::ReadOnly));
QuaZipFile archived(&testUnzip);
QVERIFY(archived.open(QIODevice::ReadOnly, "WrongPassword"));
QByteArray originalData = original.readAll();
QByteArray archivedData = archived.readAll();
QVERIFY(archivedData != originalData);
}
testUnzip.close();
QCOMPARE(testUnzip.getZipError(), UNZ_OK);
// clean up
removeTestFiles(fileNames);
testFile.remove();
}
void TestQuaZipFile::bytesAvailable_data()
{
QTest::addColumn("zipName");
QTest::addColumn("fileNames");
QTest::newRow("simple") << "test.zip" << (
QStringList() << "test0.txt" << "testdir1/test1.txt"
<< "testdir2/test2.txt" << "testdir2/subdir/test2sub.txt");
}
void TestQuaZipFile::bytesAvailable()
{
QFETCH(QString, zipName);
QFETCH(QStringList, fileNames);
QDir curDir;
if (!createTestFiles(fileNames)) {
QFAIL("Couldn't create test files");
}
if (!JlCompress::compressDir(zipName, "tmp")) {
QFAIL("Couldn't create test archive");
}
QuaZip testZip(zipName);
QVERIFY(testZip.open(QuaZip::mdUnzip));
foreach (QString fileName, fileNames) {
QFileInfo fileInfo("tmp/" + fileName);
QVERIFY(testZip.setCurrentFile(fileName));
QuaZipFile zipFile(&testZip);
QVERIFY(zipFile.open(QIODevice::ReadOnly));
QCOMPARE(zipFile.bytesAvailable(), fileInfo.size());
QCOMPARE(zipFile.read(1).size(), 1);
QCOMPARE(zipFile.bytesAvailable(), fileInfo.size() - 1);
QCOMPARE(zipFile.read(fileInfo.size() - 1).size(),
static_cast(fileInfo.size() - 1));
QCOMPARE(zipFile.bytesAvailable(), (qint64) 0);
}
removeTestFiles(fileNames);
testZip.close();
curDir.remove(zipName);
}
void TestQuaZipFile::atEnd_data()
{
bytesAvailable_data();
}
void TestQuaZipFile::atEnd()
{
QFETCH(QString, zipName);
QFETCH(QStringList, fileNames);
QDir curDir;
if (!createTestFiles(fileNames)) {
QFAIL("Couldn't create test files");
}
if (!JlCompress::compressDir(zipName, "tmp")) {
QFAIL("Couldn't create test archive");
}
QuaZip testZip(zipName);
QVERIFY(testZip.open(QuaZip::mdUnzip));
foreach (QString fileName, fileNames) {
QFileInfo fileInfo("tmp/" + fileName);
QVERIFY(testZip.setCurrentFile(fileName));
QuaZipFile zipFile(&testZip);
QVERIFY(zipFile.open(QIODevice::ReadOnly));
QCOMPARE(zipFile.atEnd(), false);
QCOMPARE(zipFile.read(1).size(), 1);
QCOMPARE(zipFile.atEnd(), false);
QCOMPARE(zipFile.read(fileInfo.size() - 1).size(),
static_cast(fileInfo.size() - 1));
QCOMPARE(zipFile.atEnd(), true);
}
removeTestFiles(fileNames);
testZip.close();
curDir.remove(zipName);
}
void TestQuaZipFile::pos_data()
{
bytesAvailable_data();
}
void TestQuaZipFile::pos()
{
QFETCH(QString, zipName);
QFETCH(QStringList, fileNames);
QDir curDir;
if (!createTestFiles(fileNames)) {
QFAIL("Couldn't create test files");
}
if (!JlCompress::compressDir(zipName, "tmp")) {
QFAIL("Couldn't create test archive");
}
QuaZip testZip(zipName);
QVERIFY(testZip.open(QuaZip::mdUnzip));
foreach (QString fileName, fileNames) {
QFileInfo fileInfo("tmp/" + fileName);
QVERIFY(testZip.setCurrentFile(fileName));
QuaZipFile zipFile(&testZip);
QVERIFY(zipFile.open(QIODevice::ReadOnly));
QCOMPARE(zipFile.pos(), (qint64) 0);
QCOMPARE(zipFile.read(1).size(), 1);
QCOMPARE(zipFile.pos(), (qint64) 1);
QCOMPARE(zipFile.read(fileInfo.size() - 1).size(),
static_cast(fileInfo.size() - 1));
QCOMPARE(zipFile.pos(), fileInfo.size());
}
removeTestFiles(fileNames);
testZip.close();
curDir.remove(zipName);
}
void TestQuaZipFile::getZip()
{
QuaZip testZip;
QuaZipFile f1(&testZip);
QCOMPARE(f1.getZip(), &testZip);
QuaZipFile f2("doesntexist.zip", "someFile");
QCOMPARE(f2.getZip(), static_cast(NULL));
f2.setZip(&testZip);
QCOMPARE(f2.getZip(), &testZip);
}
void TestQuaZipFile::setZipName()
{
QString testFileName = "testZipName.txt";
QString testZipName = "testZipName.zip";
QVERIFY(createTestFiles(QStringList() << testFileName));
QVERIFY(createTestArchive(testZipName, QStringList() << testFileName));
QuaZipFile testFile;
testFile.setZipName(testZipName);
QCOMPARE(testFile.getZipName(), testZipName);
testFile.setFileName(testFileName);
QVERIFY(testFile.open(QIODevice::ReadOnly));
testFile.close();
removeTestFiles(QStringList() << testFileName);
QDir curDir;
curDir.remove(testZipName);
}
void TestQuaZipFile::getFileInfo()
{
QuaZipFileInfo info32;
QuaZipFileInfo64 info64;
QString testFileName = "testZipName.txt";
QStringList testFiles;
testFiles << testFileName;
QString testZipName = "testZipName.zip";
QVERIFY(createTestFiles(testFiles));
QVERIFY(createTestArchive(testZipName, testFiles));
QuaZipFile testFile;
testFile.setZipName(testZipName);
testFile.setFileName(testFileName);
QVERIFY(testFile.open(QIODevice::ReadOnly));
QVERIFY(testFile.getFileInfo(&info32));
QVERIFY(testFile.getFileInfo(&info64));
QCOMPARE(info32.name, info64.name);
QCOMPARE(info32.versionCreated, info64.versionCreated);
QCOMPARE(info32.versionNeeded, info64.versionNeeded);
QCOMPARE(info32.flags, info64.flags);
QCOMPARE(info32.method, info64.method);
QCOMPARE(info32.dateTime, info64.dateTime);
QCOMPARE(info32.crc, info64.crc);
QCOMPARE(info32.compressedSize,
static_cast(info64.compressedSize));
QCOMPARE(info32.uncompressedSize,
static_cast(info64.uncompressedSize));
QCOMPARE(info32.diskNumberStart, info64.diskNumberStart);
QCOMPARE(info32.internalAttr, info64.internalAttr);
QCOMPARE(info32.externalAttr, info64.externalAttr);
QCOMPARE(info32.comment, info64.comment);
QCOMPARE(info32.extra, info64.extra);
testFile.close();
removeTestFiles(testFiles);
QDir curDir;
curDir.remove(testZipName);
}
void TestQuaZipFile::setFileName()
{
QString testFileName = "testZipName.txt";
QString testZipName = "testZipName.zip";
QVERIFY(createTestFiles(QStringList() << testFileName));
QVERIFY(createTestArchive(testZipName, QStringList() << testFileName));
QuaZipFile testFile(testZipName);
testFile.setFileName(testFileName.toUpper());
#ifdef Q_OS_WIN
QVERIFY(testFile.open(QIODevice::ReadOnly));
testFile.close();
#else
QVERIFY(!testFile.open(QIODevice::ReadOnly));
#endif
testFile.setFileName(testFileName.toUpper(), QuaZip::csInsensitive);
QCOMPARE(testFile.getCaseSensitivity(), QuaZip::csInsensitive);
QVERIFY(testFile.open(QIODevice::ReadOnly));
QCOMPARE(testFile.getActualFileName(), testFileName);
testFile.close();
testFile.setFileName(testFileName.toUpper(), QuaZip::csSensitive);
QCOMPARE(testFile.getFileName(), testFileName.toUpper());
QCOMPARE(testFile.getActualFileName(), QString());
QVERIFY(!testFile.open(QIODevice::ReadOnly));
testFile.setFileName(testFileName);
removeTestFiles(QStringList() << testFileName);
QDir curDir;
curDir.remove(testZipName);
}
void TestQuaZipFile::constructorDestructor()
{
// Just test that all constructors and destructors are available.
// (So there are none that are declared but not defined.)
QuaZip testZip;
QuaZipFile *f1 = new QuaZipFile();
delete f1; // test D0 destructor
QObject parent;
QuaZipFile f2(&testZip, &parent);
QuaZipFile f3(&parent);
QuaZipFile f4("zipName.zip");
QuaZipFile f5("zipName.zip", "fileName.txt", QuaZip::csDefault, &parent);
}
void TestQuaZipFile::setFileAttrs()
{
QuaZip testZip("setFileAttrs.zip");
QVERIFY(testZip.open(QuaZip::mdCreate));
QuaZipFile zipFile(&testZip);
QuaZipNewInfo newInfo("testPerm.txt");
newInfo.setPermissions(QFile::ReadOwner);
QVERIFY(zipFile.open(QIODevice::WriteOnly, newInfo));
zipFile.close();
QString fileTestAttr = "testAttr.txt";
QStringList fileNames;
fileNames << fileTestAttr;
QVERIFY(createTestFiles(fileNames));
newInfo.name = fileTestAttr;
newInfo.setFileDateTime("tmp/" + fileTestAttr);
newInfo.setFilePermissions("tmp/" + fileTestAttr);
QVERIFY(zipFile.open(QIODevice::WriteOnly, newInfo));
zipFile.close();
testZip.close();
QuaZipFileInfo64 info;
{
QuaZipFile readFilePerm("setFileAttrs.zip", "testPerm.txt");
QVERIFY(readFilePerm.open(QIODevice::ReadOnly));
QVERIFY(readFilePerm.getFileInfo(&info));
QCOMPARE(info.getPermissions(), QFile::ReadOwner);
readFilePerm.close();
}
{
QuaZipFile readFileAttrs("setFileAttrs.zip", "testAttr.txt");
QVERIFY(readFileAttrs.open(QIODevice::ReadOnly));
QVERIFY(readFileAttrs.getFileInfo(&info));
QFileInfo srcInfo("tmp/" + fileTestAttr);
QFile::Permissions usedPermissions =
QFile::WriteOwner | QFile::ReadOwner | QFile::ExeOwner |
QFile::WriteGroup | QFile::ReadGroup | QFile::ExeGroup |
QFile::WriteOther | QFile::ReadOther | QFile::ExeOther;
QCOMPARE(info.getPermissions() & usedPermissions,
srcInfo.permissions() & usedPermissions);
// I really hope Qt 6 will use quint64 for time_t!
quint64 newTime = info.dateTime.toTime_t();
quint64 oldTime = srcInfo.lastModified().toTime_t();
// ZIP uses weird format with 2 second precision
QCOMPARE(newTime / 2, oldTime / 2);
readFileAttrs.close();
}
removeTestFiles(fileNames);
QDir().remove(testZip.getZipName());
}
void TestQuaZipFile::largeFile()
{
QDir curDir;
QVERIFY(curDir.mkpath("tmp"));
QFile fakeLargeFile("tmp/large.zip");
QVERIFY(fakeLargeFile.open(QIODevice::WriteOnly));
QDataStream ds(&fakeLargeFile);
ds.setByteOrder(QDataStream::LittleEndian);
QList localOffsets;
const int numFiles = 2; // name fixed to 5 bytes, so MAX 10 FILES!!!
for (int i = 0; i < numFiles; ++i) {
localOffsets.append(fakeLargeFile.pos());
QBuffer extra;
extra.open(QIODevice::WriteOnly);
QDataStream es(&extra);
es.setByteOrder(QDataStream::LittleEndian);
// prepare extra
es << static_cast(0x0001u); // zip64
es << static_cast(16); // extra data size
es << static_cast(0); // uncompressed size
es << static_cast(0); // compressed size
// now the local header
ds << static_cast(0x04034b50u); // local magic
ds << static_cast(45); // version needed
ds << static_cast(0); // flags
ds << static_cast(0); // method
ds << static_cast(0); // time 00:00:00
ds << static_cast(0x21); // date 1980-01-01
ds << static_cast(0); // CRC-32
ds << static_cast(0xFFFFFFFFu); // compressed size
ds << static_cast(0xFFFFFFFFu); // uncompressed size
ds << static_cast(5); // name length
ds << static_cast(extra.size()); // extra length
ds.writeRawData("file", 4); // name
ds << static_cast('0' + i); // name (contd.)
ds.writeRawData(extra.buffer(), extra.size());
}
// central dir:
qint64 centralStart = fakeLargeFile.pos();
for (int i = 0; i < numFiles; ++i) {
QBuffer extra;
extra.open(QIODevice::WriteOnly);
QDataStream es(&extra);
es.setByteOrder(QDataStream::LittleEndian);
// prepare extra
es << static_cast(0x0001u); // zip64
es << static_cast(24); // extra data size
es << static_cast(0); // uncompressed size
es << static_cast(0); // compressed size
es << static_cast(localOffsets[i]);
// now the central dir header
ds << static_cast(0x02014b50u); // central magic
ds << static_cast(45); // version made by
ds << static_cast(45); // version needed
ds << static_cast(0); // flags
ds << static_cast(0); // method
ds << static_cast(0); // time 00:00:00
ds << static_cast(0x21); // date 1980-01-01
ds << static_cast(0); // CRC-32
ds << static_cast(0xFFFFFFFFu); // compressed size
ds << static_cast(0xFFFFFFFFu); // uncompressed size
ds << static_cast(5); // name length
ds << static_cast(extra.size()); // extra length
ds << static_cast(0); // comment length
ds << static_cast(0); // disk number
ds << static_cast(0); // internal attrs
ds << static_cast(0); // external attrs
ds << static_cast(0xFFFFFFFFu); // local offset
ds.writeRawData("file", 4); // name
ds << static_cast('0' + i); // name (contd.)
ds.writeRawData(extra.buffer(), extra.size());
}
qint64 centralEnd = fakeLargeFile.pos();
// zip64 end
ds << static_cast(0x06064b50); // zip64 end magic
ds << static_cast(44); // size of the zip64 end
ds << static_cast(45); // version made by
ds << static_cast(45); // version needed
ds << static_cast(0); // disk number
ds << static_cast(0); // central dir disk number
ds << static_cast(2); // number of entries on disk
ds << static_cast(2); // total number of entries
ds << static_cast(centralEnd - centralStart); // size
ds << static_cast(centralStart); // offset
// zip64 locator
ds << static_cast(0x07064b50); // zip64 locator magic
ds << static_cast(0); // disk number
ds << static_cast(centralEnd); // offset
ds << static_cast(1); // number of disks
// zip32 end
ds << static_cast(0x06054b50); // end magic
ds << static_cast(0); // disk number
ds << static_cast(0); // central dir disk number
ds << static_cast(2); // number of entries
ds << static_cast(0xFFFFFFFFu); // central dir size
ds << static_cast(0xFFFFFFFFu); // central dir offset
ds << static_cast(0); // comment length
fakeLargeFile.close();
QuaZip fakeLargeZip("tmp/large.zip");
QVERIFY(fakeLargeZip.open(QuaZip::mdUnzip));
QCOMPARE(fakeLargeZip.getFileInfoList().size(), numFiles);
QCOMPARE(fakeLargeZip.getFileInfoList()[0].uncompressedSize,
static_cast(0));
fakeLargeZip.close();
curDir.remove("tmp/large.zip");
}