2005-02-10 09:06:08 -05:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Name: tests/archive/ziptest.cpp
|
|
|
|
// Purpose: Test the zip classes
|
|
|
|
// Author: Mike Wetherell
|
|
|
|
// RCS-ID: $Id$
|
|
|
|
// Copyright: (c) 2004 Mike Wetherell
|
|
|
|
// Licence: wxWindows licence
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#include "testprec.h"
|
|
|
|
|
|
|
|
#ifdef __BORLANDC__
|
|
|
|
# pragma hdrstop
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef WX_PRECOMP
|
|
|
|
# include "wx/wx.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if wxUSE_STREAMS && wxUSE_ZIPSTREAM
|
|
|
|
|
|
|
|
#include "archivetest.h"
|
|
|
|
#include "wx/zipstrm.h"
|
|
|
|
|
|
|
|
using std::string;
|
|
|
|
using std::auto_ptr;
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// ArchiveTestCase<wxZipClassFactory> could be used directly, but instead this
|
|
|
|
// derived class is used so that zip specific features can be tested.
|
|
|
|
|
|
|
|
class ZipTestCase : public ArchiveTestCase<wxZipClassFactory>
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ZipTestCase(string name,
|
|
|
|
int options,
|
|
|
|
const wxString& archiver = wxEmptyString,
|
|
|
|
const wxString& unarchiver = wxEmptyString)
|
|
|
|
:
|
2005-12-18 08:58:55 -05:00
|
|
|
ArchiveTestCase<wxZipClassFactory>(name, new wxZipClassFactory,
|
2005-02-10 09:06:08 -05:00
|
|
|
options, archiver, unarchiver),
|
|
|
|
m_count(0)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
void OnCreateArchive(wxZipOutputStream& zip);
|
|
|
|
|
|
|
|
void OnArchiveExtracted(wxZipInputStream& zip, int expectedTotal);
|
|
|
|
|
|
|
|
void OnCreateEntry(wxZipOutputStream& zip,
|
|
|
|
TestEntry& testEntry,
|
|
|
|
wxZipEntry *entry);
|
|
|
|
|
|
|
|
void OnEntryExtracted(wxZipEntry& entry,
|
|
|
|
const TestEntry& testEntry,
|
|
|
|
wxZipInputStream *arc);
|
|
|
|
|
|
|
|
void OnSetNotifier(EntryT& entry);
|
|
|
|
|
|
|
|
int m_count;
|
|
|
|
wxString m_comment;
|
|
|
|
};
|
|
|
|
|
|
|
|
void ZipTestCase::OnCreateArchive(wxZipOutputStream& zip)
|
|
|
|
{
|
2009-07-23 16:30:22 -04:00
|
|
|
m_comment << wxT("Comment for test ") << m_id;
|
2005-02-10 09:06:08 -05:00
|
|
|
zip.SetComment(m_comment);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ZipTestCase::OnArchiveExtracted(wxZipInputStream& zip, int expectedTotal)
|
|
|
|
{
|
|
|
|
CPPUNIT_ASSERT(zip.GetComment() == m_comment);
|
|
|
|
CPPUNIT_ASSERT(zip.GetTotalEntries() == expectedTotal);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ZipTestCase::OnCreateEntry(wxZipOutputStream& zip,
|
|
|
|
TestEntry& testEntry,
|
|
|
|
wxZipEntry *entry)
|
|
|
|
{
|
|
|
|
zip.SetLevel((m_id + m_count) % 10);
|
|
|
|
|
|
|
|
if (entry) {
|
|
|
|
switch ((m_id + m_count) % 5) {
|
|
|
|
case 0:
|
|
|
|
{
|
2009-07-23 16:30:22 -04:00
|
|
|
wxString comment = wxT("Comment for ") + entry->GetName();
|
2005-02-10 09:06:08 -05:00
|
|
|
entry->SetComment(comment);
|
|
|
|
// lowercase the expected result, and the notifier should do
|
|
|
|
// the same for the zip entries when ModifyArchive() runs
|
|
|
|
testEntry.SetComment(comment.Lower());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
entry->SetMethod(wxZIP_METHOD_STORE);
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
entry->SetMethod(wxZIP_METHOD_DEFLATE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
entry->SetIsText(testEntry.IsText());
|
|
|
|
}
|
|
|
|
|
|
|
|
m_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ZipTestCase::OnEntryExtracted(wxZipEntry& entry,
|
|
|
|
const TestEntry& testEntry,
|
|
|
|
wxZipInputStream *arc)
|
|
|
|
{
|
|
|
|
// provide some context for the error message so that we know which
|
|
|
|
// iteration of the loop we were on
|
2009-07-23 16:30:22 -04:00
|
|
|
wxString name = wxT(" '") + entry.GetName() + wxT("'");
|
2005-02-10 09:06:08 -05:00
|
|
|
string error_entry(name.mb_str());
|
|
|
|
string error_context(" failed for entry" + error_entry);
|
|
|
|
|
|
|
|
CPPUNIT_ASSERT_MESSAGE("GetComment" + error_context,
|
|
|
|
entry.GetComment() == testEntry.GetComment());
|
|
|
|
|
|
|
|
// for seekable streams, GetNextEntry() doesn't read the local header so
|
|
|
|
// call OpenEntry() to do it
|
|
|
|
if (arc && (m_options & PipeIn) == 0 && entry.IsDir())
|
|
|
|
arc->OpenEntry(entry);
|
|
|
|
|
|
|
|
CPPUNIT_ASSERT_MESSAGE("IsText" + error_context,
|
|
|
|
entry.IsText() == testEntry.IsText());
|
|
|
|
|
|
|
|
CPPUNIT_ASSERT_MESSAGE("Extra/LocalExtra mismatch for entry" + error_entry,
|
|
|
|
(entry.GetExtraLen() != 0 && entry.GetLocalExtraLen() != 0) ||
|
|
|
|
(entry.GetExtraLen() == 0 && entry.GetLocalExtraLen() == 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
// check the notifier mechanism by using it to fold the entry comments to
|
|
|
|
// lowercase
|
|
|
|
//
|
|
|
|
class ZipNotifier : public wxZipNotifier
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
void OnEntryUpdated(wxZipEntry& entry);
|
|
|
|
};
|
|
|
|
|
|
|
|
void ZipNotifier::OnEntryUpdated(wxZipEntry& entry)
|
|
|
|
{
|
|
|
|
entry.SetComment(entry.GetComment().Lower());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ZipTestCase::OnSetNotifier(EntryT& entry)
|
|
|
|
{
|
|
|
|
static ZipNotifier notifier;
|
|
|
|
entry.SetNotifier(notifier);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// 'zip - -' produces local headers without the size field set. This is a
|
|
|
|
// case not covered by all the other tests, so this class tests it as a
|
|
|
|
// special case
|
|
|
|
|
|
|
|
class ZipPipeTestCase : public CppUnit::TestCase
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ZipPipeTestCase(string name, int options) :
|
2005-12-18 08:58:55 -05:00
|
|
|
CppUnit::TestCase(TestId::MakeId() + name),
|
|
|
|
m_options(options),
|
|
|
|
m_id(TestId::GetId())
|
|
|
|
{ }
|
2005-02-10 09:06:08 -05:00
|
|
|
|
|
|
|
protected:
|
|
|
|
void runTest();
|
|
|
|
int m_options;
|
2005-12-18 08:58:55 -05:00
|
|
|
int m_id;
|
2005-02-10 09:06:08 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
void ZipPipeTestCase::runTest()
|
|
|
|
{
|
|
|
|
TestOutputStream out(m_options);
|
|
|
|
|
2009-07-23 16:30:22 -04:00
|
|
|
wxString testdata = wxT("test data to pipe through zip");
|
|
|
|
wxString cmd = wxT("echo ") + testdata + wxT(" | zip -q - -");
|
2005-02-10 09:06:08 -05:00
|
|
|
|
|
|
|
{
|
|
|
|
PFileInputStream in(cmd);
|
|
|
|
if (in.Ok())
|
|
|
|
out.Write(in);
|
|
|
|
}
|
|
|
|
|
2005-12-18 08:58:55 -05:00
|
|
|
TestInputStream in(out, m_id % ((m_options & PipeIn) ? 4 : 3));
|
2005-02-10 09:06:08 -05:00
|
|
|
wxZipInputStream zip(in);
|
|
|
|
|
|
|
|
auto_ptr<wxZipEntry> entry(zip.GetNextEntry());
|
|
|
|
CPPUNIT_ASSERT(entry.get() != NULL);
|
|
|
|
|
|
|
|
if ((m_options & PipeIn) == 0)
|
|
|
|
CPPUNIT_ASSERT(entry->GetSize() != wxInvalidOffset);
|
|
|
|
|
|
|
|
char buf[64];
|
|
|
|
size_t len = zip.Read(buf, sizeof(buf) - 1).LastRead();
|
|
|
|
|
|
|
|
while (len > 0 && buf[len - 1] <= 32)
|
|
|
|
--len;
|
|
|
|
buf[len] = 0;
|
|
|
|
|
|
|
|
CPPUNIT_ASSERT(zip.Eof());
|
|
|
|
CPPUNIT_ASSERT(wxString(buf, *wxConvCurrent) == testdata);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Zip suite
|
|
|
|
|
|
|
|
class ziptest : public ArchiveTestSuite
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ziptest();
|
|
|
|
static CppUnit::Test *suite() { return (new ziptest)->makeSuite(); }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
ArchiveTestSuite *makeSuite();
|
|
|
|
|
2005-12-18 08:58:55 -05:00
|
|
|
CppUnit::Test *makeTest(string descr, int options,
|
2005-02-10 09:06:08 -05:00
|
|
|
bool genericInterface, const wxString& archiver,
|
|
|
|
const wxString& unarchiver);
|
|
|
|
};
|
|
|
|
|
|
|
|
ziptest::ziptest()
|
|
|
|
: ArchiveTestSuite("zip")
|
|
|
|
{
|
2009-07-23 16:30:22 -04:00
|
|
|
AddArchiver(wxT("zip -qr %s *"));
|
|
|
|
AddUnArchiver(wxT("unzip -q %s"));
|
2005-02-10 09:06:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
ArchiveTestSuite *ziptest::makeSuite()
|
|
|
|
{
|
|
|
|
ArchiveTestSuite::makeSuite();
|
|
|
|
|
2009-04-04 18:12:04 -04:00
|
|
|
#if !defined WXARC_NO_POPEN && !defined __WXMSW__
|
|
|
|
// If have popen then can check the piped output of 'zip - -'.
|
|
|
|
// The gnuwin32 build of infozip does work for this, e.g.:
|
|
|
|
// C:\>echo test data to pipe through zip | zip -q > foo.zip
|
|
|
|
// doesn't produce a valid zip, so disabled for now.
|
2009-07-23 16:30:22 -04:00
|
|
|
if (IsInPath(wxT("zip")))
|
2005-02-10 09:06:08 -05:00
|
|
|
for (int options = 0; options <= PipeIn; options += PipeIn) {
|
2009-07-23 16:30:22 -04:00
|
|
|
string name = Description(wxT("ZipPipeTestCase"), options,
|
|
|
|
false, wxT(""), wxT("zip -q - -"));
|
2005-02-10 09:06:08 -05:00
|
|
|
addTest(new ZipPipeTestCase(name, options));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
CppUnit::Test *ziptest::makeTest(
|
|
|
|
string descr,
|
|
|
|
int options,
|
|
|
|
bool genericInterface,
|
|
|
|
const wxString& archiver,
|
|
|
|
const wxString& unarchiver)
|
|
|
|
{
|
|
|
|
// unzip doesn't support piping in the zip
|
|
|
|
if ((options & PipeIn) && !unarchiver.empty())
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (genericInterface)
|
2009-06-06 19:04:50 -04:00
|
|
|
{
|
2005-02-10 09:06:08 -05:00
|
|
|
return new ArchiveTestCase<wxArchiveClassFactory>(
|
2005-12-18 08:58:55 -05:00
|
|
|
descr, new wxZipClassFactory,
|
2005-02-10 09:06:08 -05:00
|
|
|
options, archiver, unarchiver);
|
2009-06-06 19:04:50 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return new ZipTestCase(descr, options, archiver, unarchiver);
|
2005-02-10 09:06:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
CPPUNIT_TEST_SUITE_REGISTRATION(ziptest);
|
|
|
|
CPPUNIT_TEST_SUITE_NAMED_REGISTRATION(ziptest, "archive");
|
|
|
|
CPPUNIT_TEST_SUITE_NAMED_REGISTRATION(ziptest, "archive/zip");
|
|
|
|
|
|
|
|
#endif // wxUSE_STREAMS && wxUSE_ZIPSTREAM
|