X-Git-Url: https://perl5.git.perl.org/perl5.git/blobdiff_plain/cb79f740dae4d41cfe556ac0e57a6e7afcd0fb6f..c2a42ab2c12f5756cbe8c9a43ef6babb6044695a:/t/re/reg_mesg.t diff --git a/t/re/reg_mesg.t b/t/re/reg_mesg.t index 74b264a..c0ede1c 100644 --- a/t/re/reg_mesg.t +++ b/t/re/reg_mesg.t @@ -15,12 +15,29 @@ use strict; ## arrays below. The {#} is a meta-marker -- it marks where the marker should ## go. ## +## Returns empty string if that is what is expected. Otherwise, handles +## either a scalar, turning it into a single element array; or a ref to an +## array, adjusting each element. If called in array context, returns an +## array, otherwise the join of all elements + sub fixup_expect { - my $expect = shift; - $expect =~ s/{\#}/<-- HERE/; - $expect =~ s/{\#}/ <-- HERE /; - $expect .= " at "; - return $expect; + my $expect_ref = shift; + return if $expect_ref eq ""; + + my @expect; + if (ref $expect_ref) { + @expect = @$expect_ref; + } + else { + @expect = $expect_ref; + } + + foreach my $element (@expect) { + $element =~ s/{\#}/in regex; marked by <-- HERE in/; + $element =~ s/{\#}/ <-- HERE /; + $element .= " at "; + } + return wantarray ? @expect : join "", @expect; } my $inf_m1 = ($Config::Config{reg_infty} || 32767) - 1; @@ -31,115 +48,270 @@ my $inf_p1 = $inf_m1 + 2; ## my @death = ( - '/[[=foo=]]/' => 'POSIX syntax [= =] is reserved for future extensions in regex; marked by {#} in m/[[=foo=]{#}]/', + '/[[=foo=]]/' => 'POSIX syntax [= =] is reserved for future extensions {#} m/[[=foo=]{#}]/', '/(?<= .*)/' => 'Variable length lookbehind not implemented in regex m/(?<= .*)/', '/(?<= x{1000})/' => 'Lookbehind longer than 255 not implemented in regex m/(?<= x{1000})/', - '/(?@)/' => 'Sequence (?@...) not implemented in regex; marked by {#} in m/(?@{#})/', - - '/(?{ 1/' => 'Sequence (?{...}) not terminated or not {}-balanced in regex; marked by {#} in m/(?{{#} 1/', + '/(?@)/' => 'Sequence (?@...) not implemented {#} m/(?@{#})/', - '/(?(1x))/' => 'Switch condition not recognized in regex; marked by {#} in m/(?(1x{#}))/', + '/(?{ 1/' => 'Missing right curly or square bracket', - '/(?(1)x|y|z)/' => 'Switch (?(condition)... contains too many branches in regex; marked by {#} in m/(?(1)x|y|{#}z)/', + '/(?(1x))/' => 'Switch condition not recognized {#} m/(?(1x{#}))/', - '/(?(x)y|x)/' => 'Unknown switch condition (?(x) in regex; marked by {#} in m/(?({#}x)y|x)/', + '/(?(1)x|y|z)/' => 'Switch (?(condition)... contains too many branches {#} m/(?(1)x|y|{#}z)/', - '/(?/' => 'Sequence (? incomplete in regex; marked by {#} in m/(?{#}/', + '/(?(x)y|x)/' => 'Unknown switch condition (?(x) {#} m/(?({#}x)y|x)/', - '/(?;x/' => 'Sequence (?;...) not recognized in regex; marked by {#} in m/(?;{#}x/', - '/(?<;x/' => 'Sequence (?<;...) not recognized in regex; marked by {#} in m/(?<;{#}x/', + '/(?/' => 'Sequence (? incomplete {#} m/(?{#}/', - '/(?\ix/' => 'Sequence (?\...) not recognized in regex; marked by {#} in m/(?\{#}ix/', - '/(?\mx/' => 'Sequence (?\...) not recognized in regex; marked by {#} in m/(?\{#}mx/', - '/(?\:x/' => 'Sequence (?\...) not recognized in regex; marked by {#} in m/(?\{#}:x/', - '/(?\=x/' => 'Sequence (?\...) not recognized in regex; marked by {#} in m/(?\{#}=x/', - '/(?\!x/' => 'Sequence (?\...) not recognized in regex; marked by {#} in m/(?\{#}!x/', - '/(?\<=x/' => 'Sequence (?\...) not recognized in regex; marked by {#} in m/(?\{#}<=x/', - '/(?\ 'Sequence (?\...) not recognized in regex; marked by {#} in m/(?\{#}x/' => 'Sequence (?\...) not recognized in regex; marked by {#} in m/(?\{#}>x/', - '/(?^-i:foo)/' => 'Sequence (?^-...) not recognized in regex; marked by {#} in m/(?^-{#}i:foo)/', - '/(?^-i)foo/' => 'Sequence (?^-...) not recognized in regex; marked by {#} in m/(?^-{#}i)foo/', - '/(?^d:foo)/' => 'Sequence (?^d...) not recognized in regex; marked by {#} in m/(?^d{#}:foo)/', - '/(?^d)foo/' => 'Sequence (?^d...) not recognized in regex; marked by {#} in m/(?^d{#})foo/', - '/(?^lu:foo)/' => 'Sequence (?^lu...) not recognized in regex; marked by {#} in m/(?^lu{#}:foo)/', - '/(?^lu)foo/' => 'Sequence (?^lu...) not recognized in regex; marked by {#} in m/(?^lu{#})foo/', + '/(?;x/' => 'Sequence (?;...) not recognized {#} m/(?;{#}x/', + '/(?<;x/' => 'Group name must start with a non-digit word character {#} m/(?<;{#}x/', + '/(?\ix/' => 'Sequence (?\...) not recognized {#} m/(?\{#}ix/', + '/(?\mx/' => 'Sequence (?\...) not recognized {#} m/(?\{#}mx/', + '/(?\:x/' => 'Sequence (?\...) not recognized {#} m/(?\{#}:x/', + '/(?\=x/' => 'Sequence (?\...) not recognized {#} m/(?\{#}=x/', + '/(?\!x/' => 'Sequence (?\...) not recognized {#} m/(?\{#}!x/', + '/(?\<=x/' => 'Sequence (?\...) not recognized {#} m/(?\{#}<=x/', + '/(?\ 'Sequence (?\...) not recognized {#} m/(?\{#}x/' => 'Sequence (?\...) not recognized {#} m/(?\{#}>x/', + '/(?^-i:foo)/' => 'Sequence (?^-...) not recognized {#} m/(?^-{#}i:foo)/', + '/(?^-i)foo/' => 'Sequence (?^-...) not recognized {#} m/(?^-{#}i)foo/', + '/(?^d:foo)/' => 'Sequence (?^d...) not recognized {#} m/(?^d{#}:foo)/', + '/(?^d)foo/' => 'Sequence (?^d...) not recognized {#} m/(?^d{#})foo/', + '/(?^lu:foo)/' => 'Regexp modifiers "l" and "u" are mutually exclusive {#} m/(?^lu{#}:foo)/', + '/(?^lu)foo/' => 'Regexp modifiers "l" and "u" are mutually exclusive {#} m/(?^lu{#})foo/', +'/(?da:foo)/' => 'Regexp modifiers "d" and "a" are mutually exclusive {#} m/(?da{#}:foo)/', +'/(?lil:foo)/' => 'Regexp modifier "l" may not appear twice {#} m/(?lil{#}:foo)/', +'/(?aaia:foo)/' => 'Regexp modifier "a" may appear a maximum of twice {#} m/(?aaia{#}:foo)/', +'/(?i-l:foo)/' => 'Regexp modifier "l" may not appear after the "-" {#} m/(?i-l{#}:foo)/', - '/((x)/' => 'Unmatched ( in regex; marked by {#} in m/({#}(x)/', + '/((x)/' => 'Unmatched ( {#} m/({#}(x)/', - "/x{$inf_p1}/" => "Quantifier in {,} bigger than $inf_m1 in regex; marked by {#} in m/x{{#}$inf_p1}/", + "/x{$inf_p1}/" => "Quantifier in {,} bigger than $inf_m1 {#} m/x{{#}$inf_p1}/", - '/x{3,1}/' => 'Can\'t do {n,m} with n > m in regex; marked by {#} in m/x{3,1}{#}/', - '/x**/' => 'Nested quantifiers in regex; marked by {#} in m/x**{#}/', + '/x**/' => 'Nested quantifiers {#} m/x**{#}/', - '/x[/' => 'Unmatched [ in regex; marked by {#} in m/x[{#}/', + '/x[/' => 'Unmatched [ {#} m/x[{#}/', - '/*/', => 'Quantifier follows nothing in regex; marked by {#} in m/*{#}/', + '/*/', => 'Quantifier follows nothing {#} m/*{#}/', - '/\p{x/' => 'Missing right brace on \p{} in regex; marked by {#} in m/\p{{#}x/', + '/\p{x/' => 'Missing right brace on \p{} {#} m/\p{{#}x/', - '/[\p{x]/' => 'Missing right brace on \p{} in regex; marked by {#} in m/[\p{{#}x]/', + '/[\p{x]/' => 'Missing right brace on \p{} {#} m/[\p{{#}x]/', - '/(x)\2/' => 'Reference to nonexistent group in regex; marked by {#} in m/(x)\2{#}/', + '/(x)\2/' => 'Reference to nonexistent group {#} m/(x)\2{#}/', 'my $m = "\\\"; $m =~ $m', => 'Trailing \ in regex m/\/', - '/\x{1/' => 'Missing right brace on \x{} in regex; marked by {#} in m/\x{{#}1/', - - '/[\x{X]/' => 'Missing right brace on \x{} in regex; marked by {#} in m/[\x{{#}X]/', - - '/[[:barf:]]/' => 'POSIX class [:barf:] unknown in regex; marked by {#} in m/[[:barf:]{#}]/', - - '/[[=barf=]]/' => 'POSIX syntax [= =] is reserved for future extensions in regex; marked by {#} in m/[[=barf=]{#}]/', - - '/[[.barf.]]/' => 'POSIX syntax [. .] is reserved for future extensions in regex; marked by {#} in m/[[.barf.]{#}]/', - - '/[z-a]/' => 'Invalid [] range "z-a" in regex; marked by {#} in m/[z-a{#}]/', - - '/\p/' => 'Empty \p{} in regex; marked by {#} in m/\p{#}/', - - '/\P{}/' => 'Empty \P{} in regex; marked by {#} in m/\P{{#}}/', + '/\x{1/' => 'Missing right brace on \x{} {#} m/\x{1{#}/', + '/\x{X/' => 'Missing right brace on \x{} {#} m/\x{{#}X/', + + '/[\x{X]/' => 'Missing right brace on \x{} {#} m/[\x{{#}X]/', + '/[\x{A]/' => 'Missing right brace on \x{} {#} m/[\x{A{#}]/', + + '/\o{1/' => 'Missing right brace on \o{ {#} m/\o{1{#}/', + '/\o{X/' => 'Missing right brace on \o{ {#} m/\o{{#}X/', + + '/[\o{X]/' => 'Missing right brace on \o{ {#} m/[\o{{#}X]/', + '/[\o{7]/' => 'Missing right brace on \o{ {#} m/[\o{7{#}]/', + + '/[[:barf:]]/' => 'POSIX class [:barf:] unknown {#} m/[[:barf:]{#}]/', + + '/[[=barf=]]/' => 'POSIX syntax [= =] is reserved for future extensions {#} m/[[=barf=]{#}]/', + + '/[[.barf.]]/' => 'POSIX syntax [. .] is reserved for future extensions {#} m/[[.barf.]{#}]/', + + '/[z-a]/' => 'Invalid [] range "z-a" {#} m/[z-a{#}]/', + + '/\p/' => 'Empty \p{} {#} m/\p{#}/', + + '/\P{}/' => 'Empty \P{} {#} m/\P{{#}}/', + '/(?[[[:word]]])/' => "Unmatched ':' in POSIX class {#} m/(?[[[:word{#}]]])/", + '/(?[[:word]])/' => "Unmatched ':' in POSIX class {#} m/(?[[:word{#}]])/", + '/(?[[[:digit: ])/' => "Unmatched '[' in POSIX class {#} m/(?[[[:digit:{#} ])/", + '/(?[[:digit: ])/' => "Unmatched '[' in POSIX class {#} m/(?[[:digit:{#} ])/", + '/(?[[[::]]])/' => "POSIX class [::] unknown {#} m/(?[[[::]{#}]])/", + '/(?[[[:w:]]])/' => "POSIX class [:w:] unknown {#} m/(?[[[:w:]{#}]])/", + '/(?[[:w:]])/' => "POSIX class [:w:] unknown {#} m/(?[[:w:]{#}])/", + '/(?[a])/' => 'Unexpected character {#} m/(?[a{#}])/', + '/(?[\t])/l' => '(?[...]) not valid in locale {#} m/(?[{#}\t])/', + '/(?[ + \t ])/' => 'Unexpected binary operator \'+\' with no preceding operand {#} m/(?[ +{#} \t ])/', + '/(?[ \cK - ( + \t ) ])/' => 'Unexpected binary operator \'+\' with no preceding operand {#} m/(?[ \cK - ( +{#} \t ) ])/', + '/(?[ \cK ( \t ) ])/' => 'Unexpected \'(\' with no preceding operator {#} m/(?[ \cK ({#} \t ) ])/', + '/(?[ \cK \t ])/' => 'Operand with no preceding operator {#} m/(?[ \cK \t{#} ])/', + '/(?[ \0004 ])/' => 'Need exactly 3 octal digits {#} m/(?[ \0004 {#}])/', + '/(?[ \05 ])/' => 'Need exactly 3 octal digits {#} m/(?[ \05 {#}])/', + '/(?[ \o{1038} ])/' => 'Non-octal character {#} m/(?[ \o{1038{#}} ])/', + '/(?[ \o{} ])/' => 'Number with no digits {#} m/(?[ \o{}{#} ])/', + '/(?[ \x{defg} ])/' => 'Non-hex character {#} m/(?[ \x{defg{#}} ])/', + '/(?[ \xabcdef ])/' => 'Use \\x{...} for more than two hex characters {#} m/(?[ \xabc{#}def ])/', + '/(?[ \x{} ])/' => 'Number with no digits {#} m/(?[ \x{}{#} ])/', + '/(?[ \cK + ) ])/' => 'Unexpected \')\' {#} m/(?[ \cK + ){#} ])/', + '/(?[ \cK + ])/' => 'Incomplete expression within \'(?[ ])\' {#} m/(?[ \cK + {#}])/', + '/(?[ \p{foo} ])/' => 'Property \'foo\' is unknown {#} m/(?[ \p{foo}{#} ])/', + '/(?[ \p{ foo = bar } ])/' => 'Property \'foo = bar\' is unknown {#} m/(?[ \p{ foo = bar }{#} ])/', + '/(?[ \8 ])/' => 'Unrecognized escape \8 in character class {#} m/(?[ \8{#} ])/', + '/(?[ \t ]/' => 'Syntax error in (?[...]) in regex m/(?[ \t ]/', + '/(?[ [ \t ]/' => 'Syntax error in (?[...]) in regex m/(?[ [ \t ]/', + '/(?[ \t ] ]/' => 'Syntax error in (?[...]) in regex m/(?[ \t ] ]/', + '/(?[ [ ] ]/' => 'Syntax error in (?[...]) in regex m/(?[ [ ] ]/', + '/(?[ \t + \e # This was supposed to be a comment ])/' => 'Syntax error in (?[...]) in regex m/(?[ \t + \e # This was supposed to be a comment ])/', + '/(?[ ])/' => 'Incomplete expression within \'(?[ ])\' {#} m/(?[ {#}])/', + 'm/(?[[a-\d]])/' => 'False [] range "a-\d" {#} m/(?[[a-\d{#}]])/', + 'm/(?[[\w-x]])/' => 'False [] range "\w-" {#} m/(?[[\w-{#}x]])/', + 'm/(?[[a-\pM]])/' => 'False [] range "a-\pM" {#} m/(?[[a-\pM{#}]])/', + 'm/(?[[\pM-x]])/' => 'False [] range "\pM-" {#} m/(?[[\pM-{#}x]])/', + 'm/(?[[\N{LATIN CAPITAL LETTER A WITH MACRON AND GRAVE}]])/' => '\N{} in character class restricted to one character {#} m/(?[[\N{U+100.300{#}}]])/', + 'm/(?[ \p{Digit} & (?(?[ \p{Thai} | \p{Lao} ]))])/' => 'Sequence (?(...) not recognized {#} m/(?[ \p{Digit} & (?({#}?[ \p{Thai} | \p{Lao} ]))])/', + 'm/(?[ \p{Digit} & (?:(?[ \p{Thai} | \p{Lao} ]))])/' => 'Expecting \'(?flags:(?[...\' {#} m/(?[ \p{Digit} & (?{#}:(?[ \p{Thai} | \p{Lao} ]))])/', + 'm/\o{/' => 'Missing right brace on \o{ {#} m/\o{{#}/', + 'm/\o/' => 'Missing braces on \o{} {#} m/\o{#}/', + 'm/\o{}/' => 'Number with no digits {#} m/\o{}{#}/', + 'm/[\o{]/' => 'Missing right brace on \o{ {#} m/[\o{{#}]/', + 'm/[\o]/' => 'Missing braces on \o{} {#} m/[\o{#}]/', + 'm/[\o{}]/' => 'Number with no digits {#} m/[\o{}{#}]/', + 'm/(?^-i:foo)/' => 'Sequence (?^-...) not recognized {#} m/(?^-{#}i:foo)/', ); +# Tests involving a user-defined charnames translator are in pat_advanced.t + +# In the following arrays of warnings, the value can be an array of things to +# expect. If the empty string, it means no warning should be raised. ## -## Key-value pairs of code/error of code that should have non-fatal warnings. +## Key-value pairs of code/error of code that should have non-fatal regexp warnings. ## my @warning = ( - 'm/\b*/' => '\b* matches null string many times in regex; marked by {#} in m/\b*{#}/', - - 'm/[:blank:]/' => 'POSIX syntax [: :] belongs inside character classes in regex; marked by {#} in m/[:blank:]{#}/', + 'm/\b*/' => '\b* matches null string many times {#} m/\b*{#}/', + + 'm/[:blank:]/' => 'POSIX syntax [: :] belongs inside character classes {#} m/[:blank:]{#}/', + + "m'[\\y]'" => 'Unrecognized escape \y in character class passed through {#} m/[\y{#}]/', + + 'm/[a-\d]/' => 'False [] range "a-\d" {#} m/[a-\d{#}]/', + 'm/[\w-x]/' => 'False [] range "\w-" {#} m/[\w-{#}x]/', + 'm/[a-\pM]/' => 'False [] range "a-\pM" {#} m/[a-\pM{#}]/', + 'm/[\pM-x]/' => 'False [] range "\pM-" {#} m/[\pM-{#}x]/', + "m'\\y'" => 'Unrecognized escape \y passed through {#} m/\y{#}/', + '/x{3,1}/' => 'Quantifier {n,m} with n > m can\'t match {#} m/x{3,1}{#}/', + '/\08/' => '\'\08\' resolved to \'\o{0}8\' {#} m/\08{#}/', + '/\018/' => '\'\018\' resolved to \'\o{1}8\' {#} m/\018{#}/', + '/[\08]/' => '\'\08\' resolved to \'\o{0}8\' {#} m/[\08{#}]/', + '/[\018]/' => '\'\018\' resolved to \'\o{1}8\' {#} m/[\018{#}]/', + '/\87/' => 'Unrecognized escape \8 passed through {#} m/\8{#}7/', + '/a\87/' => 'Unrecognized escape \8 passed through {#} m/a\8{#}7/', + '/a\97/' => 'Unrecognized escape \9 passed through {#} m/a\9{#}7/', + '/(?=a)*/' => '(?=a)* matches null string many times {#} m/(?=a)*{#}/', + 'my $x = \'\m\'; qr/a$x/' => 'Unrecognized escape \m passed through {#} m/a\m{#}/', + '/\q/' => 'Unrecognized escape \q passed through {#} m/\q{#}/', + '/\q{/' => 'Unrecognized escape \q{ passed through {#} m/\q{{#}/', + '/(?=a){1,3}/' => 'Quantifier unexpected on zero-length expression {#} m/(?=a){1,3}{#}/', + '/\_/' => "", + '/[\_\0]/' => "", + '/[\07]/' => "", + '/[\006]/' => "", + '/[\0005]/' => "", + '/[\8\9]/' => ['Unrecognized escape \8 in character class passed through {#} m/[\8{#}\9]/', + 'Unrecognized escape \9 in character class passed through {#} m/[\8\9{#}]/', + ], + '/[:alpha:]/' => 'POSIX syntax [: :] belongs inside character classes {#} m/[:alpha:]{#}/', + '/[:zog:]/' => 'POSIX syntax [: :] belongs inside character classes {#} m/[:zog:]{#}/', + '/[.zog.]/' => 'POSIX syntax [. .] belongs inside character classes {#} m/[.zog.]{#}/', + '/[a-b]/' => "", + '/[a-\d]/' => 'False [] range "a-\d" {#} m/[a-\d{#}]/', + '/[\d-b]/' => 'False [] range "\d-" {#} m/[\d-{#}b]/', + '/[\s-\d]/' => 'False [] range "\s-" {#} m/[\s-{#}\d]/', + '/[\d-\s]/' => 'False [] range "\d-" {#} m/[\d-{#}\s]/', + '/[a-[:digit:]]/' => 'False [] range "a-[:digit:]" {#} m/[a-[:digit:]{#}]/', + '/[[:digit:]-b]/' => 'False [] range "[:digit:]-" {#} m/[[:digit:]-{#}b]/', + '/[[:alpha:]-[:digit:]]/' => 'False [] range "[:alpha:]-" {#} m/[[:alpha:]-{#}[:digit:]]/', + '/[[:digit:]-[:alpha:]]/' => 'False [] range "[:digit:]-" {#} m/[[:digit:]-{#}[:alpha:]]/', + '/[a\zb]/' => 'Unrecognized escape \z in character class passed through {#} m/[a\z{#}b]/', + '/(?c)/' => 'Useless (?c) - use /gc modifier {#} m/(?c{#})/', + '/(?-c)/' => 'Useless (?-c) - don\'t use /gc modifier {#} m/(?-c{#})/', + '/(?g)/' => 'Useless (?g) - use /g modifier {#} m/(?g{#})/', + '/(?-g)/' => 'Useless (?-g) - don\'t use /g modifier {#} m/(?-g{#})/', + '/(?o)/' => 'Useless (?o) - use /o modifier {#} m/(?o{#})/', + '/(?-o)/' => 'Useless (?-o) - don\'t use /o modifier {#} m/(?-o{#})/', + '/(?g-o)/' => [ 'Useless (?g) - use /g modifier {#} m/(?g{#}-o)/', + 'Useless (?-o) - don\'t use /o modifier {#} m/(?g-o{#})/', + ], + '/(?g-c)/' => [ 'Useless (?g) - use /g modifier {#} m/(?g{#}-c)/', + 'Useless (?-c) - don\'t use /gc modifier {#} m/(?g-c{#})/', + ], + # (?c) means (?g) error won't be thrown + '/(?o-cg)/' => [ 'Useless (?o) - use /o modifier {#} m/(?o{#}-cg)/', + 'Useless (?-c) - don\'t use /gc modifier {#} m/(?o-c{#}g)/', + ], + '/(?ogc)/' => [ 'Useless (?o) - use /o modifier {#} m/(?o{#}gc)/', + 'Useless (?g) - use /g modifier {#} m/(?og{#}c)/', + 'Useless (?c) - use /gc modifier {#} m/(?ogc{#})/', + ], +); - "m'[\\y]'" => 'Unrecognized escape \y in character class passed through in regex; marked by {#} in m/[\y{#}]/', +my @experimental_regex_sets = ( + '/(?[ \t ])/' => 'The regex_sets feature is experimental {#} m/(?[{#} \t ])/', +); - 'm/[a-\d]/' => 'False [] range "a-\d" in regex; marked by {#} in m/[a-\d{#}]/', - 'm/[\w-x]/' => 'False [] range "\w-" in regex; marked by {#} in m/[\w-{#}x]/', - 'm/[a-\pM]/' => 'False [] range "a-\pM" in regex; marked by {#} in m/[a-\pM{#}]/', - 'm/[\pM-x]/' => 'False [] range "\pM-" in regex; marked by {#} in m/[\pM-{#}x]/', - "m'\\y'" => 'Unrecognized escape \y passed through in regex; marked by {#} in m/\y{#}/', +my @deprecated = ( + '/a\b{cde/' => '"\b{" is deprecated; use "\b\{" or "\b[{]" instead {#} m/a\{#}b{cde/', + '/a\B{cde/' => '"\B{" is deprecated; use "\B\{" or "\B[{]" instead {#} m/a\{#}B{cde/', + 'use utf8; /(?x)\……\…/' => 'Escape literal pattern white space under /x {#} m/(?x)\……{#}\…/', ); while (my ($regex, $expect) = splice @death, 0, 2) { my $expect = fixup_expect($expect); + no warnings 'experimental::regex_sets'; # skip the utf8 test on EBCDIC since they do not die next if $::IS_EBCDIC && $regex =~ /utf8/; warning_is(sub { $_ = "x"; eval $regex; - like($@, qr/\Q$expect/); - }, undef, "$regex died without any other warnings"); + like($@, qr/\Q$expect/, $regex); + }, undef, "... and died without any other warnings"); } -while (my ($regex, $expect) = splice @warning, 0, 2) { - my $expect = fixup_expect($expect); - warning_like(sub { - $_ = "x"; - eval $regex; - is($@, '', "$regex did not die"); - }, qr/\Q$expect/); +foreach my $ref (\@warning, \@experimental_regex_sets, \@deprecated) { + my $warning_type = ($ref == \@warning) + ? 'regexp' + : ($ref == \@deprecated) + ? 'regexp, deprecated' + : 'experimental::regex_sets'; + while (my ($regex, $expect) = splice @$ref, 0, 2) { + my @expect = fixup_expect($expect); + { + $_ = "x"; + no warnings; + eval $regex; + } + if (is($@, "", "$regex did not die")) { + my @got = capture_warnings(sub { + $_ = "x"; + eval $regex }); + my $count = @expect; + if (! is(scalar @got, scalar @expect, "... and gave expected number ($count) of warnings")) { + if (@got < @expect) { + $count = @got; + note "Expected warnings not gotten:\n\t" . join "\n\t", @expect[$count .. $#expect]; + } + else { + note "Unexpected warnings gotten:\n\t" . join("\n\t", @got[$count .. $#got]); + } + } + foreach my $i (0 .. $count - 1) { + if (like($got[$i], qr/\Q$expect[$i]/, "... and gave expected warning[$i]")) { + ok (0 == capture_warnings(sub { + $_ = "x"; + eval "no warnings '$warning_type'; $regex;" } + ), + "... and turning off '$warning_type' warnings suppressed it"); + } + } + } + } } done_testing();