You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdepim/libksieve/tests/parsertest.cpp

669 lines
18 KiB

/* -*- c++ -*-
tests/parsertest.cpp
This file is part of the testsuite of KSieve,
the KDE internet mail/usenet news message filtering library.
Copyright (c) 2003 Marc Mutz <mutz@kde.org>
KSieve is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License, version 2, as
published by the Free Software Foundation.
KSieve 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
General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
In addition, as a special exception, the copyright holders give
permission to link the code of this program with any edition of
the TQt library by Trolltech AS, Norway (or with modified versions
of TQt that use the same license as TQt), and distribute linked
combinations including the two. You must obey the GNU General
Public License in all respects for all of the code used other than
TQt. If you modify this file, you may extend this exception to
your version of the file, but you are not obligated to do so. If
you do not wish to do so, delete this exception statement from
your version.
*/
#include <config.h>
#include <cstdlib>
#include <ksieve/parser.h>
using KSieve::Parser;
#include <ksieve/error.h>
#include <ksieve/scriptbuilder.h>
#include <tqcstring.h> // tqstrlen
#include <tqstring.h>
#include <iostream>
using std::cout;
using std::cerr;
using std::endl;
#include <cassert>
enum BuilderMethod {
TaggedArgument,
StringArgument,
NumberArgument,
CommandStart,
CommandEnd,
TestStart,
TestEnd,
TestListStart,
TestListEnd,
BlockStart,
BlockEnd,
StringListArgumentStart,
StringListEntry,
StringListArgumentEnd,
HashComment,
BracketComment,
Error,
Finished
};
static const unsigned int MAX_RESPONSES = 100;
struct TestCase {
const char * name;
const char * script;
struct Response {
BuilderMethod method;
const char * string;
bool boolean;
} responses[MAX_RESPONSES];
} testCases[] = {
//
// single commands:
//
{ "Null script",
0,
{ { Finished, 0, false } }
},
{ "Empty script",
"",
{ { Finished, 0, false } }
},
{ "WS-only script",
" \t\n\r\n",
{ { Finished, 0, false } }
},
{ "Bare hash comment",
"#comment",
{ { HashComment, "comment", false },
{ Finished, 0, false } }
},
{ "Bare bracket comment",
"/*comment*/",
{ { BracketComment, "comment", false },
{ Finished, 0, false } }
},
{ "Bare command",
"command;",
{ { CommandStart, "command", false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "Bare command - missing semicolon",
"command",
{ { CommandStart, "command", false },
{ Error, "MissingSemicolonOrBlock", false } }
},
{ "surrounded by bracket comments",
"/*comment*/command/*comment*/;/*comment*/",
{ { BracketComment, "comment", false },
{ CommandStart, "command", false },
{ BracketComment, "comment", false },
{ CommandEnd, 0, false },
{ BracketComment, "comment", false },
{ Finished, 0, false } }
},
{ "surrounded by hash comments",
"#comment\ncommand#comment\n;#comment",
{ { HashComment, "comment", false },
{ CommandStart, "command", false },
{ HashComment, "comment", false },
{ CommandEnd, 0, false },
{ HashComment, "comment", false },
{ Finished, 0, false } }
},
{ "single tagged argument",
"command :tag;",
{ { CommandStart, "command", false },
{ TaggedArgument, "tag", false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single tagged argument - missing semicolon",
"command :tag",
{ { CommandStart, "command", false },
{ TaggedArgument, "tag", false },
{ Error, "MissingSemicolonOrBlock", false } }
},
{ "single string argument - quoted string",
"command \"string\";",
{ { CommandStart, "command", false },
{ StringArgument, "string", false /*quoted*/ },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single string argument - multi-line string",
"command text:\nstring\n.\n;",
{ { CommandStart, "command", false },
{ StringArgument, "string", true /*multiline*/ },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single number argument - 100",
"command 100;",
{ { CommandStart, "command", false },
{ NumberArgument, "100 ", false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single number argument - 100k",
"command 100k;",
{ { CommandStart, "command", false },
{ NumberArgument, "102400k", false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single number argument - 100M",
"command 100M;",
{ { CommandStart, "command", false },
{ NumberArgument, "104857600M", false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single number argument - 2G",
"command 2G;",
{ { CommandStart, "command", false },
{ NumberArgument, "2147483648G", false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
#if SIZEOF_UNSIGNED_LONG == 8
# define ULONG_MAX_STRING "18446744073709551615"
# define ULONG_MAXP1_STRING "18446744073709551616"
#elif SIZEOF_UNSIGNED_LONG == 4
# define ULONG_MAX_STRING "4294967295"
# define ULONG_MAXP1_STRING "4G"
#else
# error sizeof( unsigned long ) != 4 && sizeof( unsigned long ) != 8 ???
#endif
{ "single number argument - ULONG_MAX + 1",
"command " ULONG_MAXP1_STRING ";",
{ { CommandStart, "command", false },
{ Error, "NumberOutOfRange", false } }
},
{ "single number argument - ULONG_MAX",
"command " ULONG_MAX_STRING ";",
{ { CommandStart, "command", false },
{ NumberArgument, ULONG_MAX_STRING " ", false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single one-element string list argument - quoted string",
"command [\"string\"];",
{ { CommandStart, "command", false },
{ StringListArgumentStart, 0, false },
{ StringListEntry, "string", false /*quoted*/ },
{ StringListArgumentEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single one-element string list argument - multi-line string",
"command [text:\nstring\n.\n];",
{ { CommandStart, "command", false },
{ StringListArgumentStart, 0, false },
{ StringListEntry, "string", true /*multiline*/ },
{ StringListArgumentEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single two-element string list argument - quoted strings",
"command [\"string\",\"string\"];",
{ { CommandStart, "command", false },
{ StringListArgumentStart, 0, false },
{ StringListEntry, "string", false /*quoted*/ },
{ StringListEntry, "string", false /*quoted*/ },
{ StringListArgumentEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single two-element string list argument - multi-line strings",
"command [text:\nstring\n.\n,text:\nstring\n.\n];",
{ { CommandStart, "command", false },
{ StringListArgumentStart, 0, false },
{ StringListEntry, "string", true /*multiline*/ },
{ StringListEntry, "string", true /*multiline*/ },
{ StringListArgumentEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single two-element string list argument - quoted + multi-line strings",
"command [\"string\",text:\nstring\n.\n];",
{ { CommandStart, "command", false },
{ StringListArgumentStart, 0, false },
{ StringListEntry, "string", false /*quoted*/ },
{ StringListEntry, "string", true /*multiline*/ },
{ StringListArgumentEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single two-element string list argument - multi-line + quoted strings",
"command [text:\nstring\n.\n,\"string\"];",
{ { CommandStart, "command", false },
{ StringListArgumentStart, 0, false },
{ StringListEntry, "string", true /*multiline*/ },
{ StringListEntry, "string", false /*quoted*/ },
{ StringListArgumentEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "single bare test argument",
"command test;",
{ { CommandStart, "command", false },
{ TestStart, "test", false },
{ TestEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "one-element test list argument",
"command(test);",
{ { CommandStart, "command", false },
{ TestListStart, 0, false },
{ TestStart, "test", false },
{ TestEnd, 0, false },
{ TestListEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "two-element test list argument",
"command(test,test);",
{ { CommandStart, "command", false },
{ TestListStart, 0, false },
{ TestStart, "test", false },
{ TestEnd, 0, false },
{ TestStart, "test", false },
{ TestEnd, 0, false },
{ TestListEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "zero-element block",
"command{}",
{ { CommandStart, "command", false },
{ BlockStart, 0, false },
{ BlockEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "one-element block",
"command{command;}",
{ { CommandStart, "command", false },
{ BlockStart, 0, false },
{ CommandStart, "command", false },
{ CommandEnd, 0, false },
{ BlockEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "two-element block",
"command{command;command;}",
{ { CommandStart, "command", false },
{ BlockStart, 0, false },
{ CommandStart, "command", false },
{ CommandEnd, 0, false },
{ CommandStart, "command", false },
{ CommandEnd, 0, false },
{ BlockEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
{ "command with a test with a test with a test",
"command test test test;",
{ { CommandStart, "command", false },
{ TestStart, "test", false },
{ TestStart, "test", false },
{ TestStart, "test", false },
{ TestEnd, 0, false },
{ TestEnd, 0, false },
{ TestEnd, 0, false },
{ CommandEnd, 0, false },
{ Finished, 0, false } }
},
};
static const int numTestCases = sizeof testCases / sizeof *testCases ;
// Prints out the parse tree in XML-like format. For visual inspection
// (manual tests).
class PrintingScriptBuilder : public KSieve::ScriptBuilder {
public:
PrintingScriptBuilder()
: KSieve::ScriptBuilder(), indent( 0 )
{
write( "<script type=\"application/sieve\">" );
++indent;
}
virtual ~PrintingScriptBuilder() {}
void taggedArgument( const TQString & tag ) {
write( "tag", tag );
}
void stringArgument( const TQString & string, bool multiLine, const TQString & /*fixme*/ ) {
write( multiLine ? "string type=\"multiline\"" : "string type=\"quoted\"", string );
}
void numberArgument( unsigned long number, char quantifier ) {
const TQString txt = "number" + ( quantifier ? TQString(" quantifier=\"%1\"").arg( quantifier ) : TQString() ) ;
write( txt.latin1(), TQString::number( number ) );
}
void commandStart( const TQString & identifier ) {
write( "<command>" );
++indent;
write( "identifier", identifier );
}
void commandEnd() {
--indent;
write( "</command>" );
}
void testStart( const TQString & identifier ) {
write( "<test>" );
++indent;
write( "identifier", identifier );
}
void testEnd() {
--indent;
write( "</test>" );
}
void testListStart() {
write( "<testlist>" );
++indent;
}
void testListEnd() {
--indent;
write( "</testlist>" );
}
void blockStart() {
write( "<block>" );
++indent;
}
void blockEnd() {
--indent;
write( "</block>" );
}
void stringListArgumentStart() {
write( "<stringlist>" );
++indent;
}
void stringListArgumentEnd() {
--indent;
write( "</stringlist>" );
}
void stringListEntry( const TQString & string, bool multiline, const TQString & hashComment ) {
stringArgument( string, multiline, hashComment );
}
void hashComment( const TQString & comment ) {
write( "comment type=\"hash\"", comment );
}
void bracketComment( const TQString & comment ) {
write( "comment type=\"bracket\"", comment );
}
void lineFeed() {
write( "<crlf/>" );
}
void error( const KSieve::Error & error ) {
indent = 0;
write( ("Error: " + error.asString()).latin1() );
}
void finished() {
--indent;
write( "</script>" );
}
private:
int indent;
void write( const char * msg ) {
for ( int i = 2*indent ; i > 0 ; --i )
cout << " ";
cout << msg << endl;
}
void write( const TQCString & key, const TQString & value ) {
if ( value.isEmpty() ) {
write( "<" + key + "/>" );
return;
}
write( "<" + key + ">" );
++indent;
write( value.utf8().data() );
--indent;
write( "</" + key + ">" );
}
};
// verifes that methods get called with expected arguments (and in
// expected sequence) as specified by the TestCase. For automated
// tests.
class VerifyingScriptBuilder : public KSieve::ScriptBuilder {
public:
VerifyingScriptBuilder( const TestCase & testCase )
: KSieve::ScriptBuilder(),
mNextResponse( 0 ), mTestCase( testCase ), mOk( true )
{
}
virtual ~VerifyingScriptBuilder() {}
bool ok() const { return mOk; }
void taggedArgument( const TQString & tag ) {
checkIs( TaggedArgument );
checkEquals( tag );
++mNextResponse;
}
void stringArgument( const TQString & string, bool multiline, const TQString & /*fixme*/ ) {
checkIs( StringArgument );
checkEquals( string );
checkEquals( multiline );
++mNextResponse;
}
void numberArgument( unsigned long number, char quantifier ) {
checkIs( NumberArgument );
checkEquals( TQString::number( number ) + ( quantifier ? quantifier : ' ' ) );
++mNextResponse;
}
void commandStart( const TQString & identifier ) {
checkIs( CommandStart );
checkEquals( identifier );
++mNextResponse;
}
void commandEnd() {
checkIs( CommandEnd );
++mNextResponse;
}
void testStart( const TQString & identifier ) {
checkIs( TestStart );
checkEquals( identifier );
++mNextResponse;
}
void testEnd() {
checkIs( TestEnd );
++mNextResponse;
}
void testListStart() {
checkIs( TestListStart );
++mNextResponse;
}
void testListEnd() {
checkIs( TestListEnd );
++mNextResponse;
}
void blockStart() {
checkIs( BlockStart );
++mNextResponse;
}
void blockEnd() {
checkIs( BlockEnd );
++mNextResponse;
}
void stringListArgumentStart() {
checkIs( StringListArgumentStart );
++mNextResponse;
}
void stringListEntry( const TQString & string, bool multiLine, const TQString & /*fixme*/ ) {
checkIs( StringListEntry );
checkEquals( string );
checkEquals( multiLine );
++mNextResponse;
}
void stringListArgumentEnd() {
checkIs( StringListArgumentEnd );
++mNextResponse;
}
void hashComment( const TQString & comment ) {
checkIs( HashComment );
checkEquals( comment );
++mNextResponse;
}
void bracketComment( const TQString & comment ) {
checkIs( BracketComment );
checkEquals( comment );
++mNextResponse;
}
void lineFeed() {
// FIXME
}
void error( const KSieve::Error & error ) {
checkIs( Error );
checkEquals( TQString( KSieve::Error::typeToString( error.type() ) ) );
++mNextResponse;
}
void finished() {
checkIs( Finished );
//++mNextResponse (no!)
}
private:
const TestCase::Response & currentResponse() const {
assert( mNextResponse <= MAX_RESPONSES );
return mTestCase.responses[mNextResponse];
}
void checkIs( BuilderMethod m ) {
if ( currentResponse().method != m ) {
cerr << " expected method " << (int)currentResponse().method
<< ", got " << (int)m;
mOk = false;
}
}
void checkEquals( const TQString & s ) {
if ( s != TQString::fromUtf8( currentResponse().string ) ) {
cerr << " expected string arg \""
<< ( currentResponse().string ? currentResponse().string : "<null>" )
<< "\", got \"" << ( s.isNull() ? "<null>" : s.utf8().data() ) << "\"";
mOk = false;
}
}
void checkEquals( bool b ) {
if ( b != currentResponse().boolean ) {
cerr << " expected boolean arg <" << currentResponse().boolean
<< ">, got <" << b << ">";
mOk = false;
}
}
unsigned int mNextResponse;
const TestCase & mTestCase;
bool mOk;
};
int main( int argc, char * argv[] ) {
if ( argc == 2 ) { // manual test
const char * scursor = argv[1];
const char * const send = argv[1] + tqstrlen( argv[1] );
Parser parser( scursor, send );
PrintingScriptBuilder psb;
parser.setScriptBuilder( &psb );
if ( parser.parse() )
cout << "ok" << endl;
else
cout << "bad" << endl;
} else if ( argc == 1 ) { // automated test
bool success = true;
for ( int i = 0 ; i < numTestCases ; ++i ) {
const TestCase & t = testCases[i];
cerr << t.name << ":";
VerifyingScriptBuilder v( t );
Parser p( t.script, t.script + tqstrlen( t.script ) );
p.setScriptBuilder( &v );
const bool ok = p.parse();
if ( v.ok() )
if ( ok )
cerr << " ok";
else
cerr << " xfail";
else
success = false;
cerr << endl;
}
if ( !success )
exit( 1 );
} else { // usage error
cerr << "usage: parsertest [ <string> ]" << endl;
exit( 1 );
}
return 0;
}