29 namespace _{ char x[] = #x; } \
30 using x = xtd::parse::string< decltype(_::x), _::x>;
38 #define STRING(_name, _value) \
39 namespace _{ char _name[] = _value; } \
40 using _name = xtd::parse::string< decltype(_::_name), _::_name>;
50 #define CHARACTER_(_name, _value) \
52 using _name = xtd::parse::character<_value>
61 #define REGEX(_name, _value) \
62 namespace _{ char _name[] = _value; } \
63 using _name = xtd::parse::regex< decltype(_::_name), _::_name>;
72 using pointer_type = std::shared_ptr<rule_base>;
73 using vector_type = std::vector<pointer_type>;
78 template <
typename ... _ChildRuleTs>
79 explicit rule_base(_ChildRuleTs&& ... oChildRules) : _Items{ std::forward<_ChildRuleTs>(oChildRules)... }{}
86 virtual bool isa(
const std::type_info& oType)
const = 0;
88 virtual const std::type_info&
type()
const = 0;
92 const vector_type&
items()
const{
return _Items; }
104 template <
typename _DeclT,
typename _ImplT = _DeclT>
107 using decl_type = _DeclT;
108 using impl_type = _ImplT;
111 template <
typename ... _ChildRuleTs>
112 explicit rule(_ChildRuleTs&& ... oChildRules) :
rule_base(std::forward<_ChildRuleTs>(oChildRules)...){}
114 virtual ~rule() =
default;
116 virtual bool isa(
const std::type_info& oType)
const override{
120 virtual const std::type_info&
type()
const override{
121 return typeid(_DeclT);
131 template <
typename ...>
class and_;
136 template <
typename ... _ChildRuleTs>
137 explicit and_(_ChildRuleTs&& ... oChildRules) : _super_t(std::forward<_ChildRuleTs>(oChildRules)...){}
140 template <
typename _HeadT,
typename ... _TailT>
class and_<_HeadT, _TailT...> :
public rule<and_<_HeadT, _TailT...>>{
142 using _super_t =
rule<
and_<_HeadT, _TailT...>>;
143 template <
typename ... _ChildRuleTs>
144 explicit and_(_ChildRuleTs&& ... oChildRules) : _super_t(std::forward<_ChildRuleTs>(oChildRules)...){}
153 template <
typename ...>
class or_;
158 template <
typename ... _ChildRuleTs>
159 explicit or_(_ChildRuleTs&& ... oChildRules) : _super_t(std::forward<_ChildRuleTs>(oChildRules)...){}
162 template <
typename _HeadT,
typename ... _TailT>
class or_<_HeadT, _TailT...> :
public rule<or_<_HeadT, _TailT...>>{
164 using _super_t =
rule<
or_<_HeadT, _TailT...>>;
165 template <
typename ... _ChildRuleTs>
166 explicit or_(_ChildRuleTs&& ... oChildRules) : _super_t(std::forward<_ChildRuleTs>(oChildRules)...){}
179 template <
typename ... _ChildRuleTs>
180 explicit one_or_more_(_ChildRuleTs&& ... oChildRules) :
_super_t(std::forward<_ChildRuleTs>(oChildRules)...){}
193 template <
typename ... _ChildRuleTs>
194 explicit zero_or_more_(_ChildRuleTs&& ... oChildRules) :
_super_t(std::forward<_ChildRuleTs>(oChildRules)...){}
205 template <
typename ... _ChildRuleTs>
206 explicit zero_or_one_(_ChildRuleTs&& ... oChildRules) :
_super_t(std::forward<_ChildRuleTs>(oChildRules)...){}
224 template <
typename _Ty, _Ty &>
class string;
226 template <
size_t _len,
char(&_str)[_len]>
class string<char[_len], _str> :
public rule<string<char[_len], _str>>{
229 static constexpr
size_t length = _len;
243 template <
typename _Ty, _Ty &>
class regex;
245 template <
size_t _len,
char(&_str)[_len]>
class regex<char[_len], _str> :
public rule<regex<char[_len], _str>>{
248 static constexpr
size_t length = _len;
249 explicit regex(
const std::string& newval) : _super_t(), _value(newval){}
250 const std::string& value()
const{
return _value; }
263 template <
typename _DeclT,
typename _ImplT,
bool _IgnoreCase,
typename _WhitespaceT>
class parse_helper;
266 template <
typename _DeclT,
size_t _len,
char(&_str)[_len],
typename _WhitespaceT>
267 class parse_helper<_DeclT, parse::string<char[_len], _str>, false, _WhitespaceT>{
269 template <
typename _IteratorT>
static rule_base::pointer_type parse(_IteratorT& begin, _IteratorT& end){
270 _IteratorT oCurr = begin;
272 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
274 rule_base::pointer_type oRet(
nullptr);
275 for (
size_t i = 0; (i < (_len-1)) && (oCurr < end); ++i, ++oCurr){
276 if (_str[i] != *oCurr){
277 return rule_base::pointer_type(
nullptr);
290 if (oCurr < end && isalnum(*oCurr) && isalnum(_str[_len - 2])){
291 return rule_base::pointer_type(
nullptr);
293 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
296 return rule_base::pointer_type(
new _DeclT);
302 template <
typename _DeclT,
size_t _len,
char(&_str)[_len],
typename _WhitespaceT>
303 class parse_helper<_DeclT, parse::string<char[_len], _str>, true, _WhitespaceT>{
306 template <
typename _IteratorT>
307 static rule_base::pointer_type parse(_IteratorT& begin, _IteratorT& end){
308 _IteratorT oCurr = begin;
310 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
312 rule_base::pointer_type oRet(
nullptr);
313 for (
size_t i = 0; (i < _len-1) && (oCurr < end); ++i, ++oCurr){
314 if (tolower(_str[i]) != tolower(*oCurr)){
315 return rule_base::pointer_type(
nullptr);
319 if (oCurr < end && isalnum(*oCurr) && isalnum(_str[_len - 2])){
320 return rule_base::pointer_type(
nullptr);
323 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
326 return rule_base::pointer_type(
new _DeclT);
332 template <
typename _DeclT,
size_t _len,
char(&_str)[_len],
bool _IgnoreCase,
typename _WhitespaceT>
333 class parse_helper<_DeclT, parse::regex<char[_len], _str>, _IgnoreCase, _WhitespaceT>{
336 template <
typename _IteratorT>
337 static rule_base::pointer_type parse(_IteratorT& begin, _IteratorT& end){
338 _IteratorT oCurr = begin;
340 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
342 static std::regex oRE(_str, std::regex_constants::ECMAScript | std::regex_constants::nosubs | std::regex_constants::optimize | (_IgnoreCase ? std::regex_constants::icase : std::regex_constants::optimize));
343 std::match_results<std::string::iterator> oMatch;
344 if (!std::regex_search(oCurr, end, oMatch, oRE, std::regex_constants::match_continuous | std::regex_constants::format_first_only)){
345 return rule_base::pointer_type(
nullptr);
347 oCurr += oMatch[0].length();
350 if (oCurr < end && isalnum(*oCurr) && isalnum(_str[_len - 1])){
351 return rule_base::pointer_type(
nullptr);
355 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
358 return rule_base::pointer_type(
new _DeclT(oMatch[0].str()));
365 template <
bool _IgnoreCase>
366 class parse_helper<whitespace<>, void, _IgnoreCase, void>{
368 template <
typename _IteratorT>
369 static bool parse(_IteratorT&, _IteratorT&){
return false; }
372 template <
char _HeadCH,
char... _TailCH,
bool _IgnoreCase>
373 class parse_helper<whitespace<_HeadCH, _TailCH...>, void, _IgnoreCase, void>{
375 template <
typename _IteratorT>
376 static bool parse(_IteratorT& begin, _IteratorT& end){
377 _IteratorT oCurr = begin;
380 if (*oCurr == _HeadCH){
382 }
else if (parse_helper<whitespace<_TailCH...>,
void, _IgnoreCase,
void>::parse(oCurr, end)){
394 template <
typename _DeclT,
char _Ch,
typename _WhitespaceT>
395 class parse_helper<_DeclT, character<_Ch>, true, _WhitespaceT>{
398 template <
typename _IteratorT>
399 static rule_base::pointer_type parse(_IteratorT& begin, _IteratorT& end){
400 _IteratorT oCurr = begin;
401 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
402 if (lower_case<char, _Ch>::value != tolower(*oCurr)){
403 return rule_base::pointer_type(
nullptr);
406 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
408 return rule_base::pointer_type(
new character<_Ch>);
413 template <
typename _DeclT,
char _Ch,
typename _WhitespaceT>
414 class parse_helper<_DeclT, character<_Ch>, false, _WhitespaceT>{
417 template <
typename _IteratorT>
418 static rule_base::pointer_type parse(_IteratorT& begin, _IteratorT& end){
419 _IteratorT oCurr = begin;
420 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
421 if (_Ch != oCurr[0]){
422 return rule_base::pointer_type(
nullptr);
425 parse_helper< _WhitespaceT, void, true, void>::parse(oCurr, end);
427 return rule_base::pointer_type(
new character<_Ch>);
434 template <
typename _DeclT,
bool _IgnoreCase,
typename _WhitespaceT >
435 class parse_helper < _DeclT, parse::and_<>, _IgnoreCase, _WhitespaceT>{
438 template <
typename _IteratorT,
typename ... _ChildRuleTs>
439 static rule_base::pointer_type parse(_IteratorT& , _IteratorT& , _ChildRuleTs&& ... oChildRules){
440 return rule_base::pointer_type(
new _DeclT(std::forward<_ChildRuleTs>(oChildRules)...));
445 template <
typename _DeclT,
typename _HeadT,
typename ... _TailT,
bool _IgnoreCase,
typename _WhitespaceT >
446 class parse_helper < _DeclT, parse::and_<_HeadT, _TailT...>, _IgnoreCase, _WhitespaceT>{
449 template <
typename _IteratorT,
typename ... _ChildRuleTs>
450 static rule_base::pointer_type parse(_IteratorT& begin, _IteratorT& end, _ChildRuleTs&& ... oChildRules){
451 _IteratorT oBegin = begin;
452 auto oItem = parse_helper<_HeadT, typename _HeadT::impl_type, _IgnoreCase, _WhitespaceT>::parse(oBegin, end);
456 oItem = parse_helper<_DeclT, parse::and_<_TailT...>, _IgnoreCase, _WhitespaceT>::parse(oBegin, end, std::forward<_ChildRuleTs>(oChildRules)..., oItem);
467 template <
typename _DeclT,
bool _IgnoreCase,
typename _WhitespaceT >
468 class parse_helper < _DeclT, parse::or_<>, _IgnoreCase, _WhitespaceT>{
471 template <
typename _IteratorT,
typename ... _ChildRuleTs>
472 static rule_base::pointer_type parse(_IteratorT& , _IteratorT& , _ChildRuleTs ... ){
473 return rule_base::pointer_type(
nullptr);
478 template <
typename _DeclT,
typename _HeadT,
typename ... _TailT,
bool _IgnoreCase,
typename _WhitespaceT >
479 class parse_helper < _DeclT, parse::or_<_HeadT, _TailT...>, _IgnoreCase, _WhitespaceT>{
482 template <
typename _IteratorT,
typename ... _ChildRuleTs>
483 static rule_base::pointer_type parse(_IteratorT& begin, _IteratorT& end, _ChildRuleTs&& ... oChildRules){
484 _IteratorT oBegin = begin;
485 auto oItem = parse_helper<_HeadT, typename _HeadT::impl_type, _IgnoreCase, _WhitespaceT>::parse(oBegin, end);
488 return rule_base::pointer_type(
new _DeclT(oItem));
490 oItem = parse_helper<_DeclT, parse::or_<_TailT...>, _IgnoreCase, _WhitespaceT>::parse(oBegin, end, std::forward<_ChildRuleTs>(oChildRules)..., oItem);
502 template <
typename _DeclT,
typename _Ty,
bool _IgnoreCase,
typename _WhitespaceT >
503 class parse_helper < _DeclT, parse::zero_or_more_<_Ty>, _IgnoreCase, _WhitespaceT>{
506 template <
typename _IteratorT,
typename ... _ChildRuleTs>
507 static rule_base::pointer_type parse(_IteratorT& begin, _IteratorT& end, _ChildRuleTs&& ... oChildRules){
508 _IteratorT oBegin = begin;
509 auto oItem = parse_helper<_Ty, typename _Ty::impl_type, _IgnoreCase, _WhitespaceT>::parse(oBegin, end);
511 return rule_base::pointer_type(
new _DeclT(std::forward<_ChildRuleTs>(oChildRules)...));
513 return parse_helper<_DeclT, parse::zero_or_more_<_Ty>, _IgnoreCase, _WhitespaceT>::parse(oBegin, end, std::forward<_ChildRuleTs>(oChildRules)..., oItem);
527 template <
typename _RuleT,
bool _IgnoreCase = false,
typename _WhitespaceT = xtd::parse::whitespace<>>
class parser {
535 template <
typename _IteratorT>
static parse::rule_base::pointer_type
parse(_IteratorT begin, _IteratorT end) {
536 _IteratorT oBegin = begin;
537 _IteratorT oEnd = end;
538 return parse::_::parse_helper<_RuleT, typename _RuleT::impl_type, _IgnoreCase, _WhitespaceT>::parse(oBegin, oEnd);
Represents a parse algorithm where all specified elements parse are contiguously present in the input...
Represents a parse algorithm where where the terminal or rule is repeated one or more times in the in...
virtual const std::type_info & type() const =0
Gets the type info of the concrete implementation.
Character terminal parsing algorithm.
Curiously recurring template pattern to simplify creation of rule_base implementations Rules and term...
Represents a parse algorithm where one of the listed elements is present in the input stream One of t...
virtual bool isa(const std::type_info &oType) const override
Determines if the interface is implemented by a concrete type.
list of whitespace characters to ignore in the input stream.
Represents a parse algorithm where the terminal or rule repeats exactly zero or one time in the input...
host, target and build configurations and settings Various components are purpose built for specific ...
virtual const std::type_info & type() const override
Gets the type info of the concrete implementation.
virtual bool isa(const std::type_info &oType) const =0
Determines if the interface is implemented by a concrete type.
regular expression parsing algorithm.
rule_base(_ChildRuleTs &&...oChildRules)
Constructor.
static parse::rule_base::pointer_type parse(_IteratorT begin, _IteratorT end)
Parses text.
Represents a parse algorithm where the terminal or rule repeats zero or more times in the input strea...
const vector_type & items() const
Accessor for child parse elements.
Base class of both rules and terminals Though rules and terminals are technically different they shar...
String terminal parsing algorithm.