#!./perl # # This is a home for regular expression tests that don't fit into # the format supported by re/regexp.t. If you want to add a test # that does fit that format, add it to re/re_tests, not here. use strict; use warnings; use 5.010; sub run_tests; $| = 1; BEGIN { chdir 't' if -d 't'; @INC = ('../lib','.'); do "re/ReTest.pl" or die $@; } plan tests => 123; # Update this when adding/deleting tests. run_tests() unless caller; # # Tests start here. # sub run_tests { { local $Message = "Call code from qr //"; local $_ = 'var="foo"'; $a = qr/(?{++$b})/; $b = 7; ok /$a$a/ && $b eq '9'; my $c="$a"; ok /$a$a/ && $b eq '11'; undef $@; eval {/$c/}; ok $@ && $@ =~ /not allowed at runtime/; use re "eval"; /$a$c$a/; iseq $b, '14'; our $lex_a = 43; our $lex_b = 17; our $lex_c = 27; my $lex_res = ($lex_b =~ qr/$lex_b(?{ $lex_c = $lex_a++ })/); iseq $lex_res, 1; iseq $lex_a, 44; iseq $lex_c, 43; no re "eval"; undef $@; my $match = eval { /$a$c$a/ }; ok $@ && $@ =~ /Eval-group not allowed/ && !$match; iseq $b, '14'; $lex_a = 2; $lex_a = 43; $lex_b = 17; $lex_c = 27; $lex_res = ($lex_b =~ qr/17(?{ $lex_c = $lex_a++ })/); iseq $lex_res, 1; iseq $lex_a, 44; iseq $lex_c, 43; } { our $a = bless qr /foo/ => 'Foo'; ok 'goodfood' =~ $a, "Reblessed qr // matches"; iseq $a, '(?-xism:foo)', "Reblessed qr // stringifies"; my $x = "\x{3fe}"; my $z = my $y = "\317\276"; # Byte representation of $x $a = qr /$x/; ok $x =~ $a, "UTF-8 interpolation in qr //"; ok "a$a" =~ $x, "Stringified qr // preserves UTF-8"; ok "a$x" =~ /^a$a\z/, "Interpolated qr // preserves UTF-8"; ok "a$x" =~ /^a(??{$a})\z/, "Postponed interpolation of qr // preserves UTF-8"; { local $BugId = '17776'; iseq length qr /##/x, 12, "## in qr // doesn't corrupt memory"; } { use re 'eval'; ok "$x$x" =~ /^$x(??{$x})\z/, "Postponed UTF-8 string in UTF-8 re matches UTF-8"; ok "$y$x" =~ /^$y(??{$x})\z/, "Postponed UTF-8 string in non-UTF-8 re matches UTF-8"; ok "$y$x" !~ /^$y(??{$y})\z/, "Postponed non-UTF-8 string in non-UTF-8 re doesn't match UTF-8"; ok "$x$x" !~ /^$x(??{$y})\z/, "Postponed non-UTF-8 string in UTF-8 re doesn't match UTF-8"; ok "$y$y" =~ /^$y(??{$y})\z/, "Postponed non-UTF-8 string in non-UTF-8 re matches non-UTF8"; ok "$x$y" =~ /^$x(??{$y})\z/, "Postponed non-UTF-8 string in UTF-8 re matches non-UTF8"; $y = $z; # Reset $y after upgrade. ok "$x$y" !~ /^$x(??{$x})\z/, "Postponed UTF-8 string in UTF-8 re doesn't match non-UTF-8"; ok "$y$y" !~ /^$y(??{$x})\z/, "Postponed UTF-8 string in non-UTF-8 re doesn't match non-UTF-8"; } } { use re 'eval'; local $Message = 'Test if $^N and $+ work in (?{{})'; our @ctl_n = (); our @plus = (); our $nested_tags; $nested_tags = qr{ < ((\w)+) (?{ push @ctl_n, (defined $^N ? $^N : "undef"); push @plus, (defined $+ ? $+ : "undef"); }) > (??{$nested_tags})* </\s* \w+ \s*> }x; my $c = 0; for my $test ( # Test structure: # [ Expected result, Regex, Expected value(s) of $^N, Expected value(s) of $+ ] [ 1, qr#^$nested_tags$#, "bla blubb bla", "a b a" ], [ 1, qr#^($nested_tags)$#, "bla blubb <bla><blubb></blubb></bla>", "a b a" ], [ 1, qr#^(|)$nested_tags$#, "bla blubb bla", "a b a" ], [ 1, qr#^(?:|)$nested_tags$#, "bla blubb bla", "a b a" ], [ 1, qr#^<(bl|bla)>$nested_tags<(/\1)>$#, "blubb /bla", "b /bla" ], [ 1, qr#(??{"(|)"})$nested_tags$#, "bla blubb bla", "a b a" ], [ 1, qr#^(??{"(bla|)"})$nested_tags$#, "bla blubb bla", "a b a" ], [ 1, qr#^(??{"(|)"})(??{$nested_tags})$#, "bla blubb undef", "a b undef" ], [ 1, qr#^(??{"(?:|)"})$nested_tags$#, "bla blubb bla", "a b a" ], [ 1, qr#^((??{"(?:bla|)"}))((??{$nested_tags}))$#, "bla blubb <bla><blubb></blubb></bla>", "a b <bla><blubb></blubb></bla>" ], [ 1, qr#^((??{"(?!)?"}))((??{$nested_tags}))$#, "bla blubb <bla><blubb></blubb></bla>", "a b <bla><blubb></blubb></bla>" ], [ 1, qr#^((??{"(?:|<(/?bla)>)"}))((??{$nested_tags}))\1$#, "bla blubb <bla><blubb></blubb></bla>", "a b <bla><blubb></blubb></bla>" ], [ 0, qr#^((??{"(?!)"}))?((??{$nested_tags}))(?!)$#, "bla blubb undef", "a b undef" ], ) { #"#silence vim highlighting $c++; @ctl_n = (); @plus = (); my $match = (("<bla><blubb></blubb></bla>" =~ $test->[1]) ? 1 : 0); push @ctl_n, (defined $^N ? $^N : "undef"); push @plus, (defined $+ ? $+ : "undef"); ok($test->[0] == $match, "match $c"); if ($test->[0] != $match) { # unset @ctl_n and @plus @ctl_n = @plus = (); } iseq("@ctl_n", $test->[2], "ctl_n $c"); iseq("@plus", $test->[3], "plus $c"); } } { use re 'eval'; local $BugId = '56194'; our $f; local $f; $f = sub { defined $_[0] ? $_[0] : "undef"; }; ok("123" =~ m/^(\d)(((??{1 + $^N})))+$/); our @ctl_n; our @plus; my $re = qr#(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1}))*(?{$^N})#; my $re2 = qr#(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1}))*(?{$^N})(|a(b)c|def)(??{"$^R"})#; my $re3 = qr#(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1})){2}(?{$^N})(|a(b)c|def)(??{"$^R"})#; our $re5; local $re5 = qr#(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1})){2}(?{$^N})#; my $re6 = qr#(??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1})#; my $re7 = qr#(??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1})#; my $re8 = qr/(\d+)/; my $c = 0; for my $test ( # Test structure: # [ # String to match # Regex too match # Expected values of $^N # Expected values of $+ # Expected values of $1, $2, $3, $4 and $5 # ] [ "1233", qr#^(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1}))+(??{$^N})$#, "1 2 3 3", "1 2 3 3", "\$1 = 1, \$2 = 3, \$3 = undef, \$4 = undef, \$5 = undef", ], [ "1233", qr#^(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1}))+(abc|def|)?(??{$+})$#, "1 2 3 3", "1 2 3 3", "\$1 = 1, \$2 = 3, \$3 = undef, \$4 = undef, \$5 = undef", ], [ "1233", qr#^(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1}))+(|abc|def)?(??{$+})$#, "1 2 3 3", "1 2 3 3", "\$1 = 1, \$2 = 3, \$3 = undef, \$4 = undef, \$5 = undef", ], [ "1233", qr#^(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1}))+(abc|def|)?(??{$^N})$#, "1 2 3 3", "1 2 3 3", "\$1 = 1, \$2 = 3, \$3 = undef, \$4 = undef, \$5 = undef", ], [ "1233", qr#^(1)((??{ push @ctl_n, $f->($^N); push @plus, $f->($+); $^N + 1}))+(|abc|def)?(??{$^N})$#, "1 2 3 3", "1 2 3 3", "\$1 = 1, \$2 = 3, \$3 = undef, \$4 = undef, \$5 = undef", ], [ "123abc3", qr#^($re)(|a(b)c|def)(??{$^R})$#, "1 2 3 abc", "1 2 3 b", "\$1 = 123, \$2 = 1, \$3 = 3, \$4 = abc, \$5 = b", ], [ "123abc3", qr#^($re2)$#, "1 2 3 123abc3", "1 2 3 b", "\$1 = 123abc3, \$2 = 1, \$3 = 3, \$4 = abc, \$5 = b", ], [ "123abc3", qr#^($re3)$#, "1 2 123abc3", "1 2 b", "\$1 = 123abc3, \$2 = 1, \$3 = 3, \$4 = abc, \$5 = b", ], [ "123abc3", qr#^(??{$re5})(|abc|def)(??{"$^R"})$#, "1 2 abc", "1 2 abc", "\$1 = abc, \$2 = undef, \$3 = undef, \$4 = undef, \$5 = undef", ], [ "123abc3", qr#^(??{$re5})(|a(b)c|def)(??{"$^R"})$#, "1 2 abc", "1 2 b", "\$1 = abc, \$2 = b, \$3 = undef, \$4 = undef, \$5 = undef", ], [ "1234", qr#^((\d+)((??{push @ctl_n, $f->($^N); push @plus, $f->($+);$^N + 1}))((??{push @ctl_n, $f->($^N); push @plus, $f->($+);$^N + 1}))((??{push @ctl_n, $f->($^N); push @plus, $f->($+);$^N + 1})))$#, "1234 123 12 1 2 3 1234", "1234 123 12 1 2 3 4", "\$1 = 1234, \$2 = 1, \$3 = 2, \$4 = 3, \$5 = 4", ], [ "1234556", qr#^(\d+)($re6)($re6)($re6)$re6(($re6)$re6)$#, "1234556 123455 12345 1234 123 12 1 2 3 4 4 5 56", "1234556 123455 12345 1234 123 12 1 2 3 4 4 5 5", "\$1 = 1, \$2 = 2, \$3 = 3, \$4 = 4, \$5 = 56", ], [ "12345562", qr#^((??{$re8}))($re7)($re7)($re7)$re7($re7)($re7(\2))$#, "12345562 1234556 123455 12345 1234 123 12 1 2 3 4 4 5 62", "12345562 1234556 123455 12345 1234 123 12 1 2 3 4 4 5 2", "\$1 = 1, \$2 = 2, \$3 = 3, \$4 = 4, \$5 = 5", ], ) { $c++; @ctl_n = (); @plus = (); undef $^R; my $match = $test->[0] =~ $test->[1]; my $str = join(", ", '$1 = '.$f->($1), '$2 = '.$f->($2), '$3 = '.$f->($3), '$4 = '.$f->($4),'$5 = '.$f->($5)); push @ctl_n, $f->($^N); push @plus, $f->($+); ok($match, "match $c"); if (not $match) { # unset $str, @ctl_n and @plus $str = ""; @ctl_n = @plus = (); } iseq("@ctl_n", $test->[2], "ctl_n $c"); iseq("@plus", $test->[3], "plus $c"); iseq($str, $test->[4], "str $c"); } SKIP: { if ($] le '5.010') { skip "test segfaults on perl < 5.10", 4; } @ctl_n = (); @plus = (); our $re4; local $re4 = qr#(1)((??{push @ctl_n, $f->($^N); push @plus, $f->($+);$^N + 1})){2}(?{$^N})(|abc|def)(??{"$^R"})#; undef $^R; my $match = "123abc3" =~ m/^(??{$re4})$/; my $str = join(", ", '$1 = '.$f->($1), '$2 = '.$f->($2), '$3 = '.$f->($3), '$4 = '.$f->($4),'$5 = '.$f->($5),'$^R = '.$f->($^R)); push @ctl_n, $f->($^N); push @plus, $f->($+); ok($match); if (not $match) { # unset $str @ctl_n = (); @plus = (); $str = ""; } iseq("@ctl_n", "1 2 undef"); iseq("@plus", "1 2 undef"); iseq($str, "\$1 = undef, \$2 = undef, \$3 = undef, \$4 = undef, \$5 = undef, \$^R = undef"); } } } # End of sub run_tests 1;