2
0
mirror of https://github.com/boostorg/spirit.git synced 2026-01-19 04:42:11 +00:00
Files
spirit/test/x3/rule3.cpp
Nana Sakisaka 3df9ca3788 Modernize x3::action (#815)
`x3::action` now uses concepts to dispatch `f(ctx)` or `f()`, depending on 
whether `f` accepts such signature.

The dispatching implementation is moved from `call.hpp` to `action.hpp`,
 as the function is only used from the `x3::action` class.

`operator/`: Deprecated. The symbol `/` normally means "ordered choice"
in PEG, and is irrelevant to semantic actions. Furthermore, using C++'s 
`operator/` for this purpose may introduce surprising behavior when it's 
mixed with ordinary PEG operators, for instance, the unary `operator+`, 
due to precedence.
2025-09-10 07:55:14 +09:00

179 lines
4.3 KiB
C++

/*=============================================================================
Copyright (c) 2001-2012 Joel de Guzman
Copyright (c) 2025 Nana Sakisaka
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 "test.hpp"
#include <boost/spirit/home/x3.hpp>
#include <boost/fusion/include/adapt_struct.hpp>
#include <boost/fusion/include/std_pair.hpp>
#include <boost/variant.hpp>
#include <string>
#include <vector>
#include <cstring>
#include <iostream>
#ifdef _MSC_VER
// bogus https://developercommunity.visualstudio.com/t/buggy-warning-c4709/471956
# pragma warning(disable: 4709) // comma operator within array index expression
#endif
using boost::spirit::x3::_val;
namespace x3 = boost::spirit::x3;
struct f
{
template <typename Context>
void operator()(Context const& ctx) const
{
_val(ctx) += _attr(ctx);
}
};
struct stationary
{
explicit stationary(int i) : val{i} {}
stationary(stationary const&) = delete;
stationary& operator=(int i) { val = i; return *this; }
int val;
};
namespace check_stationary {
boost::spirit::x3::rule<class a_r, stationary> const a;
boost::spirit::x3::rule<class b_r, stationary> const b;
auto const a_def = '{' >> boost::spirit::x3::int_ >> '}';
auto const b_def = a;
BOOST_SPIRIT_X3_DEFINE(a)
BOOST_SPIRIT_X3_DEFINE(b)
}
namespace check_recursive {
using node_t = boost::make_recursive_variant<
int,
std::vector<boost::recursive_variant_>
>::type;
boost::spirit::x3::rule<class grammar_r, node_t> const grammar;
auto const grammar_def = '[' >> grammar % ',' >> ']' | boost::spirit::x3::int_;
BOOST_SPIRIT_X3_DEFINE(grammar)
}
namespace check_recursive_scoped {
using check_recursive::node_t;
x3::rule<class intvec_r, node_t> const intvec;
auto const grammar = intvec = '[' >> intvec % ',' >> ']' | x3::int_;
}
struct recursive_tuple
{
int value;
std::vector<recursive_tuple> children;
};
BOOST_FUSION_ADAPT_STRUCT(recursive_tuple,
value, children)
// regression test for #461
namespace check_recursive_tuple {
x3::rule<class grammar_r, recursive_tuple> const grammar;
auto const grammar_def = x3::int_ >> ('{' >> grammar % ',' >> '}' | x3::eps);
BOOST_SPIRIT_X3_DEFINE(grammar)
BOOST_SPIRIT_X3_INSTANTIATE(decltype(grammar), char const*, x3::unused_type)
}
int main()
{
using spirit_test::test_attr;
using spirit_test::test;
using namespace boost::spirit::x3::standard;
using boost::spirit::x3::rule;
using boost::spirit::x3::lit;
using boost::spirit::x3::eps;
using boost::spirit::x3::unused_type;
// synth attribute value-init
{
std::string s;
typedef rule<class r, std::string> rule_type;
auto rdef = rule_type()
= alpha [f()]
;
BOOST_TEST(test_attr("abcdef", +rdef, s));
BOOST_TEST(s == "abcdef");
}
// synth attribute value-init
{
std::string s;
typedef rule<class r, std::string> rule_type;
auto rdef = rule_type() =
alpha[([](auto& ctx) {
_val(ctx) += _attr(ctx);
})]
;
BOOST_TEST(test_attr("abcdef", +rdef, s));
BOOST_TEST(s == "abcdef");
}
{
auto r = rule<class r_id, int>{} = eps[([] (auto& ctx) {
using boost::spirit::x3::_val;
static_assert(
std::is_same_v<std::decay_t<decltype(_val(ctx))>, unused_type>,
"Attribute must not be synthesized"
);
})];
BOOST_TEST(test("", r));
}
// ensure no unneeded synthesization, copying and moving occurred
{
stationary st { 0 };
BOOST_TEST(test_attr("{42}", check_stationary::b, st));
BOOST_TEST_EQ(st.val, 42);
}
{
using namespace check_recursive;
node_t v;
BOOST_TEST(test_attr("[4,2]", grammar, v));
BOOST_TEST((node_t{std::vector<node_t>{{4}, {2}}} == v));
}
{
using namespace check_recursive_scoped;
node_t v;
BOOST_TEST(test_attr("[4,2]", grammar, v));
BOOST_TEST((node_t{std::vector<node_t>{{4}, {2}}} == v));
}
return boost::report_errors();
}