| //===----------------------------------------------------------------------===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| // <regex> |
| |
| // class regex_iterator<BidirectionalIterator, charT, traits> |
| |
| // regex_iterator operator++(int); |
| |
| #include <regex> |
| #include <cassert> |
| #include <iterator> |
| #include "test_macros.h" |
| |
| void validate_prefixes(const std::regex& empty_matching_pattern) { |
| const char source[] = "abc"; |
| |
| std::cregex_iterator i(source, source + 3, empty_matching_pattern); |
| assert(!i->prefix().matched); |
| assert(i->prefix().length() == 0); |
| assert(i->prefix().first == source); |
| assert(i->prefix().second == source); |
| |
| ++i; |
| assert(i->prefix().matched); |
| assert(i->prefix().length() == 1); |
| assert(i->prefix().first == source); |
| assert(i->prefix().second == source + 1); |
| assert(i->prefix().str() == "a"); |
| |
| ++i; |
| assert(i->prefix().matched); |
| assert(i->prefix().length() == 1); |
| assert(i->prefix().first == source + 1); |
| assert(i->prefix().second == source + 2); |
| assert(i->prefix().str() == "b"); |
| |
| ++i; |
| assert(i->prefix().matched); |
| assert(i->prefix().length() == 1); |
| assert(i->prefix().first == source + 2); |
| assert(i->prefix().second == source + 3); |
| assert(i->prefix().str() == "c"); |
| |
| ++i; |
| assert(i == std::cregex_iterator()); |
| } |
| |
| void test_prefix_adjustment() { |
| // Check that we correctly adjust the match prefix when dealing with zero-length matches -- this is explicitly |
| // required by the Standard ([re.regiter.incr]: "In all cases in which the call to `regex_search` returns true, |
| // `match.prefix().first` shall be equal to the previous value of `match[0].second`"). For a pattern that matches |
| // empty sequences, there is an implicit zero-length match between every character in a string -- make sure the |
| // prefix of each of these matches (except the first one) is the preceding character. |
| |
| // An empty pattern produces zero-length matches. |
| validate_prefixes(std::regex("")); |
| // Any character repeated zero or more times can produce zero-length matches. |
| validate_prefixes(std::regex("X*")); |
| validate_prefixes(std::regex("X{0,3}")); |
| } |
| |
| int main(int, char**) { |
| { |
| std::regex phone_numbers("\\d{3}-\\d{4}"); |
| const char phone_book[] = "555-1234, 555-2345, 555-3456"; |
| std::cregex_iterator i(std::begin(phone_book), std::end(phone_book), phone_numbers); |
| std::cregex_iterator i2 = i; |
| assert(i != std::cregex_iterator()); |
| assert(i2 != std::cregex_iterator()); |
| assert((*i).size() == 1); |
| assert((*i).position() == 0); |
| assert((*i).str() == "555-1234"); |
| assert((*i2).size() == 1); |
| assert((*i2).position() == 0); |
| assert((*i2).str() == "555-1234"); |
| i++; |
| assert(i != std::cregex_iterator()); |
| assert(i2 != std::cregex_iterator()); |
| assert((*i).size() == 1); |
| assert((*i).position() == 10); |
| assert((*i).str() == "555-2345"); |
| assert((*i2).size() == 1); |
| assert((*i2).position() == 0); |
| assert((*i2).str() == "555-1234"); |
| i++; |
| assert(i != std::cregex_iterator()); |
| assert(i2 != std::cregex_iterator()); |
| assert((*i).size() == 1); |
| assert((*i).position() == 20); |
| assert((*i).str() == "555-3456"); |
| assert((*i2).size() == 1); |
| assert((*i2).position() == 0); |
| assert((*i2).str() == "555-1234"); |
| i++; |
| assert(i == std::cregex_iterator()); |
| assert(i2 != std::cregex_iterator()); |
| assert((*i2).size() == 1); |
| assert((*i2).position() == 0); |
| assert((*i2).str() == "555-1234"); |
| } |
| { |
| std::regex phone_numbers("\\d{3}-\\d{4}"); |
| const char phone_book[] = "555-1234, 555-2345, 555-3456"; |
| std::cregex_iterator i(std::begin(phone_book), std::end(phone_book), phone_numbers); |
| std::cregex_iterator i2 = i; |
| assert(i != std::cregex_iterator()); |
| assert(i2 != std::cregex_iterator()); |
| assert((*i).size() == 1); |
| assert((*i).position() == 0); |
| assert((*i).str() == "555-1234"); |
| assert((*i2).size() == 1); |
| assert((*i2).position() == 0); |
| assert((*i2).str() == "555-1234"); |
| ++i; |
| assert(i != std::cregex_iterator()); |
| assert(i2 != std::cregex_iterator()); |
| assert((*i).size() == 1); |
| assert((*i).position() == 10); |
| assert((*i).str() == "555-2345"); |
| assert((*i2).size() == 1); |
| assert((*i2).position() == 0); |
| assert((*i2).str() == "555-1234"); |
| ++i; |
| assert(i != std::cregex_iterator()); |
| assert(i2 != std::cregex_iterator()); |
| assert((*i).size() == 1); |
| assert((*i).position() == 20); |
| assert((*i).str() == "555-3456"); |
| assert((*i2).size() == 1); |
| assert((*i2).position() == 0); |
| assert((*i2).str() == "555-1234"); |
| ++i; |
| assert(i == std::cregex_iterator()); |
| assert(i2 != std::cregex_iterator()); |
| assert((*i2).size() == 1); |
| assert((*i2).position() == 0); |
| assert((*i2).str() == "555-1234"); |
| } |
| { // https://llvm.org/PR33681 |
| std::regex rex(".*"); |
| const char foo[] = "foo"; |
| // The -1 is because we don't want the implicit null from the array. |
| std::cregex_iterator i(std::begin(foo), std::end(foo) - 1, rex); |
| std::cregex_iterator e; |
| assert(i != e); |
| assert((*i).size() == 1); |
| assert((*i).str() == "foo"); |
| |
| ++i; |
| assert(i != e); |
| assert((*i).size() == 1); |
| assert((*i).str() == ""); |
| |
| ++i; |
| assert(i == e); |
| } |
| |
| test_prefix_adjustment(); |
| |
| return 0; |
| } |