xpressive/test/misc1.cpp
2006-01-04 02:37:26 +00:00

244 lines
7.0 KiB
C++
Executable File

///////////////////////////////////////////////////////////////////////////////
// misc1.hpp
//
// Copyright 2004 Eric Niebler. Distributed under the Boost
// Software License, Version 1.0. (See accompanying file
// LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
#include <iostream>
#include <boost/xpressive/xpressive.hpp>
#include <boost/xpressive/traits/cpp_regex_traits.hpp>
#include "./test_minimal.hpp"
using namespace boost::xpressive;
void test1()
{
// make sure the following compiles:
sregex a = _;
sregex b = _;
sregex c = a >> b;
c = a | b;
c = !a;
c = *a;
c = +a;
}
///////////////////////////////////////////////////////////////////////////////
// test for basic_regex in a keep
//
void test2()
{
std::locale loc;
std::string str("Its a mad Mad mAd maD world");
sregex word = +_w;
sregex sentence = imbue(loc)(*(keep(word) >> +_s) >> word);
smatch what;
BOOST_CHECK(regex_match(str, what, sentence));
BOOST_CHECK(what.nested_results().size() == 7);
BOOST_CHECK(what(word, 0)[0] == "Its");
BOOST_CHECK(what(word, 1)[0] == "a");
BOOST_CHECK(what(word, 2)[0] == "mad");
BOOST_CHECK(what(word, 3)[0] == "Mad");
BOOST_CHECK(what(word, 4)[0] == "mAd");
BOOST_CHECK(what(word, 5)[0] == "maD");
BOOST_CHECK(what(word, 6)[0] == "world");
}
///////////////////////////////////////////////////////////////////////////////
// test for a simple non-recursive grammar
//
void test3()
{
// test for a simple regex grammar
std::string buffer =
"FROGGIE\r\n"
"Volume = 1\r\n"
"Other1= 2\r\n"
"Channel=3\r\n"
"Other =4\r\n"
"\r\n"
"FROGGIE\r\n"
"Volume = 5\r\n"
"Other1= 6\r\n"
"Channel=7\r\n"
"Other =8\r\n"
"\r\n"
"FROGGIE\r\n"
"Volume = 9\r\n"
"Other1= 0\r\n"
"Channel=10\r\n"
"\r\n";
mark_tag name(1), value(2);
sregex name_value_pair_ =
(name= +alnum) >> *_s >> "=" >> *_s >>
(value= +_d) >> *_s >> _ln;
sregex message_ =
*_s >> "FROGGIE" >> _ln >> +name_value_pair_ >> _ln;
sregex re_ = +message_;
smatch::nested_results_type::const_iterator msg, nvp;
smatch tmpwhat;
BOOST_REQUIRE(regex_search(buffer, tmpwhat, re_));
// for giggles, make a deep-copy of the tree of results
smatch what = tmpwhat;
BOOST_REQUIRE(3 == what.nested_results().size());
msg = what.nested_results().begin();
BOOST_REQUIRE(4 == msg->nested_results().size());
nvp = msg->nested_results().begin();
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Volume" == (*nvp)[name]);
BOOST_CHECK("1" == (*nvp)[value]);
++nvp;
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Other1" == (*nvp)[name]);
BOOST_CHECK("2" == (*nvp)[value]);
++nvp;
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Channel" == (*nvp)[name]);
BOOST_CHECK("3" == (*nvp)[value]);
++nvp;
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Other" == (*nvp)[name]);
BOOST_CHECK("4" == (*nvp)[value]);
++msg;
BOOST_REQUIRE(4 == msg->nested_results().size());
nvp = msg->nested_results().begin();
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Volume" == (*nvp)[name]);
BOOST_CHECK("5" == (*nvp)[value]);
++nvp;
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Other1" == (*nvp)[name]);
BOOST_CHECK("6" == (*nvp)[value]);
++nvp;
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Channel" == (*nvp)[name]);
BOOST_CHECK("7" == (*nvp)[value]);
++nvp;
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Other" == (*nvp)[name]);
BOOST_CHECK("8" == (*nvp)[value]);
++msg;
BOOST_REQUIRE(3 == msg->nested_results().size());
nvp = msg->nested_results().begin();
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Volume" == (*nvp)[name]);
BOOST_CHECK("9" == (*nvp)[value]);
++nvp;
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Other1" == (*nvp)[name]);
BOOST_CHECK("0" == (*nvp)[value]);
++nvp;
BOOST_REQUIRE(3 == nvp->size());
BOOST_CHECK("Channel" == (*nvp)[name]);
BOOST_CHECK("10" == (*nvp)[value]);
}
///////////////////////////////////////////////////////////////////////////////
// test for a self-recursive regex
//
void test4()
{
sregex parentheses;
parentheses // A balanced set of parentheses ...
= '(' // is an opening parenthesis ...
>> // followed by ...
*( // zero or more ...
keep( +~(set='(',')') ) // of a bunch of things that are not parentheses ...
| // or ...
by_ref(parentheses) // a balanced set of parentheses
) // (ooh, recursion!) ...
>> // followed by ...
')' // a closing parenthesis
;
smatch what;
smatch::nested_results_type::const_iterator pwhat, pwhat2;
std::string str( "blah blah( a(b)c (c(e)f (g)h )i (j)6 )blah" );
BOOST_REQUIRE(regex_search(str, what, parentheses));
BOOST_REQUIRE(1 == what.size());
BOOST_CHECK("( a(b)c (c(e)f (g)h )i (j)6 )" == what[0]);
BOOST_REQUIRE(3 == what.nested_results().size());
pwhat = what.nested_results().begin();
BOOST_REQUIRE(1 == pwhat->size());
BOOST_CHECK("(b)" == (*pwhat)[0]);
++pwhat;
BOOST_REQUIRE(1 == pwhat->size());
BOOST_CHECK("(c(e)f (g)h )" == (*pwhat)[0]);
BOOST_REQUIRE(2 == pwhat->nested_results().size());
pwhat2 = pwhat->nested_results().begin();
BOOST_REQUIRE(1 == pwhat2->size());
BOOST_CHECK("(e)" == (*pwhat2)[0]);
++pwhat2;
BOOST_REQUIRE(1 == pwhat2->size());
BOOST_CHECK("(g)" == (*pwhat2)[0]);
++pwhat;
BOOST_REQUIRE(1 == pwhat->size());
BOOST_CHECK("(j)" == (*pwhat)[0]);
}
///////////////////////////////////////////////////////////////////////////////
// test for a sub-match scoping
//
void test5()
{
sregex inner = sregex::compile( "(.)\\1" );
sregex outer = (s1= _) >> inner >> s1;
std::string abba("ABBA");
BOOST_CHECK(regex_match(abba, outer));
}
///////////////////////////////////////////////////////////////////////////////
// Ye olde calculator. Test recursive grammar.
//
void test6()
{
sregex group, factor, term, expression;
group = '(' >> by_ref(expression) >> ')';
factor = +_d | group;
term = factor >> *(('*' >> factor) | ('/' >> factor));
expression = term >> *(('+' >> term) | ('-' >> term));
smatch what;
std::string str("foo 9*(10+3) bar");
BOOST_REQUIRE(regex_search(str, what, expression));
BOOST_CHECK("9*(10+3)" == what[0]);
}
///////////////////////////////////////////////////////////////////////////////
// test_main
//
int test_main( int, char*[] )
{
test1();
test2();
test3();
test4();
test5();
test6();
return 0;
}