mirror of
https://github.com/boostorg/xpressive.git
synced 2025-05-11 13:23:56 +00:00
186 lines
5.6 KiB
C++
Executable File
186 lines
5.6 KiB
C++
Executable File
///////////////////////////////////////////////////////////////////////////////
|
|
// test_actions.cpp
|
|
//
|
|
// 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)
|
|
|
|
#define BOOST_XPRESSIVE_BETTER_ERRORS
|
|
|
|
#include <map>
|
|
#include <list>
|
|
#include <stack>
|
|
#include <numeric>
|
|
#include <boost/xpressive/xpressive_static.hpp>
|
|
#include <boost/xpressive/regex_actions.hpp>
|
|
#include <boost/test/unit_test.hpp>
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// test1
|
|
// simple action which builds a string
|
|
void test1()
|
|
{
|
|
using namespace boost::xpressive;
|
|
|
|
std::string result;
|
|
std::string str("foo bar baz foo bar baz");
|
|
sregex rx = (+_w)[ result += _ ] >> *(' ' >> (+_w)[ result += ',' + _ ]);
|
|
|
|
if(!regex_match(str, rx))
|
|
{
|
|
BOOST_ERROR("oops");
|
|
}
|
|
else
|
|
{
|
|
BOOST_CHECK_EQUAL(result, "foo,bar,baz,foo,bar,baz");
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// test2
|
|
// test backtracking over actions
|
|
void test2()
|
|
{
|
|
using namespace boost::xpressive;
|
|
|
|
std::string result;
|
|
std::string str("foo bar baz foo bar baz");
|
|
sregex rx = (+_w)[ result += _ ] >> *(' ' >> (+_w)[ result += ',' + _ ]) >> repeat<4>(_);
|
|
|
|
if(!regex_match(str, rx))
|
|
{
|
|
BOOST_ERROR("oops");
|
|
}
|
|
else
|
|
{
|
|
BOOST_CHECK_EQUAL(result, "foo,bar,baz,foo,bar");
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// test3
|
|
// cast string to int, push back into list
|
|
void test3()
|
|
{
|
|
using namespace boost::xpressive;
|
|
|
|
std::list<int> result;
|
|
std::string str("1 23 456 7890");
|
|
sregex rx = (+_d)[ push_back( result, as<int>(_) ) ]
|
|
>> *(' ' >> (+_w)[ push_back( result, as<int>(_) ) ]);
|
|
|
|
if(!regex_match(str, rx))
|
|
{
|
|
BOOST_ERROR("oops");
|
|
}
|
|
else
|
|
{
|
|
BOOST_REQUIRE_EQUAL(result.size(), 4u);
|
|
BOOST_CHECK_EQUAL(*result.begin(), 1);
|
|
BOOST_CHECK_EQUAL(*++result.begin(), 23);
|
|
BOOST_CHECK_EQUAL(*++++result.begin(), 456);
|
|
BOOST_CHECK_EQUAL(*++++++result.begin(), 7890);
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// test4
|
|
// build a map of strings to integers
|
|
void test4()
|
|
{
|
|
using namespace boost::xpressive;
|
|
|
|
std::map<std::string, int> result;
|
|
std::string str("aaa=>1 bbb=>23 ccc=>456");
|
|
sregex pair = ( (s1= +_w) >> "=>" >> (s2= +_d) )[ ref(result)[s1] = as<int>(s2) ];
|
|
sregex rx = pair >> *(+_s >> pair);
|
|
|
|
if(!regex_match(str, rx))
|
|
{
|
|
BOOST_ERROR("oops");
|
|
}
|
|
else
|
|
{
|
|
BOOST_REQUIRE_EQUAL(result.size(), 3u);
|
|
BOOST_CHECK_EQUAL(result["aaa"], 1);
|
|
BOOST_CHECK_EQUAL(result["bbb"], 23);
|
|
BOOST_CHECK_EQUAL(result["ccc"], 456);
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// test5
|
|
// calculator that calculates. This is just silly, but hey.
|
|
void test5()
|
|
{
|
|
using namespace boost::xpressive;
|
|
|
|
int left = 0, right = 0;
|
|
std::stack<int> stack;
|
|
std::string str("4+5*(1+1)");
|
|
|
|
sregex group, factor, term, expression;
|
|
|
|
group = '(' >> by_ref(expression) >> ')';
|
|
factor = (+_d)[ push(stack, as<int>(_)) ] | group;
|
|
term = factor >> *(
|
|
('*' >> factor)
|
|
[ ref(right) = top(stack)
|
|
, pop(stack)
|
|
, ref(left) = top(stack)
|
|
, pop(stack)
|
|
, push(stack, ref(left) * ref(right))
|
|
]
|
|
| ('/' >> factor)
|
|
[ ref(right) = top(stack)
|
|
, pop(stack)
|
|
, ref(left) = top(stack)
|
|
, pop(stack)
|
|
, push(stack, ref(left) / ref(right))
|
|
]
|
|
);
|
|
expression = term >> *(
|
|
('+' >> term)
|
|
[ ref(right) = top(stack)
|
|
, pop(stack)
|
|
, ref(left) = top(stack)
|
|
, pop(stack)
|
|
, push(stack, ref(left) + ref(right))
|
|
]
|
|
| ('-' >> term)
|
|
[ ref(right) = top(stack)
|
|
, pop(stack)
|
|
, ref(left) = top(stack)
|
|
, pop(stack)
|
|
, push(stack, ref(left) - ref(right))
|
|
]
|
|
);
|
|
|
|
if(!regex_match(str, expression))
|
|
{
|
|
BOOST_ERROR("oops");
|
|
}
|
|
else
|
|
{
|
|
BOOST_REQUIRE_EQUAL(stack.size(), 1u);
|
|
BOOST_CHECK_EQUAL(stack.top(), 14);
|
|
}
|
|
}
|
|
|
|
using namespace boost::unit_test;
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// init_unit_test_suite
|
|
//
|
|
test_suite* init_unit_test_suite( int argc, char* argv[] )
|
|
{
|
|
test_suite *test = BOOST_TEST_SUITE("test_actions");
|
|
test->add(BOOST_TEST_CASE(&test1));
|
|
test->add(BOOST_TEST_CASE(&test2));
|
|
test->add(BOOST_TEST_CASE(&test3));
|
|
test->add(BOOST_TEST_CASE(&test4));
|
|
test->add(BOOST_TEST_CASE(&test5));
|
|
return test;
|
|
}
|
|
|