11#include "interval.hpp"
25#include <netinet/in.h>
37#pragma warning(disable: 4100)
38#elif defined(__GNUC__)
39#pragma GCC diagnostic push
40#pragma GCC diagnostic ignored "-Wunknown-pragmas"
41#pragma GCC diagnostic ignored "-Wunused-parameter"
44#define ENUM_FLAG_OPERATOR(T,X) \
45inline T operator X (const T lhs, const T rhs) { return static_cast<T>(static_cast<std::underlying_type_t<T>>(lhs) X static_cast<std::underlying_type_t<T>>(rhs)); } \
46inline T operator X (const T lhs, const std::underlying_type_t<T> rhs) { return static_cast<T>(static_cast<std::underlying_type_t<T>>(lhs) X rhs); } \
47inline T operator X (const std::underlying_type_t<T> lhs, const T rhs) { return static_cast<T>(lhs X static_cast<std::underlying_type_t<T>>(rhs)); } \
48inline T& operator X= (T& lhs, const T rhs) { return lhs = lhs X rhs; } \
49inline T& operator X= (T& lhs, const std::underlying_type_t<T> rhs) { return lhs = lhs X rhs; }
50#define ENUM_FLAGS(T, type) \
52inline T operator ~ (T t) { return (T) (~static_cast<std::underlying_type_t <T>>(t)); } \
53ENUM_FLAG_OPERATOR(T,|) \
54ENUM_FLAG_OPERATOR(T,^) \
55ENUM_FLAG_OPERATOR(T,&) \
59#elif defined(__APPLE__)
60#define s6_words __u6_addr.__u6_addr16
62#define s6_words s6_addr16
72 constexpr int match_default = 0;
73 constexpr int match_case_insensitive = 0x1;
74 constexpr int match_multiline = 0x2;
87 _In_reads_or_z_opt_(end)
const T* text,
88 _In_
size_t start = 0,
89 _In_
size_t end = SIZE_MAX,
90 _In_
int flags = match_default)
92 for (
size_t i = start; i < end && text[i]; i++)
93 if (match(text, i, end, flags))
99 _In_reads_or_z_opt_(end)
const T* text,
100 _In_
size_t start = 0,
101 _In_
size_t end = SIZE_MAX,
102 _In_
int flags = match_default)
104 return do_match(text, start, end, flags);
108 _In_
const std::basic_string_view<T, std::char_traits<T>> text,
109 _In_
size_t start = 0,
110 _In_
size_t end = SIZE_MAX,
111 _In_
int flags = match_default)
113 return match(text.data(), start, std::min<size_t>(end, text.size()), flags);
116 virtual void invalidate()
124 virtual bool do_match(
125 _In_reads_or_z_opt_(end)
const T* text,
126 _In_
size_t start = 0,
127 _In_
size_t end = SIZE_MAX,
128 _In_
int flags = match_default) = 0;
131 template <
class T_out =
wchar_t>
132 const T_out* next_sgml_cp(_In_
const char* text, _In_
size_t start, _In_
size_t end, _Out_
size_t& chr_end, _Out_ T_out(&buf)[5])
134 if (text[start] ==
'&') {
136 const auto& ctype = std::use_facet<std::ctype<T>>(m_locale);
137 for (chr_end = start + 1;; chr_end++) {
138 if (chr_end >= end || text[chr_end] == 0) {
142 if (text[chr_end] ==
';') {
145 size_t n = chr_end - start - 1;
146 auto entity_w = utf32_to_wstr(sgml2uni(text + start + 1, n, buf32), buf);
154 else if (text[chr_end] ==
'&' || ctype.is(ctype.space, text[chr_end])) {
160 buf[0] = text[start];
167 std::locale m_locale;
170 using parser = basic_parser<char>;
171 using wparser = basic_parser<wchar_t>;
173 using tparser = wparser;
175 using tparser = parser;
177 using sgml_parser = basic_parser<char>;
186 virtual bool do_match(
187 _In_reads_or_z_opt_(end)
const T* text,
188 _In_
size_t start = 0,
189 _In_
size_t end = SIZE_MAX,
190 _In_
int flags = match_default)
192 stdex_assert(text || start >= end);
193 if (start < end && text[start]) {
194 this->interval.
start = this->interval.
end = start;
221 virtual bool do_match(
222 _In_reads_or_z_opt_(end)
const T* text,
223 _In_
size_t start = 0,
224 _In_
size_t end = SIZE_MAX,
225 _In_
int flags = match_default)
227 stdex_assert(text || start >= end);
228 if (start < end && text[start]) {
229 this->interval.
end = (this->interval.
start = start) + 1;
254 virtual bool do_match(
255 _In_reads_or_z_(end)
const char* text,
256 _In_
size_t start = 0,
257 _In_
size_t end = SIZE_MAX,
258 _In_
int flags = match_default)
260 stdex_assert(text || start >= end);
261 if (start < end && text[start]) {
262 if (text[start] ==
'&') {
264 const auto& ctype = std::use_facet<std::ctype<char>>(m_locale);
265 for (this->interval.
end = start + 1; this->interval.end < end && text[this->interval.end]; this->interval.end++)
266 if (text[this->interval.
end] ==
';') {
267 this->interval.
end++;
268 this->interval.
start = start;
271 else if (text[this->interval.
end] ==
'&' || ctype.is(ctype.space, text[this->interval.end]))
275 this->interval.
end = (this->interval.
start = start) + 1;
290 basic_cu(T chr,
bool invert =
false, _In_
const std::locale&
locale = std::locale()) :
297 virtual bool do_match(
298 _In_reads_or_z_opt_(end)
const T* text,
299 _In_
size_t start = 0,
300 _In_
size_t end = SIZE_MAX,
301 _In_
int flags = match_default)
303 stdex_assert(text || start >= end);
304 if (start < end && text[start]) {
306 if (flags & match_case_insensitive) {
307 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
308 r = ctype.tolower(text[start]) == ctype.tolower(m_chr);
311 r = text[start] == m_chr;
312 if ((r && !m_invert) || (!r && m_invert)) {
313 this->interval.
end = (this->interval.
start = start) + 1;
339 sgml_cp(
const char* chr,
size_t count = SIZE_MAX,
bool invert =
false, _In_
const std::locale&
locale = std::locale()) :
343 stdex_assert(chr || !count);
346 m_chr.assign(count ? next_sgml_cp(chr, 0, count, chr_end, buf) : L
"");
350 virtual bool do_match(
351 _In_reads_or_z_(end)
const char* text,
352 _In_
size_t start = 0,
353 _In_
size_t end = SIZE_MAX,
354 _In_
int flags = match_default)
356 stdex_assert(text || start >= end);
357 if (start < end && text[start]) {
359 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
360 bool r = ((flags & match_case_insensitive) ?
361 stdex::strnicmp(chr, stdex::strlen(chr), m_chr.data(), m_chr.size(), m_locale) :
362 stdex::strncmp(chr, stdex::strlen(chr), m_chr.data(), m_chr.size())) == 0;
363 if ((r && !m_invert) || (!r && m_invert)) {
364 this->interval.
start = start;
389 virtual bool do_match(
390 _In_reads_or_z_opt_(end)
const T* text,
391 _In_
size_t start = 0,
392 _In_
size_t end = SIZE_MAX,
393 _In_
int flags = match_default)
395 stdex_assert(text || start >= end);
396 if (start < end && text[start]) {
398 ((flags & match_multiline) || !stdex::islbreak(text[start])) &&
399 std::use_facet<std::ctype<T>>(this->m_locale).is(std::ctype_base::space, text[start]);
400 if ((r && !m_invert) || (!r && m_invert)) {
401 this->interval.
end = (this->interval.
start = start) + 1;
431 virtual bool do_match(
432 _In_reads_or_z_(end)
const char* text,
433 _In_
size_t start = 0,
434 _In_
size_t end = SIZE_MAX,
435 _In_
int flags = match_default)
437 stdex_assert(text || start >= end);
438 if (start < end && text[start]) {
440 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
441 const wchar_t* chr_end = chr + stdex::strlen(chr);
443 ((flags & match_multiline) || !stdex::islbreak(chr, SIZE_MAX)) &&
444 std::use_facet<std::ctype<wchar_t>>(m_locale).scan_not(std::ctype_base::space, chr, chr_end) == chr_end;
445 if ((r && !m_invert) || (!r && m_invert)) {
446 this->interval.
start = start;
469 virtual bool do_match(
470 _In_reads_or_z_opt_(end)
const T* text,
471 _In_
size_t start = 0,
472 _In_
size_t end = SIZE_MAX,
473 _In_
int flags = match_default)
475 stdex_assert(text || start >= end);
476 if (start < end && text[start]) {
477 bool r = std::use_facet<std::ctype<T>>(this->m_locale).is(std::ctype_base::punct, text[start]);
478 if ((r && !m_invert) || (!r && m_invert)) {
479 this->interval.
end = (this->interval.
start = start) + 1;
509 virtual bool do_match(
510 _In_reads_or_z_(end)
const char* text,
511 _In_
size_t start = 0,
512 _In_
size_t end = SIZE_MAX,
513 _In_
int flags = match_default)
515 stdex_assert(text || start >= end);
516 if (start < end && text[start]) {
518 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
519 const wchar_t* chr_end = chr + stdex::strlen(chr);
520 bool r = std::use_facet<std::ctype<wchar_t>>(m_locale).scan_not(std::ctype_base::punct, chr, chr_end) == chr_end;
521 if ((r && !m_invert) || (!r && m_invert)) {
522 this->interval.
start = start;
544 virtual bool do_match(
545 _In_reads_or_z_opt_(end)
const T* text,
546 _In_
size_t start = 0,
547 _In_
size_t end = SIZE_MAX,
548 _In_
int flags = match_default)
550 stdex_assert(text || start >= end);
551 if (start < end && text[start]) {
553 ((flags & match_multiline) || !stdex::islbreak(text[start])) &&
554 std::use_facet<std::ctype<T>>(this->m_locale).is(std::ctype_base::space | std::ctype_base::punct, text[start]);
555 if ((r && !m_invert) || (!r && m_invert)) {
556 this->interval.
end = (this->interval.
start = start) + 1;
586 virtual bool do_match(
587 _In_reads_or_z_(end)
const char* text,
588 _In_
size_t start = 0,
589 _In_
size_t end = SIZE_MAX,
590 _In_
int flags = match_default)
592 stdex_assert(text || start >= end);
593 if (start < end && text[start]) {
595 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
596 const wchar_t* chr_end = chr + stdex::strlen(chr);
598 ((flags & match_multiline) || !stdex::islbreak(chr, SIZE_MAX)) &&
599 std::use_facet<std::ctype<wchar_t>>(m_locale).scan_not(std::ctype_base::space | std::ctype_base::punct, chr, chr_end) == chr_end;
600 if ((r && !m_invert) || (!r && m_invert)) {
601 this->interval.
start = start;
617 basic_bol(
bool invert =
false) : m_invert(invert) {}
620 virtual bool do_match(
621 _In_reads_or_z_opt_(end)
const T* text,
622 _In_
size_t start = 0,
623 _In_
size_t end = SIZE_MAX,
624 _In_
int flags = match_default)
626 stdex_assert(text || !end);
627 stdex_assert(text || start >= end);
628 bool r = start == 0 || (start <= end && stdex::islbreak(text[start - 1]));
629 if ((r && !m_invert) || (!r && m_invert)) {
630 this->interval.
end = this->interval.
start = start;
656 basic_eol(
bool invert =
false) : m_invert(invert) {}
659 virtual bool do_match(
660 _In_reads_or_z_opt_(end)
const T* text,
661 _In_
size_t start = 0,
662 _In_
size_t end = SIZE_MAX,
663 _In_
int flags = match_default)
665 stdex_assert(text || start >= end);
666 bool r = start >= end || !text[start] || stdex::islbreak(text[start]);
667 if ((r && !m_invert) || (!r && m_invert)) {
668 this->interval.
end = this->interval.
start = start;
691 basic_set(
bool invert =
false, _In_
const std::locale&
locale = std::locale()) :
693 hit_offset(SIZE_MAX),
697 virtual void invalidate()
699 hit_offset = SIZE_MAX;
706 virtual bool do_match(
707 _In_reads_or_z_opt_(end)
const T* text,
708 _In_
size_t start = 0,
709 _In_
size_t end = SIZE_MAX,
710 _In_
int flags = match_default) = 0;
723 _In_reads_or_z_(count)
const T* set,
724 _In_
size_t count = SIZE_MAX,
725 _In_
bool invert =
false,
726 _In_
const std::locale&
locale = std::locale()) :
730 m_set.assign(set, set + stdex::strnlen(set, count));
734 virtual bool do_match(
735 _In_reads_or_z_opt_(end)
const T* text,
736 _In_
size_t start = 0,
737 _In_
size_t end = SIZE_MAX,
738 _In_
int flags = match_default)
740 stdex_assert(text || start >= end);
741 if (start < end && text[start]) {
742 const T* set = m_set.data();
743 size_t r = (flags & match_case_insensitive) ?
744 stdex::strnichr(set, m_set.size(), text[start], this->m_locale) :
745 stdex::strnchr(set, m_set.size(), text[start]);
746 if ((r != stdex::npos && !this->m_invert) || (r == stdex::npos && this->m_invert)) {
747 this->hit_offset = r;
748 this->interval.
end = (this->interval.
start = start) + 1;
752 this->hit_offset = SIZE_MAX;
757 std::basic_string<T> m_set;
774 sgml_cp_set(
const char* set,
size_t count = SIZE_MAX,
bool invert =
false, _In_
const std::locale&
locale = std::locale()) :
778 m_set = sgml2str(set, count);
782 virtual bool do_match(
783 _In_reads_or_z_(end)
const char* text,
784 _In_
size_t start = 0,
785 _In_
size_t end = SIZE_MAX,
786 _In_
int flags = match_default)
788 stdex_assert(text || start >= end);
789 if (start < end && text[start]) {
791 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
792 const wchar_t* set = m_set.data();
793 size_t r = (flags & match_case_insensitive) ?
794 stdex::strnistr(set, m_set.size(), chr, m_locale) :
795 stdex::strnstr(set, m_set.size(), chr);
796 if ((r != stdex::npos && !m_invert) || (r == stdex::npos && m_invert)) {
798 this->interval.
start = start;
802 hit_offset = SIZE_MAX;
818 _In_reads_or_z_(count)
const T* str,
819 _In_
size_t count = SIZE_MAX,
820 _In_
const std::locale&
locale = std::locale()) :
822 m_str(str, str + stdex::strnlen(str, count))
826 virtual bool do_match(
827 _In_reads_or_z_opt_(end)
const T* text,
828 _In_
size_t start = 0,
829 _In_
size_t end = SIZE_MAX,
830 _In_
int flags = match_default)
832 stdex_assert(text || start >= end);
835 n = std::min<size_t>(end - start, m);
836 bool r = ((flags & match_case_insensitive) ?
837 stdex::strnicmp(text + start, n, m_str.data(), m, this->m_locale) :
838 stdex::strncmp(text + start, n, m_str.data(), m)) == 0;
840 this->interval.
end = (this->interval.
start = start) + n;
847 std::basic_string<T> m_str;
864 sgml_string(
const char* str,
size_t count = SIZE_MAX, _In_
const std::locale&
locale = std::locale()) :
866 m_str(sgml2str(str, count))
870 virtual bool do_match(
871 _In_reads_or_z_(end)
const char* text,
872 _In_
size_t start = 0,
873 _In_
size_t end = SIZE_MAX,
874 _In_
int flags = match_default)
876 stdex_assert(text || start >= end);
877 const wchar_t* str = m_str.data();
878 const bool case_insensitive = flags & match_case_insensitive ? true :
false;
879 const auto& ctype = std::use_facet<std::ctype<wchar_t>>(m_locale);
880 for (this->interval.
end = start;;) {
882 this->interval.
start = start;
885 if (this->interval.
end >= end || !text[this->interval.end]) {
890 const wchar_t* chr = next_sgml_cp(text, this->interval.
end, end, this->interval.end, buf);
891 for (; *chr; ++str, ++chr) {
893 (case_insensitive ? ctype.tolower(*str) != ctype.tolower(*chr) : *str != *chr))
920 virtual bool do_match(
921 _In_reads_or_z_opt_(end)
const T* text,
922 _In_
size_t start = 0,
923 _In_
size_t end = SIZE_MAX,
924 _In_
int flags = match_default)
926 stdex_assert(text || start >= end);
927 this->interval.
start = this->interval.
end = start;
928 for (
size_t i = 0; ; i++) {
931 if (!
m_el->match(text, this->interval.end, end, flags)) {
936 if (
m_el->interval.end == this->interval.end) {
940 this->interval.
end =
m_el->interval.end;
946 std::shared_ptr<basic_parser<T>>
m_el;
974 _In_
const std::locale&
locale = std::locale()) :
977 stdex_assert(el || !count);
978 m_collection.reserve(count);
979 for (
size_t i = 0; i < count; i++)
980 m_collection.push_back(el[i]);
985 _In_
const std::locale&
locale = std::locale()) :
987 m_collection(std::move(collection))
990 virtual void invalidate()
992 for (
auto& el : m_collection)
998 std::vector<std::shared_ptr<basic_parser<T>>> m_collection;
1009 _In_count_(count)
const std::shared_ptr<
basic_parser<T>>* el =
nullptr,
1010 _In_
size_t count = 0,
1011 _In_
const std::locale&
locale = std::locale()) :
1017 _In_
const std::locale&
locale = std::locale()) :
1022 virtual bool do_match(
1023 _In_reads_or_z_opt_(end)
const T* text,
1024 _In_
size_t start = 0,
1025 _In_
size_t end = SIZE_MAX,
1026 _In_
int flags = match_default)
1028 stdex_assert(text || start >= end);
1029 this->interval.
end = start;
1030 for (
auto i = this->m_collection.begin(); i != this->m_collection.end(); ++i) {
1031 if (!(*i)->match(text, this->interval.end, end, flags)) {
1032 for (++i; i != this->m_collection.end(); ++i)
1037 this->interval.
end = (*i)->interval.end;
1039 this->interval.
start = start;
1062 hit_offset(SIZE_MAX)
1067 _In_count_(count)
const std::shared_ptr<
basic_parser<T>>* el =
nullptr,
1068 _In_
size_t count = 0,
1069 _In_
const std::locale&
locale = std::locale()) :
1071 hit_offset(SIZE_MAX)
1076 _In_
const std::locale&
locale = std::locale()) :
1078 hit_offset(SIZE_MAX)
1081 virtual void invalidate()
1083 hit_offset = SIZE_MAX;
1090 virtual bool do_match(
1091 _In_reads_or_z_opt_(end)
const T* text,
1092 _In_
size_t start = 0,
1093 _In_
size_t end = SIZE_MAX,
1094 _In_
int flags = match_default)
1096 stdex_assert(text || start >= end);
1098 for (
auto i = this->m_collection.begin(); i != this->m_collection.end(); ++i, ++hit_offset) {
1099 if ((*i)->match(text, start, end, flags)) {
1101 for (++i; i != this->m_collection.end(); ++i)
1106 hit_offset = SIZE_MAX;
1124 template <
class T,
class T_parser = basic_
string<T>>
1129 _In_reads_(count)
const T* str_z =
nullptr,
1130 _In_
size_t count = 0,
1131 _In_
const std::locale&
locale = std::locale()) :
1134 build(str_z, count);
1141 va_start(params, str);
1150 va_start(params, str);
1156 void build(_In_reads_(count)
const T* str_z, _In_
size_t count)
1158 stdex_assert(str_z || !count);
1163 offset < count && str_z[offset];
1164 offset += stdex::strnlen(str_z + offset, count - offset) + 1, ++n);
1165 this->m_collection.reserve(n);
1168 offset < count && str_z[offset];
1169 offset += stdex::strnlen(str_z + offset, count - offset) + 1)
1170 this->m_collection.push_back(std::move(std::make_shared<T_parser>(str_z + offset, count - offset, this->m_locale)));
1174 void build(_In_z_
const T* str, _In_ va_list params)
1178 this->m_collection.push_back(std::move(std::make_shared<T_parser>(str, SIZE_MAX, this->m_locale)));
1179 (p = va_arg(params,
const T*)) !=
nullptr;
1180 this->m_collection.push_back(std::move(std::make_shared<T_parser>(p, SIZE_MAX, this->m_locale))));
1201 _In_count_(count)
const std::shared_ptr<
basic_parser<T>>* el =
nullptr,
1202 _In_
size_t count = 0,
1203 _In_
const std::locale&
locale = std::locale()) :
1209 _In_
const std::locale&
locale = std::locale()) :
1214 virtual bool do_match(
1215 _In_reads_or_z_opt_(end)
const T* text,
1216 _In_
size_t start = 0,
1217 _In_
size_t end = SIZE_MAX,
1218 _In_
int flags = match_default)
1220 stdex_assert(text || start >= end);
1221 for (
auto& el : this->m_collection)
1223 if (match_recursively(text, start, end, flags)) {
1224 this->interval.
start = start;
1231 bool match_recursively(
1232 _In_reads_or_z_opt_(end)
const T* text,
1233 _In_
size_t start = 0,
1234 _In_
size_t end = SIZE_MAX,
1235 _In_
int flags = match_default)
1237 bool all_matched =
true;
1238 for (
auto& el : this->m_collection) {
1239 if (!el->interval) {
1241 all_matched =
false;
1242 if (el->match(text, start, end, flags)) {
1244 if (match_recursively(text, el->interval.end, end, flags)) {
1253 this->interval.
end = start;
1281 virtual void invalidate()
1309 _In_
const std::locale&
locale = std::locale()) :
1324 virtual bool do_match(
1325 _In_reads_or_z_opt_(end)
const T* text,
1326 _In_
size_t start = 0,
1327 _In_
size_t end = SIZE_MAX,
1328 _In_
int flags = match_default)
1330 stdex_assert(text || start >= end);
1331 for (this->interval.
end = start, this->value = 0; this->interval.end < end && text[this->interval.end];) {
1333 if (m_digit_0->match(text, this->interval.end, end, flags)) { dig = 0; this->interval.
end = m_digit_0->interval.end; }
1334 else if (m_digit_1->match(text, this->interval.end, end, flags)) { dig = 1; this->interval.
end = m_digit_1->interval.end; }
1335 else if (m_digit_2->match(text, this->interval.end, end, flags)) { dig = 2; this->interval.
end = m_digit_2->interval.end; }
1336 else if (m_digit_3->match(text, this->interval.end, end, flags)) { dig = 3; this->interval.
end = m_digit_3->interval.end; }
1337 else if (m_digit_4->match(text, this->interval.end, end, flags)) { dig = 4; this->interval.
end = m_digit_4->interval.end; }
1338 else if (m_digit_5->match(text, this->interval.end, end, flags)) { dig = 5; this->interval.
end = m_digit_5->interval.end; }
1339 else if (m_digit_6->match(text, this->interval.end, end, flags)) { dig = 6; this->interval.
end = m_digit_6->interval.end; }
1340 else if (m_digit_7->match(text, this->interval.end, end, flags)) { dig = 7; this->interval.
end = m_digit_7->interval.end; }
1341 else if (m_digit_8->match(text, this->interval.end, end, flags)) { dig = 8; this->interval.
end = m_digit_8->interval.end; }
1342 else if (m_digit_9->match(text, this->interval.end, end, flags)) { dig = 9; this->interval.
end = m_digit_9->interval.end; }
1347 this->interval.
start = start;
1354 std::shared_ptr<basic_parser<T>>
1386 _In_
const std::locale&
locale = std::locale()) :
1391 m_separator(separator)
1394 virtual void invalidate()
1405 virtual bool do_match(
1406 _In_reads_or_z_opt_(end)
const T* text,
1407 _In_
size_t start = 0,
1408 _In_
size_t end = SIZE_MAX,
1409 _In_
int flags = match_default)
1411 stdex_assert(text || start >= end);
1412 if (m_digits->match(text, start, end, flags)) {
1414 this->
value = m_digits->value;
1417 this->interval.
start = start;
1418 this->interval.
end = m_digits->interval.end;
1419 if (m_digits->interval.size() <= 3) {
1421 size_t hit_offset = SIZE_MAX;
1422 while (m_separator->match(text, this->interval.end, end, flags) &&
1423 (hit_offset == SIZE_MAX || hit_offset == m_separator->hit_offset) &&
1424 m_digits->match(text, m_separator->interval.end, end, flags) &&
1425 m_digits->interval.size() == 3)
1428 this->
value = this->
value * 1000 + m_digits->value;
1431 this->interval.
end = m_digits->interval.end;
1432 hit_offset = m_separator->hit_offset;
1443 std::shared_ptr<basic_integer10<T>> m_digits;
1444 std::shared_ptr<basic_set<T>> m_separator;
1447 using integer10ts = basic_integer10ts<char>;
1448 using winteger10ts = basic_integer10ts<wchar_t>;
1450 using tinteger10ts = winteger10ts;
1452 using tinteger10ts = integer10ts;
1454 using sgml_integer10ts = basic_integer10ts<char>;
1480 _In_
const std::locale&
locale = std::locale()) :
1492 m_digit_10(digit_10),
1493 m_digit_11(digit_11),
1494 m_digit_12(digit_12),
1495 m_digit_13(digit_13),
1496 m_digit_14(digit_14),
1497 m_digit_15(digit_15)
1501 virtual bool do_match(
1502 _In_reads_or_z_opt_(end)
const T* text,
1503 _In_
size_t start = 0,
1504 _In_
size_t end = SIZE_MAX,
1505 _In_
int flags = match_default)
1507 stdex_assert(text || start >= end);
1508 for (this->interval.
end = start, this->value = 0; this->interval.end < end && text[this->interval.end];) {
1510 if (m_digit_0->match(text, this->interval.end, end, flags)) { dig = 0; this->interval.
end = m_digit_0->interval.end; }
1511 else if (m_digit_1->match(text, this->interval.end, end, flags)) { dig = 1; this->interval.
end = m_digit_1->interval.end; }
1512 else if (m_digit_2->match(text, this->interval.end, end, flags)) { dig = 2; this->interval.
end = m_digit_2->interval.end; }
1513 else if (m_digit_3->match(text, this->interval.end, end, flags)) { dig = 3; this->interval.
end = m_digit_3->interval.end; }
1514 else if (m_digit_4->match(text, this->interval.end, end, flags)) { dig = 4; this->interval.
end = m_digit_4->interval.end; }
1515 else if (m_digit_5->match(text, this->interval.end, end, flags)) { dig = 5; this->interval.
end = m_digit_5->interval.end; }
1516 else if (m_digit_6->match(text, this->interval.end, end, flags)) { dig = 6; this->interval.
end = m_digit_6->interval.end; }
1517 else if (m_digit_7->match(text, this->interval.end, end, flags)) { dig = 7; this->interval.
end = m_digit_7->interval.end; }
1518 else if (m_digit_8->match(text, this->interval.end, end, flags)) { dig = 8; this->interval.
end = m_digit_8->interval.end; }
1519 else if (m_digit_9->match(text, this->interval.end, end, flags)) { dig = 9; this->interval.
end = m_digit_9->interval.end; }
1520 else if (m_digit_10->match(text, this->interval.end, end, flags)) { dig = 10; this->interval.
end = m_digit_10->interval.end; }
1521 else if (m_digit_11->match(text, this->interval.end, end, flags)) { dig = 11; this->interval.
end = m_digit_11->interval.end; }
1522 else if (m_digit_12->match(text, this->interval.end, end, flags)) { dig = 12; this->interval.
end = m_digit_12->interval.end; }
1523 else if (m_digit_13->match(text, this->interval.end, end, flags)) { dig = 13; this->interval.
end = m_digit_13->interval.end; }
1524 else if (m_digit_14->match(text, this->interval.end, end, flags)) { dig = 14; this->interval.
end = m_digit_14->interval.end; }
1525 else if (m_digit_15->match(text, this->interval.end, end, flags)) { dig = 15; this->interval.
end = m_digit_15->interval.end; }
1530 this->interval.
start = start;
1537 std::shared_ptr<basic_parser<T>>
1582 _In_
const std::locale&
locale = std::locale()) :
1586 m_digit_10(digit_10),
1587 m_digit_50(digit_50),
1588 m_digit_100(digit_100),
1589 m_digit_500(digit_500),
1590 m_digit_1000(digit_1000),
1591 m_digit_5000(digit_5000),
1592 m_digit_10000(digit_10000)
1596 virtual bool do_match(
1597 _In_reads_or_z_opt_(end)
const T* text,
1598 _In_
size_t start = 0,
1599 _In_
size_t end = SIZE_MAX,
1600 _In_
int flags = match_default)
1602 stdex_assert(text || start >= end);
1604 dig[5] = { SIZE_MAX, SIZE_MAX, SIZE_MAX, SIZE_MAX, SIZE_MAX },
1607 for (this->interval.
end = start, this->value = 0; this->interval.end < end && text[this->interval.end]; dig[3] = dig[2], dig[2] = dig[1], dig[1] = dig[0], this->interval.end = end2) {
1608 if (m_digit_1 && m_digit_1->match(text, this->interval.end, end, flags)) { dig[0] = 1; end2 = m_digit_1->interval.end; }
1609 else if (m_digit_5 && m_digit_5->match(text, this->interval.end, end, flags)) { dig[0] = 5; end2 = m_digit_5->interval.end; }
1610 else if (m_digit_10 && m_digit_10->match(text, this->interval.end, end, flags)) { dig[0] = 10; end2 = m_digit_10->interval.end; }
1611 else if (m_digit_50 && m_digit_50->match(text, this->interval.end, end, flags)) { dig[0] = 50; end2 = m_digit_50->interval.end; }
1612 else if (m_digit_100 && m_digit_100->match(text, this->interval.end, end, flags)) { dig[0] = 100; end2 = m_digit_100->interval.end; }
1613 else if (m_digit_500 && m_digit_500->match(text, this->interval.end, end, flags)) { dig[0] = 500; end2 = m_digit_500->interval.end; }
1614 else if (m_digit_1000 && m_digit_1000->match(text, this->interval.end, end, flags)) { dig[0] = 1000; end2 = m_digit_1000->interval.end; }
1615 else if (m_digit_5000 && m_digit_5000->match(text, this->interval.end, end, flags)) { dig[0] = 5000; end2 = m_digit_5000->interval.end; }
1616 else if (m_digit_10000 && m_digit_10000->match(text, this->interval.end, end, flags)) { dig[0] = 10000; end2 = m_digit_10000->interval.end; }
1620 if (dig[4] == SIZE_MAX) dig[4] = dig[0];
1622 if (dig[3] == dig[2] && dig[2] == dig[1] && dig[1] == dig[0] && dig[0] != dig[4]) {
1626 if (dig[0] <= dig[1]) {
1628 this->
value += dig[0];
1631 (dig[1] == 1 && (dig[0] == 5 || dig[0] == 10)) ||
1632 (dig[1] == 10 && (dig[0] == 50 || dig[0] == 100)) ||
1633 (dig[1] == 100 && (dig[0] == 500 || dig[0] == 1000)) ||
1634 (dig[1] == 1000 && (dig[0] == 5000 || dig[0] == 10000)))
1637 if (dig[2] < dig[0]) {
1641 this->
value -= dig[1];
1645 this->
value += dig[0];
1653 this->interval.
start = start;
1660 std::shared_ptr<basic_parser<T>>
1692 _In_
const std::locale&
locale = std::locale()) :
1694 numerator(_numerator),
1695 fraction_line(_fraction_line),
1696 denominator(_denominator)
1699 virtual void invalidate()
1701 numerator->invalidate();
1702 fraction_line->invalidate();
1703 denominator->invalidate();
1707 std::shared_ptr<basic_parser<T>> numerator;
1708 std::shared_ptr<basic_parser<T>> fraction_line;
1709 std::shared_ptr<basic_parser<T>> denominator;
1712 virtual bool do_match(
1713 _In_reads_or_z_opt_(end)
const T* text,
1714 _In_
size_t start = 0,
1715 _In_
size_t end = SIZE_MAX,
1716 _In_
int flags = match_default)
1718 stdex_assert(text || start >= end);
1719 if (numerator->match(text, start, end, flags) &&
1720 fraction_line->match(text, numerator->interval.end, end, flags) &&
1721 denominator->match(text, fraction_line->interval.end, end, flags))
1723 this->interval.
start = start;
1724 this->interval.
end = denominator->interval.end;
1727 numerator->invalidate();
1728 fraction_line->invalidate();
1729 denominator->invalidate();
1756 _In_
const std::locale&
locale = std::locale()) :
1759 separator(_separator),
1764 virtual void invalidate()
1767 separator->invalidate();
1768 guest->invalidate();
1772 std::shared_ptr<basic_parser<T>> home;
1773 std::shared_ptr<basic_parser<T>> separator;
1774 std::shared_ptr<basic_parser<T>> guest;
1777 virtual bool do_match(
1778 _In_reads_or_z_opt_(end)
const T* text,
1779 _In_
size_t start = 0,
1780 _In_
size_t end = SIZE_MAX,
1781 _In_
int flags = match_default)
1783 stdex_assert(text || start >= end);
1784 this->interval.
end = start;
1786 const int space_match_flags = flags & ~match_multiline;
1788 if (home->match(text, this->interval.end, end, flags))
1789 this->interval.
end = home->interval.end;
1793 for (; m_space->match(text, this->interval.
end, end, space_match_flags); this->interval.end = m_space->interval.end);
1795 if (separator->match(text, this->interval.end, end, flags))
1796 this->interval.
end = separator->interval.end;
1800 for (; m_space->match(text, this->interval.
end, end, space_match_flags); this->interval.end = m_space->interval.end);
1802 if (guest->match(text, this->interval.end, end, flags))
1803 this->interval.
end = guest->interval.end;
1807 this->interval.
start = start;
1812 separator->invalidate();
1813 guest->invalidate();
1818 std::shared_ptr<basic_parser<T>> m_space;
1842 _In_
const std::locale&
locale = std::locale()) :
1850 virtual void invalidate()
1865 virtual bool do_match(
1866 _In_reads_or_z_opt_(end)
const T* text,
1867 _In_
size_t start = 0,
1868 _In_
size_t end = SIZE_MAX,
1869 _In_
int flags = match_default)
1871 stdex_assert(text || start >= end);
1872 this->interval.
end = start;
1893 if (
number->match(text, this->interval.end, end, flags)) {
1894 this->interval.
start = start;
1895 this->interval.
end =
number->interval.end;
1907 using signed_numeral = basic_signed_numeral<char>;
1908 using wsigned_numeral = basic_signed_numeral<wchar_t>;
1910 using tsigned_numeral = wsigned_numeral;
1912 using tsigned_numeral = signed_numeral;
1914 using sgml_signed_numeral = basic_signed_numeral<char>;
1930 _In_
const std::locale&
locale = std::locale()) :
1940 virtual void invalidate()
1957 virtual bool do_match(
1958 _In_reads_or_z_opt_(end)
const T* text,
1959 _In_
size_t start = 0,
1960 _In_
size_t end = SIZE_MAX,
1961 _In_
int flags = match_default)
1963 stdex_assert(text || start >= end);
1964 this->interval.
end = start;
1988 const int space_match_flags = flags & ~match_multiline;
1989 if (
integer->match(text, this->interval.end, end, flags) &&
1990 m_space->match(text,
integer->interval.end, end, space_match_flags))
1992 for (this->interval.
end = m_space->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
1993 if (
fraction->match(text, this->interval.end, end, flags)) {
1994 this->interval.
start = start;
1999 this->interval.
start = start;
2005 if (
fraction->match(text, this->interval.end, end, flags)) {
2007 this->interval.
start = start;
2013 if (
integer->match(text, this->interval.end, end, flags)) {
2015 this->interval.
start = start;
2029 std::shared_ptr<basic_parser<T>> m_space;
2032 using mixed_numeral = basic_mixed_numeral<char>;
2033 using wmixed_numeral = basic_mixed_numeral<wchar_t>;
2035 using tmixed_numeral = wmixed_numeral;
2037 using tmixed_numeral = mixed_numeral;
2039 using sgml_mixed_numeral = basic_mixed_numeral<char>;
2059 _In_
const std::locale&
locale = std::locale()) :
2071 value(std::numeric_limits<double>::quiet_NaN())
2074 virtual void invalidate()
2086 value = std::numeric_limits<double>::quiet_NaN();
2103 virtual bool do_match(
2104 _In_reads_or_z_opt_(end)
const T* text,
2105 _In_
size_t start = 0,
2106 _In_
size_t end = SIZE_MAX,
2107 _In_
int flags = match_default)
2109 stdex_assert(text || start >= end);
2110 this->interval.
end = start;
2133 if (
integer->match(text, this->interval.end, end, flags))
2144 if (
integer->interval.empty() &&
2190 double e = (double)
exponent->value;
2193 value *= pow(10.0, e);
2196 this->interval.
start = start;
2201 using scientific_numeral = basic_scientific_numeral<char>;
2202 using wscientific_numeral = basic_scientific_numeral<wchar_t>;
2204 using tscientific_numeral = wscientific_numeral;
2206 using tscientific_numeral = scientific_numeral;
2208 using sgml_scientific_numeral = basic_scientific_numeral<char>;
2225 _In_
const std::locale&
locale = std::locale()) :
2236 virtual void invalidate()
2257 virtual bool do_match(
2258 _In_reads_or_z_opt_(end)
const T* text,
2259 _In_
size_t start = 0,
2260 _In_
size_t end = SIZE_MAX,
2261 _In_
int flags = match_default)
2263 stdex_assert(text || start >= end);
2264 this->interval.
end = start;
2266 if (
positive_sign->match(text, this->interval.end, end, flags)) {
2271 else if (
negative_sign->match(text, this->interval.end, end, flags)) {
2276 else if (
special_sign->match(text, this->interval.end, end, flags)) {
2287 if (
currency->match(text, this->interval.end, end, flags))
2300 if (
integer->match(text, this->interval.end, end, flags))
2310 if (
integer->interval.empty() &&
2325 this->interval.
start = start;
2330 using monetary_numeral = basic_monetary_numeral<char>;
2331 using wmonetary_numeral = basic_monetary_numeral<wchar_t>;
2333 using tmonetary_numeral = wmonetary_numeral;
2335 using tmonetary_numeral = monetary_numeral;
2337 using sgml_monetary_numeral = basic_monetary_numeral<char>;
2358 _In_
const std::locale&
locale = std::locale()) :
2370 m_separator(separator)
2375 virtual void invalidate()
2393 virtual bool do_match(
2394 _In_reads_or_z_opt_(end)
const T* text,
2395 _In_
size_t start = 0,
2396 _In_
size_t end = SIZE_MAX,
2397 _In_
int flags = match_default)
2399 stdex_assert(text || start >= end);
2400 this->interval.
end = start;
2404 for (i = 0; i < 4; i++) {
2406 if (m_separator->match(text, this->interval.end, end, flags))
2407 this->interval.
end = m_separator->interval.end;
2413 bool is_empty =
true;
2415 for (x = 0; this->interval.
end < end && text[this->interval.
end];) {
2416 size_t dig, digit_end;
2417 if (m_digit_0->match(text, this->interval.end, end, flags)) { dig = 0; digit_end = m_digit_0->interval.end; }
2418 else if (m_digit_1->match(text, this->interval.end, end, flags)) { dig = 1; digit_end = m_digit_1->interval.end; }
2419 else if (m_digit_2->match(text, this->interval.end, end, flags)) { dig = 2; digit_end = m_digit_2->interval.end; }
2420 else if (m_digit_3->match(text, this->interval.end, end, flags)) { dig = 3; digit_end = m_digit_3->interval.end; }
2421 else if (m_digit_4->match(text, this->interval.end, end, flags)) { dig = 4; digit_end = m_digit_4->interval.end; }
2422 else if (m_digit_5->match(text, this->interval.end, end, flags)) { dig = 5; digit_end = m_digit_5->interval.end; }
2423 else if (m_digit_6->match(text, this->interval.end, end, flags)) { dig = 6; digit_end = m_digit_6->interval.end; }
2424 else if (m_digit_7->match(text, this->interval.end, end, flags)) { dig = 7; digit_end = m_digit_7->interval.end; }
2425 else if (m_digit_8->match(text, this->interval.end, end, flags)) { dig = 8; digit_end = m_digit_8->interval.end; }
2426 else if (m_digit_9->match(text, this->interval.end, end, flags)) { dig = 9; digit_end = m_digit_9->interval.end; }
2428 size_t x_n = x * 10 + dig;
2431 this->interval.
end = digit_end;
2440 value.s_addr = (
value.s_addr << 8) | (uint8_t)x;
2445 HE2BE(
reinterpret_cast<uint32_t&
>(
value.s_addr));
2446 this->interval.
start = start;
2454 std::shared_ptr<basic_parser<T>>
2465 std::shared_ptr<basic_parser<T>> m_separator;
2468 using ipv4_address = basic_ipv4_address<char>;
2469 using wipv4_address = basic_ipv4_address<wchar_t>;
2471 using tipv4_address = wipv4_address;
2473 using tipv4_address = ipv4_address;
2475 using sgml_ipv4_address = basic_ipv4_address<char>;
2487 virtual bool do_match(
2488 _In_reads_or_z_opt_(end)
const T* text,
2489 _In_
size_t start = 0,
2490 _In_
size_t end = SIZE_MAX,
2491 _In_
int flags = match_default)
2493 stdex_assert(text || start >= end);
2494 if (start < end && text[start]) {
2495 if (text[start] ==
'-' ||
2496 text[start] ==
'_' ||
2497 text[start] ==
':' ||
2498 std::use_facet<std::ctype<T>>(this->m_locale).is(std::ctype_base::alnum, text[start]))
2500 this->interval.
end = (this->interval.
start = start) + 1;
2526 virtual bool do_match(
2527 _In_reads_or_z_(end)
const char* text,
2528 _In_
size_t start = 0,
2529 _In_
size_t end = SIZE_MAX,
2530 _In_
int flags = match_default)
2532 stdex_assert(text || start >= end);
2533 if (start < end && text[start]) {
2535 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
2536 const wchar_t* chr_end = chr + stdex::strlen(chr);
2537 if (((chr[0] == L
'-' ||
2539 chr[0] == L
':') && chr[1] == 0) ||
2540 std::use_facet<std::ctype<wchar_t>>(m_locale).scan_not(std::ctype_base::alnum, chr, chr_end) == chr_end)
2542 this->interval.
start = start;
2576 _In_
const std::shared_ptr<
basic_parser<T>>& scope_id_separator =
nullptr,
2578 _In_
const std::locale&
locale = std::locale()) :
2590 m_digit_10(digit_10),
2591 m_digit_11(digit_11),
2592 m_digit_12(digit_12),
2593 m_digit_13(digit_13),
2594 m_digit_14(digit_14),
2595 m_digit_15(digit_15),
2596 m_separator(separator),
2597 m_scope_id_separator(scope_id_separator),
2603 virtual void invalidate()
2631 virtual bool do_match(
2632 _In_reads_or_z_opt_(end)
const T* text,
2633 _In_
size_t start = 0,
2634 _In_
size_t end = SIZE_MAX,
2635 _In_
int flags = match_default)
2637 stdex_assert(text || start >= end);
2638 this->interval.
end = start;
2641 size_t i, compaction_i = SIZE_MAX, compaction_start = start;
2642 for (i = 0; i < 8; i++) {
2643 bool is_empty =
true;
2645 if (m_separator->match(text, this->interval.end, end, flags)) {
2647 this->interval.
end = m_separator->interval.end;
2648 if (m_separator->match(text, this->interval.end, end, flags)) {
2650 if (compaction_i == SIZE_MAX) {
2653 compaction_start = m_separator->interval.start;
2654 this->interval.
end = m_separator->interval.end;
2673 for (x = 0; this->interval.
end < end && text[this->interval.
end];) {
2674 size_t dig, digit_end;
2675 if (m_digit_0->match(text, this->interval.end, end, flags)) { dig = 0; digit_end = m_digit_0->interval.end; }
2676 else if (m_digit_1->match(text, this->interval.end, end, flags)) { dig = 1; digit_end = m_digit_1->interval.end; }
2677 else if (m_digit_2->match(text, this->interval.end, end, flags)) { dig = 2; digit_end = m_digit_2->interval.end; }
2678 else if (m_digit_3->match(text, this->interval.end, end, flags)) { dig = 3; digit_end = m_digit_3->interval.end; }
2679 else if (m_digit_4->match(text, this->interval.end, end, flags)) { dig = 4; digit_end = m_digit_4->interval.end; }
2680 else if (m_digit_5->match(text, this->interval.end, end, flags)) { dig = 5; digit_end = m_digit_5->interval.end; }
2681 else if (m_digit_6->match(text, this->interval.end, end, flags)) { dig = 6; digit_end = m_digit_6->interval.end; }
2682 else if (m_digit_7->match(text, this->interval.end, end, flags)) { dig = 7; digit_end = m_digit_7->interval.end; }
2683 else if (m_digit_8->match(text, this->interval.end, end, flags)) { dig = 8; digit_end = m_digit_8->interval.end; }
2684 else if (m_digit_9->match(text, this->interval.end, end, flags)) { dig = 9; digit_end = m_digit_9->interval.end; }
2685 else if (m_digit_10->match(text, this->interval.end, end, flags)) { dig = 10; digit_end = m_digit_10->interval.end; }
2686 else if (m_digit_11->match(text, this->interval.end, end, flags)) { dig = 11; digit_end = m_digit_11->interval.end; }
2687 else if (m_digit_12->match(text, this->interval.end, end, flags)) { dig = 12; digit_end = m_digit_12->interval.end; }
2688 else if (m_digit_13->match(text, this->interval.end, end, flags)) { dig = 13; digit_end = m_digit_13->interval.end; }
2689 else if (m_digit_14->match(text, this->interval.end, end, flags)) { dig = 14; digit_end = m_digit_14->interval.end; }
2690 else if (m_digit_15->match(text, this->interval.end, end, flags)) { dig = 15; digit_end = m_digit_15->interval.end; }
2692 size_t x_n = x * 16 + dig;
2693 if (x_n <= 0xffff) {
2695 this->interval.
end = digit_end;
2702 if (compaction_i != SIZE_MAX) {
2709 HE2BE(
reinterpret_cast<uint16_t&
>(this->value.s6_words[i]));
2712 if (compaction_i != SIZE_MAX) {
2715 for (j = 8, k = i; k > compaction_i;) {
2716 this->value.s6_words[--j] = this->value.s6_words[--k];
2719 for (; j > compaction_i;) {
2720 this->value.s6_words[--j] = 0;
2728 if (m_scope_id_separator && m_scope_id_separator->match(text, this->interval.end, end, flags) &&
2729 scope_id &&
scope_id->match(text, m_scope_id_separator->interval.end, end, flags))
2734 this->interval.
start = start;
2742 std::shared_ptr<basic_parser<T>>
2759 std::shared_ptr<basic_parser<T>> m_separator, m_scope_id_separator;
2762 using ipv6_address = basic_ipv6_address<char>;
2763 using wipv6_address = basic_ipv6_address<wchar_t>;
2765 using tipv6_address = wipv6_address;
2767 using tipv6_address = ipv6_address;
2769 using sgml_ipv6_address = basic_ipv6_address<char>;
2779 _In_
bool allow_idn,
2780 _In_
const std::locale&
locale = std::locale()) :
2782 m_allow_idn(allow_idn),
2789 virtual bool do_match(
2790 _In_reads_or_z_opt_(end)
const T* text,
2791 _In_
size_t start = 0,
2792 _In_
size_t end = SIZE_MAX,
2793 _In_
int flags = match_default)
2795 stdex_assert(text || start >= end);
2796 if (start < end && text[start]) {
2797 if ((
'A' <= text[start] && text[start] <=
'Z') ||
2798 (
'a' <= text[start] && text[start] <=
'z') ||
2799 (
'0' <= text[start] && text[start] <=
'9'))
2801 else if (text[start] ==
'-')
2803 else if (m_allow_idn && std::use_facet<std::ctype<T>>(this->m_locale).is(std::ctype_base::alnum, text[start]))
2809 this->interval.
end = (this->interval.
start = start) + 1;
2819 using dns_domain_char = basic_dns_domain_char<char>;
2820 using wdns_domain_char = basic_dns_domain_char<wchar_t>;
2822 using tdns_domain_char = wdns_domain_char;
2824 using tdns_domain_char = dns_domain_char;
2834 _In_
bool allow_idn,
2835 _In_
const std::locale&
locale = std::locale()) :
2840 virtual bool do_match(
2841 _In_reads_or_z_(end)
const char* text,
2842 _In_
size_t start = 0,
2843 _In_
size_t end = SIZE_MAX,
2844 _In_
int flags = match_default)
2846 stdex_assert(text || start >= end);
2847 if (start < end && text[start]) {
2849 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
2850 const wchar_t* chr_end = chr + stdex::strlen(chr);
2851 if (((
'A' <= chr[0] && chr[0] <=
'Z') ||
2852 (
'a' <= chr[0] && chr[0] <=
'z') ||
2853 (
'0' <= chr[0] && chr[0] <=
'9')) && chr[1] == 0)
2855 else if (chr[0] ==
'-' && chr[1] == 0)
2857 else if (m_allow_idn && std::use_facet<std::ctype<wchar_t>>(m_locale).scan_not(std::ctype_base::alnum, chr, chr_end) == chr_end)
2863 this->interval.
start = start;
2879 _In_
bool allow_absolute,
2882 _In_
const std::locale&
locale = std::locale()) :
2885 m_domain_char(domain_char),
2886 m_separator(separator)
2890 virtual bool do_match(
2891 _In_reads_or_z_opt_(end)
const T* text,
2892 _In_
size_t start = 0,
2893 _In_
size_t end = SIZE_MAX,
2894 _In_
int flags = match_default)
2896 stdex_assert(text || start >= end);
2897 size_t i = start, count;
2898 for (count = 0; i < end && text[i] && count < 127; count++) {
2899 if (m_domain_char->match(text, i, end, flags) &&
2900 m_domain_char->allow_on_edge)
2903 this->interval.
end = i = m_domain_char->interval.end;
2904 while (i < end && text[i]) {
2905 if (m_domain_char->allow_on_edge &&
2906 m_separator->match(text, i, end, flags))
2910 this->interval.
end = i = m_separator->interval.end;
2912 this->interval.
end = i;
2913 i = m_separator->interval.end;
2917 if (m_domain_char->match(text, i, end, flags)) {
2918 if (m_domain_char->allow_on_edge)
2919 this->interval.
end = i = m_domain_char->interval.end;
2921 i = m_domain_char->interval.end;
2924 this->interval.
start = start;
2933 this->interval.
start = start;
2941 std::shared_ptr<basic_dns_domain_char<T>> m_domain_char;
2942 std::shared_ptr<basic_parser<T>> m_separator;
2964 virtual bool do_match(
2965 _In_reads_or_z_opt_(end)
const T* text,
2966 _In_
size_t start = 0,
2967 _In_
size_t end = SIZE_MAX,
2968 _In_
int flags = match_default)
2970 stdex_assert(text || start >= end);
2971 if (start < end && text[start]) {
2972 if (text[start] ==
'-' ||
2973 text[start] ==
'.' ||
2974 text[start] ==
'_' ||
2975 text[start] ==
'~' ||
2976 text[start] ==
'%' ||
2977 text[start] ==
'!' ||
2978 text[start] ==
'$' ||
2979 text[start] ==
'&' ||
2980 text[start] ==
'\'' ||
2983 text[start] ==
'*' ||
2984 text[start] ==
'+' ||
2985 text[start] ==
',' ||
2986 text[start] ==
';' ||
2987 text[start] ==
'=' ||
2988 std::use_facet<std::ctype<T>>(this->m_locale).is(std::ctype_base::alnum, text[start]))
2990 this->interval.
end = (this->interval.
start = start) + 1;
3016 virtual bool do_match(
3017 _In_reads_or_z_(end)
const char* text,
3018 _In_
size_t start = 0,
3019 _In_
size_t end = SIZE_MAX,
3020 _In_
int flags = match_default)
3022 stdex_assert(text || start >= end);
3023 if (start < end && text[start]) {
3025 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
3026 const wchar_t* chr_end = chr + stdex::strlen(chr);
3027 if (((chr[0] == L
'-' ||
3042 chr[0] == L
'=') && chr[1] == 0) ||
3043 std::use_facet<std::ctype<wchar_t>>(m_locale).scan_not(std::ctype_base::alnum, chr, chr_end) == chr_end)
3045 this->interval.
start = start;
3065 virtual bool do_match(
3066 _In_reads_or_z_opt_(end)
const T* text,
3067 _In_
size_t start = 0,
3068 _In_
size_t end = SIZE_MAX,
3069 _In_
int flags = match_default)
3071 stdex_assert(text || start >= end);
3072 if (start < end && text[start]) {
3073 if (text[start] ==
'-' ||
3074 text[start] ==
'.' ||
3075 text[start] ==
'_' ||
3076 text[start] ==
'~' ||
3077 text[start] ==
'%' ||
3078 text[start] ==
'!' ||
3079 text[start] ==
'$' ||
3080 text[start] ==
'&' ||
3081 text[start] ==
'\'' ||
3082 text[start] ==
'(' ||
3083 text[start] ==
')' ||
3084 text[start] ==
'*' ||
3085 text[start] ==
'+' ||
3086 text[start] ==
',' ||
3087 text[start] ==
';' ||
3088 text[start] ==
'=' ||
3089 text[start] ==
':' ||
3090 std::use_facet<std::ctype<T>>(this->m_locale).is(std::ctype_base::alnum, text[start]))
3092 this->interval.
end = (this->interval.
start = start) + 1;
3118 virtual bool do_match(
3119 _In_reads_or_z_(end)
const char* text,
3120 _In_
size_t start = 0,
3121 _In_
size_t end = SIZE_MAX,
3122 _In_
int flags = match_default)
3124 stdex_assert(text || start >= end);
3125 if (start < end && text[start]) {
3127 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
3128 const wchar_t* chr_end = chr + stdex::strlen(chr);
3129 if (((chr[0] == L
'-' ||
3145 chr[0] == L
':') && chr[1] == 0) ||
3146 std::use_facet<std::ctype<wchar_t>>(m_locale).scan_not(std::ctype_base::alnum, chr, chr_end) == chr_end)
3148 this->interval.
start = start;
3167 virtual bool do_match(
3168 _In_reads_or_z_opt_(end)
const T* text,
3169 _In_
size_t start = 0,
3170 _In_
size_t end = SIZE_MAX,
3171 _In_
int flags = match_default)
3173 stdex_assert(text || start >= end);
3174 if (start < end && text[start]) {
3175 if (text[start] ==
'/' ||
3176 text[start] ==
'-' ||
3177 text[start] ==
'.' ||
3178 text[start] ==
'_' ||
3179 text[start] ==
'~' ||
3180 text[start] ==
'%' ||
3181 text[start] ==
'!' ||
3182 text[start] ==
'$' ||
3183 text[start] ==
'&' ||
3184 text[start] ==
'\'' ||
3185 text[start] ==
'(' ||
3186 text[start] ==
')' ||
3187 text[start] ==
'*' ||
3188 text[start] ==
'+' ||
3189 text[start] ==
',' ||
3190 text[start] ==
';' ||
3191 text[start] ==
'=' ||
3192 text[start] ==
':' ||
3193 text[start] ==
'@' ||
3194 text[start] ==
'?' ||
3195 text[start] ==
'#' ||
3196 std::use_facet<std::ctype<T>>(this->m_locale).is(std::ctype_base::alnum, text[start]))
3198 this->interval.
end = (this->interval.
start = start) + 1;
3224 virtual bool do_match(
3225 _In_reads_or_z_(end)
const char* text,
3226 _In_
size_t start = 0,
3227 _In_
size_t end = SIZE_MAX,
3228 _In_
int flags = match_default)
3230 stdex_assert(text || start >= end);
3231 if (start < end && text[start]) {
3233 const wchar_t* chr = next_sgml_cp(text, start, end, this->interval.
end, buf);
3234 const wchar_t* chr_end = chr + stdex::strlen(chr);
3235 if (((chr[0] == L
'/' ||
3255 chr[0] == L
'#') && chr[1] == 0) ||
3256 std::use_facet<std::ctype<wchar_t>>(m_locale).scan_not(std::ctype_base::alnum, chr, chr_end) == chr_end)
3258 this->interval.
start = start;
3278 _In_
const std::locale&
locale = std::locale()) :
3280 m_path_char(path_char),
3281 m_query_start(query_start),
3282 m_bookmark_start(bookmark_start)
3285 virtual void invalidate()
3301 virtual bool do_match(
3302 _In_reads_or_z_opt_(end)
const T* text,
3303 _In_
size_t start = 0,
3304 _In_
size_t end = SIZE_MAX,
3305 _In_
int flags = match_default)
3307 stdex_assert(text || start >= end);
3309 this->interval.
end = start;
3317 if (this->interval.
end >= end || !text[this->interval.end])
3319 if (m_query_start->match(text, this->interval.end, end, flags)) {
3320 path.
end = this->interval.
end;
3321 query.
start = this->interval.
end = m_query_start->interval.end;
3323 if (this->interval.
end >= end || !text[this->interval.end]) {
3324 query.
end = this->interval.
end;
3327 if (m_bookmark_start->match(text, this->interval.end, end, flags)) {
3328 query.
end = this->interval.
end;
3329 bookmark.
start = this->interval.
end = m_bookmark_start->interval.end;
3331 if (this->interval.
end >= end || !text[this->interval.end]) {
3332 bookmark.
end = this->interval.
end;
3335 if (m_path_char->match(text, this->interval.end, end, flags))
3336 this->interval.
end = m_path_char->interval.end;
3338 bookmark.
end = this->interval.
end;
3342 this->interval.
start = start;
3345 if (m_path_char->match(text, this->interval.end, end, flags))
3346 this->interval.
end = m_path_char->interval.end;
3348 query.
end = this->interval.
end;
3352 this->interval.
start = start;
3355 if (m_bookmark_start->match(text, this->interval.end, end, flags)) {
3356 path.
end = this->interval.
end;
3357 bookmark.
start = this->interval.
end = m_bookmark_start->interval.end;
3359 if (this->interval.
end >= end || !text[this->interval.end]) {
3360 bookmark.
end = this->interval.
end;
3363 if (m_path_char->match(text, this->interval.end, end, flags))
3364 this->interval.
end = m_path_char->interval.end;
3366 bookmark.
end = this->interval.
end;
3370 this->interval.
start = start;
3373 if (m_path_char->match(text, this->interval.end, end, flags))
3374 this->interval.
end = m_path_char->interval.end;
3380 path.
end = this->interval.
end;
3381 this->interval.
start = start;
3393 std::shared_ptr<basic_parser<T>> m_path_char;
3394 std::shared_ptr<basic_parser<T>> m_query_start;
3395 std::shared_ptr<basic_parser<T>> m_bookmark_start;
3431 _In_
const std::locale&
locale = std::locale()) :
3433 http_scheme(_http_scheme),
3434 ftp_scheme(_ftp_scheme),
3435 mailto_scheme(_mailto_scheme),
3436 file_scheme(_file_scheme),
3439 username(_username),
3440 password(_password),
3442 m_ip_lbracket(ip_lbracket),
3443 m_ip_rbracket(ip_rbracket),
3444 ipv4_host(_ipv4_host),
3445 ipv6_host(_ipv6_host),
3446 dns_host(_dns_host),
3451 virtual void invalidate()
3453 http_scheme->invalidate();
3454 ftp_scheme->invalidate();
3455 mailto_scheme->invalidate();
3456 file_scheme->invalidate();
3457 username->invalidate();
3458 password->invalidate();
3459 ipv4_host->invalidate();
3460 ipv6_host->invalidate();
3461 dns_host->invalidate();
3467 std::shared_ptr<basic_parser<T>> http_scheme;
3468 std::shared_ptr<basic_parser<T>> ftp_scheme;
3469 std::shared_ptr<basic_parser<T>> mailto_scheme;
3470 std::shared_ptr<basic_parser<T>> file_scheme;
3471 std::shared_ptr<basic_parser<T>> username;
3472 std::shared_ptr<basic_parser<T>> password;
3473 std::shared_ptr<basic_parser<T>> ipv4_host;
3474 std::shared_ptr<basic_parser<T>> ipv6_host;
3475 std::shared_ptr<basic_parser<T>> dns_host;
3476 std::shared_ptr<basic_parser<T>> port;
3477 std::shared_ptr<basic_parser<T>> path;
3480 virtual bool do_match(
3481 _In_reads_or_z_opt_(end)
const T* text,
3482 _In_
size_t start = 0,
3483 _In_
size_t end = SIZE_MAX,
3484 _In_
int flags = match_default)
3486 stdex_assert(text || start >= end);
3488 this->interval.
end = start;
3490 if (http_scheme->match(text, this->interval.end, end, flags) &&
3491 m_colon->match(text, http_scheme->interval.end, end, flags) &&
3492 m_slash->match(text, m_colon->interval.end, end, flags) &&
3493 m_slash->match(text, m_slash->interval.end, end, flags))
3496 this->interval.
end = m_slash->interval.end;
3497 ftp_scheme->invalidate();
3498 mailto_scheme->invalidate();
3499 file_scheme->invalidate();
3501 else if (ftp_scheme->match(text, this->interval.end, end, flags) &&
3502 m_colon->match(text, ftp_scheme->interval.end, end, flags) &&
3503 m_slash->match(text, m_colon->interval.end, end, flags) &&
3504 m_slash->match(text, m_slash->interval.end, end, flags))
3507 this->interval.
end = m_slash->interval.end;
3508 http_scheme->invalidate();
3509 mailto_scheme->invalidate();
3510 file_scheme->invalidate();
3512 else if (mailto_scheme->match(text, this->interval.end, end, flags) &&
3513 m_colon->match(text, mailto_scheme->interval.end, end, flags))
3516 this->interval.
end = m_colon->interval.end;
3517 http_scheme->invalidate();
3518 ftp_scheme->invalidate();
3519 file_scheme->invalidate();
3521 else if (file_scheme->match(text, this->interval.end, end, flags) &&
3522 m_colon->match(text, file_scheme->interval.end, end, flags) &&
3523 m_slash->match(text, m_colon->interval.end, end, flags) &&
3524 m_slash->match(text, m_slash->interval.end, end, flags))
3527 this->interval.
end = m_slash->interval.end;
3528 http_scheme->invalidate();
3529 ftp_scheme->invalidate();
3530 mailto_scheme->invalidate();
3534 http_scheme->invalidate();
3535 ftp_scheme->invalidate();
3536 mailto_scheme->invalidate();
3537 file_scheme->invalidate();
3540 if (ftp_scheme->interval) {
3541 if (username->match(text, this->interval.end, end, flags)) {
3542 if (m_colon->match(text, username->interval.end, end, flags) &&
3543 password->match(text, m_colon->interval.end, end, flags) &&
3544 m_at->match(text, password->interval.end, end, flags))
3547 this->interval.
end = m_at->interval.end;
3549 else if (m_at->match(text, this->interval.end, end, flags)) {
3551 this->interval.
end = m_at->interval.end;
3552 password->invalidate();
3555 username->invalidate();
3556 password->invalidate();
3560 username->invalidate();
3561 password->invalidate();
3564 if (ipv4_host->match(text, this->interval.end, end, flags)) {
3566 this->interval.
end = ipv4_host->interval.end;
3567 ipv6_host->invalidate();
3568 dns_host->invalidate();
3571 m_ip_lbracket->match(text, this->interval.end, end, flags) &&
3572 ipv6_host->match(text, m_ip_lbracket->interval.end, end, flags) &&
3573 m_ip_rbracket->match(text, ipv6_host->interval.end, end, flags))
3576 this->interval.
end = m_ip_rbracket->interval.end;
3577 ipv4_host->invalidate();
3578 dns_host->invalidate();
3580 else if (dns_host->match(text, this->interval.end, end, flags)) {
3582 this->interval.
end = dns_host->interval.end;
3583 ipv4_host->invalidate();
3584 ipv6_host->invalidate();
3591 if (m_colon->match(text, this->interval.end, end, flags) &&
3592 port->match(text, m_colon->interval.end, end, flags))
3595 this->interval.
end = port->interval.end;
3600 if (path->match(text, this->interval.end, end, flags)) {
3602 this->interval.
end = path->interval.end;
3605 this->interval.
start = start;
3609 if (mailto_scheme->interval) {
3610 if (username->match(text, this->interval.end, end, flags) &&
3611 m_at->match(text, username->interval.end, end, flags))
3614 this->interval.
end = m_at->interval.end;
3621 if (m_ip_lbracket->match(text, this->interval.end, end, flags) &&
3622 ipv4_host->match(text, m_ip_lbracket->interval.end, end, flags) &&
3623 m_ip_rbracket->match(text, ipv4_host->interval.end, end, flags))
3626 this->interval.
end = m_ip_rbracket->interval.end;
3627 ipv6_host->invalidate();
3628 dns_host->invalidate();
3631 m_ip_lbracket->match(text, this->interval.end, end, flags) &&
3632 ipv6_host->match(text, m_ip_lbracket->interval.end, end, flags) &&
3633 m_ip_rbracket->match(text, ipv6_host->interval.end, end, flags))
3636 this->interval.
end = m_ip_rbracket->interval.end;
3637 ipv4_host->invalidate();
3638 dns_host->invalidate();
3640 else if (dns_host->match(text, this->interval.end, end, flags)) {
3642 this->interval.
end = dns_host->interval.end;
3643 ipv4_host->invalidate();
3644 ipv6_host->invalidate();
3651 password->invalidate();
3654 this->interval.
start = start;
3658 if (file_scheme->interval) {
3659 if (path->match(text, this->interval.end, end, flags)) {
3661 this->interval.
end = path->interval.end;
3664 username->invalidate();
3665 password->invalidate();
3666 ipv4_host->invalidate();
3667 ipv6_host->invalidate();
3668 dns_host->invalidate();
3670 this->interval.
start = start;
3677 if (http_scheme->interval &&
3678 username->match(text, this->interval.end, end, flags))
3680 if (m_colon->match(text, username->interval.end, end, flags) &&
3681 password->match(text, m_colon->interval.end, end, flags) &&
3682 m_at->match(text, password->interval.end, end, flags))
3685 this->interval.
end = m_at->interval.end;
3687 else if (m_at->match(text, username->interval.end, end, flags)) {
3689 this->interval.
end = m_at->interval.end;
3690 password->invalidate();
3693 username->invalidate();
3694 password->invalidate();
3698 username->invalidate();
3699 password->invalidate();
3702 if (ipv4_host->match(text, this->interval.end, end, flags)) {
3704 this->interval.
end = ipv4_host->interval.end;
3705 ipv6_host->invalidate();
3706 dns_host->invalidate();
3709 m_ip_lbracket->match(text, this->interval.end, end, flags) &&
3710 ipv6_host->match(text, m_ip_lbracket->interval.end, end, flags) &&
3711 m_ip_rbracket->match(text, ipv6_host->interval.end, end, flags))
3714 this->interval.
end = m_ip_rbracket->interval.end;
3715 ipv4_host->invalidate();
3716 dns_host->invalidate();
3718 else if (dns_host->match(text, this->interval.end, end, flags)) {
3720 this->interval.
end = dns_host->interval.end;
3721 ipv4_host->invalidate();
3722 ipv6_host->invalidate();
3729 if (m_colon->match(text, this->interval.end, end, flags) &&
3730 port->match(text, m_colon->interval.end, end, flags))
3733 this->interval.
end = port->interval.end;
3738 if (path->match(text, this->interval.end, end, flags)) {
3740 this->interval.
end = path->interval.end;
3743 this->interval.
start = start;
3747 std::shared_ptr<basic_parser<T>> m_colon;
3748 std::shared_ptr<basic_parser<T>> m_slash;
3749 std::shared_ptr<basic_parser<T>> m_at;
3750 std::shared_ptr<basic_parser<T>> m_ip_lbracket;
3751 std::shared_ptr<basic_parser<T>> m_ip_rbracket;
3778 _In_
const std::locale&
locale = std::locale()) :
3780 username(_username),
3782 m_ip_lbracket(ip_lbracket),
3783 m_ip_rbracket(ip_rbracket),
3784 ipv4_host(_ipv4_host),
3785 ipv6_host(_ipv6_host),
3789 virtual void invalidate()
3791 username->invalidate();
3792 ipv4_host->invalidate();
3793 ipv6_host->invalidate();
3794 dns_host->invalidate();
3798 std::shared_ptr<basic_parser<T>> username;
3799 std::shared_ptr<basic_parser<T>> ipv4_host;
3800 std::shared_ptr<basic_parser<T>> ipv6_host;
3801 std::shared_ptr<basic_parser<T>> dns_host;
3804 virtual bool do_match(
3805 _In_reads_or_z_opt_(end)
const T* text,
3806 _In_
size_t start = 0,
3807 _In_
size_t end = SIZE_MAX,
3808 _In_
int flags = match_default)
3810 stdex_assert(text || start >= end);
3812 if (username->match(text, start, end, flags) &&
3813 m_at->match(text, username->interval.end, end, flags))
3816 if (m_ip_lbracket->match(text, m_at->interval.end, end, flags) &&
3817 ipv4_host->match(text, m_ip_lbracket->interval.end, end, flags) &&
3818 m_ip_rbracket->match(text, ipv4_host->interval.end, end, flags))
3821 this->interval.
end = m_ip_rbracket->interval.end;
3822 ipv6_host->invalidate();
3823 dns_host->invalidate();
3826 m_ip_lbracket->match(text, m_at->interval.end, end, flags) &&
3827 ipv6_host->match(text, m_ip_lbracket->interval.end, end, flags) &&
3828 m_ip_rbracket->match(text, ipv6_host->interval.end, end, flags))
3831 this->interval.
end = m_ip_rbracket->interval.end;
3832 ipv4_host->invalidate();
3833 dns_host->invalidate();
3835 else if (dns_host->match(text, m_at->interval.end, end, flags)) {
3837 this->interval.
end = dns_host->interval.end;
3838 ipv4_host->invalidate();
3839 ipv6_host->invalidate();
3843 this->interval.
start = start;
3852 std::shared_ptr<basic_parser<T>> m_at;
3853 std::shared_ptr<basic_parser<T>> m_ip_lbracket;
3854 std::shared_ptr<basic_parser<T>> m_ip_rbracket;
3879 _In_
const std::locale&
locale = std::locale()) :
3888 virtual void invalidate()
3894 mouth->invalidate();
3899 std::shared_ptr<basic_parser<T>>
apex;
3900 std::shared_ptr<basic_parser<T>>
eyes;
3901 std::shared_ptr<basic_parser<T>>
nose;
3905 virtual bool do_match(
3906 _In_reads_or_z_opt_(end)
const T* text,
3907 _In_
size_t start = 0,
3908 _In_
size_t end = SIZE_MAX,
3909 _In_
int flags = match_default)
3911 stdex_assert(text || start >= end);
3917 mouth->invalidate();
3918 this->interval.
start = start;
3923 this->interval.
end = start;
3925 if (
apex &&
apex->match(text, this->interval.end, end, flags))
3926 this->interval.
end =
apex->interval.end;
3928 if (
eyes->match(text, this->interval.end, end, flags)) {
3929 if (
nose &&
nose->match(text,
eyes->interval.end, end, flags) &&
3930 mouth->match(text,
nose->interval.end, end, flags))
3933 start_mouth =
mouth->interval.start,
3934 hit_offset =
mouth->hit_offset;
3936 for (this->interval.
end =
mouth->interval.end;
mouth->match(text, this->interval.end, end, flags) &&
mouth->hit_offset == hit_offset; this->interval.end =
mouth->interval.end);
3937 mouth->interval.start = start_mouth;
3938 mouth->interval.end = this->interval.
end;
3939 this->interval.
start = start;
3942 if (
mouth->match(text,
eyes->interval.end, end, flags)) {
3944 start_mouth =
mouth->interval.start,
3945 hit_offset =
mouth->hit_offset;
3947 for (this->interval.
end =
mouth->interval.end;
mouth->match(text, this->interval.end, end, flags) &&
mouth->hit_offset == hit_offset; this->interval.end =
mouth->interval.end);
3949 mouth->interval.start = start_mouth;
3950 mouth->interval.end = this->interval.
end;
3951 this->interval.
start = start;
3960 mouth->invalidate();
3966 using emoticon = basic_emoticon<char>;
3967 using wemoticon = basic_emoticon<wchar_t>;
3969 using temoticon = wemoticon;
3971 using temoticon = emoticon;
3973 using sgml_emoticon = basic_emoticon<char>;
3978 enum date_format_t {
3979 date_format_none = 0,
3980 date_format_dmy = 0x1,
3981 date_format_mdy = 0x2,
3982 date_format_ymd = 0x4,
3983 date_format_ym = 0x8,
3984 date_format_my = 0x10,
3985 date_format_dm = 0x20,
3986 date_format_md = 0x40,
3997 _In_
int format_mask,
4003 _In_
const std::locale&
locale = std::locale()) :
4005 format(date_format_none),
4006 m_format_mask(format_mask),
4010 m_separator(separator),
4014 virtual void invalidate()
4016 if (day) day->invalidate();
4017 if (month) month->invalidate();
4018 if (year) year->invalidate();
4019 format = date_format_none;
4023 date_format_t format;
4024 std::shared_ptr<basic_integer<T>> day;
4025 std::shared_ptr<basic_integer<T>> month;
4026 std::shared_ptr<basic_integer<T>> year;
4029 virtual bool do_match(
4030 _In_reads_or_z_opt_(end)
const T* text,
4031 _In_
size_t start = 0,
4032 _In_
size_t end = SIZE_MAX,
4033 _In_
int flags = match_default)
4035 stdex_assert(text || start >= end);
4037 const int space_match_flags = flags & ~match_multiline;
4038 if ((m_format_mask & date_format_dmy) == date_format_dmy) {
4039 if (day->match(text, start, end, flags)) {
4040 for (this->interval.
end = day->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4041 if (m_separator->match(text, this->interval.end, end, flags)) {
4042 size_t hit_offset = m_separator->hit_offset;
4043 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4044 if (month->match(text, this->interval.end, end, flags)) {
4045 for (this->interval.
end = month->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4046 if (m_separator->match(text, this->interval.end, end, flags) &&
4047 m_separator->hit_offset == hit_offset)
4049 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4050 if (year->match(text, this->interval.end, end, flags) &&
4051 is_valid(day->value, month->value))
4053 this->interval.
start = start;
4054 this->interval.
end = year->interval.end;
4055 format = date_format_dmy;
4064 if ((m_format_mask & date_format_mdy) == date_format_mdy) {
4065 if (month->match(text, start, end, flags)) {
4066 for (this->interval.
end = month->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4067 if (m_separator->match(text, this->interval.end, end, flags)) {
4068 size_t hit_offset = m_separator->hit_offset;
4069 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4070 if (day->match(text, this->interval.end, end, flags)) {
4071 for (this->interval.
end = day->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4072 if (m_separator->match(text, this->interval.end, end, flags) &&
4073 m_separator->hit_offset == hit_offset)
4075 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4076 if (year->match(text, this->interval.end, end, flags) &&
4077 is_valid(day->value, month->value))
4079 this->interval.
start = start;
4080 this->interval.
end = year->interval.end;
4081 format = date_format_mdy;
4090 if ((m_format_mask & date_format_ymd) == date_format_ymd) {
4091 if (year->match(text, start, end, flags)) {
4092 for (this->interval.
end = year->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4093 if (m_separator->match(text, this->interval.end, end, flags)) {
4094 size_t hit_offset = m_separator->hit_offset;
4095 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4096 if (month->match(text, this->interval.end, end, flags)) {
4097 for (this->interval.
end = month->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4098 if (m_separator->match(text, this->interval.end, end, flags) &&
4099 m_separator->hit_offset == hit_offset)
4101 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4102 if (day->match(text, this->interval.end, end, flags) &&
4103 is_valid(day->value, month->value))
4105 this->interval.
start = start;
4106 this->interval.
end = day->interval.end;
4107 format = date_format_ymd;
4116 if ((m_format_mask & date_format_ym) == date_format_ym) {
4117 if (year->match(text, start, end, flags)) {
4118 for (this->interval.
end = year->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4119 if (m_separator->match(text, this->interval.end, end, flags)) {
4120 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4121 if (month->match(text, this->interval.end, end, flags) &&
4122 is_valid(SIZE_MAX, month->value))
4124 if (day) day->invalidate();
4125 this->interval.
start = start;
4126 this->interval.
end = month->interval.end;
4127 format = date_format_ym;
4134 if ((m_format_mask & date_format_my) == date_format_my) {
4135 if (month->match(text, start, end, flags)) {
4136 for (this->interval.
end = month->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4137 if (m_separator->match(text, this->interval.end, end, flags)) {
4138 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4139 if (year->match(text, this->interval.end, end, flags) &&
4140 is_valid(SIZE_MAX, month->value))
4142 if (day) day->invalidate();
4143 this->interval.
start = start;
4144 this->interval.
end = year->interval.end;
4145 format = date_format_my;
4152 if ((m_format_mask & date_format_dm) == date_format_dm) {
4153 if (day->match(text, start, end, flags)) {
4154 for (this->interval.
end = day->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4155 if (m_separator->match(text, this->interval.end, end, flags)) {
4156 size_t hit_offset = m_separator->hit_offset;
4157 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4158 if (month->match(text, this->interval.end, end, flags) &&
4159 is_valid(day->value, month->value))
4161 if (year) year->invalidate();
4162 this->interval.
start = start;
4163 for (this->interval.
end = month->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4164 if (m_separator->match(text, this->interval.end, end, flags) &&
4165 m_separator->hit_offset == hit_offset)
4166 this->interval.
end = m_separator->interval.end;
4168 this->interval.
end = month->interval.end;
4169 format = date_format_dm;
4176 if ((m_format_mask & date_format_md) == date_format_md) {
4177 if (month->match(text, start, end, flags)) {
4178 for (this->interval.
end = month->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4179 if (m_separator->match(text, this->interval.end, end, flags)) {
4180 size_t hit_offset = m_separator->hit_offset;
4181 for (this->interval.
end = m_separator->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4182 if (day->match(text, this->interval.end, end, flags) &&
4183 is_valid(day->value, month->value))
4185 if (year) year->invalidate();
4186 this->interval.
start = start;
4187 for (this->interval.
end = day->interval.end; m_space->match(text, this->interval.end, end, space_match_flags); this->interval.end = m_space->interval.end);
4188 if (m_separator->match(text, this->interval.end, end, flags) &&
4189 m_separator->hit_offset == hit_offset)
4190 this->interval.
end = m_separator->interval.end;
4192 this->interval.
end = day->interval.end;
4193 format = date_format_md;
4200 if (day) day->invalidate();
4201 if (month) month->invalidate();
4202 if (year) year->invalidate();
4203 format = date_format_none;
4208 static bool is_valid(
size_t day,
size_t month)
4210 if (month == SIZE_MAX) {
4214 if (day == SIZE_MAX) {
4227 return 1 <= day && day <= 31;
4229 return 1 <= day && day <= 29;
4234 return 1 <= day && day <= 30;
4241 std::shared_ptr<basic_set<T>> m_separator;
4242 std::shared_ptr<basic_parser<T>> m_space;
4268 _In_
const std::locale&
locale = std::locale()) :
4273 millisecond(_millisecond),
4274 m_separator(separator),
4275 m_millisecond_separator(millisecond_separator)
4278 virtual void invalidate()
4281 minute->invalidate();
4282 if (second) second->invalidate();
4283 if (millisecond) millisecond->invalidate();
4287 std::shared_ptr<basic_integer10<T>> hour;
4288 std::shared_ptr<basic_integer10<T>> minute;
4289 std::shared_ptr<basic_integer10<T>> second;
4290 std::shared_ptr<basic_integer10<T>> millisecond;
4293 virtual bool do_match(
4294 _In_reads_or_z_opt_(end)
const T* text,
4295 _In_
size_t start = 0,
4296 _In_
size_t end = SIZE_MAX,
4297 _In_
int flags = match_default)
4299 stdex_assert(text || start >= end);
4301 if (hour->match(text, start, end, flags) &&
4302 m_separator->match(text, hour->interval.end, end, flags) &&
4303 minute->match(text, m_separator->interval.end, end, flags) &&
4307 size_t hit_offset = m_separator->hit_offset;
4308 if (m_separator->match(text, minute->interval.end, end, flags) &&
4309 m_separator->hit_offset == hit_offset &&
4310 second && second->match(text, m_separator->interval.end, end, flags) &&
4314 if (m_millisecond_separator && m_millisecond_separator->match(text, second->interval.end, end, flags) &&
4315 millisecond && millisecond->match(text, m_millisecond_separator->interval.end, end, flags) &&
4316 millisecond->value < 1000)
4319 this->interval.
end = millisecond->interval.end;
4322 if (millisecond) millisecond->invalidate();
4323 this->interval.
end = second->interval.end;
4327 if (second) second->invalidate();
4328 if (millisecond) millisecond->invalidate();
4329 this->interval.
end = minute->interval.end;
4331 this->interval.
start = start;
4336 minute->invalidate();
4337 if (second) second->invalidate();
4338 if (millisecond) millisecond->invalidate();
4343 std::shared_ptr<basic_set<T>> m_separator;
4344 std::shared_ptr<basic_parser<T>> m_millisecond_separator;
4371 _In_
const std::locale&
locale = std::locale()) :
4374 degree_separator(_degree_separator),
4376 minute_separator(_minute_separator),
4378 second_separator(_second_separator),
4382 virtual void invalidate()
4384 degree->invalidate();
4385 degree_separator->invalidate();
4386 minute->invalidate();
4387 minute_separator->invalidate();
4388 if (second) second->invalidate();
4389 if (second_separator) second_separator->invalidate();
4390 if (decimal) decimal->invalidate();
4394 std::shared_ptr<basic_integer10<T>> degree;
4395 std::shared_ptr<basic_parser<T>> degree_separator;
4396 std::shared_ptr<basic_integer10<T>> minute;
4397 std::shared_ptr<basic_parser<T>> minute_separator;
4398 std::shared_ptr<basic_integer10<T>> second;
4399 std::shared_ptr<basic_parser<T>> second_separator;
4400 std::shared_ptr<basic_parser<T>> decimal;
4403 virtual bool do_match(
4404 _In_reads_or_z_opt_(end)
const T* text,
4405 _In_
size_t start = 0,
4406 _In_
size_t end = SIZE_MAX,
4407 _In_
int flags = match_default)
4409 stdex_assert(text || start >= end);
4411 this->interval.
end = start;
4413 if (degree->match(text, this->interval.end, end, flags) &&
4414 degree_separator->match(text, degree->interval.end, end, flags))
4417 this->interval.
end = degree_separator->interval.end;
4420 degree->invalidate();
4421 degree_separator->invalidate();
4424 if (minute->match(text, this->interval.end, end, flags) &&
4425 minute->value < 60 &&
4426 minute_separator->match(text, minute->interval.end, end, flags))
4429 this->interval.
end = minute_separator->interval.end;
4432 minute->invalidate();
4433 minute_separator->invalidate();
4436 if (second && second->match(text, this->interval.end, end, flags) &&
4440 this->interval.
end = second->interval.end;
4441 if (second_separator && second_separator->match(text, this->interval.end, end, flags))
4442 this->interval.
end = second_separator->interval.end;
4444 if (second_separator) second_separator->invalidate();
4447 if (second) second->invalidate();
4448 if (second_separator) second_separator->invalidate();
4451 if (degree->interval.start < degree->interval.end ||
4452 minute->interval.start < minute->interval.end ||
4453 (second && second->interval.start < second->interval.end))
4455 if (decimal && decimal->match(text, this->interval.end, end, flags)) {
4457 this->interval.
end = decimal->interval.end;
4460 decimal->invalidate();
4461 this->interval.
start = start;
4464 if (decimal) decimal->invalidate();
4489 _In_
const std::shared_ptr<
basic_set<T>>& lparenthesis,
4490 _In_
const std::shared_ptr<
basic_set<T>>& rparenthesis,
4493 _In_
const std::locale&
locale = std::locale()) :
4496 m_plus_sign(plus_sign),
4497 m_lparenthesis(lparenthesis),
4498 m_rparenthesis(rparenthesis),
4499 m_separator(separator),
4503 virtual void invalidate()
4512 virtual bool do_match(
4513 _In_reads_or_z_opt_(end)
const T* text,
4514 _In_
size_t start = 0,
4515 _In_
size_t end = SIZE_MAX,
4516 _In_
int flags = match_default)
4518 stdex_assert(text || start >= end);
4520 size_t safe_digit_end = start, safe_value_size = 0;
4521 bool has_digits =
false, after_digit =
false, in_parentheses =
false, after_parentheses =
false;
4522 const int space_match_flags = flags & ~match_multiline;
4524 this->interval.
end = start;
4526 m_lparenthesis->invalidate();
4527 m_rparenthesis->invalidate();
4529 if (m_plus_sign && m_plus_sign->match(text, this->interval.end, end, flags)) {
4530 value.append(text + m_plus_sign->interval.start, text + m_plus_sign->interval.end);
4531 safe_value_size =
value.size();
4532 this->interval.
end = m_plus_sign->interval.end;
4536 stdex_assert(text || this->interval.
end >= end);
4537 if (this->interval.
end >= end || !text[this->interval.end])
4539 if (m_digit->match(text, this->interval.end, end, flags)) {
4541 value.append(text + m_digit->interval.start, text + m_digit->interval.end);
4542 this->interval.
end = m_digit->interval.end;
4543 if (!in_parentheses) {
4544 safe_digit_end = this->interval.
end;
4545 safe_value_size =
value.size();
4549 after_parentheses =
false;
4552 m_lparenthesis && !m_lparenthesis->interval &&
4553 m_rparenthesis && !m_rparenthesis->interval &&
4554 m_lparenthesis->match(text, this->interval.end, end, flags))
4557 value.append(text + m_lparenthesis->interval.start, m_lparenthesis->interval.size());
4558 this->interval.
end = m_lparenthesis->interval.end;
4559 in_parentheses =
true;
4560 after_digit =
false;
4561 after_parentheses =
false;
4565 m_rparenthesis && !m_rparenthesis->interval &&
4566 m_rparenthesis->match(text, this->interval.end, end, flags) &&
4567 m_lparenthesis->hit_offset == m_rparenthesis->hit_offset)
4570 value.append(text + m_rparenthesis->interval.start, text + m_rparenthesis->interval.end);
4571 this->interval.
end = m_rparenthesis->interval.end;
4572 safe_digit_end = this->interval.
end;
4573 safe_value_size =
value.size();
4574 in_parentheses =
false;
4575 after_digit =
false;
4576 after_parentheses =
true;
4581 !after_parentheses &&
4582 m_separator && m_separator->match(text, this->interval.end, end, flags))
4585 this->interval.
end = m_separator->interval.end;
4586 after_digit =
false;
4587 after_parentheses =
false;
4590 (after_digit || after_parentheses) &&
4591 m_space && m_space->match(text, this->interval.end, end, space_match_flags))
4594 this->interval.
end = m_space->interval.end;
4595 after_digit =
false;
4596 after_parentheses =
false;
4602 value.erase(safe_value_size);
4603 this->interval.
start = start;
4604 this->interval.
end = safe_digit_end;
4612 std::shared_ptr<basic_parser<T>> m_digit;
4613 std::shared_ptr<basic_parser<T>> m_plus_sign;
4614 std::shared_ptr<basic_set<T>> m_lparenthesis;
4615 std::shared_ptr<basic_set<T>> m_rparenthesis;
4616 std::shared_ptr<basic_parser<T>> m_separator;
4617 std::shared_ptr<basic_parser<T>> m_space;
4620 using phone_number = basic_phone_number<char>;
4621 using wphone_number = basic_phone_number<wchar_t>;
4623 using tphone_number = wphone_number;
4625 using tphone_number = phone_number;
4627 using sgml_phone_number = basic_phone_number<char>;
4640 _In_
const std::locale&
locale = std::locale()) :
4650 virtual void invalidate()
4665 virtual bool do_match(
4666 _In_reads_or_z_opt_(end)
const T* text,
4667 _In_
size_t start = 0,
4668 _In_
size_t end = SIZE_MAX,
4669 _In_
int flags = match_default)
4671 stdex_assert(text || start >= end);
4672 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
4673 const bool case_insensitive = flags & match_case_insensitive ? true :
false;
4679 static const country_t s_countries[] = {
4680 { {
'A',
'D' }, {}, 24 },
4681 { {
'A',
'E' }, {}, 23 },
4682 { {
'A',
'L' }, {}, 28 },
4683 { {
'A',
'O' }, {}, 25 },
4684 { {
'A',
'T' }, {}, 20 },
4685 { {
'A',
'Z' }, {}, 28 },
4686 { {
'B',
'A' }, {
'3',
'9' }, 20},
4687 { {
'B',
'E' }, {}, 16 },
4688 { {
'B',
'F' }, {}, 28 },
4689 { {
'B',
'G' }, {}, 22 },
4690 { {
'B',
'H' }, {}, 22 },
4691 { {
'B',
'I' }, {}, 27 },
4692 { {
'B',
'J' }, {}, 28 },
4693 { {
'B',
'R' }, {}, 29 },
4694 { {
'B',
'Y' }, {}, 28 },
4695 { {
'C',
'F' }, {}, 27 },
4696 { {
'C',
'G' }, {}, 27 },
4697 { {
'C',
'H' }, {}, 21 },
4698 { {
'C',
'I' }, {}, 28 },
4699 { {
'C',
'M' }, {}, 27 },
4700 { {
'C',
'R' }, {}, 22 },
4701 { {
'C',
'V' }, {}, 25 },
4702 { {
'C',
'Y' }, {}, 28 },
4703 { {
'C',
'Z' }, {}, 24 },
4704 { {
'D',
'E' }, {}, 22 },
4705 { {
'D',
'J' }, {}, 27 },
4706 { {
'D',
'K' }, {}, 18 },
4707 { {
'D',
'O' }, {}, 28 },
4708 { {
'D',
'Z' }, {}, 26 },
4709 { {
'E',
'E' }, {}, 20 },
4710 { {
'E',
'G' }, {}, 29 },
4711 { {
'E',
'S' }, {}, 24 },
4712 { {
'F',
'I' }, {}, 18 },
4713 { {
'F',
'O' }, {}, 18 },
4714 { {
'F',
'R' }, {}, 27 },
4715 { {
'G',
'A' }, {}, 27 },
4716 { {
'G',
'B' }, {}, 22 },
4717 { {
'G',
'E' }, {}, 22 },
4718 { {
'G',
'I' }, {}, 23 },
4719 { {
'G',
'L' }, {}, 18 },
4720 { {
'G',
'Q' }, {}, 27 },
4721 { {
'G',
'R' }, {}, 27 },
4722 { {
'G',
'T' }, {}, 28 },
4723 { {
'G',
'W' }, {}, 25 },
4724 { {
'H',
'N' }, {}, 28 },
4725 { {
'H',
'R' }, {}, 21 },
4726 { {
'H',
'U' }, {}, 28 },
4727 { {
'I',
'E' }, {}, 22 },
4728 { {
'I',
'L' }, {}, 23 },
4729 { {
'I',
'Q' }, {}, 23 },
4730 { {
'I',
'R' }, {}, 26 },
4731 { {
'I',
'S' }, {}, 26 },
4732 { {
'I',
'T' }, {}, 27 },
4733 { {
'J',
'O' }, {}, 30 },
4734 { {
'K',
'M' }, {}, 27 },
4735 { {
'K',
'W' }, {}, 30 },
4736 { {
'K',
'Z' }, {}, 20 },
4737 { {
'L',
'B' }, {}, 28 },
4738 { {
'L',
'C' }, {}, 32 },
4739 { {
'L',
'I' }, {}, 21 },
4740 { {
'L',
'T' }, {}, 20 },
4741 { {
'L',
'U' }, {}, 20 },
4742 { {
'L',
'V' }, {}, 21 },
4743 { {
'L',
'Y' }, {}, 25 },
4744 { {
'M',
'A' }, {}, 28 },
4745 { {
'M',
'C' }, {}, 27 },
4746 { {
'M',
'D' }, {}, 24 },
4747 { {
'M',
'E' }, {
'2',
'5' }, 22 },
4748 { {
'M',
'G' }, {}, 27 },
4749 { {
'M',
'K' }, {
'0',
'7' }, 19 },
4750 { {
'M',
'L' }, {}, 28 },
4751 { {
'M',
'R' }, {
'1',
'3' }, 27},
4752 { {
'M',
'T' }, {}, 31 },
4753 { {
'M',
'U' }, {}, 30 },
4754 { {
'M',
'Z' }, {}, 25 },
4755 { {
'N',
'E' }, {}, 28 },
4756 { {
'N',
'I' }, {}, 32 },
4757 { {
'N',
'L' }, {}, 18 },
4758 { {
'N',
'O' }, {}, 15 },
4759 { {
'P',
'K' }, {}, 24 },
4760 { {
'P',
'L' }, {}, 28 },
4761 { {
'P',
'S' }, {}, 29 },
4762 { {
'P',
'T' }, {
'5',
'0' }, 25 },
4763 { {
'Q',
'A' }, {}, 29 },
4764 { {
'R',
'O' }, {}, 24 },
4765 { {
'R',
'S' }, {
'3',
'5' }, 22 },
4766 { {
'R',
'U' }, {}, 33 },
4767 { {
'S',
'A' }, {}, 24 },
4768 { {
'S',
'C' }, {}, 31 },
4769 { {
'S',
'D' }, {}, 18 },
4770 { {
'S',
'E' }, {}, 24 },
4771 { {
'S',
'I' }, {
'5',
'6' }, 19 },
4772 { {
'S',
'K' }, {}, 24 },
4773 { {
'S',
'M' }, {}, 27 },
4774 { {
'S',
'N' }, {}, 28 },
4775 { {
'S',
'T' }, {}, 25 },
4776 { {
'S',
'V' }, {}, 28 },
4777 { {
'T',
'D' }, {}, 27 },
4778 { {
'T',
'G' }, {}, 28 },
4779 { {
'T',
'L' }, {
'3',
'8' }, 23},
4780 { {
'T',
'N' }, {
'5',
'9' }, 24 },
4781 { {
'T',
'R' }, {}, 26 },
4782 { {
'U',
'A' }, {}, 29 },
4783 { {
'V',
'A' }, {}, 22 },
4784 { {
'V',
'G' }, {}, 24 },
4785 { {
'X',
'K' }, {}, 20 },
4787 const country_t* country_desc =
nullptr;
4788 size_t n, available, next, bban_length;
4791 this->interval.
end = start;
4792 for (
size_t i = 0; i < 2; ++i, ++this->interval.
end) {
4793 if (this->interval.
end >= end || !text[this->interval.end])
4795 T chr = case_insensitive ? ctype.toupper(text[this->interval.
end]) : text[this->interval.
end];
4796 if (chr <
'A' ||
'Z' < chr)
4798 this->country[i] = chr;
4800 for (
size_t l = 0, r = _countof(s_countries);;) {
4803 size_t m = (l + r) / 2;
4804 const country_t& c = s_countries[m];
4805 if (c.country[0] < this->country[0] || (c.country[0] == this->country[0] && c.country[1] < this->country[1]))
4807 else if (this->country[0] < c.country[0] || (this->country[0] == c.country[0] && this->country[1] < c.country[1]))
4814 this->country[2] = 0;
4816 for (
size_t i = 0; i < 2; ++i, ++this->interval.
end) {
4817 if (this->interval.
end >= end || text[this->interval.end] <
'0' ||
'9' < text[this->interval.end])
4819 this->check_digits[i] = text[this->interval.
end];
4821 this->check_digits[2] = 0;
4823 if ((country_desc->check_digits[0] && this->check_digits[0] != country_desc->check_digits[0]) ||
4824 (country_desc->check_digits[1] && this->check_digits[1] != country_desc->check_digits[1]))
4827 bban_length = country_desc->length - 4;
4828 for (n = 0; n < bban_length;) {
4829 if (this->interval.
end >= end || !text[this->interval.end])
4831 if (m_space && m_space->match(text, this->interval.end, end, flags)) {
4832 this->interval.
end = m_space->interval.end;
4835 T chr = case_insensitive ? ctype.toupper(text[this->interval.
end]) : text[this->interval.
end];
4836 if ((
'0' <= chr && chr <=
'9') || (
'A' <= chr && chr <=
'Z')) {
4837 this->bban[n++] = chr;
4838 this->interval.
end++;
4848 for (
size_t i = 0; ; ++i) {
4849 if (!this->bban[i]) {
4850 for (i = 0; i < 2; ++i) {
4851 if (
'A' <= this->country[i] && this->country[i] <=
'J') {
4852 normalized[available++] =
'1';
4853 normalized[available++] =
'0' + this->country[i] -
'A';
4855 else if (
'K' <= this->country[i] && this->country[i] <=
'T') {
4856 normalized[available++] =
'2';
4857 normalized[available++] =
'0' + this->country[i] -
'K';
4859 else if (
'U' <= this->country[i] && this->country[i] <=
'Z') {
4860 normalized[available++] =
'3';
4861 normalized[available++] =
'0' + this->country[i] -
'U';
4864 normalized[available++] = this->check_digits[0];
4865 normalized[available++] = this->check_digits[1];
4866 normalized[available] = 0;
4869 if (
'0' <= this->bban[i] && this->bban[i] <=
'9')
4870 normalized[available++] = this->bban[i];
4871 else if (
'A' <= this->bban[i] && this->bban[i] <=
'J') {
4872 normalized[available++] =
'1';
4873 normalized[available++] =
'0' + this->bban[i] -
'A';
4875 else if (
'K' <= this->bban[i] && this->bban[i] <=
'T') {
4876 normalized[available++] =
'2';
4877 normalized[available++] =
'0' + this->bban[i] -
'K';
4879 else if (
'U' <= this->bban[i] && this->bban[i] <=
'Z') {
4880 normalized[available++] =
'3';
4881 normalized[available++] =
'0' + this->bban[i] -
'U';
4886 nominator = stdex::strtou32(normalized, 9, &next, 10);
4889 if (!normalized[next]) {
4890 this->is_valid = nominator == 1;
4893 size_t digit_count = nominator == 0 ? 0 : nominator < 10 ? 1 : 2;
4894 for (; digit_count < 9 && normalized[next]; ++next, ++digit_count)
4895 nominator = nominator * 10 +
static_cast<uint32_t
>(normalized[next] -
'0');
4898 this->interval.
start = start;
4906 std::shared_ptr<basic_parser<T>> m_space;
4909 using iban = basic_iban<char>;
4910 using wiban = basic_iban<wchar_t>;
4912 using tiban = wiban;
4916 using sgml_iban = basic_iban<char>;
4929 _In_
const std::locale&
locale = std::locale()) :
4933 this->check_digits[0] = 0;
4935 this->is_valid =
false;
4938 virtual void invalidate()
4940 this->check_digits[0] = 0;
4942 this->is_valid =
false;
4951 virtual bool do_match(
4952 _In_reads_or_z_opt_(end)
const T* text,
4953 _In_
size_t start = 0,
4954 _In_
size_t end = SIZE_MAX,
4955 _In_
int flags = match_default)
4957 stdex_assert(text || start >= end);
4958 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
4959 const bool case_insensitive = flags & match_case_insensitive ? true :
false;
4960 size_t n, available, next;
4963 this->interval.
end = start;
4964 if (this->interval.
end + 1 >= end ||
4965 (case_insensitive ? ctype.toupper(text[this->interval.end]) : text[this->interval.end]) !=
'R' ||
4966 (case_insensitive ? ctype.toupper(text[this->interval.end + 1]) : text[this->interval.end + 1]) !=
'F')
4968 this->interval.
end += 2;
4970 for (
size_t i = 0; i < 2; ++i, ++this->interval.
end) {
4971 if (this->interval.
end >= end || text[this->interval.end] <
'0' ||
'9' < text[this->interval.end])
4973 this->check_digits[i] = text[this->interval.
end];
4975 this->check_digits[2] = 0;
4978 if (m_space && m_space->match(text, this->interval.end, end, flags))
4979 this->interval.
end = m_space->interval.end;
4980 for (
size_t j = 0; j < 4; ++j) {
4981 if (this->interval.
end >= end || !text[this->interval.end])
4983 T chr = case_insensitive ? ctype.toupper(text[this->interval.
end]) : text[this->interval.
end];
4984 if ((
'0' <= chr && chr <=
'9') || (
'A' <= chr && chr <=
'Z')) {
4987 this->reference[n++] = chr;
4988 this->interval.
end++;
4997 this->reference[_countof(this->reference) - 1] = 0;
4998 for (
size_t i = n, j = _countof(this->reference) - 1; i;)
4999 this->reference[--j] = this->reference[--i];
5000 for (
size_t j = _countof(this->reference) - 1 - n; j;)
5001 this->reference[--j] =
'0';
5006 for (
size_t i = 0; ; ++i) {
5007 if (!this->reference[i]) {
5008 normalized[available++] =
'2';
5009 normalized[available++] =
'7';
5010 normalized[available++] =
'1';
5011 normalized[available++] =
'5';
5012 normalized[available++] = this->check_digits[0];
5013 normalized[available++] = this->check_digits[1];
5014 normalized[available] = 0;
5017 if (
'0' <= this->reference[i] && this->reference[i] <=
'9')
5018 normalized[available++] = this->reference[i];
5019 else if (
'A' <= this->reference[i] && this->reference[i] <=
'J') {
5020 normalized[available++] =
'1';
5021 normalized[available++] =
'0' + this->reference[i] -
'A';
5023 else if (
'K' <= this->reference[i] && this->reference[i] <=
'T') {
5024 normalized[available++] =
'2';
5025 normalized[available++] =
'0' + this->reference[i] -
'K';
5027 else if (
'U' <= this->reference[i] && this->reference[i] <=
'Z') {
5028 normalized[available++] =
'3';
5029 normalized[available++] =
'0' + this->reference[i] -
'U';
5034 nominator = stdex::strtou32(normalized, 9, &next, 10);
5037 if (!normalized[next]) {
5038 this->is_valid = nominator == 1;
5041 size_t digit_count = nominator == 0 ? 0 : nominator < 10 ? 1 : 2;
5042 for (; digit_count < 9 && normalized[next]; ++next, ++digit_count)
5043 nominator = nominator * 10 +
static_cast<uint32_t
>(normalized[next] -
'0');
5046 this->interval.
start = start;
5054 std::shared_ptr<basic_parser<T>> m_space;
5057 using creditor_reference = basic_creditor_reference<char>;
5058 using wcreditor_reference = basic_creditor_reference<wchar_t>;
5060 using tcreditor_reference = wcreditor_reference;
5062 using tcreditor_reference = creditor_reference;
5064 using sgml_creditor_reference = basic_creditor_reference<char>;
5078 virtual bool do_match(
5079 _In_reads_or_z_opt_(end)
const T* text,
5080 _In_
size_t start = 0,
5081 _In_
size_t end = SIZE_MAX,
5082 _In_
int flags = match_default)
5084 stdex_assert(text || start >= end);
5085 this->interval.
end = start;
5087 if (this->interval.
end >= end || !text[this->interval.end])
5089 if (
'0' <= text[this->interval.
end] && text[this->interval.end] <=
'9')
5090 this->interval.
end++;
5095 this->interval.
start = start;
5124 virtual bool do_match(
5125 _In_reads_or_z_opt_(end)
const T* text,
5126 _In_
size_t start = 0,
5127 _In_
size_t end = SIZE_MAX,
5128 _In_
int flags = match_default)
5130 stdex_assert(text || start >= end);
5131 if (start < end && text[start] ==
'-') {
5132 this->interval.
end = (this->interval.
start = start) + 1;
5162 _In_
const std::locale&
locale = std::locale()) :
5174 virtual void invalidate()
5177 this->
part1.invalidate();
5178 this->
part2.invalidate();
5179 this->
part3.invalidate();
5180 this->is_valid =
false;
5191 virtual bool do_match(
5192 _In_reads_or_z_opt_(end)
const T* text,
5193 _In_
size_t start = 0,
5194 _In_
size_t end = SIZE_MAX,
5195 _In_
int flags = match_default)
5197 stdex_assert(text || start >= end);
5198 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
5199 const bool case_insensitive = flags & match_case_insensitive ? true :
false;
5201 this->interval.
end = start;
5202 if (this->interval.
end + 1 >= end ||
5203 (case_insensitive ? ctype.toupper(text[this->interval.end]) : text[this->interval.end]) !=
'S' ||
5204 (case_insensitive ? ctype.toupper(text[this->interval.end + 1]) : text[this->interval.end + 1]) !=
'I')
5206 this->interval.
end += 2;
5208 for (
size_t i = 0; i < 2; ++i, ++this->interval.
end) {
5209 if (this->interval.
end >= end || text[this->interval.end] <
'0' ||
'9' < text[this->interval.end])
5211 this->model[i] = text[this->interval.
end];
5215 this->part1.invalidate();
5216 this->part2.invalidate();
5217 this->part3.invalidate();
5218 if (this->model[0] ==
'9' && this->model[1] ==
'9') {
5220 this->interval.
start = start;
5224 if (m_space && m_space->match(text, this->interval.end, end, flags))
5225 this->interval.
end = m_space->interval.end;
5227 this->part1.match(text, this->interval.
end, end, flags) &&
5228 this->m_delimiter.match(text, this->part1.
interval.
end, end, flags) &&
5229 this->part2.match(text, this->m_delimiter.interval.end, end, flags) &&
5230 this->m_delimiter.match(text, this->part2.
interval.
end, end, flags) &&
5231 this->part3.match(text, this->m_delimiter.interval.end, end, flags);
5233 this->interval.
start = start;
5241 this->interval.
end = start + 4;
5243 if (this->model[0] ==
'0' && this->model[1] ==
'0')
5254 else if (this->model[0] ==
'0' && this->model[1] ==
'1')
5260 text + this->part1.
interval.
start, this->part1.interval.size(),
5261 text + this->part2.interval.start, this->part2.interval.size(),
5262 text + this->part3.interval.start, this->part3.interval.size()) :
5267 text + this->part1.
interval.
start, this->part1.interval.size(),
5268 text + this->part2.interval.start, this->part2.interval.size()) :
5271 check11(text + this->part1.
interval.
start, this->part1.interval.size()) :
5273 else if (this->model[0] ==
'0' && this->model[1] ==
'2')
5278 check11(text + this->part2.
interval.
start, this->part2.interval.size()) &&
5279 check11(text + this->part3.
interval.
start, this->part3.interval.size()) :
5281 else if (this->model[0] ==
'0' && this->model[1] ==
'3')
5286 check11(text + this->part1.
interval.
start, this->part1.interval.size()) &&
5287 check11(text + this->part2.
interval.
start, this->part2.interval.size()) &&
5288 check11(text + this->part3.
interval.
start, this->part3.interval.size()) :
5290 else if (this->model[0] ==
'0' && this->model[1] ==
'4')
5295 check11(text + this->part1.
interval.
start, this->part1.interval.size()) &&
5296 check11(text + this->part3.
interval.
start, this->part3.interval.size()) :
5298 else if ((this->model[0] ==
'0' || this->model[0] ==
'5') && this->model[1] ==
'5')
5303 check11(text + this->part1.
interval.
start, this->part1.interval.size()) :
5307 check11(text + this->part1.
interval.
start, this->part1.interval.size()) :
5310 check11(text + this->part1.
interval.
start, this->part1.interval.size()) :
5312 else if (this->model[0] ==
'0' && this->model[1] ==
'6')
5318 text + this->part2.
interval.
start, this->part2.interval.size(),
5319 text + this->part3.interval.start, this->part3.interval.size()) :
5323 check11(text + this->part2.
interval.
start, this->part2.interval.size()) :
5325 else if (this->model[0] ==
'0' && this->model[1] ==
'7')
5330 check11(text + this->part2.
interval.
start, this->part2.interval.size()) :
5334 check11(text + this->part2.
interval.
start, this->part2.interval.size()) :
5336 else if (this->model[0] ==
'0' && this->model[1] ==
'8')
5342 text + this->part1.
interval.
start, this->part1.interval.size(),
5343 text + this->part2.interval.start, this->part2.interval.size()) &&
5344 check11(text + this->part3.
interval.
start, this->part3.interval.size()) :
5346 else if (this->model[0] ==
'0' && this->model[1] ==
'9')
5352 text + this->part1.
interval.
start, this->part1.interval.size(),
5353 text + this->part2.interval.start, this->part2.interval.size()) :
5358 text + this->part1.
interval.
start, this->part1.interval.size(),
5359 text + this->part2.interval.start, this->part2.interval.size()) :
5362 check11(text + this->part1.
interval.
start, this->part1.interval.size()) :
5364 else if (this->model[0] ==
'1' && this->model[1] ==
'0')
5369 check11(text + this->part1.
interval.
start, this->part1.interval.size()) &&
5371 text + this->part2.
interval.
start, this->part2.interval.size(),
5372 text + this->part3.interval.start, this->part3.interval.size()) :
5376 check11(text + this->part1.
interval.
start, this->part1.interval.size()) &&
5377 check11(text + this->part2.
interval.
start, this->part2.interval.size()) :
5380 (this->model[0] ==
'1' && (this->model[1] ==
'1' || this->model[1] ==
'8' || this->model[1] ==
'9')) ||
5381 ((this->model[0] ==
'2' || this->model[0] ==
'3') && this->model[1] ==
'8') ||
5382 (this->model[0] ==
'4' && (this->model[1] ==
'0' || this->model[1] ==
'1' || this->model[1] ==
'8' || this->model[1] ==
'9')) ||
5383 (this->model[0] ==
'5' && (this->model[1] ==
'1' || this->model[1] ==
'8')))
5388 check11(text + this->part1.
interval.
start, this->part1.interval.size()) &&
5389 check11(text + this->part2.
interval.
start, this->part2.interval.size()) :
5393 check11(text + this->part1.
interval.
start, this->part1.interval.size()) &&
5394 check11(text + this->part2.
interval.
start, this->part2.interval.size()) :
5396 else if (this->model[0] ==
'1' && this->model[1] ==
'2')
5402 check11(text + this->part1.
interval.
start, this->part1.interval.size()) :
5404 else if ((this->model[0] ==
'2' || this->model[0] ==
'3') && this->model[1] ==
'1')
5410 check11(text + this->part1.
interval.
start, this->part1.interval.size()) :
5421 static bool check11(
5422 _In_count_(num_part1)
const T*
part1, _In_
size_t num_part1)
5424 stdex_assert(
part1 && num_part1 >= 1);
5425 uint32_t nominator = 0, ponder = 2;
5426 for (
size_t i = num_part1 - 1; i--; ++ponder)
5427 nominator +=
static_cast<uint32_t
>(
part1[i] -
'0') * ponder;
5428 uint8_t control = 11 -
static_cast<uint8_t
>(nominator % 11);
5431 return control ==
part1[num_part1 - 1] -
'0';
5434 static bool check11(
5435 _In_count_(num_part1)
const T*
part1, _In_
size_t num_part1,
5436 _In_count_(num_part2)
const T*
part2, _In_
size_t num_part2)
5438 stdex_assert(
part1 || !num_part1);
5439 stdex_assert(
part2 && num_part2 >= 1);
5440 uint32_t nominator = 0, ponder = 2;
5441 for (
size_t i = num_part2 - 1; i--; ++ponder)
5442 nominator +=
static_cast<uint32_t
>(
part2[i] -
'0') * ponder;
5443 for (
size_t i = num_part1; i--; ++ponder)
5444 nominator +=
static_cast<uint32_t
>(
part1[i] -
'0') * ponder;
5445 uint8_t control = 11 -
static_cast<uint8_t
>(nominator % 11);
5448 return control ==
part2[num_part2 - 1] -
'0';
5451 static bool check11(
5452 _In_count_(num_part1)
const T*
part1, _In_
size_t num_part1,
5453 _In_count_(num_part2)
const T*
part2, _In_
size_t num_part2,
5454 _In_count_(num_part3)
const T*
part3, _In_
size_t num_part3)
5456 stdex_assert(
part1 || !num_part1);
5457 stdex_assert(
part2 || !num_part2);
5458 stdex_assert(
part3 && num_part3 >= 1);
5459 uint32_t nominator = 0, ponder = 2;
5460 for (
size_t i = num_part3 - 1; i--; ++ponder)
5461 nominator +=
static_cast<uint32_t
>(
part3[i] -
'0') * ponder;
5462 for (
size_t i = num_part2; i--; ++ponder)
5463 nominator +=
static_cast<uint32_t
>(
part2[i] -
'0') * ponder;
5464 for (
size_t i = num_part1; i--; ++ponder)
5465 nominator +=
static_cast<uint32_t
>(
part1[i] -
'0') * ponder;
5466 uint8_t control = 11 -
static_cast<uint8_t
>(nominator % 11);
5469 return control ==
part2[num_part3 - 1] -
'0';
5472 std::shared_ptr<basic_parser<T>> m_space;
5473 basic_si_reference_delimiter<T> m_delimiter;
5476 using si_reference = basic_si_reference<char>;
5477 using wsi_reference = basic_si_reference<wchar_t>;
5479 using tsi_reference = wsi_reference;
5481 using tsi_reference = si_reference;
5483 using sgml_si_reference = basic_si_reference<char>;
5496 _In_
const std::locale&
locale = std::locale()) :
5505 virtual void invalidate()
5516 virtual bool do_match(
5517 _In_reads_or_z_opt_(end)
const T* text,
5518 _In_
size_t start = 0,
5519 _In_
size_t end = SIZE_MAX,
5520 _In_
int flags = match_default)
5522 stdex_assert(text || start >= end);
5526 this->interval.
end = start;
5528 const int element_match_flags = flags & ~match_case_insensitive;
5530 if (m_element->match(text, this->interval.end, end, element_match_flags)) {
5531 this->interval.
end = m_element->interval.end;
5532 while (m_digit->match(text, this->interval.end, end, flags)) {
5533 this->interval.
end = m_digit->interval.end;
5538 if (m_sign->match(text, this->interval.end, end, flags)) {
5539 this->interval.
end = m_sign->interval.end;
5542 this->interval.
start = start;
5552 std::shared_ptr<basic_parser<T>> m_element;
5553 std::shared_ptr<basic_parser<T>> m_digit;
5554 std::shared_ptr<basic_parser<T>> m_sign;
5572 virtual bool do_match(
5573 _In_reads_or_z_(end)
const char* text,
5574 _In_
size_t start = 0,
5575 _In_
size_t end = SIZE_MAX,
5576 _In_
int flags = match_default)
5578 stdex_assert(text || start >= end);
5579 this->interval.
end = start;
5581 stdex_assert(text || this->interval.
end >= end);
5582 if (this->interval.
end < end && text[this->interval.end]) {
5583 if (text[this->interval.
end] ==
'\r') {
5584 this->interval.
end++;
5585 if (this->interval.
end < end && text[this->interval.end] ==
'\n') {
5586 this->interval.
start = start;
5587 this->interval.
end++;
5591 else if (text[this->interval.
end] ==
'\n') {
5592 this->interval.
start = start;
5593 this->interval.
end++;
5608 virtual bool do_match(
5609 _In_reads_or_z_(end)
const char* text,
5610 _In_
size_t start = 0,
5611 _In_
size_t end = SIZE_MAX,
5612 _In_
int flags = match_default)
5614 stdex_assert(text || start >= end);
5615 this->interval.
end = start;
5616 if (m_line_break.match(text, this->interval.end, end, flags)) {
5618 if (this->interval.
end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])) {
5619 this->interval.
start = start;
5620 this->interval.
end++;
5621 while (this->interval.
end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])) this->interval.
end++;
5625 else if (this->interval.
end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])) {
5626 this->interval.
start = start;
5627 this->interval.
end++;
5628 while (this->interval.
end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])) this->interval.
end++;
5644 virtual bool do_match(
5645 _In_reads_or_z_(end)
const char* text,
5646 _In_
size_t start = 0,
5647 _In_
size_t end = SIZE_MAX,
5648 _In_
int flags = match_default)
5650 stdex_assert(text || start >= end);
5651 this->interval.
end = start;
5653 stdex_assert(text || this->interval.
end >= end);
5654 if (m_space.match(text, this->interval.end, end, flags)) {
5655 this->interval.
start = start;
5659 else if (this->interval.
end < end && text[this->interval.end] && text[this->interval.end] >= 0x20) {
5660 this->interval.
start = start;
5661 this->interval.
end++;
5677 virtual bool do_match(
5678 _In_reads_or_z_(end)
const char* text,
5679 _In_
size_t start = 0,
5680 _In_
size_t end = SIZE_MAX,
5681 _In_
int flags = match_default)
5683 stdex_assert(text || start >= end);
5684 this->interval.
end = start;
5686 if (this->interval.
end < end && text[this->interval.end]) {
5687 if ((
unsigned int)text[this->interval.
end] < 0x20 ||
5688 (
unsigned int)text[this->interval.end] == 0x7f ||
5689 text[this->interval.end] ==
'(' ||
5690 text[this->interval.end] ==
')' ||
5691 text[this->interval.end] ==
'<' ||
5692 text[this->interval.end] ==
'>' ||
5693 text[this->interval.end] ==
'@' ||
5694 text[this->interval.end] ==
',' ||
5695 text[this->interval.end] ==
';' ||
5696 text[this->interval.end] ==
':' ||
5697 text[this->interval.end] ==
'\\' ||
5698 text[this->interval.end] ==
'\"' ||
5699 text[this->interval.end] ==
'/' ||
5700 text[this->interval.end] ==
'[' ||
5701 text[this->interval.end] ==
']' ||
5702 text[this->interval.end] ==
'?' ||
5703 text[this->interval.end] ==
'=' ||
5704 text[this->interval.end] ==
'{' ||
5705 text[this->interval.end] ==
'}' ||
5706 stdex::isspace(text[this->interval.end]))
5709 this->interval.
end++;
5715 this->interval.
start = start;
5731 virtual void invalidate()
5735 parser::invalidate();
5741 virtual bool do_match(
5742 _In_reads_or_z_(end)
const char* text,
5743 _In_
size_t start = 0,
5744 _In_
size_t end = SIZE_MAX,
5745 _In_
int flags = match_default)
5747 stdex_assert(text || start >= end);
5748 this->interval.
end = start;
5749 if (this->interval.
end < end && text[this->interval.end] !=
'"')
5751 this->interval.
end++;
5754 stdex_assert(text || this->interval.
end >= end);
5755 if (this->interval.
end < end && text[this->interval.end]) {
5756 if (text[this->interval.
end] ==
'"') {
5758 this->interval.
end++;
5761 else if (text[this->interval.
end] ==
'\\') {
5762 this->interval.
end++;
5763 if (this->interval.
end < end && text[this->interval.end]) {
5764 this->interval.
end++;
5769 else if (m_chr.match(text, this->interval.end, end, flags))
5770 this->interval.
end++;
5777 this->interval.
start = start;
5794 virtual void invalidate()
5796 string.invalidate();
5798 parser::invalidate();
5805 virtual bool do_match(
5806 _In_reads_or_z_(end)
const char* text,
5807 _In_
size_t start = 0,
5808 _In_
size_t end = SIZE_MAX,
5809 _In_
int flags = match_default)
5811 stdex_assert(text || start >= end);
5812 this->interval.
end = start;
5813 if (
string.match(text, this->interval.
end, end, flags)) {
5815 this->interval.
end =
string.interval.end;
5816 this->interval.
start = start;
5819 else if (
token.match(text, this->interval.end, end, flags)) {
5820 string.invalidate();
5822 this->interval.
start = start;
5838 virtual void invalidate()
5842 parser::invalidate();
5849 virtual bool do_match(
5850 _In_reads_or_z_(end)
const char* text,
5851 _In_
size_t start = 0,
5852 _In_
size_t end = SIZE_MAX,
5853 _In_
int flags = match_default)
5855 stdex_assert(text || start >= end);
5856 this->interval.
end = start;
5857 if (
name.match(text, this->interval.end, end, flags))
5861 while (m_space.match(text, this->interval.end, end, flags))
5863 stdex_assert(text || this->interval.
end >= end);
5864 if (this->interval.
end < end && text[this->interval.end] ==
'=')
5865 this->interval.
end++;
5867 while (m_space.match(text, this->interval.end, end, flags))
5869 if (
value.match(text, this->interval.end, end, flags))
5873 this->interval.
start = start;
5890 virtual bool do_match(
5891 _In_reads_or_z_(end)
const char* text,
5892 _In_
size_t start = 0,
5893 _In_
size_t end = SIZE_MAX,
5894 _In_
int flags = match_default)
5896 stdex_assert(text || start >= end);
5897 if (start + 2 < end &&
5898 text[start] ==
'*' &&
5899 text[start + 1] ==
'/' &&
5900 text[start + 2] ==
'*')
5902 this->interval.
end = (this->interval.
start = start) + 3;
5905 else if (start < end && text[start] ==
'*') {
5906 this->interval.
end = (this->interval.
start = start) + 1;
5922 virtual void invalidate()
5925 subtype.invalidate();
5926 parser::invalidate();
5933 virtual bool do_match(
5934 _In_reads_or_z_(end)
const char* text,
5935 _In_
size_t start = 0,
5936 _In_
size_t end = SIZE_MAX,
5937 _In_
int flags = match_default)
5939 stdex_assert(text || start >= end);
5940 this->interval.
end = start;
5941 if (type.match(text, this->interval.end, end, flags))
5945 while (m_space.match(text, this->interval.end, end, flags))
5947 if (this->interval.
end < end && text[this->interval.end] ==
'/')
5948 this->interval.
end++;
5951 while (m_space.match(text, this->interval.end, end, flags))
5953 if (subtype.match(text, this->interval.end, end, flags))
5957 this->interval.
start = start;
5974 virtual void invalidate()
5977 http_media_range::invalidate();
5980 std::list<http_parameter> params;
5983 virtual bool do_match(
5984 _In_reads_or_z_(end)
const char* text,
5985 _In_
size_t start = 0,
5986 _In_
size_t end = SIZE_MAX,
5987 _In_
int flags = match_default)
5989 stdex_assert(text || start >= end);
5990 if (!http_media_range::do_match(text, start, end, flags))
5994 if (this->interval.
end < end && text[this->interval.end]) {
5995 if (m_space.match(text, this->interval.end, end, flags))
5997 else if (text[this->interval.
end] ==
';') {
5998 this->interval.
end++;
5999 while (m_space.match(text, this->interval.end, end, flags))
6002 if (param.match(text, this->interval.end, end, flags)) {
6003 this->interval.
end = param.interval.end;
6004 params.push_back(std::move(param));
6015 this->interval.
end = params.empty() ? subtype.
interval.
end : params.back().interval.end;
6030 virtual bool do_match(
6031 _In_reads_or_z_(end)
const char* text,
6032 _In_
size_t start = 0,
6033 _In_
size_t end = SIZE_MAX,
6034 _In_
int flags = match_default)
6036 stdex_assert(text || start >= end);
6037 this->interval.
end = start;
6039 if (this->interval.
end < end && text[this->interval.end]) {
6040 if ((
unsigned int)text[this->interval.
end] < 0x20 ||
6041 (
unsigned int)text[this->interval.end] == 0x7f ||
6042 text[this->interval.end] ==
':' ||
6043 text[this->interval.end] ==
'/' ||
6044 stdex::isspace(text[this->interval.end]))
6047 this->interval.
end++;
6053 this->interval.
start = start;
6072 virtual void invalidate()
6075 parser::invalidate();
6081 virtual bool do_match(
6082 _In_reads_or_z_(end)
const char* text,
6083 _In_
size_t start = 0,
6084 _In_
size_t end = SIZE_MAX,
6085 _In_
int flags = match_default)
6087 stdex_assert(text || start >= end);
6089 this->interval.
end = start;
6091 if (this->interval.
end < end && text[this->interval.end]) {
6092 if (
'0' <= text[this->interval.
end] && text[this->interval.end] <=
'9') {
6093 size_t _value =
static_cast<size_t>(value) * 10 +
static_cast<size_t>(text[this->interval.
end] -
'0');
6094 if (_value > UINT16_MAX) {
6099 value = (uint16_t)_value;
6100 this->interval.
end++;
6109 this->interval.
start = start;
6123 virtual bool do_match(
6124 _In_reads_or_z_(end)
const char* text,
6125 _In_
size_t start = 0,
6126 _In_
size_t end = SIZE_MAX,
6127 _In_
int flags = match_default)
6129 stdex_assert(text || start >= end);
6130 this->interval.
end = start;
6132 if (this->interval.
end < end && text[this->interval.end]) {
6133 if ((
unsigned int)text[this->interval.
end] < 0x20 ||
6134 (
unsigned int)text[this->interval.end] == 0x7f ||
6135 text[this->interval.end] ==
'?' ||
6136 text[this->interval.end] ==
'/' ||
6137 stdex::isspace(text[this->interval.end]))
6140 this->interval.
end++;
6145 this->interval.
start = start;
6156 virtual void invalidate()
6159 parser::invalidate();
6165 virtual bool do_match(
6166 _In_reads_or_z_(end)
const char* text,
6167 _In_
size_t start = 0,
6168 _In_
size_t end = SIZE_MAX,
6169 _In_
int flags = match_default)
6171 stdex_assert(text || start >= end);
6173 this->interval.
end = start;
6175 stdex_assert(text || this->interval.
end >= end);
6176 if (this->interval.
end < end && text[this->interval.end] !=
'/')
6178 this->interval.
end++;
6179 s.match(text, this->interval.
end, end, flags);
6181 this->interval.
end = s.interval.end;
6183 if (this->interval.
end < end && text[this->interval.end]) {
6184 if (text[this->interval.
end] ==
'/') {
6185 this->interval.
end++;
6186 s.match(text, this->interval.
end, end, flags);
6188 this->interval.
end = s.interval.end;
6196 this->interval.
start = start;
6211 virtual void invalidate()
6217 parser::invalidate();
6224 virtual bool do_match(
6225 _In_reads_or_z_(end)
const char* text,
6226 _In_
size_t start = 0,
6227 _In_
size_t end = SIZE_MAX,
6228 _In_
int flags = match_default)
6230 stdex_assert(text || start >= end);
6231 this->interval.
end = start;
6234 if (this->interval.
end < end && text[this->interval.end]) {
6235 if ((
unsigned int)text[this->interval.
end] < 0x20 ||
6236 (
unsigned int)text[this->interval.end] == 0x7f ||
6237 text[this->interval.end] ==
'&' ||
6238 text[this->interval.end] ==
'=' ||
6239 stdex::isspace(text[this->interval.end]))
6242 this->interval.
end++;
6248 name.
end = this->interval.
end;
6251 if (text[this->interval.
end] ==
'=') {
6252 this->interval.
end++;
6255 if (this->interval.
end < end && text[this->interval.end]) {
6256 if ((
unsigned int)text[this->interval.
end] < 0x20 ||
6257 (
unsigned int)text[this->interval.end] == 0x7f ||
6258 text[this->interval.end] ==
'&' ||
6259 stdex::isspace(text[this->interval.end]))
6262 this->interval.
end++;
6267 value.
end = this->interval.
end;
6273 this->interval.
start = start;
6293 virtual void invalidate()
6295 server.invalidate();
6299 parser::invalidate();
6305 std::list<http_url_parameter> params;
6308 virtual bool do_match(
6309 _In_reads_or_z_(end)
const char* text,
6310 _In_
size_t start = 0,
6311 _In_
size_t end = SIZE_MAX,
6312 _In_
int flags = match_default)
6314 stdex_assert(text || start >= end);
6315 this->interval.
end = start;
6317 if (this->interval.
end + 7 <= end && stdex::strnicmp(text + this->interval.end, 7,
"http://", SIZE_MAX, m_locale) == 0) {
6318 this->interval.
end += 7;
6319 if (server.match(text, this->interval.end, end, flags))
6323 if (this->interval.
end < end && text[this->interval.end] ==
':') {
6324 this->interval.
end++;
6325 if (port.match(text, this->interval.end, end, flags))
6334 server.invalidate();
6339 if (path.match(text, this->interval.end, end, flags))
6346 if (this->interval.
end < end && text[this->interval.end] ==
'?') {
6347 this->interval.
end++;
6349 if (this->interval.
end < end && text[this->interval.end]) {
6350 if ((
unsigned int)text[this->interval.
end] < 0x20 ||
6351 (
unsigned int)text[this->interval.end] == 0x7f ||
6352 stdex::isspace(text[this->interval.end]))
6354 else if (text[this->interval.
end] ==
'&')
6355 this->interval.
end++;
6358 if (param.match(text, this->interval.end, end, flags)) {
6359 this->interval.
end = param.interval.end;
6360 params.push_back(std::move(param));
6371 this->interval.
start = start;
6386 virtual void invalidate()
6389 parser::invalidate();
6392 std::vector<stdex::interval<size_t>> components;
6395 virtual bool do_match(
6396 _In_reads_or_z_(end)
const char* text,
6397 _In_
size_t start = 0,
6398 _In_
size_t end = SIZE_MAX,
6399 _In_
int flags = match_default)
6401 stdex_assert(text || start >= end);
6402 this->interval.
end = start;
6405 if (this->interval.
end < end && text[this->interval.end]) {
6407 k.
end = this->interval.
end;
6409 if (k.
end < end && text[k.
end]) {
6410 if (stdex::isalpha(text[k.
end]))
6418 if (this->interval.
end < k.
end) {
6420 this->interval.
end = k.
end;
6421 components.push_back(k);
6425 if (this->interval.
end < end && text[this->interval.end] ==
'-')
6426 this->interval.
end++;
6433 if (!components.empty()) {
6434 this->interval.
start = start;
6435 this->interval.
end = components.back().end;
6454 virtual void invalidate()
6457 parser::invalidate();
6463 virtual bool do_match(
6464 _In_reads_or_z_(end)
const char* text,
6465 _In_
size_t start = 0,
6466 _In_
size_t end = SIZE_MAX,
6467 _In_
int flags = match_default)
6469 stdex_assert(text || start >= end);
6470 size_t celi_del = 0, decimalni_del = 0, decimalni_del_n = 1;
6471 this->interval.
end = start;
6473 if (this->interval.
end < end && text[this->interval.end]) {
6474 if (
'0' <= text[this->interval.
end] && text[this->interval.end] <=
'9') {
6475 celi_del = celi_del * 10 +
static_cast<size_t>(text[this->interval.
end] -
'0');
6476 this->interval.
end++;
6478 else if (text[this->interval.
end] ==
'.') {
6479 this->interval.
end++;
6481 if (this->interval.
end < end && text[this->interval.end]) {
6482 if (
'0' <= text[this->interval.
end] && text[this->interval.end] <=
'9') {
6483 decimalni_del = decimalni_del * 10 +
static_cast<size_t>(text[this->interval.
end] -
'0');
6484 decimalni_del_n *= 10;
6485 this->interval.
end++;
6502 value = (float)((
double)celi_del + (
double)decimalni_del / decimalni_del_n);
6503 this->interval.
start = start;
6518 virtual bool do_match(
6519 _In_reads_or_z_(end)
const char* text,
6520 _In_
size_t start = 0,
6521 _In_
size_t end = SIZE_MAX,
6522 _In_
int flags = match_default)
6524 stdex_assert(text || end <= start);
6525 if (start < end && text[start] ==
'*') {
6526 this->interval.
end = (this->interval.
start = start) + 1;
6537 template <
class T,
class T_asterisk = http_asterisk>
6546 virtual void invalidate()
6548 asterisk.invalidate();
6550 factor.invalidate();
6551 parser::invalidate();
6554 T_asterisk asterisk;
6559 virtual bool do_match(
6560 _In_reads_or_z_(end)
const char* text,
6561 _In_
size_t start = 0,
6562 _In_
size_t end = SIZE_MAX,
6563 _In_
int flags = match_default)
6565 stdex_assert(text || start >= end);
6566 size_t konec_vrednosti;
6567 this->interval.
end = start;
6568 if (asterisk.match(text, this->interval.end, end, flags)) {
6569 this->interval.
end = konec_vrednosti = asterisk.interval.end;
6572 else if (value.match(text, this->interval.end, end, flags)) {
6573 this->interval.
end = konec_vrednosti = value.interval.end;
6574 asterisk.invalidate();
6577 asterisk.invalidate();
6583 while (this->interval.
end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])) this->interval.
end++;
6584 if (this->interval.
end < end && text[this->interval.end] ==
';') {
6585 this->interval.
end++;
6586 while (this->interval.
end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])) this->interval.
end++;
6587 if (this->interval.
end < end && (text[this->interval.end] ==
'q' || text[this->interval.end] ==
'Q')) {
6588 this->interval.
end++;
6589 while (this->interval.
end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])) this->interval.
end++;
6590 if (this->interval.
end < end && text[this->interval.end] ==
'=') {
6591 this->interval.
end++;
6592 while (this->interval.
end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])) this->interval.
end++;
6593 if (factor.match(text, this->interval.end, end, flags))
6599 factor.invalidate();
6600 this->interval.
end = konec_vrednosti;
6602 this->interval.
start = start;
6613 virtual void invalidate()
6617 parser::invalidate();
6624 virtual bool do_match(
6625 _In_reads_or_z_(end)
const char* text,
6626 _In_
size_t start = 0,
6627 _In_
size_t end = SIZE_MAX,
6628 _In_
int flags = match_default)
6630 stdex_assert(text || start >= end);
6631 this->interval.
end = start;
6632 if (this->interval.
end < end && text[this->interval.end] ==
'$')
6633 this->interval.
end++;
6636 if (name.match(text, this->interval.end, end, flags))
6640 while (m_space.match(text, this->interval.end, end, flags))
6642 if (this->interval.
end < end && text[this->interval.end] ==
'=')
6643 this->interval.
end++;
6646 while (m_space.match(text, this->interval.end, end, flags))
6648 if (value.match(text, this->interval.end, end, flags))
6652 this->interval.
start = start;
6669 virtual void invalidate()
6674 parser::invalidate();
6682 virtual bool do_match(
6683 _In_reads_or_z_(end)
const char* text,
6684 _In_
size_t start = 0,
6685 _In_
size_t end = SIZE_MAX,
6686 _In_
int flags = match_default)
6688 stdex_assert(text || start >= end);
6689 this->interval.
end = start;
6690 if (
name.match(text, this->interval.end, end, flags))
6694 while (m_space.match(text, this->interval.end, end, flags))
6696 if (this->interval.
end < end && text[this->interval.end] ==
'=')
6697 this->interval.
end++;
6700 while (m_space.match(text, this->interval.end, end, flags))
6702 if (
value.match(text, this->interval.end, end, flags))
6708 if (this->interval.
end < end && text[this->interval.end]) {
6709 if (m_space.match(text, this->interval.end, end, flags))
6711 else if (text[this->interval.
end] ==
';') {
6712 this->interval.
end++;
6713 while (m_space.match(text, this->interval.end, end, flags))
6716 if (param.match(text, this->interval.end, end, flags)) {
6717 this->interval.
end = param.interval.end;
6718 params.push_back(std::move(param));
6729 this->interval.
start = start;
6747 virtual void invalidate()
6753 parser::invalidate();
6760 virtual bool do_match(
6761 _In_reads_or_z_(end)
const char* text,
6762 _In_
size_t start = 0,
6763 _In_
size_t end = SIZE_MAX,
6764 _In_
int flags = match_default)
6766 stdex_assert(text || start >= end);
6767 this->interval.
end = start;
6770 if (this->interval.
end < end && text[this->interval.end]) {
6771 if (text[this->interval.
end] ==
'/') {
6772 type.
end = this->interval.
end;
6773 this->interval.
end++;
6774 version.
start = this->interval.
end;
6776 if (this->interval.
end < end && text[this->interval.end]) {
6777 if (stdex::isspace(text[this->interval.
end])) {
6778 version.
end = this->interval.
end;
6782 this->interval.
end++;
6785 version.
end = this->interval.
end;
6791 else if (stdex::isspace(text[this->interval.
end])) {
6792 type.
end = this->interval.
end;
6796 this->interval.
end++;
6799 type.
end = this->interval.
end;
6804 this->interval.
start = start;
6827 virtual void invalidate()
6831 version_maj.
start = 1;
6832 version_maj.
end = 0;
6833 version_min.
start = 1;
6834 version_min.
end = 0;
6836 parser::invalidate();
6845 virtual bool do_match(
6846 _In_reads_or_z_(end)
const char* text,
6847 _In_
size_t start = 0,
6848 _In_
size_t end = SIZE_MAX,
6849 _In_
int flags = match_default)
6851 stdex_assert(text || start >= end);
6852 this->interval.
end = start;
6855 if (this->interval.
end < end && text[this->interval.end]) {
6856 if (text[this->interval.
end] ==
'/') {
6857 type.
end = this->interval.
end;
6858 this->interval.
end++;
6861 else if (stdex::isspace(text[this->interval.
end]))
6864 this->interval.
end++;
6867 type.
end = this->interval.
end;
6871 version_maj.
start = this->interval.
end;
6873 if (this->interval.
end < end && text[this->interval.end]) {
6874 if (text[this->interval.
end] ==
'.') {
6875 version_maj.
end = this->interval.
end;
6876 this->interval.
end++;
6877 version_min.
start = this->interval.
end;
6879 if (this->interval.
end < end && text[this->interval.end]) {
6880 if (stdex::isspace(text[this->interval.
end])) {
6881 version_min.
end = this->interval.
end;
6883 (uint16_t)strtoui(text + version_maj.
start, version_maj.
size(),
nullptr, 10) * 0x100 +
6884 (uint16_t)strtoui(text + version_min.
start, version_min.
size(),
nullptr, 10);
6888 this->interval.
end++;
6895 else if (stdex::isspace(text[this->interval.
end])) {
6896 version_maj.
end = this->interval.
end;
6897 version_min.
start = 1;
6898 version_min.
end = 0;
6899 version = (uint16_t)strtoui(text + version_maj.
start, version_maj.
size(),
nullptr, 10) * 0x100;
6903 this->interval.
end++;
6908 this->interval.
start = start;
6929 virtual void invalidate()
6934 protocol.invalidate();
6935 parser::invalidate();
6943 virtual bool do_match(
6944 _In_reads_or_z_(end)
const char* text,
6945 _In_
size_t start = 0,
6946 _In_
size_t end = SIZE_MAX,
6947 _In_
int flags = match_default)
6949 stdex_assert(text || start >= end);
6950 this->interval.
end = start;
6953 if (m_line_break.match(text, this->interval.end, end, flags))
6955 else if (this->interval.
end < end && text[this->interval.end]) {
6956 if (stdex::isspace(text[this->interval.
end]))
6957 this->interval.
end++;
6966 if (m_line_break.match(text, this->interval.end, end, flags))
6968 else if (this->interval.
end < end && text[this->interval.end]) {
6969 if (stdex::isspace(text[this->interval.
end])) {
6970 verb.
end = this->interval.
end;
6971 this->interval.
end++;
6975 this->interval.
end++;
6982 if (m_line_break.match(text, this->interval.end, end, flags))
6984 else if (this->interval.
end < end && text[this->interval.end]) {
6985 if (stdex::isspace(text[this->interval.
end]))
6986 this->interval.
end++;
6993 if (
url.match(text, this->interval.end, end, flags))
6998 protocol.invalidate();
7000 if (m_line_break.match(text, this->interval.end, end, flags)) {
7004 else if (this->interval.
end < end && text[this->interval.end]) {
7005 if (stdex::isspace(text[this->interval.
end]))
7006 this->interval.
end++;
7014 if (m_line_break.match(text, this->interval.end, end, flags)) {
7018 else if (protocol.match(text, this->interval.end, end, flags)) {
7027 if (m_line_break.match(text, this->interval.end, end, flags)) {
7031 else if (this->interval.
end < end && text[this->interval.end])
7032 this->interval.
end++;
7038 this->interval.
start = start;
7055 virtual void invalidate()
7061 parser::invalidate();
7068 virtual bool do_match(
7069 _In_reads_or_z_(end)
const char* text,
7070 _In_
size_t start = 0,
7071 _In_
size_t end = SIZE_MAX,
7072 _In_
int flags = match_default)
7074 stdex_assert(text || start >= end);
7075 this->interval.
end = start;
7077 if (m_line_break.match(text, this->interval.end, end, flags) ||
7078 (this->interval.end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end])))
7082 if (m_line_break.match(text, this->interval.end, end, flags))
7084 else if (this->interval.
end < end && text[this->interval.end]) {
7085 if (stdex::isspace(text[this->interval.
end])) {
7086 name.
end = this->interval.
end;
7087 this->interval.
end++;
7089 if (m_line_break.match(text, this->interval.end, end, flags))
7091 else if (this->interval.
end < end && text[this->interval.end]) {
7092 if (stdex::isspace(text[this->interval.
end]))
7093 this->interval.
end++;
7100 if (this->interval.
end < end && text[this->interval.end] ==
':') {
7101 this->interval.
end++;
7108 else if (text[this->interval.
end] ==
':') {
7109 name.
end = this->interval.
end;
7110 this->interval.
end++;
7114 this->interval.
end++;
7119 value.
start = SIZE_MAX;
7122 if (m_line_break.match(text, this->interval.end, end, flags)) {
7124 if (!m_line_break.match(text, this->interval.end, end, flags) &&
7125 this->interval.end < end && text[this->interval.end] && stdex::isspace(text[this->interval.end]))
7126 this->interval.
end++;
7130 else if (this->interval.
end < end && text[this->interval.end]) {
7131 if (stdex::isspace(text[this->interval.
end]))
7132 this->interval.
end++;
7134 if (value.
start == SIZE_MAX) value.
start = this->interval.
end;
7135 value.
end = ++this->interval.
end;
7141 this->interval.
start = start;
7155 template <
class KEY,
class T>
7160 _In_reads_or_z_(end)
const char* text,
7161 _In_
size_t start = 0,
7162 _In_
size_t end = SIZE_MAX,
7163 _In_
int flags = match_default)
7165 while (start < end) {
7166 while (start < end && text[start] && stdex::isspace(text[start])) start++;
7167 if (start < end && text[start] ==
',') {
7169 while (start < end&& text[start] && stdex::isspace(text[start])) start++;
7172 if (el.match(text, start, end, flags)) {
7173 start = el.interval.end;
7174 T::insert(std::move(el));
7184 constexpr bool operator()(
const T& a,
const T& b)
const noexcept
7186 return a.factor.value > b.factor.value;
7193 template <
class T,
class AX = std::allocator<T>>
7215 _In_
const std::locale&
locale = std::locale()) :
7230 virtual void invalidate()
7236 std::basic_string<T> value;
7239 virtual bool do_match(
7240 _In_reads_or_z_opt_(end)
const T* text,
7241 _In_
size_t start = 0,
7242 _In_
size_t end = SIZE_MAX,
7243 _In_
int flags = match_default)
7245 stdex_assert(text || start >= end);
7246 this->interval.
end = start;
7247 if (m_quote->match(text, this->interval.end, end, flags)) {
7248 this->interval.
end = m_quote->interval.end;
7251 if (m_quote->match(text, this->interval.end, end, flags)) {
7252 this->interval.
start = start;
7253 this->interval.
end = m_quote->interval.end;
7256 if (m_escape->match(text, this->interval.end, end, flags)) {
7257 if (m_quote->match(text, m_escape->interval.end, end, flags)) {
7258 value +=
'"'; this->interval.
end = m_quote->interval.end;
7261 if (m_sol->match(text, m_escape->interval.end, end, flags)) {
7262 value +=
'/'; this->interval.
end = m_sol->interval.end;
7265 if (m_bs->match(text, m_escape->interval.end, end, flags)) {
7266 value +=
'\b'; this->interval.
end = m_bs->interval.end;
7269 if (m_ff->match(text, m_escape->interval.end, end, flags)) {
7270 value +=
'\f'; this->interval.
end = m_ff->interval.end;
7273 if (m_lf->match(text, m_escape->interval.end, end, flags)) {
7274 value +=
'\n'; this->interval.
end = m_lf->interval.end;
7277 if (m_cr->match(text, m_escape->interval.end, end, flags)) {
7278 value +=
'\r'; this->interval.
end = m_cr->interval.end;
7281 if (m_htab->match(text, m_escape->interval.end, end, flags)) {
7282 value +=
'\t'; this->interval.
end = m_htab->interval.end;
7286 m_uni->match(text, m_escape->interval.end, end, flags) &&
7287 m_hex->match(text, m_uni->interval.end, std::min(m_uni->interval.end + 4, end), flags | match_case_insensitive) &&
7288 m_hex->interval.size() == 4 )
7290 stdex_assert(m_hex->value <= 0xffff);
7291 if (
sizeof(T) == 1) {
7292 if (m_hex->value > 0x7ff) {
7293 value += (T)(0xe0 | ((m_hex->value >> 12) & 0x0f));
7294 value += (T)(0x80 | ((m_hex->value >> 6) & 0x3f));
7295 value += (T)(0x80 | (m_hex->value & 0x3f));
7297 else if (m_hex->value > 0x7f) {
7298 value += (T)(0xc0 | ((m_hex->value >> 6) & 0x1f));
7299 value += (T)(0x80 | (m_hex->value & 0x3f));
7302 value += (T)(m_hex->value & 0x7f);
7305 value += (T)m_hex->value;
7306 this->interval.
end = m_hex->interval.end;
7309 if (m_escape->match(text, m_escape->interval.end, end, flags)) {
7310 value +=
'\\'; this->interval.
end = m_escape->interval.end;
7314 if (m_chr->match(text, this->interval.end, end, flags)) {
7315 value.append(text + m_chr->interval.start, m_chr->interval.size());
7316 this->interval.
end = m_chr->interval.end;
7327 std::shared_ptr<basic_parser<T>> m_quote;
7328 std::shared_ptr<basic_parser<T>> m_chr;
7329 std::shared_ptr<basic_parser<T>> m_escape;
7330 std::shared_ptr<basic_parser<T>> m_sol;
7331 std::shared_ptr<basic_parser<T>> m_bs;
7332 std::shared_ptr<basic_parser<T>> m_ff;
7333 std::shared_ptr<basic_parser<T>> m_lf;
7334 std::shared_ptr<basic_parser<T>> m_cr;
7335 std::shared_ptr<basic_parser<T>> m_htab;
7336 std::shared_ptr<basic_parser<T>> m_uni;
7337 std::shared_ptr<basic_integer16<T>> m_hex;
7355 virtual void invalidate()
7364 virtual bool do_match(
7365 _In_reads_or_z_opt_(end)
const T* text,
7366 _In_
size_t start = 0,
7367 _In_
size_t end = SIZE_MAX,
7368 _In_
int flags = match_multiline)
7370 _Unreferenced_(flags);
7371 stdex_assert(text || start + 1 >= end);
7372 if (start + 1 < end &&
7373 text[start] ==
'/' &&
7374 text[start + 1] ==
'*')
7377 this->content.
start = this->interval.
end = start + 2;
7379 if (this->interval.
end >= end || !text[this->interval.end])
7381 if (this->interval.
end + 1 < end &&
7382 text[this->interval.end] ==
'*' &&
7383 text[this->interval.end + 1] ==
'/')
7386 this->content.
end = this->interval.
end;
7387 this->interval.
start = start;
7388 this->interval.
end = this->interval.
end + 2;
7391 this->interval.
end++;
7400 using css_comment = basic_css_comment<char>;
7401 using wcss_comment = basic_css_comment<wchar_t>;
7403 using tcss_comment = wcss_comment;
7405 using tcss_comment = css_comment;
7415 virtual bool do_match(
7416 _In_reads_or_z_opt_(end)
const T* text,
7417 _In_
size_t start = 0,
7418 _In_
size_t end = SIZE_MAX,
7419 _In_
int flags = match_multiline)
7421 _Unreferenced_(flags);
7422 stdex_assert(text || start + 3 >= end);
7423 if (start + 3 < end &&
7424 text[start] ==
'<' &&
7425 text[start + 1] ==
'!' &&
7426 text[start + 2] ==
'-' &&
7427 text[start + 3] ==
'-')
7429 this->interval.
start = start;
7430 this->interval.
end = start + 4;
7453 virtual bool do_match(
7454 _In_reads_or_z_opt_(end)
const T* text,
7455 _In_
size_t start = 0,
7456 _In_
size_t end = SIZE_MAX,
7457 _In_
int flags = match_multiline)
7459 _Unreferenced_(flags);
7460 stdex_assert(text || start + 2 >= end);
7461 if (start + 2 < end &&
7462 text[start] ==
'-' &&
7463 text[start + 1] ==
'-' &&
7464 text[start + 2] ==
'>')
7466 this->interval.
start = start;
7467 this->interval.
end = start + 3;
7490 virtual void invalidate()
7499 virtual bool do_match(
7500 _In_reads_or_z_opt_(end)
const T* text,
7501 _In_
size_t start = 0,
7502 _In_
size_t end = SIZE_MAX,
7503 _In_
int flags = match_multiline)
7505 _Unreferenced_(flags);
7506 this->interval.
end = start;
7507 stdex_assert(text || this->interval.
end >= end);
7508 if (this->interval.
end < end &&
7509 (text[this->interval.end] ==
'\"' || text[this->interval.end] ==
'\''))
7512 T quote = text[this->interval.
end];
7513 this->content.
start = ++this->interval.
end;
7515 if (this->interval.
end >= end || !text[this->interval.end])
7517 if (text[this->interval.
end] == quote) {
7519 this->content.
end = this->interval.
end;
7520 this->interval.
start = start;
7521 this->interval.
end++;
7524 if (this->interval.
end + 1 < end &&
7525 text[this->interval.end] ==
'\\' &&
7526 (text[this->interval.end + 1] ==
'\"' || text[this->interval.end + 1] ==
'\''))
7529 this->interval.
end = this->interval.
end + 2;
7532 this->interval.
end++;
7542 using css_string = basic_css_string<char>;
7543 using wcss_string = basic_css_string<wchar_t>;
7545 using tcss_string = wcss_string;
7547 using tcss_string = css_string;
7557 virtual void invalidate()
7566 virtual bool do_match(
7567 _In_reads_or_z_opt_(end)
const T* text,
7568 _In_
size_t start = 0,
7569 _In_
size_t end = SIZE_MAX,
7570 _In_
int flags = match_multiline)
7572 _Unreferenced_(flags);
7573 this->interval.
end = start;
7574 stdex_assert(text || this->interval.
end + 3 >= end);
7575 if (this->interval.
end + 3 < end &&
7576 (text[this->interval.end] ==
'u' || text[this->interval.end] ==
'U') &&
7577 (text[this->interval.end + 1] ==
'r' || text[this->interval.end + 1] ==
'R') &&
7578 (text[this->interval.end + 2] ==
'l' || text[this->interval.end + 2] ==
'L') &&
7579 text[this->interval.end + 3] ==
'(')
7582 this->interval.
end = this->interval.
end + 4;
7585 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
7586 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
7588 if (this->interval.
end < end &&
7589 (text[this->interval.end] ==
'\"' || text[this->interval.end] ==
'\''))
7592 T quote = text[this->interval.
end];
7593 this->content.
start = ++this->interval.
end;
7595 if (this->interval.
end >= end || !text[this->interval.end])
7597 if (text[this->interval.
end] == quote) {
7599 this->content.
end = this->interval.
end;
7600 this->interval.
end++;
7603 if (this->interval.
end + 1 < end &&
7604 text[this->interval.end] ==
'\\' &&
7605 (text[this->interval.end + 1] ==
'\"' || text[this->interval.end + 1] ==
'\''))
7608 this->interval.
end = this->interval.
end + 2;
7611 this->interval.
end++;
7615 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
7617 if (this->interval.
end < end &&
7618 text[this->interval.end] ==
')')
7621 this->interval.
start = start;
7622 this->interval.
end++;
7630 if (this->interval.
end >= end || !text[this->interval.end])
7632 if (text[this->interval.
end] ==
')') {
7634 this->interval.
start = start;
7635 this->interval.
end++;
7638 if (ctype.is(ctype.space, text[this->interval.end]))
7639 this->interval.
end++;
7641 this->content.
end = ++this->interval.
end;
7652 using css_uri = basic_css_uri<char>;
7653 using wcss_uri = basic_css_uri<wchar_t>;
7655 using tcss_uri = wcss_uri;
7657 using tcss_uri = css_uri;
7667 virtual void invalidate()
7676 virtual bool do_match(
7677 _In_reads_or_z_opt_(end)
const T* text,
7678 _In_
size_t start = 0,
7679 _In_
size_t end = SIZE_MAX,
7680 _In_
int flags = match_multiline)
7682 _Unreferenced_(flags);
7683 this->interval.
end = start;
7684 stdex_assert(text || this->interval.
end + 6 >= end);
7685 if (this->interval.
end + 6 < end &&
7686 text[this->interval.end] ==
'@' &&
7687 (text[this->interval.end + 1] ==
'i' || text[this->interval.end + 1] ==
'I') &&
7688 (text[this->interval.end + 2] ==
'm' || text[this->interval.end + 2] ==
'M') &&
7689 (text[this->interval.end + 3] ==
'p' || text[this->interval.end + 3] ==
'P') &&
7690 (text[this->interval.end + 4] ==
'o' || text[this->interval.end + 4] ==
'O') &&
7691 (text[this->interval.end + 5] ==
'r' || text[this->interval.end + 5] ==
'R') &&
7692 (text[this->interval.end + 6] ==
't' || text[this->interval.end + 6] ==
'T'))
7695 this->interval.
end = this->interval.
end + 7;
7698 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
7699 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
7701 if (this->interval.
end < end &&
7702 (text[this->interval.end] ==
'\"' || text[this->interval.end] ==
'\''))
7705 T quote = text[this->interval.
end];
7706 this->content.
start = ++this->interval.
end;
7708 if (this->interval.
end >= end || !text[this->interval.end])
7710 if (text[this->interval.
end] == quote) {
7712 this->content.
end = this->interval.
end;
7713 this->interval.
start = start;
7714 this->interval.
end++;
7717 if (this->interval.
end + 1 < end &&
7718 text[this->interval.end] ==
'\\' &&
7719 (text[this->interval.end + 1] ==
'\"' || text[this->interval.end + 1] ==
'\''))
7722 this->interval.
end = this->interval.
end + 2;
7725 this->interval.
end++;
7736 using css_import = basic_css_import<char>;
7737 using wcss_import = basic_css_import<wchar_t>;
7739 using tcss_import = wcss_import;
7741 using tcss_import = css_import;
7751 virtual void invalidate()
7764 virtual bool do_match(
7765 _In_reads_or_z_opt_(end)
const T* text,
7766 _In_
size_t start = 0,
7767 _In_
size_t end = SIZE_MAX,
7768 _In_
int flags = match_multiline)
7770 _Unreferenced_(flags);
7771 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
7773 this->interval.
end = start;
7774 this->base_type.
start = this->interval.
end;
7776 stdex_assert(text || this->interval.
end >= end);
7777 if (this->interval.
end >= end || !text[this->interval.end])
7779 if (text[this->interval.
end] ==
'/' ||
7780 text[this->interval.end] ==
';' ||
7781 ctype.is(ctype.space, text[this->interval.end]))
7783 this->interval.
end++;
7785 if (this->interval.
end <= this->base_type.start)
7787 this->base_type.
end = this->interval.
end;
7789 if (end <= this->
interval.
end || text[this->interval.end] !=
'/')
7792 this->interval.
end++;
7793 this->sub_type.
start = this->interval.
end;
7795 if (this->interval.
end >= end || !text[this->interval.end])
7797 if (text[this->interval.
end] ==
'/' ||
7798 text[this->interval.end] ==
';' ||
7799 ctype.is(ctype.space, text[this->interval.end]))
7801 this->interval.
end++;
7803 if (this->interval.
end <= this->sub_type.start)
7806 this->sub_type.
end = this->interval.
end;
7808 if (this->interval.
end < end && text[this->interval.end] ==
';') {
7809 this->interval.
end++;
7812 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
7814 if (this->interval.
end + 7 < end &&
7815 (text[this->interval.end] ==
'c' || text[this->interval.end] ==
'C') &&
7816 (text[this->interval.end + 1] ==
'h' || text[this->interval.end + 1] ==
'H') &&
7817 (text[this->interval.end + 2] ==
'a' || text[this->interval.end + 2] ==
'A') &&
7818 (text[this->interval.end + 3] ==
'r' || text[this->interval.end + 3] ==
'R') &&
7819 (text[this->interval.end + 4] ==
's' || text[this->interval.end + 4] ==
'S') &&
7820 (text[this->interval.end + 5] ==
'e' || text[this->interval.end + 5] ==
'E') &&
7821 (text[this->interval.end + 6] ==
't' || text[this->interval.end + 6] ==
'T') &&
7822 text[this->interval.end + 7] ==
'=')
7824 this->interval.
end = this->interval.
end + 8;
7825 if (this->interval.
end < end &&
7826 (text[this->interval.end] ==
'\"' || text[this->interval.end] ==
'\''))
7829 T quote = text[this->interval.
end];
7830 this->charset.
start = ++this->interval.
end;
7832 if (this->interval.
end >= end || !text[this->interval.end]) {
7837 if (text[this->interval.
end] == quote) {
7839 this->charset.
end = this->interval.
end;
7840 this->interval.
end++;
7843 this->interval.
end++;
7848 this->charset.
start = this->interval.
end;
7850 if (this->interval.
end >= end || !text[this->interval.end] ||
7851 ctype.is(ctype.space, text[this->interval.end])) {
7852 this->charset.
end = this->interval.
end;
7855 this->interval.
end++;
7860 this->interval.
start = start;
7869 using mime_type = basic_mime_type<char>;
7870 using wmime_type = basic_mime_type<wchar_t>;
7872 using tmime_type = wmime_type;
7874 using tmime_type = mime_type;
7884 virtual bool do_match(
7885 _In_reads_or_z_opt_(end)
const T* text,
7886 _In_
size_t start = 0,
7887 _In_
size_t end = SIZE_MAX,
7888 _In_
int flags = match_default)
7890 _Unreferenced_(flags);
7891 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
7892 this->interval.
end = start;
7894 stdex_assert(text || this->interval.
end >= end);
7895 if (this->interval.
end >= end || !text[this->interval.end]) {
7897 this->interval.
start = start;
7903 if (text[this->interval.
end] ==
'>' ||
7904 text[this->interval.end] ==
'=' ||
7905 (text[this->interval.end] ==
'/' && this->interval.end + 1 < end && text[this->interval.end + 1] ==
'>') ||
7906 ctype.is(ctype.space, text[this->interval.end]))
7908 this->interval.
start = start;
7911 this->interval.
end++;
7931 virtual void invalidate()
7940 virtual bool do_match(
7941 _In_reads_or_z_opt_(end)
const T* text,
7942 _In_
size_t start = 0,
7943 _In_
size_t end = SIZE_MAX,
7944 _In_
int flags = match_default)
7946 _Unreferenced_(flags);
7947 this->interval.
end = start;
7948 stdex_assert(text || this->interval.
end >= end);
7949 if (this->interval.
end < end &&
7950 (text[this->interval.end] ==
'\"' || text[this->interval.end] ==
'\''))
7953 T quote = text[this->interval.
end];
7954 this->content.
start = ++this->interval.
end;
7956 if (this->interval.
end >= end || !text[this->interval.end]) {
7962 if (text[this->interval.
end] == quote) {
7964 this->content.
end = this->interval.
end;
7965 this->interval.
start = start;
7966 this->interval.
end++;
7969 this->interval.
end++;
7974 this->content.
start = this->interval.
end;
7975 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
7977 stdex_assert(text || this->interval.
end >= end);
7978 if (this->interval.
end >= end || !text[this->interval.end]) {
7979 this->content.
end = this->interval.
end;
7980 this->interval.
start = start;
7983 if (text[this->interval.
end] ==
'>' ||
7984 (text[this->interval.end] ==
'/' && this->interval.end + 1 < end && text[this->interval.end + 1] ==
'>') ||
7985 ctype.is(ctype.space, text[this->interval.end]))
7987 this->content.
end = this->interval.
end;
7988 this->interval.
start = start;
7991 this->interval.
end++;
7996 using html_value = basic_html_value<char>;
7997 using whtml_value = basic_html_value<wchar_t>;
7999 using thtml_value = whtml_value;
8001 using thtml_value = html_value;
8007 enum class html_sequence_t {
8038 type(html_sequence_t::unknown)
8041 virtual void invalidate()
8043 this->type = html_sequence_t::unknown;
8044 this->name.invalidate();
8054 virtual bool do_match(
8055 _In_reads_or_z_opt_(end)
const T* text,
8056 _In_
size_t start = 0,
8057 _In_
size_t end = SIZE_MAX,
8058 _In_
int flags = match_multiline)
8060 stdex_assert(text || start >= end);
8061 if (start >= end || text[start] !=
'<')
8063 this->interval.
end = start + 1;
8064 if (this->interval.
end >= end || !text[this->interval.end])
8066 if (text[this->interval.
end] ==
'/' &&
8067 this->m_ident.match(text, this->interval.end + 1, end, flags))
8070 this->type = html_sequence_t::element_end;
8071 this->name = this->m_ident.interval;
8072 this->interval.
end = this->m_ident.interval.end;
8074 else if (text[this->interval.
end] ==
'!') {
8076 this->interval.
end++;
8077 if (this->interval.
end + 1 < end &&
8078 text[this->interval.end] ==
'-' &&
8079 text[this->interval.end + 1] ==
'-')
8082 this->name.
start = this->interval.
end = this->interval.
end + 2;
8084 if (this->interval.
end >= end || !text[this->interval.end])
8086 if (this->interval.
end + 2 < end &&
8087 text[this->interval.end] ==
'-' &&
8088 text[this->interval.end + 1] ==
'-' &&
8089 text[this->interval.end + 2] ==
'>')
8092 this->type = html_sequence_t::comment;
8093 this->name.
end = this->interval.
end;
8094 this->attributes.clear();
8095 this->interval.
start = start;
8096 this->interval.
end = this->interval.
end + 3;
8099 this->interval.
end++;
8102 this->type = html_sequence_t::declaration;
8103 this->name.
start = this->name.
end = this->interval.
end;
8105 else if (text[this->interval.
end] ==
'?') {
8107 this->name.
start = ++this->interval.
end;
8109 if (this->interval.
end >= end || !text[this->interval.end])
8111 if (text[this->interval.
end] ==
'>') {
8113 this->type = html_sequence_t::instruction;
8114 this->name.
end = this->interval.
end;
8115 this->attributes.clear();
8116 this->interval.
start = start;
8117 this->interval.
end++;
8120 if (this->interval.
end + 1 < end &&
8121 text[this->interval.end] ==
'?' &&
8122 text[this->interval.end + 1] ==
'>')
8125 this->type = html_sequence_t::instruction;
8126 this->name.
end = this->interval.
end;
8127 this->attributes.clear();
8128 this->interval.
start = start;
8129 this->interval.
end = this->interval.
end + 2;
8132 this->interval.
end++;
8135 else if (this->m_ident.match(text, this->interval.end, end, flags)) {
8137 this->type = html_sequence_t::element_start;
8138 this->name = this->m_ident.interval;
8139 this->interval.
end = this->m_ident.interval.end;
8146 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
8147 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
8149 this->attributes.clear();
8151 if (this->type == html_sequence_t::element_start &&
8152 this->interval.
end + 1 < end &&
8153 text[this->interval.end] ==
'/' &&
8154 text[this->interval.end + 1] ==
'>')
8157 this->type = html_sequence_t::element;
8158 this->interval.
end = this->interval.
end + 2;
8161 if (this->interval.
end < end &&
8162 text[this->interval.end] ==
'>')
8165 this->interval.
end++;
8168 if (this->type == html_sequence_t::declaration &&
8169 this->interval.
end + 1 < end &&
8170 text[this->interval.end] ==
'!' &&
8171 text[this->interval.end + 1] ==
'>')
8174 this->interval.
end = this->interval.
end + 2;
8177 if (this->type == html_sequence_t::declaration &&
8178 this->interval.
end + 1 < end &&
8179 text[this->interval.end] ==
'-' &&
8180 text[this->interval.end + 1] ==
'-')
8183 this->interval.
end = this->interval.
end + 2;
8185 if (this->interval.
end >= end || !text[this->interval.end])
8187 if (this->interval.
end + 1 < end &&
8188 text[this->interval.end] ==
'-' &&
8189 text[this->interval.end + 1] ==
'-')
8192 this->interval.
end = this->interval.
end + 2;
8195 this->interval.
end++;
8199 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
8203 if (this->interval.
end >= end || !text[this->interval.end])
8208 if (this->m_ident.match(text, this->interval.end, end, flags)) {
8209 this->attributes.push_back(std::move(
html_attribute{ this->m_ident.interval }));
8210 a = &this->attributes.back();
8212 this->interval.
end = this->m_ident.interval.end;
8216 this->interval.
end++;
8221 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
8223 if (this->interval.
end < end && text[this->interval.end] ==
'=') {
8224 this->interval.
end++;
8227 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
8229 if (this->m_value.match(text, this->interval.end, end, flags)) {
8231 a->value = this->m_value.content;
8232 this->interval.
end = this->m_value.interval.end;
8235 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
8240 a->value.invalidate();
8245 this->interval.
start = start;
8257 using html_tag = basic_html_tag<char>;
8258 using whtml_tag = basic_html_tag<wchar_t>;
8260 using thtml_tag = whtml_tag;
8262 using thtml_tag = html_tag;
8272 virtual void invalidate()
8282 _In_reads_or_z_opt_(end)
const T* text,
8283 _In_
size_t start = 0,
8284 _In_
size_t end = SIZE_MAX,
8285 _In_
int flags = match_multiline)
8287 _Unreferenced_(flags);
8288 stdex_assert(text || start + 2 >= end);
8289 if (start + 2 < end &&
8290 text[start] ==
'<' &&
8291 text[start + 1] ==
'!' &&
8292 text[start + 2] ==
'[')
8294 this->interval.
end = start + 3;
8297 const auto& ctype = std::use_facet<std::ctype<T>>(this->m_locale);
8298 for (; this->interval.
end < end && text[this->interval.
end] && ctype.is(ctype.space, text[this->interval.end]); this->interval.end++);
8300 this->condition.
start = this->condition.
end = this->interval.
end;
8303 if (this->interval.
end >= end || !text[this->interval.end])
8305 if (text[this->interval.
end] ==
'[') {
8306 this->interval.
start = start;
8307 this->interval.
end++;
8310 if (ctype.is(ctype.space, text[this->interval.end]))
8311 this->interval.
end++;
8313 this->condition.
end = ++this->interval.
end;
8323 using html_declaration_condition_start = basic_html_declaration_condition_start<char>;
8324 using whtml_declaration_condition_start = basic_html_declaration_condition_start<wchar_t>;
8326 using thtml_declaration_condition_start = whtml_declaration_condition_start;
8328 using thtml_declaration_condition_start = html_declaration_condition_start;
8338 virtual bool do_match(
8339 _In_reads_or_z_opt_(end)
const T* text,
8340 _In_
size_t start = 0,
8341 _In_
size_t end = SIZE_MAX,
8342 _In_
int flags = match_multiline)
8344 _Unreferenced_(flags);
8345 stdex_assert(text || start + 2 >= end);
8346 if (start + 2 < end &&
8347 text[start] ==
']' &&
8348 text[start + 1] ==
']' &&
8349 text[start + 2] ==
'>')
8351 this->interval.
start = start;
8352 this->interval.
end = start + 3;
8370#undef ENUM_FLAG_OPERATOR
8373#if defined(_MSC_VER)
8375#elif defined(__GNUC__)
8376#pragma GCC diagnostic pop
locale_t helper class to free_locale when going out of scope.
Definition locale.hpp:74
Test for angle in d°mm'ss.dddd form.
Definition parser.hpp:4361
Test for any code unit.
Definition parser.hpp:216
Test for beginning of line.
Definition parser.hpp:615
Test for any.
Definition parser.hpp:1058
Test for Creditor Reference.
Definition parser.hpp:4925
T reference[22]
Normalized national reference number.
Definition parser.hpp:4947
T check_digits[3]
Two check digits.
Definition parser.hpp:4946
bool is_valid
Is reference valid per ISO 7064.
Definition parser.hpp:4948
Legacy CSS comment end -->
Definition parser.hpp:7451
Legacy CSS comment start <!--
Definition parser.hpp:7413
CSS import directive.
Definition parser.hpp:7665
stdex::interval< size_t > content
content position in source
Definition parser.hpp:7673
CSS string.
Definition parser.hpp:7488
stdex::interval< size_t > content
content position in source
Definition parser.hpp:7496
URI in CSS.
Definition parser.hpp:7555
stdex::interval< size_t > content
content position in source
Definition parser.hpp:7563
Test for any code unit from a given string of code units.
Definition parser.hpp:720
Test for specific code unit.
Definition parser.hpp:288
Test for date.
Definition parser.hpp:3994
Test for valid DNS domain character.
Definition parser.hpp:2776
bool allow_on_edge
Is character allowed at the beginning or an end of a DNS domain?
Definition parser.hpp:2786
Test for DNS domain/hostname.
Definition parser.hpp:2876
bool m_allow_absolute
May DNS names end with a dot (absolute name)?
Definition parser.hpp:2940
Test for e-mail address.
Definition parser.hpp:3768
Test for emoticon.
Definition parser.hpp:3871
std::shared_ptr< basic_parser< T > > apex
apex/eyebrows/halo (e.g. O, 0)
Definition parser.hpp:3899
std::shared_ptr< basic_parser< T > > eyes
eyes (e.g. :, ;, >, |, B)
Definition parser.hpp:3900
std::shared_ptr< basic_set< T > > mouth
mouth (e.g. ), ), (, (, |, P, D, p, d)
Definition parser.hpp:3902
std::shared_ptr< basic_parser< T > > nose
nose (e.g. -, o)
Definition parser.hpp:3901
std::shared_ptr< basic_parser< T > > emoticon
emoticon as a whole (e.g. 😀, 🤔, 😶)
Definition parser.hpp:3898
Test for end of line.
Definition parser.hpp:654
Test for fraction.
Definition parser.hpp:1686
End of condition ...]]>
Definition parser.hpp:8336
Start of condition <![condition[...
Definition parser.hpp:8270
virtual bool do_match(_In_reads_or_z_opt_(end) const T *text, size_t start=0, size_t end=SIZE_MAX, int flags=match_multiline)
condition position in source
Definition parser.hpp:8281
Contiguous sequence of characters representing name of element, attribute etc.
Definition parser.hpp:7882
Tag.
Definition parser.hpp:8034
std::vector< html_attribute > attributes
tag attributes
Definition parser.hpp:8051
html_sequence_t type
tag type
Definition parser.hpp:8049
stdex::interval< size_t > name
tag name position in source
Definition parser.hpp:8050
Optionally-quoted string representing value of an attribute.
Definition parser.hpp:7929
stdex::interval< size_t > content
content position in source
Definition parser.hpp:7937
Test for International Bank Account Number.
Definition parser.hpp:4636
T bban[31]
Normalized Basic Bank Account Number.
Definition parser.hpp:4661
T country[3]
ISO 3166-1 alpha-2 country code.
Definition parser.hpp:4659
T check_digits[3]
Two check digits.
Definition parser.hpp:4660
bool is_valid
Is IBAN valid per ISO 7064.
Definition parser.hpp:4662
Test for decimal integer.
Definition parser.hpp:1296
Test for decimal integer possibly containing thousand separators.
Definition parser.hpp:1381
bool has_separators
Did integer have any separators?
Definition parser.hpp:1402
size_t digit_count
Total number of digits in integer.
Definition parser.hpp:1401
Test for hexadecimal integer.
Definition parser.hpp:1461
Base class for integer testing.
Definition parser.hpp:1274
size_t value
Calculated value of the numeral.
Definition parser.hpp:1288
Test for IPv4 address.
Definition parser.hpp:2344
stdex::interval< size_t > components[4]
Individual component intervals.
Definition parser.hpp:2389
struct in_addr value
IPv4 address value.
Definition parser.hpp:2390
Test for IPv6 address.
Definition parser.hpp:2556
std::shared_ptr< basic_parser< T > > scope_id
Scope ID (e.g. NIC index with link-local addresses)
Definition parser.hpp:2628
stdex::interval< size_t > components[8]
Individual component intervals.
Definition parser.hpp:2626
struct in6_addr value
IPv6 address value.
Definition parser.hpp:2627
Test for valid IPv6 address scope ID character.
Definition parser.hpp:2482
Test for repeating.
Definition parser.hpp:910
bool m_greedy
try to match as long sequence as possible
Definition parser.hpp:949
std::shared_ptr< basic_parser< T > > m_el
repeating element
Definition parser.hpp:946
size_t m_min_iterations
minimum number of iterations
Definition parser.hpp:947
size_t m_max_iterations
maximum number of iterations
Definition parser.hpp:948
Test for JSON string.
Definition parser.hpp:7201
MIME content type.
Definition parser.hpp:7749
stdex::interval< size_t > base_type
basic type position in source
Definition parser.hpp:7759
stdex::interval< size_t > sub_type
sub-type position in source
Definition parser.hpp:7760
stdex::interval< size_t > charset
charset position in source
Definition parser.hpp:7761
Test for mixed numeral.
Definition parser.hpp:1921
std::shared_ptr< basic_parser< T > > fraction
fraction
Definition parser.hpp:1954
std::shared_ptr< basic_parser< T > > special_sign
Special sign (e.g. plus-minus '±')
Definition parser.hpp:1952
std::shared_ptr< basic_parser< T > > negative_sign
Negative sign.
Definition parser.hpp:1951
std::shared_ptr< basic_parser< T > > positive_sign
Positive sign.
Definition parser.hpp:1950
std::shared_ptr< basic_parser< T > > integer
Integer part.
Definition parser.hpp:1953
Test for monetary numeral.
Definition parser.hpp:2215
std::shared_ptr< basic_parser< T > > positive_sign
Positive sign.
Definition parser.hpp:2248
std::shared_ptr< basic_parser< T > > decimal_separator
Decimal separator.
Definition parser.hpp:2253
std::shared_ptr< basic_parser< T > > currency
Currency part.
Definition parser.hpp:2251
std::shared_ptr< basic_parser< T > > decimal
Decimal part.
Definition parser.hpp:2254
std::shared_ptr< basic_parser< T > > integer
Integer part.
Definition parser.hpp:2252
std::shared_ptr< basic_parser< T > > negative_sign
Negative sign.
Definition parser.hpp:2249
std::shared_ptr< basic_parser< T > > special_sign
Special sign (e.g. plus-minus '±')
Definition parser.hpp:2250
"No-op" match
Definition parser.hpp:184
Base template for all parsers.
Definition parser.hpp:81
stdex::interval< size_t > interval
Region of the last match.
Definition parser.hpp:121
Test for permutation.
Definition parser.hpp:1198
Test for phone number.
Definition parser.hpp:4484
std::basic_string< T > value
Normalized phone number.
Definition parser.hpp:4509
Test for any punctuation code unit.
Definition parser.hpp:461
Test for Roman numeral.
Definition parser.hpp:1570
Test for scientific numeral.
Definition parser.hpp:2046
std::shared_ptr< basic_parser< T > > special_sign
Special sign (e.g. plus-minus '±')
Definition parser.hpp:2092
std::shared_ptr< basic_parser< T > > exponent_symbol
Exponent symbol (e.g. 'e')
Definition parser.hpp:2096
std::shared_ptr< basic_parser< T > > positive_sign
Positive sign.
Definition parser.hpp:2090
std::shared_ptr< basic_parser< T > > negative_sign
Negative sign.
Definition parser.hpp:2091
double value
Calculated value of the numeral.
Definition parser.hpp:2100
std::shared_ptr< basic_parser< T > > negative_exp_sign
Negative exponent sign (e.g. '-')
Definition parser.hpp:2098
std::shared_ptr< basic_integer< T > > decimal
Decimal part.
Definition parser.hpp:2095
std::shared_ptr< basic_parser< T > > positive_exp_sign
Positive exponent sign (e.g. '+')
Definition parser.hpp:2097
std::shared_ptr< basic_integer< T > > exponent
Exponent part.
Definition parser.hpp:2099
std::shared_ptr< basic_parser< T > > decimal_separator
Decimal separator.
Definition parser.hpp:2094
std::shared_ptr< basic_integer< T > > integer
Integer part.
Definition parser.hpp:2093
Test for match score.
Definition parser.hpp:1749
Test for sequence.
Definition parser.hpp:1006
Definition parser.hpp:689
Test for SI Reference delimiter.
Definition parser.hpp:5119
Test for SI Reference part.
Definition parser.hpp:5073
Test for SI Reference.
Definition parser.hpp:5158
basic_si_reference_part< T > part3
Reference data part 3 (P3)
Definition parser.hpp:5187
basic_si_reference_part< T > part1
Reference data part 1 (P1)
Definition parser.hpp:5185
bool is_valid
Is reference valid.
Definition parser.hpp:5188
T model[3]
Reference model.
Definition parser.hpp:5184
basic_si_reference_part< T > part2
Reference data part 2 (P2)
Definition parser.hpp:5186
Test for signed numeral.
Definition parser.hpp:1835
std::shared_ptr< basic_parser< T > > special_sign
Special sign (e.g. plus-minus '±')
Definition parser.hpp:1861
std::shared_ptr< basic_parser< T > > negative_sign
Negative sign.
Definition parser.hpp:1860
std::shared_ptr< basic_parser< T > > positive_sign
Positive sign.
Definition parser.hpp:1859
std::shared_ptr< basic_parser< T > > number
Number.
Definition parser.hpp:1862
Test for any space code unit.
Definition parser.hpp:381
Test for any space or punctuation code unit.
Definition parser.hpp:536
Test for any string.
Definition parser.hpp:1126
Test for given string.
Definition parser.hpp:815
Test for time.
Definition parser.hpp:4259
Test for valid URL password character.
Definition parser.hpp:3060
Test for valid URL path character.
Definition parser.hpp:3162
Test for URL path.
Definition parser.hpp:3272
Test for valid URL username character.
Definition parser.hpp:2959
Test for URL.
Definition parser.hpp:3412
Test for HTTP agent.
Definition parser.hpp:6745
Test for HTTP any type.
Definition parser.hpp:5888
Test for HTTP asterisk.
Definition parser.hpp:6516
Test for HTTP cookie parameter (RFC2109)
Definition parser.hpp:6611
Test for HTTP cookie (RFC2109)
Definition parser.hpp:6667
std::list< http_cookie_parameter > params
List of cookie parameters.
Definition parser.hpp:6679
http_token name
Cookie name.
Definition parser.hpp:6677
http_value value
Cookie value.
Definition parser.hpp:6678
Test for HTTP language (RFC1766)
Definition parser.hpp:6384
Test for HTTP line break (RFC2616: CRLF | LF)
Definition parser.hpp:5570
Test for HTTP parameter (RFC2616: parameter)
Definition parser.hpp:5836
http_token name
Parameter name.
Definition parser.hpp:5845
http_value value
Parameter value.
Definition parser.hpp:5846
Test for HTTP protocol.
Definition parser.hpp:6820
uint16_t version
HTTP protocol version: 0x100 = 1.0, 0x101 = 1.1...
Definition parser.hpp:6842
Test for HTTP quoted string (RFC2616: quoted-string)
Definition parser.hpp:5729
stdex::interval< size_t > content
String content (without quotes)
Definition parser.hpp:5738
Test for HTTP request.
Definition parser.hpp:6921
Test for HTTP space (RFC2616: LWS)
Definition parser.hpp:5606
Test for HTTP text character (RFC2616: TEXT)
Definition parser.hpp:5642
Test for HTTP token (RFC2616: token - tolerates non-ASCII)
Definition parser.hpp:5675
Test for HTTP URL parameter.
Definition parser.hpp:6209
Test for HTTP URL path segment.
Definition parser.hpp:6121
Test for HTTP URL path segment.
Definition parser.hpp:6154
std::vector< http_url_path_segment > segments
Path segments.
Definition parser.hpp:6162
Test for HTTP URL port.
Definition parser.hpp:6065
Test for HTTP URL server.
Definition parser.hpp:6028
Test for HTTP URL.
Definition parser.hpp:6286
Collection of HTTP values.
Definition parser.hpp:7157
Test for HTTP value (RFC2616: value)
Definition parser.hpp:5792
http_quoted_string string
Value when matched as quoted string.
Definition parser.hpp:5801
http_token token
Value when matched as token.
Definition parser.hpp:5802
Test for HTTP weight factor.
Definition parser.hpp:6447
float value
Calculated value of the weight factor.
Definition parser.hpp:6460
Test for HTTP weighted value.
Definition parser.hpp:6539
Base template for collection-holding parsers.
Definition parser.hpp:966
Test for any SGML code point.
Definition parser.hpp:249
Test for any SGML code point from a given string of SGML code points.
Definition parser.hpp:772
Test for specific SGML code point.
Definition parser.hpp:337
Test for valid DNS domain SGML character.
Definition parser.hpp:2831
Test for valid IPv6 address scope ID SGML character.
Definition parser.hpp:2521
Test for any SGML punctuation code point.
Definition parser.hpp:502
Test for any SGML space code point.
Definition parser.hpp:424
Test for any SGML space or punctuation code point.
Definition parser.hpp:579
Test for SGML given string.
Definition parser.hpp:862
Test for valid URL password SGML character.
Definition parser.hpp:3113
Test for valid URL path SGML character.
Definition parser.hpp:3219
Test for valid URL username SGML character.
Definition parser.hpp:3011
Numerical interval.
Definition interval.hpp:18
T size() const
Returns interval size.
Definition interval.hpp:47
T end
interval end
Definition interval.hpp:20
interval() noexcept
Constructs an invalid interval.
Definition interval.hpp:25
void invalidate()
Invalidates interval.
Definition interval.hpp:59
T start
interval start
Definition interval.hpp:19
Tag attribute.
Definition parser.hpp:8024
stdex::interval< size_t > name
attribute name position in source
Definition parser.hpp:8025
stdex::interval< size_t > value
attribute value position in source
Definition parser.hpp:8026
Definition parser.hpp:7183