368 lines
11 KiB
C++
368 lines
11 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// Name: tests/textfile/textfile.cpp
|
|
// Purpose: wxTextFile unit test
|
|
// Author: Vadim Zeitlin
|
|
// Created: 2006-03-31
|
|
// Copyright: (c) 2006 Vadim Zeitlin
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// headers
|
|
// ----------------------------------------------------------------------------
|
|
|
|
#include "testprec.h"
|
|
|
|
|
|
#if wxUSE_TEXTFILE
|
|
|
|
#ifndef WX_PRECOMP
|
|
#endif // WX_PRECOMP
|
|
|
|
#include "wx/ffile.h"
|
|
#include "wx/textfile.h"
|
|
|
|
#ifdef __VISUALC__
|
|
#define unlink _unlink
|
|
#endif
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// test class
|
|
// ----------------------------------------------------------------------------
|
|
|
|
class TextFileTestCase : public CppUnit::TestCase
|
|
{
|
|
public:
|
|
TextFileTestCase()
|
|
{
|
|
srand((unsigned)time(NULL));
|
|
}
|
|
|
|
virtual void tearDown() wxOVERRIDE { unlink(GetTestFileName()); }
|
|
|
|
private:
|
|
CPPUNIT_TEST_SUITE( TextFileTestCase );
|
|
CPPUNIT_TEST( ReadEmpty );
|
|
CPPUNIT_TEST( ReadDOS );
|
|
CPPUNIT_TEST( ReadDOSLast );
|
|
CPPUNIT_TEST( ReadUnix );
|
|
CPPUNIT_TEST( ReadUnixLast );
|
|
CPPUNIT_TEST( ReadMac );
|
|
CPPUNIT_TEST( ReadMacLast );
|
|
CPPUNIT_TEST( ReadMixed );
|
|
CPPUNIT_TEST( ReadMixedWithFuzzing );
|
|
CPPUNIT_TEST( ReadCRCRLF );
|
|
#if wxUSE_UNICODE
|
|
CPPUNIT_TEST( ReadUTF8 );
|
|
CPPUNIT_TEST( ReadUTF16 );
|
|
#endif // wxUSE_UNICODE
|
|
CPPUNIT_TEST( ReadBig );
|
|
CPPUNIT_TEST_SUITE_END();
|
|
|
|
void ReadEmpty();
|
|
void ReadDOS();
|
|
void ReadDOSLast();
|
|
void ReadUnix();
|
|
void ReadUnixLast();
|
|
void ReadMac();
|
|
void ReadMacLast();
|
|
void ReadMixed();
|
|
void ReadMixedWithFuzzing();
|
|
void ReadCRCRLF();
|
|
#if wxUSE_UNICODE
|
|
void ReadUTF8();
|
|
void ReadUTF16();
|
|
#endif // wxUSE_UNICODE
|
|
void ReadBig();
|
|
|
|
// return the name of the test file we use
|
|
static const char *GetTestFileName() { return "textfiletest.txt"; }
|
|
|
|
// create the test file with the given contents
|
|
static void CreateTestFile(const char *contents)
|
|
{
|
|
CreateTestFile(strlen(contents), contents);
|
|
}
|
|
|
|
// create the test file with the given contents (version must be used if
|
|
// contents contains NULs)
|
|
static void CreateTestFile(size_t len, const char *contents);
|
|
|
|
|
|
wxDECLARE_NO_COPY_CLASS(TextFileTestCase);
|
|
};
|
|
|
|
// register in the unnamed registry so that these tests are run by default
|
|
CPPUNIT_TEST_SUITE_REGISTRATION( TextFileTestCase );
|
|
|
|
// also include in its own registry so that these tests can be run alone
|
|
CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( TextFileTestCase, "TextFileTestCase" );
|
|
|
|
void TextFileTestCase::CreateTestFile(size_t len, const char *contents)
|
|
{
|
|
FILE *f = fopen(GetTestFileName(), "wb");
|
|
CPPUNIT_ASSERT( f );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( len, fwrite(contents, 1, len, f) );
|
|
CPPUNIT_ASSERT_EQUAL( 0, fclose(f) );
|
|
}
|
|
|
|
void TextFileTestCase::ReadEmpty()
|
|
{
|
|
CreateTestFile("");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName())) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( (size_t)0, f.GetLineCount() );
|
|
CPPUNIT_ASSERT( f.Eof() );
|
|
CPPUNIT_ASSERT_EQUAL( "", f.GetFirstLine() );
|
|
CPPUNIT_ASSERT_EQUAL( "", f.GetLastLine() );
|
|
}
|
|
|
|
void TextFileTestCase::ReadDOS()
|
|
{
|
|
CreateTestFile("foo\r\nbar\r\nbaz");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName())) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( (size_t)3, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Dos, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_None, f.GetLineType(2) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("bar")), f.GetLine(1) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("baz")), f.GetLastLine() );
|
|
}
|
|
|
|
void TextFileTestCase::ReadDOSLast()
|
|
{
|
|
CreateTestFile("foo\r\n");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(GetTestFileName()) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( 1, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Dos, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( "foo", f.GetFirstLine() );
|
|
}
|
|
|
|
void TextFileTestCase::ReadUnix()
|
|
{
|
|
CreateTestFile("foo\nbar\nbaz");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName())) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( (size_t)3, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Unix, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_None, f.GetLineType(2) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("bar")), f.GetLine(1) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("baz")), f.GetLastLine() );
|
|
}
|
|
|
|
void TextFileTestCase::ReadUnixLast()
|
|
{
|
|
CreateTestFile("foo\n");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(GetTestFileName()) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( 1, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Unix, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( "foo", f.GetFirstLine() );
|
|
}
|
|
|
|
void TextFileTestCase::ReadMac()
|
|
{
|
|
CreateTestFile("foo\rbar\r\rbaz");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName())) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( (size_t)4, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Mac, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Mac, f.GetLineType(1) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Mac, f.GetLineType(2) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_None, f.GetLineType(3) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("foo")), f.GetLine(0) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("bar")), f.GetLine(1) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("")), f.GetLine(2) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("baz")), f.GetLastLine() );
|
|
}
|
|
|
|
void TextFileTestCase::ReadMacLast()
|
|
{
|
|
CreateTestFile("foo\r");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(GetTestFileName()) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( 1, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Mac, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( "foo", f.GetFirstLine() );
|
|
}
|
|
|
|
void TextFileTestCase::ReadMixed()
|
|
{
|
|
CreateTestFile("foo\rbar\r\nbaz\n");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName())) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( (size_t)3, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Mac, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Dos, f.GetLineType(1) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Unix, f.GetLineType(2) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("foo")), f.GetFirstLine() );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("bar")), f.GetLine(1) );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(wxT("baz")), f.GetLastLine() );
|
|
}
|
|
|
|
void TextFileTestCase::ReadMixedWithFuzzing()
|
|
{
|
|
for ( int iteration = 0; iteration < 100; iteration++)
|
|
{
|
|
// Create a random buffer with lots of newlines. This is intended to catch
|
|
// bad parsing in unexpected situations such as the one from ReadCRCRLF()
|
|
// (which is so common it deserves a test of its own).
|
|
static const char CHOICES[] = {'\r', '\n', 'X'};
|
|
|
|
const size_t BUF_LEN = 100;
|
|
char data[BUF_LEN + 1];
|
|
data[0] = 'X';
|
|
data[BUF_LEN] = '\0';
|
|
unsigned linesCnt = 0;
|
|
for ( size_t i = 1; i < BUF_LEN; i++ )
|
|
{
|
|
char ch = CHOICES[rand() % WXSIZEOF(CHOICES)];
|
|
data[i] = ch;
|
|
if ( ch == '\r' || (ch == '\n' && data[i-1] != '\r') )
|
|
linesCnt++;
|
|
}
|
|
if (data[BUF_LEN-1] != '\r' && data[BUF_LEN-1] != '\n')
|
|
linesCnt++; // last line was unterminated
|
|
|
|
CreateTestFile(data);
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName())) );
|
|
CPPUNIT_ASSERT_EQUAL( (size_t)linesCnt, f.GetLineCount() );
|
|
}
|
|
}
|
|
|
|
void TextFileTestCase::ReadCRCRLF()
|
|
{
|
|
// Notepad may create files with CRCRLF line endings (see
|
|
// https://stackoverflow.com/questions/6998506/text-file-with-0d-0d-0a-line-breaks).
|
|
// Older versions of wx would loose all data when reading such files.
|
|
// Test that the data are read, but don't worry about empty lines in between or
|
|
// line endings. Also include a longer streak of CRs, because they can
|
|
// happen as well.
|
|
CreateTestFile("foo\r\r\nbar\r\r\r\nbaz\r\r\n");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName())) );
|
|
|
|
wxString all;
|
|
for ( wxString str = f.GetFirstLine(); !f.Eof(); str = f.GetNextLine() )
|
|
all += str;
|
|
|
|
CPPUNIT_ASSERT_EQUAL( "foobarbaz", all );
|
|
}
|
|
|
|
#if wxUSE_UNICODE
|
|
|
|
void TextFileTestCase::ReadUTF8()
|
|
{
|
|
CreateTestFile("\xd0\x9f\n"
|
|
"\xd1\x80\xd0\xb8\xd0\xb2\xd0\xb5\xd1\x82");
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName()), wxConvUTF8) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( (size_t)2, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Unix, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_None, f.GetLineType(1) );
|
|
#ifdef wxHAVE_U_ESCAPE
|
|
CPPUNIT_ASSERT_EQUAL( wxString(L"\u041f"), f.GetFirstLine() );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(L"\u0440\u0438\u0432\u0435\u0442"),
|
|
f.GetLastLine() );
|
|
#endif // wxHAVE_U_ESCAPE
|
|
}
|
|
|
|
void TextFileTestCase::ReadUTF16()
|
|
{
|
|
CreateTestFile(16,
|
|
"\x1f\x04\x0d\x00\x0a\x00"
|
|
"\x40\x04\x38\x04\x32\x04\x35\x04\x42\x04");
|
|
|
|
wxTextFile f;
|
|
wxMBConvUTF16LE conv;
|
|
CPPUNIT_ASSERT( f.Open(wxString::FromAscii(GetTestFileName()), conv) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( (size_t)2, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_Dos, f.GetLineType(0) );
|
|
CPPUNIT_ASSERT_EQUAL( wxTextFileType_None, f.GetLineType(1) );
|
|
|
|
#ifdef wxHAVE_U_ESCAPE
|
|
CPPUNIT_ASSERT_EQUAL( wxString(L"\u041f"), f.GetFirstLine() );
|
|
CPPUNIT_ASSERT_EQUAL( wxString(L"\u0440\u0438\u0432\u0435\u0442"),
|
|
f.GetLastLine() );
|
|
#endif // wxHAVE_U_ESCAPE
|
|
}
|
|
|
|
#endif // wxUSE_UNICODE
|
|
|
|
void TextFileTestCase::ReadBig()
|
|
{
|
|
static const size_t NUM_LINES = 10000;
|
|
|
|
{
|
|
wxFFile f(GetTestFileName(), "w");
|
|
for ( size_t n = 0; n < NUM_LINES; n++ )
|
|
{
|
|
fprintf(f.fp(), "Line %lu\n", (unsigned long)n + 1);
|
|
}
|
|
}
|
|
|
|
wxTextFile f;
|
|
CPPUNIT_ASSERT( f.Open(GetTestFileName()) );
|
|
|
|
CPPUNIT_ASSERT_EQUAL( NUM_LINES, f.GetLineCount() );
|
|
CPPUNIT_ASSERT_EQUAL( wxString("Line 1"), f[0] );
|
|
CPPUNIT_ASSERT_EQUAL( wxString("Line 999"), f[998] );
|
|
CPPUNIT_ASSERT_EQUAL( wxString("Line 1000"), f[999] );
|
|
CPPUNIT_ASSERT_EQUAL( wxString::Format("Line %lu", (unsigned long)NUM_LINES),
|
|
f[NUM_LINES - 1] );
|
|
}
|
|
|
|
#ifdef __LINUX__
|
|
|
|
// Check if using wxTextFile with special files, whose reported size doesn't
|
|
// correspond to the real amount of data in them, works.
|
|
TEST_CASE("wxTextFile::Special", "[textfile][linux][special-file]")
|
|
{
|
|
SECTION("/proc")
|
|
{
|
|
wxTextFile f;
|
|
CHECK( f.Open("/proc/cpuinfo") );
|
|
|
|
// /proc files seem to be always empty in LXC containers, so skip this
|
|
// check there.
|
|
if ( !IsRunningInLXC() )
|
|
CHECK( f.GetLineCount() > 1 );
|
|
}
|
|
|
|
SECTION("/sys")
|
|
{
|
|
wxTextFile f;
|
|
CHECK( f.Open("/sys/power/state") );
|
|
REQUIRE( f.GetLineCount() == 1 );
|
|
INFO( "/sys/power/state contains \"" << f[0] << "\"" );
|
|
CHECK( (f[0].find("mem") != wxString::npos || f[0].find("disk") != wxString::npos) );
|
|
}
|
|
}
|
|
|
|
#endif // __LINUX__
|
|
|
|
#endif // wxUSE_TEXTFILE
|