Конструкция boost::phoenix try_ catch_all не компилируется
Я пишу грамматику boost::spirit::qi для разбора даты.
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/phoenix.hpp>
#include <boost/date_time.hpp>
template < typename InputIterator >
struct date_rfc1123_grammar :
boost::spirit::qi::grammar< InputIterator, boost::gregorian::date()> {
typedef boost::gregorian::date value_type;
date_rfc1123_grammar() : date_rfc1123_grammar::base_type(date)
{
namespace qi = boost::spirit::qi;
namespace phx = boost::phoenix;
using qi::_pass;
using qi::_val;
using qi::_2;
using qi::_3;
using qi::_4;
_2digits = qi::uint_parser< std::uint32_t, 10, 2, 2 >();
_4digits = qi::uint_parser< std::uint32_t, 10, 4, 4 >();
date = (weekday >> ' ' >> _2digits >> ' ' >> month >> ' ' >> _4digits)
[
phx::try_[
_val = phx::construct< value_type >( _4, _3, _2 )
].catch_all[
_pass = false
]
];
}
boost::spirit::qi::rule< InputIterator, value_type()> date;
weekday_grammar weekday;
month_grammar month;
boost::spirit::qi::rule< InputIterator, std::int32_t() > _2digits;
boost::spirit::qi::rule< InputIterator, std::int32_t() > _4digits;
};
Я полагаюсь на boost::gregorian::date
конструктор для проверки аргументов и хотел бы, чтобы синтаксический анализатор потерпел неудачу в случае исключения. Но boost::phoenix::try_[ ].catch_all[ ]
Конструкция не компилируется со следующим сообщением:
/path_to_file/datetime_parse.hpp:102:8: required from ‘tip::http::grammar::parse::date_rfc1123_grammar<InputIterator>::date_rfc1123_grammar() [with InputIterator = boost::spirit::multi_pass<std::istreambuf_iterator<char, std::char_traits<char> > >]’
/path_to_file/grammar_parse_test.hpp:17:7: required from here
/usr/local/include/boost/proto/traits.hpp:341:13: error: static assertion failed: 0 == Expr::proto_arity_c
BOOST_STATIC_ASSERT(0 == Expr::proto_arity_c);
^
Без try_.catch_all
ОК скомпилировать грамматику, но я бы хотел, чтобы парсер перехватил исключение и установил _pass
флаг false, чтобы грамматика не сработала.
Информация об ОС и компиляторе:
$ uname -a
Linux zmij 3.19.0-27-generic #29-Ubuntu SMP Fri Aug 14 21:43:37 UTC 2015 x86_64 x86_64 x86_64 GNU/Linux
$ g++ -v
Thread model: posix
gcc version 4.9.2 (Ubuntu 4.9.2-10ubuntu13)
буст версия 1.58
1 ответ
Решение
Я видел это раньше, может быть локальным для определенных версий boost/compiler.
Обходной путь должен был бы включать неоперативное заявление (например, _pass=_pass
) сделать это последовательность:
date = (weekday >> ' ' >> _2digits >> ' ' >> month >> ' ' >> _4digits)
[
_pass = _pass,
phx::try_[
_val = phx::construct< value_type >( _4, _3, _2 )
].catch_all[
_pass = false
]
];
Посмотри это в прямом эфире на Колиру
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/phoenix.hpp>
#include <boost/date_time/gregorian/greg_date.hpp>
template <typename InputIterator>
struct date_rfc1123_grammar : boost::spirit::qi::grammar< InputIterator, boost::gregorian::date()>
{
typedef boost::gregorian::date value_type;
date_rfc1123_grammar() : date_rfc1123_grammar::base_type(date)
{
namespace qi = boost::spirit::qi;
namespace phx = boost::phoenix;
using qi::_pass;
using qi::_val;
using qi::_2;
using qi::_3;
using qi::_4;
date = (weekday >> ' ' >> _2digits >> ' ' >> month >> ' ' >> _4digits)
[
_pass = _pass,
phx::try_[
_val = phx::construct< value_type >( _4, _3, _2 )
].catch_all[
_pass = false
]
];
}
boost::spirit::qi::rule< InputIterator, value_type()> date;
boost::spirit::qi::rule< InputIterator, uint()> weekday, month;
boost::spirit::qi::uint_parser< std::uint32_t, 10, 2, 2 > _2digits;
boost::spirit::qi::uint_parser< std::uint32_t, 10, 4, 4 > _4digits;
};
int main() {
using It = std::string::const_iterator;
std::string const input;
date_rfc1123_grammar<It> g;
It f = input.begin(), l = input.end();
boost::gregorian::date d;
bool ok = boost::spirit::qi::parse(f, l, g, d);
return ok?1:2;
}