/* * Copyright (c) 1999, 2019, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /** * @test * @summary tests RegExp framework (use -Dseed=X to set PRNG seed) * @author Mike McCloskey * @bug 4481568 4482696 4495089 4504687 4527731 4599621 4631553 4619345 * 4630911 4672616 4711773 4727935 4750573 4792284 4803197 4757029 4808962 * 4872664 4803179 4892980 4900747 4945394 4938995 4979006 4994840 4997476 * 5013885 5003322 4988891 5098443 5110268 6173522 4829857 5027748 6376940 * 6358731 6178785 6284152 6231989 6497148 6486934 6233084 6504326 6635133 * 6350801 6676425 6878475 6919132 6931676 6948903 6990617 7014645 7039066 * 7067045 7014640 7189363 8007395 8013252 8013254 8012646 8023647 6559590 * 8027645 8035076 8039124 8035975 8074678 6854417 8143854 8147531 7071819 * 8151481 4867170 7080302 6728861 6995635 6736245 4916384 6328855 6192895 * 6345469 6988218 6693451 7006761 8140212 8143282 8158482 8176029 8184706 * 8194667 8197462 8184692 8221431 8224789 * * @library /test/lib * @library /lib/testlibrary/java/lang * @build jdk.test.lib.RandomFactory * @run main RegExTest * @key randomness */ import java.io.BufferedReader; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.InputStreamReader; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.math.BigInteger; import java.nio.CharBuffer; import java.nio.file.Files; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import java.util.Random; import java.util.Scanner; import java.util.function.Function; import java.util.function.Predicate; import java.util.regex.Matcher; import java.util.regex.MatchResult; import java.util.regex.Pattern; import java.util.regex.PatternSyntaxException; import jdk.test.lib.RandomFactory; /** * This is a test class created to check the operation of * the Pattern and Matcher classes. */ public class RegExTest { private static Random generator = RandomFactory.getRandom(); private static boolean failure = false; private static int failCount = 0; private static String firstFailure = null; /** * Main to interpret arguments and run several tests. * */ public static void main(String[] args) throws Exception { // Most of the tests are in a file processFile("TestCases.txt"); //processFile("PerlCases.txt"); processFile("BMPTestCases.txt"); processFile("SupplementaryTestCases.txt"); // These test many randomly generated char patterns bm(); slice(); // These are hard to put into the file escapes(); blankInput(); // Substitition tests on randomly generated sequences globalSubstitute(); stringbufferSubstitute(); stringbuilderSubstitute(); substitutionBasher(); substitutionBasher2(); // Canonical Equivalence ceTest(); // Anchors anchorTest(); // boolean match calls matchesTest(); lookingAtTest(); // Pattern API patternMatchesTest(); // Misc lookbehindTest(); nullArgumentTest(); backRefTest(); groupCaptureTest(); caretTest(); charClassTest(); emptyPatternTest(); findIntTest(); group0Test(); longPatternTest(); octalTest(); ampersandTest(); negationTest(); splitTest(); appendTest(); caseFoldingTest(); commentsTest(); unixLinesTest(); replaceFirstTest(); gTest(); zTest(); serializeTest(); reluctantRepetitionTest(); multilineDollarTest(); dollarAtEndTest(); caretBetweenTerminatorsTest(); // This RFE rejected in Tiger numOccurrencesTest(); javaCharClassTest(); nonCaptureRepetitionTest(); notCapturedGroupCurlyMatchTest(); escapedSegmentTest(); literalPatternTest(); literalReplacementTest(); regionTest(); toStringTest(); negatedCharClassTest(); findFromTest(); boundsTest(); unicodeWordBoundsTest(); caretAtEndTest(); wordSearchTest(); hitEndTest(); toMatchResultTest(); toMatchResultTest2(); surrogatesInClassTest(); removeQEQuotingTest(); namedGroupCaptureTest(); nonBmpClassComplementTest(); unicodePropertiesTest(); unicodeHexNotationTest(); unicodeClassesTest(); unicodeCharacterNameTest(); horizontalAndVerticalWSTest(); linebreakTest(); branchTest(); groupCurlyNotFoundSuppTest(); groupCurlyBackoffTest(); patternAsPredicate(); patternAsMatchPredicate(); invalidFlags(); embeddedFlags(); grapheme(); expoBacktracking(); invalidGroupName(); illegalRepetitionRange(); if (failure) { throw new RuntimeException("RegExTest failed, 1st failure: " + firstFailure); } else { System.err.println("OKAY: All tests passed."); } } // Utility functions private static String getRandomAlphaString(int length) { StringBuffer buf = new StringBuffer(length); for (int i=0; i 0) { failure = true; if (firstFailure == null) { firstFailure = testName; } } failCount = 0; } /** * Converts ASCII alphabet characters [A-Za-z] in the given 's' to * supplementary characters. This method does NOT fully take care * of the regex syntax. */ private static String toSupplementaries(String s) { int length = s.length(); StringBuffer sb = new StringBuffer(length * 2); for (int i = 0; i < length; ) { char c = s.charAt(i++); if (c == '\\') { sb.append(c); if (i < length) { c = s.charAt(i++); sb.append(c); if (c == 'u') { // assume no syntax error sb.append(s.charAt(i++)); sb.append(s.charAt(i++)); sb.append(s.charAt(i++)); sb.append(s.charAt(i++)); } } } else if ((c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z')) { sb.append('\ud800').append((char)('\udc00'+c)); } else { sb.append(c); } } return sb.toString(); } // Regular expression tests // This is for bug 6178785 // Test if an expected NPE gets thrown when passing in a null argument private static boolean check(Runnable test) { try { test.run(); failCount++; return false; } catch (NullPointerException npe) { return true; } } private static void nullArgumentTest() { check(() -> Pattern.compile(null)); check(() -> Pattern.matches(null, null)); check(() -> Pattern.matches("xyz", null)); check(() -> Pattern.quote(null)); check(() -> Pattern.compile("xyz").split(null)); check(() -> Pattern.compile("xyz").matcher(null)); final Matcher m = Pattern.compile("xyz").matcher("xyz"); m.matches(); check(() -> m.appendTail((StringBuffer) null)); check(() -> m.appendTail((StringBuilder)null)); check(() -> m.replaceAll((String) null)); check(() -> m.replaceAll((Function)null)); check(() -> m.replaceFirst((String)null)); check(() -> m.replaceFirst((Function) null)); check(() -> m.appendReplacement((StringBuffer)null, null)); check(() -> m.appendReplacement((StringBuilder)null, null)); check(() -> m.reset(null)); check(() -> Matcher.quoteReplacement(null)); //check(() -> m.usePattern(null)); report("Null Argument"); } // This is for bug6635133 // Test if surrogate pair in Unicode escapes can be handled correctly. private static void surrogatesInClassTest() throws Exception { Pattern pattern = Pattern.compile("[\\ud834\\udd21-\\ud834\\udd24]"); Matcher matcher = pattern.matcher("\ud834\udd22"); if (!matcher.find()) failCount++; report("Surrogate pair in Unicode escape"); } // This is for bug6990617 // Test if Pattern.RemoveQEQuoting works correctly if the octal unicode // char encoding is only 2 or 3 digits instead of 4 and the first quoted // char is an octal digit. private static void removeQEQuotingTest() throws Exception { Pattern pattern = Pattern.compile("\\011\\Q1sometext\\E\\011\\Q2sometext\\E"); Matcher matcher = pattern.matcher("\t1sometext\t2sometext"); if (!matcher.find()) failCount++; report("Remove Q/E Quoting"); } // This is for bug 4988891 // Test toMatchResult to see that it is a copy of the Matcher // that is not affected by subsequent operations on the original private static void toMatchResultTest() throws Exception { Pattern pattern = Pattern.compile("squid"); Matcher matcher = pattern.matcher( "agiantsquidofdestinyasmallsquidoffate"); matcher.find(); int matcherStart1 = matcher.start(); MatchResult mr = matcher.toMatchResult(); if (mr == matcher) failCount++; int resultStart1 = mr.start(); if (matcherStart1 != resultStart1) failCount++; matcher.find(); int matcherStart2 = matcher.start(); int resultStart2 = mr.start(); if (matcherStart2 == resultStart2) failCount++; if (resultStart1 != resultStart2) failCount++; MatchResult mr2 = matcher.toMatchResult(); if (mr == mr2) failCount++; if (mr2.start() != matcherStart2) failCount++; report("toMatchResult is a copy"); } private static void checkExpectedISE(Runnable test) { try { test.run(); failCount++; } catch (IllegalStateException x) { } catch (IndexOutOfBoundsException xx) { failCount++; } } private static void checkExpectedIOOE(Runnable test) { try { test.run(); failCount++; } catch (IndexOutOfBoundsException x) {} } // This is for bug 8074678 // Test the result of toMatchResult throws ISE if no match is availble private static void toMatchResultTest2() throws Exception { Matcher matcher = Pattern.compile("nomatch").matcher("hello world"); matcher.find(); MatchResult mr = matcher.toMatchResult(); checkExpectedISE(() -> mr.start()); checkExpectedISE(() -> mr.start(2)); checkExpectedISE(() -> mr.end()); checkExpectedISE(() -> mr.end(2)); checkExpectedISE(() -> mr.group()); checkExpectedISE(() -> mr.group(2)); matcher = Pattern.compile("(match)").matcher("there is a match"); matcher.find(); MatchResult mr2 = matcher.toMatchResult(); checkExpectedIOOE(() -> mr2.start(2)); checkExpectedIOOE(() -> mr2.end(2)); checkExpectedIOOE(() -> mr2.group(2)); report("toMatchResult2 appropriate exceptions"); } // This is for bug 5013885 // Must test a slice to see if it reports hitEnd correctly private static void hitEndTest() throws Exception { // Basic test of Slice node Pattern p = Pattern.compile("^squidattack"); Matcher m = p.matcher("squack"); m.find(); if (m.hitEnd()) failCount++; m.reset("squid"); m.find(); if (!m.hitEnd()) failCount++; // Test Slice, SliceA and SliceU nodes for (int i=0; i<3; i++) { int flags = 0; if (i==1) flags = Pattern.CASE_INSENSITIVE; if (i==2) flags = Pattern.UNICODE_CASE; p = Pattern.compile("^abc", flags); m = p.matcher("ad"); m.find(); if (m.hitEnd()) failCount++; m.reset("ab"); m.find(); if (!m.hitEnd()) failCount++; } // Test Boyer-Moore node p = Pattern.compile("catattack"); m = p.matcher("attack"); m.find(); if (!m.hitEnd()) failCount++; p = Pattern.compile("catattack"); m = p.matcher("attackattackattackcatatta"); m.find(); if (!m.hitEnd()) failCount++; // 8184706: Matching u+0d at EOL against \R should hit-end p = Pattern.compile("...\\R"); m = p.matcher("cat" + (char)0x0a); m.find(); if (m.hitEnd()) failCount++; m = p.matcher("cat" + (char)0x0d); m.find(); if (!m.hitEnd()) failCount++; m = p.matcher("cat" + (char)0x0d + (char)0x0a); m.find(); if (m.hitEnd()) failCount++; report("hitEnd"); } // This is for bug 4997476 // It is weird code submitted by customer demonstrating a regression private static void wordSearchTest() throws Exception { String testString = new String("word1 word2 word3"); Pattern p = Pattern.compile("\\b"); Matcher m = p.matcher(testString); int position = 0; int start = 0; while (m.find(position)) { start = m.start(); if (start == testString.length()) break; if (m.find(start+1)) { position = m.start(); } else { position = testString.length(); } if (testString.substring(start, position).equals(" ")) continue; if (!testString.substring(start, position-1).startsWith("word")) failCount++; } report("Customer word search"); } // This is for bug 4994840 private static void caretAtEndTest() throws Exception { // Problem only occurs with multiline patterns // containing a beginning-of-line caret "^" followed // by an expression that also matches the empty string. Pattern pattern = Pattern.compile("^x?", Pattern.MULTILINE); Matcher matcher = pattern.matcher("\r"); matcher.find(); matcher.find(); report("Caret at end"); } // This test is for 4979006 // Check to see if word boundary construct properly handles unicode // non spacing marks private static void unicodeWordBoundsTest() throws Exception { String spaces = " "; String wordChar = "a"; String nsm = "\u030a"; assert (Character.getType('\u030a') == Character.NON_SPACING_MARK); Pattern pattern = Pattern.compile("\\b"); Matcher matcher = pattern.matcher(""); // S=other B=word character N=non spacing mark .=word boundary // SS.BB.SS String input = spaces + wordChar + wordChar + spaces; twoFindIndexes(input, matcher, 2, 4); // SS.BBN.SS input = spaces + wordChar +wordChar + nsm + spaces; twoFindIndexes(input, matcher, 2, 5); // SS.BN.SS input = spaces + wordChar + nsm + spaces; twoFindIndexes(input, matcher, 2, 4); // SS.BNN.SS input = spaces + wordChar + nsm + nsm + spaces; twoFindIndexes(input, matcher, 2, 5); // SSN.BB.SS input = spaces + nsm + wordChar + wordChar + spaces; twoFindIndexes(input, matcher, 3, 5); // SS.BNB.SS input = spaces + wordChar + nsm + wordChar + spaces; twoFindIndexes(input, matcher, 2, 5); // SSNNSS input = spaces + nsm + nsm + spaces; matcher.reset(input); if (matcher.find()) failCount++; // SSN.BBN.SS input = spaces + nsm + wordChar + wordChar + nsm + spaces; twoFindIndexes(input, matcher, 3, 6); report("Unicode word boundary"); } private static void twoFindIndexes(String input, Matcher matcher, int a, int b) throws Exception { matcher.reset(input); matcher.find(); if (matcher.start() != a) failCount++; matcher.find(); if (matcher.start() != b) failCount++; } // This test is for 6284152 static void check(String regex, String input, String[] expected) { List result = new ArrayList(); Pattern p = Pattern.compile(regex); Matcher m = p.matcher(input); while (m.find()) { result.add(m.group()); } if (!Arrays.asList(expected).equals(result)) failCount++; } private static void lookbehindTest() throws Exception { //Positive check("(?<=%.{0,5})foo\\d", "%foo1\n%bar foo2\n%bar foo3\n%blahblah foo4\nfoo5", new String[]{"foo1", "foo2", "foo3"}); //boundary at end of the lookbehind sub-regex should work consistently //with the boundary just after the lookbehind sub-regex check("(?<=.*\\b)foo", "abcd foo", new String[]{"foo"}); check("(?<=.*)\\bfoo", "abcd foo", new String[]{"foo"}); check("(?]"); Matcher matcher = pattern.matcher("\u203A"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("[^fr]"); matcher = pattern.matcher("a"); if (!matcher.find()) failCount++; matcher.reset("\u203A"); if (!matcher.find()) failCount++; String s = "for"; String result[] = s.split("[^fr]"); if (!result[0].equals("f")) failCount++; if (!result[1].equals("r")) failCount++; s = "f\u203Ar"; result = s.split("[^fr]"); if (!result[0].equals("f")) failCount++; if (!result[1].equals("r")) failCount++; // Test adding to bits, subtracting a node, then adding to bits again pattern = Pattern.compile("[^f\u203Ar]"); matcher = pattern.matcher("a"); if (!matcher.find()) failCount++; matcher.reset("f"); if (matcher.find()) failCount++; matcher.reset("\u203A"); if (matcher.find()) failCount++; matcher.reset("r"); if (matcher.find()) failCount++; matcher.reset("\u203B"); if (!matcher.find()) failCount++; // Test subtracting a node, adding to bits, subtracting again pattern = Pattern.compile("[^\u203Ar\u203B]"); matcher = pattern.matcher("a"); if (!matcher.find()) failCount++; matcher.reset("\u203A"); if (matcher.find()) failCount++; matcher.reset("r"); if (matcher.find()) failCount++; matcher.reset("\u203B"); if (matcher.find()) failCount++; matcher.reset("\u203C"); if (!matcher.find()) failCount++; report("Negated Character Class"); } // This test is for 4628291 private static void toStringTest() throws Exception { Pattern pattern = Pattern.compile("b+"); if (pattern.toString() != "b+") failCount++; Matcher matcher = pattern.matcher("aaabbbccc"); String matcherString = matcher.toString(); // unspecified matcher.find(); matcherString = matcher.toString(); // unspecified matcher.region(0,3); matcherString = matcher.toString(); // unspecified matcher.reset(); matcherString = matcher.toString(); // unspecified report("toString"); } // This test is for 4808962 private static void literalPatternTest() throws Exception { int flags = Pattern.LITERAL; Pattern pattern = Pattern.compile("abc\\t$^", flags); check(pattern, "abc\\t$^", true); pattern = Pattern.compile(Pattern.quote("abc\\t$^")); check(pattern, "abc\\t$^", true); pattern = Pattern.compile("\\Qa^$bcabc\\E", flags); check(pattern, "\\Qa^$bcabc\\E", true); check(pattern, "a^$bcabc", false); pattern = Pattern.compile("\\\\Q\\\\E"); check(pattern, "\\Q\\E", true); pattern = Pattern.compile("\\Qabc\\Eefg\\\\Q\\\\Ehij"); check(pattern, "abcefg\\Q\\Ehij", true); pattern = Pattern.compile("\\\\\\Q\\\\E"); check(pattern, "\\\\\\\\", true); pattern = Pattern.compile(Pattern.quote("\\Qa^$bcabc\\E")); check(pattern, "\\Qa^$bcabc\\E", true); check(pattern, "a^$bcabc", false); pattern = Pattern.compile(Pattern.quote("\\Qabc\\Edef")); check(pattern, "\\Qabc\\Edef", true); check(pattern, "abcdef", false); pattern = Pattern.compile(Pattern.quote("abc\\Edef")); check(pattern, "abc\\Edef", true); check(pattern, "abcdef", false); pattern = Pattern.compile(Pattern.quote("\\E")); check(pattern, "\\E", true); pattern = Pattern.compile("((((abc.+?:)", flags); check(pattern, "((((abc.+?:)", true); flags |= Pattern.MULTILINE; pattern = Pattern.compile("^cat$", flags); check(pattern, "abc^cat$def", true); check(pattern, "cat", false); flags |= Pattern.CASE_INSENSITIVE; pattern = Pattern.compile("abcdef", flags); check(pattern, "ABCDEF", true); check(pattern, "AbCdEf", true); flags |= Pattern.DOTALL; pattern = Pattern.compile("a...b", flags); check(pattern, "A...b", true); check(pattern, "Axxxb", false); flags |= Pattern.CANON_EQ; Pattern p = Pattern.compile("testa\u030a", flags); check(pattern, "testa\u030a", false); check(pattern, "test\u00e5", false); // Supplementary character test flags = Pattern.LITERAL; pattern = Pattern.compile(toSupplementaries("abc\\t$^"), flags); check(pattern, toSupplementaries("abc\\t$^"), true); pattern = Pattern.compile(Pattern.quote(toSupplementaries("abc\\t$^"))); check(pattern, toSupplementaries("abc\\t$^"), true); pattern = Pattern.compile(toSupplementaries("\\Qa^$bcabc\\E"), flags); check(pattern, toSupplementaries("\\Qa^$bcabc\\E"), true); check(pattern, toSupplementaries("a^$bcabc"), false); pattern = Pattern.compile(Pattern.quote(toSupplementaries("\\Qa^$bcabc\\E"))); check(pattern, toSupplementaries("\\Qa^$bcabc\\E"), true); check(pattern, toSupplementaries("a^$bcabc"), false); pattern = Pattern.compile(Pattern.quote(toSupplementaries("\\Qabc\\Edef"))); check(pattern, toSupplementaries("\\Qabc\\Edef"), true); check(pattern, toSupplementaries("abcdef"), false); pattern = Pattern.compile(Pattern.quote(toSupplementaries("abc\\Edef"))); check(pattern, toSupplementaries("abc\\Edef"), true); check(pattern, toSupplementaries("abcdef"), false); pattern = Pattern.compile(toSupplementaries("((((abc.+?:)"), flags); check(pattern, toSupplementaries("((((abc.+?:)"), true); flags |= Pattern.MULTILINE; pattern = Pattern.compile(toSupplementaries("^cat$"), flags); check(pattern, toSupplementaries("abc^cat$def"), true); check(pattern, toSupplementaries("cat"), false); flags |= Pattern.DOTALL; // note: this is case-sensitive. pattern = Pattern.compile(toSupplementaries("a...b"), flags); check(pattern, toSupplementaries("a...b"), true); check(pattern, toSupplementaries("axxxb"), false); flags |= Pattern.CANON_EQ; String t = toSupplementaries("test"); p = Pattern.compile(t + "a\u030a", flags); check(pattern, t + "a\u030a", false); check(pattern, t + "\u00e5", false); report("Literal pattern"); } // This test is for 4803179 // This test is also for 4808962, replacement parts private static void literalReplacementTest() throws Exception { int flags = Pattern.LITERAL; Pattern pattern = Pattern.compile("abc", flags); Matcher matcher = pattern.matcher("zzzabczzz"); String replaceTest = "$0"; String result = matcher.replaceAll(replaceTest); if (!result.equals("zzzabczzz")) failCount++; matcher.reset(); String literalReplacement = matcher.quoteReplacement(replaceTest); result = matcher.replaceAll(literalReplacement); if (!result.equals("zzz$0zzz")) failCount++; matcher.reset(); replaceTest = "\\t$\\$"; literalReplacement = matcher.quoteReplacement(replaceTest); result = matcher.replaceAll(literalReplacement); if (!result.equals("zzz\\t$\\$zzz")) failCount++; // Supplementary character test pattern = Pattern.compile(toSupplementaries("abc"), flags); matcher = pattern.matcher(toSupplementaries("zzzabczzz")); replaceTest = "$0"; result = matcher.replaceAll(replaceTest); if (!result.equals(toSupplementaries("zzzabczzz"))) failCount++; matcher.reset(); literalReplacement = matcher.quoteReplacement(replaceTest); result = matcher.replaceAll(literalReplacement); if (!result.equals(toSupplementaries("zzz$0zzz"))) failCount++; matcher.reset(); replaceTest = "\\t$\\$"; literalReplacement = matcher.quoteReplacement(replaceTest); result = matcher.replaceAll(literalReplacement); if (!result.equals(toSupplementaries("zzz\\t$\\$zzz"))) failCount++; // IAE should be thrown if backslash or '$' is the last character // in replacement string try { "\uac00".replaceAll("\uac00", "$"); failCount++; } catch (IllegalArgumentException iie) { } catch (Exception e) { failCount++; } try { "\uac00".replaceAll("\uac00", "\\"); failCount++; } catch (IllegalArgumentException iie) { } catch (Exception e) { failCount++; } report("Literal replacement"); } // This test is for 4757029 private static void regionTest() throws Exception { Pattern pattern = Pattern.compile("abc"); Matcher matcher = pattern.matcher("abcdefabc"); matcher.region(0,9); if (!matcher.find()) failCount++; if (!matcher.find()) failCount++; matcher.region(0,3); if (!matcher.find()) failCount++; matcher.region(3,6); if (matcher.find()) failCount++; matcher.region(0,2); if (matcher.find()) failCount++; expectRegionFail(matcher, 1, -1); expectRegionFail(matcher, -1, -1); expectRegionFail(matcher, -1, 1); expectRegionFail(matcher, 5, 3); expectRegionFail(matcher, 5, 12); expectRegionFail(matcher, 12, 12); pattern = Pattern.compile("^abc$"); matcher = pattern.matcher("zzzabczzz"); matcher.region(0,9); if (matcher.find()) failCount++; matcher.region(3,6); if (!matcher.find()) failCount++; matcher.region(3,6); matcher.useAnchoringBounds(false); if (matcher.find()) failCount++; // Supplementary character test pattern = Pattern.compile(toSupplementaries("abc")); matcher = pattern.matcher(toSupplementaries("abcdefabc")); matcher.region(0,9*2); if (!matcher.find()) failCount++; if (!matcher.find()) failCount++; matcher.region(0,3*2); if (!matcher.find()) failCount++; matcher.region(1,3*2); if (matcher.find()) failCount++; matcher.region(3*2,6*2); if (matcher.find()) failCount++; matcher.region(0,2*2); if (matcher.find()) failCount++; matcher.region(0,2*2+1); if (matcher.find()) failCount++; expectRegionFail(matcher, 1*2, -1); expectRegionFail(matcher, -1, -1); expectRegionFail(matcher, -1, 1*2); expectRegionFail(matcher, 5*2, 3*2); expectRegionFail(matcher, 5*2, 12*2); expectRegionFail(matcher, 12*2, 12*2); pattern = Pattern.compile(toSupplementaries("^abc$")); matcher = pattern.matcher(toSupplementaries("zzzabczzz")); matcher.region(0,9*2); if (matcher.find()) failCount++; matcher.region(3*2,6*2); if (!matcher.find()) failCount++; matcher.region(3*2+1,6*2); if (matcher.find()) failCount++; matcher.region(3*2,6*2-1); if (matcher.find()) failCount++; matcher.region(3*2,6*2); matcher.useAnchoringBounds(false); if (matcher.find()) failCount++; report("Regions"); } private static void expectRegionFail(Matcher matcher, int index1, int index2) { try { matcher.region(index1, index2); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Correct result } catch (IllegalStateException ise) { // Correct result } } // This test is for 4803197 private static void escapedSegmentTest() throws Exception { Pattern pattern = Pattern.compile("\\Qdir1\\dir2\\E"); check(pattern, "dir1\\dir2", true); pattern = Pattern.compile("\\Qdir1\\dir2\\\\E"); check(pattern, "dir1\\dir2\\", true); pattern = Pattern.compile("(\\Qdir1\\dir2\\\\E)"); check(pattern, "dir1\\dir2\\", true); // Supplementary character test pattern = Pattern.compile(toSupplementaries("\\Qdir1\\dir2\\E")); check(pattern, toSupplementaries("dir1\\dir2"), true); pattern = Pattern.compile(toSupplementaries("\\Qdir1\\dir2")+"\\\\E"); check(pattern, toSupplementaries("dir1\\dir2\\"), true); pattern = Pattern.compile(toSupplementaries("(\\Qdir1\\dir2")+"\\\\E)"); check(pattern, toSupplementaries("dir1\\dir2\\"), true); report("Escaped segment"); } // This test is for 4792284 private static void nonCaptureRepetitionTest() throws Exception { String input = "abcdefgh;"; String[] patterns = new String[] { "(?:\\w{4})+;", "(?:\\w{8})*;", "(?:\\w{2}){2,4};", "(?:\\w{4}){2,};", // only matches the ".*?(?:\\w{5})+;", // specified minimum ".*?(?:\\w{9})*;", // number of reps - OK "(?:\\w{4})+?;", // lazy repetition - OK "(?:\\w{4})++;", // possessive repetition - OK "(?:\\w{2,}?)+;", // non-deterministic - OK "(\\w{4})+;", // capturing group - OK }; for (int i = 0; i < patterns.length; i++) { // Check find() check(patterns[i], 0, input, input, true); // Check matches() Pattern p = Pattern.compile(patterns[i]); Matcher m = p.matcher(input); if (m.matches()) { if (!m.group(0).equals(input)) failCount++; } else { failCount++; } } report("Non capturing repetition"); } // This test is for 6358731 private static void notCapturedGroupCurlyMatchTest() throws Exception { Pattern pattern = Pattern.compile("(abc)+|(abcd)+"); Matcher matcher = pattern.matcher("abcd"); if (!matcher.matches() || matcher.group(1) != null || !matcher.group(2).equals("abcd")) { failCount++; } report("Not captured GroupCurly"); } // This test is for 4706545 private static void javaCharClassTest() throws Exception { for (int i=0; i<1000; i++) { char c = (char)generator.nextInt(); check("{javaLowerCase}", c, Character.isLowerCase(c)); check("{javaUpperCase}", c, Character.isUpperCase(c)); check("{javaUpperCase}+", c, Character.isUpperCase(c)); check("{javaTitleCase}", c, Character.isTitleCase(c)); check("{javaDigit}", c, Character.isDigit(c)); check("{javaDefined}", c, Character.isDefined(c)); check("{javaLetter}", c, Character.isLetter(c)); check("{javaLetterOrDigit}", c, Character.isLetterOrDigit(c)); check("{javaJavaIdentifierStart}", c, Character.isJavaIdentifierStart(c)); check("{javaJavaIdentifierPart}", c, Character.isJavaIdentifierPart(c)); check("{javaUnicodeIdentifierStart}", c, Character.isUnicodeIdentifierStart(c)); check("{javaUnicodeIdentifierPart}", c, Character.isUnicodeIdentifierPart(c)); check("{javaIdentifierIgnorable}", c, Character.isIdentifierIgnorable(c)); check("{javaSpaceChar}", c, Character.isSpaceChar(c)); check("{javaWhitespace}", c, Character.isWhitespace(c)); check("{javaISOControl}", c, Character.isISOControl(c)); check("{javaMirrored}", c, Character.isMirrored(c)); } // Supplementary character test for (int i=0; i<1000; i++) { int c = generator.nextInt(Character.MAX_CODE_POINT - Character.MIN_SUPPLEMENTARY_CODE_POINT) + Character.MIN_SUPPLEMENTARY_CODE_POINT; check("{javaLowerCase}", c, Character.isLowerCase(c)); check("{javaUpperCase}", c, Character.isUpperCase(c)); check("{javaUpperCase}+", c, Character.isUpperCase(c)); check("{javaTitleCase}", c, Character.isTitleCase(c)); check("{javaDigit}", c, Character.isDigit(c)); check("{javaDefined}", c, Character.isDefined(c)); check("{javaLetter}", c, Character.isLetter(c)); check("{javaLetterOrDigit}", c, Character.isLetterOrDigit(c)); check("{javaJavaIdentifierStart}", c, Character.isJavaIdentifierStart(c)); check("{javaJavaIdentifierPart}", c, Character.isJavaIdentifierPart(c)); check("{javaUnicodeIdentifierStart}", c, Character.isUnicodeIdentifierStart(c)); check("{javaUnicodeIdentifierPart}", c, Character.isUnicodeIdentifierPart(c)); check("{javaIdentifierIgnorable}", c, Character.isIdentifierIgnorable(c)); check("{javaSpaceChar}", c, Character.isSpaceChar(c)); check("{javaWhitespace}", c, Character.isWhitespace(c)); check("{javaISOControl}", c, Character.isISOControl(c)); check("{javaMirrored}", c, Character.isMirrored(c)); } report("Java character classes"); } // This test is for 4523620 /* private static void numOccurrencesTest() throws Exception { Pattern pattern = Pattern.compile("aaa"); if (pattern.numOccurrences("aaaaaa", false) != 2) failCount++; if (pattern.numOccurrences("aaaaaa", true) != 4) failCount++; pattern = Pattern.compile("^"); if (pattern.numOccurrences("aaaaaa", false) != 1) failCount++; if (pattern.numOccurrences("aaaaaa", true) != 1) failCount++; report("Number of Occurrences"); } */ // This test is for 4776374 private static void caretBetweenTerminatorsTest() throws Exception { int flags1 = Pattern.DOTALL; int flags2 = Pattern.DOTALL | Pattern.UNIX_LINES; int flags3 = Pattern.DOTALL | Pattern.UNIX_LINES | Pattern.MULTILINE; int flags4 = Pattern.DOTALL | Pattern.MULTILINE; check("^....", flags1, "test\ntest", "test", true); check(".....^", flags1, "test\ntest", "test", false); check(".....^", flags1, "test\n", "test", false); check("....^", flags1, "test\r\n", "test", false); check("^....", flags2, "test\ntest", "test", true); check("....^", flags2, "test\ntest", "test", false); check(".....^", flags2, "test\n", "test", false); check("....^", flags2, "test\r\n", "test", false); check("^....", flags3, "test\ntest", "test", true); check(".....^", flags3, "test\ntest", "test\n", true); check(".....^", flags3, "test\u0085test", "test\u0085", false); check(".....^", flags3, "test\n", "test", false); check(".....^", flags3, "test\r\n", "test", false); check("......^", flags3, "test\r\ntest", "test\r\n", true); check("^....", flags4, "test\ntest", "test", true); check(".....^", flags3, "test\ntest", "test\n", true); check(".....^", flags4, "test\u0085test", "test\u0085", true); check(".....^", flags4, "test\n", "test\n", false); check(".....^", flags4, "test\r\n", "test\r", false); // Supplementary character test String t = toSupplementaries("test"); check("^....", flags1, t+"\n"+t, t, true); check(".....^", flags1, t+"\n"+t, t, false); check(".....^", flags1, t+"\n", t, false); check("....^", flags1, t+"\r\n", t, false); check("^....", flags2, t+"\n"+t, t, true); check("....^", flags2, t+"\n"+t, t, false); check(".....^", flags2, t+"\n", t, false); check("....^", flags2, t+"\r\n", t, false); check("^....", flags3, t+"\n"+t, t, true); check(".....^", flags3, t+"\n"+t, t+"\n", true); check(".....^", flags3, t+"\u0085"+t, t+"\u0085", false); check(".....^", flags3, t+"\n", t, false); check(".....^", flags3, t+"\r\n", t, false); check("......^", flags3, t+"\r\n"+t, t+"\r\n", true); check("^....", flags4, t+"\n"+t, t, true); check(".....^", flags3, t+"\n"+t, t+"\n", true); check(".....^", flags4, t+"\u0085"+t, t+"\u0085", true); check(".....^", flags4, t+"\n", t+"\n", false); check(".....^", flags4, t+"\r\n", t+"\r", false); report("Caret between terminators"); } // This test is for 4727935 private static void dollarAtEndTest() throws Exception { int flags1 = Pattern.DOTALL; int flags2 = Pattern.DOTALL | Pattern.UNIX_LINES; int flags3 = Pattern.DOTALL | Pattern.MULTILINE; check("....$", flags1, "test\n", "test", true); check("....$", flags1, "test\r\n", "test", true); check(".....$", flags1, "test\n", "test\n", true); check(".....$", flags1, "test\u0085", "test\u0085", true); check("....$", flags1, "test\u0085", "test", true); check("....$", flags2, "test\n", "test", true); check(".....$", flags2, "test\n", "test\n", true); check(".....$", flags2, "test\u0085", "test\u0085", true); check("....$", flags2, "test\u0085", "est\u0085", true); check("....$.blah", flags3, "test\nblah", "test\nblah", true); check(".....$.blah", flags3, "test\n\nblah", "test\n\nblah", true); check("....$blah", flags3, "test\nblah", "!!!!", false); check(".....$blah", flags3, "test\nblah", "!!!!", false); // Supplementary character test String t = toSupplementaries("test"); String b = toSupplementaries("blah"); check("....$", flags1, t+"\n", t, true); check("....$", flags1, t+"\r\n", t, true); check(".....$", flags1, t+"\n", t+"\n", true); check(".....$", flags1, t+"\u0085", t+"\u0085", true); check("....$", flags1, t+"\u0085", t, true); check("....$", flags2, t+"\n", t, true); check(".....$", flags2, t+"\n", t+"\n", true); check(".....$", flags2, t+"\u0085", t+"\u0085", true); check("....$", flags2, t+"\u0085", toSupplementaries("est\u0085"), true); check("....$."+b, flags3, t+"\n"+b, t+"\n"+b, true); check(".....$."+b, flags3, t+"\n\n"+b, t+"\n\n"+b, true); check("....$"+b, flags3, t+"\n"+b, "!!!!", false); check(".....$"+b, flags3, t+"\n"+b, "!!!!", false); report("Dollar at End"); } // This test is for 4711773 private static void multilineDollarTest() throws Exception { Pattern findCR = Pattern.compile("$", Pattern.MULTILINE); Matcher matcher = findCR.matcher("first bit\nsecond bit"); matcher.find(); if (matcher.start(0) != 9) failCount++; matcher.find(); if (matcher.start(0) != 20) failCount++; // Supplementary character test matcher = findCR.matcher(toSupplementaries("first bit\n second bit")); // double BMP chars matcher.find(); if (matcher.start(0) != 9*2) failCount++; matcher.find(); if (matcher.start(0) != 20*2) failCount++; report("Multiline Dollar"); } private static void reluctantRepetitionTest() throws Exception { Pattern p = Pattern.compile("1(\\s\\S+?){1,3}?[\\s,]2"); check(p, "1 word word word 2", true); check(p, "1 wor wo w 2", true); check(p, "1 word word 2", true); check(p, "1 word 2", true); check(p, "1 wo w w 2", true); check(p, "1 wo w 2", true); check(p, "1 wor w 2", true); p = Pattern.compile("([a-z])+?c"); Matcher m = p.matcher("ababcdefdec"); check(m, "ababc"); // Supplementary character test p = Pattern.compile(toSupplementaries("([a-z])+?c")); m = p.matcher(toSupplementaries("ababcdefdec")); check(m, toSupplementaries("ababc")); report("Reluctant Repetition"); } private static Pattern serializedPattern(Pattern p) throws Exception { ByteArrayOutputStream baos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(baos); oos.writeObject(p); oos.close(); try (ObjectInputStream ois = new ObjectInputStream( new ByteArrayInputStream(baos.toByteArray()))) { return (Pattern)ois.readObject(); } } private static void serializeTest() throws Exception { String patternStr = "(b)"; String matchStr = "b"; Pattern pattern = Pattern.compile(patternStr); Pattern serializedPattern = serializedPattern(pattern); Matcher matcher = serializedPattern.matcher(matchStr); if (!matcher.matches()) failCount++; if (matcher.groupCount() != 1) failCount++; pattern = Pattern.compile("a(?-i)b", Pattern.CASE_INSENSITIVE); serializedPattern = serializedPattern(pattern); if (!serializedPattern.matcher("Ab").matches()) failCount++; if (serializedPattern.matcher("AB").matches()) failCount++; report("Serialization"); } private static void gTest() { Pattern pattern = Pattern.compile("\\G\\w"); Matcher matcher = pattern.matcher("abc#x#x"); matcher.find(); matcher.find(); matcher.find(); if (matcher.find()) failCount++; pattern = Pattern.compile("\\GA*"); matcher = pattern.matcher("1A2AA3"); matcher.find(); if (matcher.find()) failCount++; pattern = Pattern.compile("\\GA*"); matcher = pattern.matcher("1A2AA3"); if (!matcher.find(1)) failCount++; matcher.find(); if (matcher.find()) failCount++; report("\\G"); } private static void zTest() { Pattern pattern = Pattern.compile("foo\\Z"); // Positives check(pattern, "foo\u0085", true); check(pattern, "foo\u2028", true); check(pattern, "foo\u2029", true); check(pattern, "foo\n", true); check(pattern, "foo\r", true); check(pattern, "foo\r\n", true); // Negatives check(pattern, "fooo", false); check(pattern, "foo\n\r", false); pattern = Pattern.compile("foo\\Z", Pattern.UNIX_LINES); // Positives check(pattern, "foo", true); check(pattern, "foo\n", true); // Negatives check(pattern, "foo\r", false); check(pattern, "foo\u0085", false); check(pattern, "foo\u2028", false); check(pattern, "foo\u2029", false); report("\\Z"); } private static void replaceFirstTest() { Pattern pattern = Pattern.compile("(ab)(c*)"); Matcher matcher = pattern.matcher("abccczzzabcczzzabccc"); if (!matcher.replaceFirst("test").equals("testzzzabcczzzabccc")) failCount++; matcher.reset("zzzabccczzzabcczzzabccczzz"); if (!matcher.replaceFirst("test").equals("zzztestzzzabcczzzabccczzz")) failCount++; matcher.reset("zzzabccczzzabcczzzabccczzz"); String result = matcher.replaceFirst("$1"); if (!result.equals("zzzabzzzabcczzzabccczzz")) failCount++; matcher.reset("zzzabccczzzabcczzzabccczzz"); result = matcher.replaceFirst("$2"); if (!result.equals("zzzccczzzabcczzzabccczzz")) failCount++; pattern = Pattern.compile("a*"); matcher = pattern.matcher("aaaaaaaaaa"); if (!matcher.replaceFirst("test").equals("test")) failCount++; pattern = Pattern.compile("a+"); matcher = pattern.matcher("zzzaaaaaaaaaa"); if (!matcher.replaceFirst("test").equals("zzztest")) failCount++; // Supplementary character test pattern = Pattern.compile(toSupplementaries("(ab)(c*)")); matcher = pattern.matcher(toSupplementaries("abccczzzabcczzzabccc")); if (!matcher.replaceFirst(toSupplementaries("test")) .equals(toSupplementaries("testzzzabcczzzabccc"))) failCount++; matcher.reset(toSupplementaries("zzzabccczzzabcczzzabccczzz")); if (!matcher.replaceFirst(toSupplementaries("test")). equals(toSupplementaries("zzztestzzzabcczzzabccczzz"))) failCount++; matcher.reset(toSupplementaries("zzzabccczzzabcczzzabccczzz")); result = matcher.replaceFirst("$1"); if (!result.equals(toSupplementaries("zzzabzzzabcczzzabccczzz"))) failCount++; matcher.reset(toSupplementaries("zzzabccczzzabcczzzabccczzz")); result = matcher.replaceFirst("$2"); if (!result.equals(toSupplementaries("zzzccczzzabcczzzabccczzz"))) failCount++; pattern = Pattern.compile(toSupplementaries("a*")); matcher = pattern.matcher(toSupplementaries("aaaaaaaaaa")); if (!matcher.replaceFirst(toSupplementaries("test")).equals(toSupplementaries("test"))) failCount++; pattern = Pattern.compile(toSupplementaries("a+")); matcher = pattern.matcher(toSupplementaries("zzzaaaaaaaaaa")); if (!matcher.replaceFirst(toSupplementaries("test")).equals(toSupplementaries("zzztest"))) failCount++; report("Replace First"); } private static void unixLinesTest() { Pattern pattern = Pattern.compile(".*"); Matcher matcher = pattern.matcher("aa\u2028blah"); matcher.find(); if (!matcher.group(0).equals("aa")) failCount++; pattern = Pattern.compile(".*", Pattern.UNIX_LINES); matcher = pattern.matcher("aa\u2028blah"); matcher.find(); if (!matcher.group(0).equals("aa\u2028blah")) failCount++; pattern = Pattern.compile("[az]$", Pattern.MULTILINE | Pattern.UNIX_LINES); matcher = pattern.matcher("aa\u2028zz"); check(matcher, "a\u2028", false); // Supplementary character test pattern = Pattern.compile(".*"); matcher = pattern.matcher(toSupplementaries("aa\u2028blah")); matcher.find(); if (!matcher.group(0).equals(toSupplementaries("aa"))) failCount++; pattern = Pattern.compile(".*", Pattern.UNIX_LINES); matcher = pattern.matcher(toSupplementaries("aa\u2028blah")); matcher.find(); if (!matcher.group(0).equals(toSupplementaries("aa\u2028blah"))) failCount++; pattern = Pattern.compile(toSupplementaries("[az]$"), Pattern.MULTILINE | Pattern.UNIX_LINES); matcher = pattern.matcher(toSupplementaries("aa\u2028zz")); check(matcher, toSupplementaries("a\u2028"), false); report("Unix Lines"); } private static void commentsTest() { int flags = Pattern.COMMENTS; Pattern pattern = Pattern.compile("aa \\# aa", flags); Matcher matcher = pattern.matcher("aa#aa"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("aa # blah", flags); matcher = pattern.matcher("aa"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("aa blah", flags); matcher = pattern.matcher("aablah"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("aa # blah blech ", flags); matcher = pattern.matcher("aa"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("aa # blah\n ", flags); matcher = pattern.matcher("aa"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("aa # blah\nbc # blech", flags); matcher = pattern.matcher("aabc"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("aa # blah\nbc# blech", flags); matcher = pattern.matcher("aabc"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("aa # blah\nbc\\# blech", flags); matcher = pattern.matcher("aabc#blech"); if (!matcher.matches()) failCount++; // Supplementary character test pattern = Pattern.compile(toSupplementaries("aa \\# aa"), flags); matcher = pattern.matcher(toSupplementaries("aa#aa")); if (!matcher.matches()) failCount++; pattern = Pattern.compile(toSupplementaries("aa # blah"), flags); matcher = pattern.matcher(toSupplementaries("aa")); if (!matcher.matches()) failCount++; pattern = Pattern.compile(toSupplementaries("aa blah"), flags); matcher = pattern.matcher(toSupplementaries("aablah")); if (!matcher.matches()) failCount++; pattern = Pattern.compile(toSupplementaries("aa # blah blech "), flags); matcher = pattern.matcher(toSupplementaries("aa")); if (!matcher.matches()) failCount++; pattern = Pattern.compile(toSupplementaries("aa # blah\n "), flags); matcher = pattern.matcher(toSupplementaries("aa")); if (!matcher.matches()) failCount++; pattern = Pattern.compile(toSupplementaries("aa # blah\nbc # blech"), flags); matcher = pattern.matcher(toSupplementaries("aabc")); if (!matcher.matches()) failCount++; pattern = Pattern.compile(toSupplementaries("aa # blah\nbc# blech"), flags); matcher = pattern.matcher(toSupplementaries("aabc")); if (!matcher.matches()) failCount++; pattern = Pattern.compile(toSupplementaries("aa # blah\nbc\\# blech"), flags); matcher = pattern.matcher(toSupplementaries("aabc#blech")); if (!matcher.matches()) failCount++; report("Comments"); } private static void caseFoldingTest() { // bug 4504687 int flags = Pattern.CASE_INSENSITIVE | Pattern.UNICODE_CASE; Pattern pattern = Pattern.compile("aa", flags); Matcher matcher = pattern.matcher("ab"); if (matcher.matches()) failCount++; pattern = Pattern.compile("aA", flags); matcher = pattern.matcher("ab"); if (matcher.matches()) failCount++; pattern = Pattern.compile("aa", flags); matcher = pattern.matcher("aB"); if (matcher.matches()) failCount++; matcher = pattern.matcher("Ab"); if (matcher.matches()) failCount++; // ASCII "a" // Latin-1 Supplement "a" + grave // Cyrillic "a" String[] patterns = new String[] { //single "a", "\u00e0", "\u0430", //slice "ab", "\u00e0\u00e1", "\u0430\u0431", //class single "[a]", "[\u00e0]", "[\u0430]", //class range "[a-b]", "[\u00e0-\u00e5]", "[\u0430-\u0431]", //back reference "(a)\\1", "(\u00e0)\\1", "(\u0430)\\1" }; String[] texts = new String[] { "A", "\u00c0", "\u0410", "AB", "\u00c0\u00c1", "\u0410\u0411", "A", "\u00c0", "\u0410", "B", "\u00c2", "\u0411", "aA", "\u00e0\u00c0", "\u0430\u0410" }; boolean[] expected = new boolean[] { true, false, false, true, false, false, true, false, false, true, false, false, true, false, false }; flags = Pattern.CASE_INSENSITIVE; for (int i = 0; i < patterns.length; i++) { pattern = Pattern.compile(patterns[i], flags); matcher = pattern.matcher(texts[i]); if (matcher.matches() != expected[i]) { System.out.println("<1> Failed at " + i); failCount++; } } flags = Pattern.CASE_INSENSITIVE | Pattern.UNICODE_CASE; for (int i = 0; i < patterns.length; i++) { pattern = Pattern.compile(patterns[i], flags); matcher = pattern.matcher(texts[i]); if (!matcher.matches()) { System.out.println("<2> Failed at " + i); failCount++; } } // flag unicode_case alone should do nothing flags = Pattern.UNICODE_CASE; for (int i = 0; i < patterns.length; i++) { pattern = Pattern.compile(patterns[i], flags); matcher = pattern.matcher(texts[i]); if (matcher.matches()) { System.out.println("<3> Failed at " + i); failCount++; } } // Special cases: i, I, u+0131 and u+0130 flags = Pattern.UNICODE_CASE | Pattern.CASE_INSENSITIVE; pattern = Pattern.compile("[h-j]+", flags); if (!pattern.matcher("\u0131\u0130").matches()) failCount++; report("Case Folding"); } private static void appendTest() { Pattern pattern = Pattern.compile("(ab)(cd)"); Matcher matcher = pattern.matcher("abcd"); String result = matcher.replaceAll("$2$1"); if (!result.equals("cdab")) failCount++; String s1 = "Swap all: first = 123, second = 456"; String s2 = "Swap one: first = 123, second = 456"; String r = "$3$2$1"; pattern = Pattern.compile("([a-z]+)( *= *)([0-9]+)"); matcher = pattern.matcher(s1); result = matcher.replaceAll(r); if (!result.equals("Swap all: 123 = first, 456 = second")) failCount++; matcher = pattern.matcher(s2); if (matcher.find()) { StringBuffer sb = new StringBuffer(); matcher.appendReplacement(sb, r); matcher.appendTail(sb); result = sb.toString(); if (!result.equals("Swap one: 123 = first, second = 456")) failCount++; } // Supplementary character test pattern = Pattern.compile(toSupplementaries("(ab)(cd)")); matcher = pattern.matcher(toSupplementaries("abcd")); result = matcher.replaceAll("$2$1"); if (!result.equals(toSupplementaries("cdab"))) failCount++; s1 = toSupplementaries("Swap all: first = 123, second = 456"); s2 = toSupplementaries("Swap one: first = 123, second = 456"); r = toSupplementaries("$3$2$1"); pattern = Pattern.compile(toSupplementaries("([a-z]+)( *= *)([0-9]+)")); matcher = pattern.matcher(s1); result = matcher.replaceAll(r); if (!result.equals(toSupplementaries("Swap all: 123 = first, 456 = second"))) failCount++; matcher = pattern.matcher(s2); if (matcher.find()) { StringBuffer sb = new StringBuffer(); matcher.appendReplacement(sb, r); matcher.appendTail(sb); result = sb.toString(); if (!result.equals(toSupplementaries("Swap one: 123 = first, second = 456"))) failCount++; } report("Append"); } private static void splitTest() { Pattern pattern = Pattern.compile(":"); String[] result = pattern.split("foo:and:boo", 2); if (!result[0].equals("foo")) failCount++; if (!result[1].equals("and:boo")) failCount++; // Supplementary character test Pattern patternX = Pattern.compile(toSupplementaries("X")); result = patternX.split(toSupplementaries("fooXandXboo"), 2); if (!result[0].equals(toSupplementaries("foo"))) failCount++; if (!result[1].equals(toSupplementaries("andXboo"))) failCount++; CharBuffer cb = CharBuffer.allocate(100); cb.put("foo:and:boo"); cb.flip(); result = pattern.split(cb); if (!result[0].equals("foo")) failCount++; if (!result[1].equals("and")) failCount++; if (!result[2].equals("boo")) failCount++; // Supplementary character test CharBuffer cbs = CharBuffer.allocate(100); cbs.put(toSupplementaries("fooXandXboo")); cbs.flip(); result = patternX.split(cbs); if (!result[0].equals(toSupplementaries("foo"))) failCount++; if (!result[1].equals(toSupplementaries("and"))) failCount++; if (!result[2].equals(toSupplementaries("boo"))) failCount++; String source = "0123456789"; for (int limit=-2; limit<3; limit++) { for (int x=0; x<10; x++) { result = source.split(Integer.toString(x), limit); int expectedLength = limit < 1 ? 2 : limit; if ((limit == 0) && (x == 9)) { // expected dropping of "" if (result.length != 1) failCount++; if (!result[0].equals("012345678")) { failCount++; } } else { if (result.length != expectedLength) { failCount++; } if (!result[0].equals(source.substring(0,x))) { if (limit != 1) { failCount++; } else { if (!result[0].equals(source.substring(0,10))) { failCount++; } } } if (expectedLength > 1) { // Check segment 2 if (!result[1].equals(source.substring(x+1,10))) failCount++; } } } } // Check the case for no match found for (int limit=-2; limit<3; limit++) { result = source.split("e", limit); if (result.length != 1) failCount++; if (!result[0].equals(source)) failCount++; } // Check the case for limit == 0, source = ""; // split() now returns 0-length for empty source "" see #6559590 source = ""; result = source.split("e", 0); if (result.length != 1) failCount++; if (!result[0].equals(source)) failCount++; // Check both split() and splitAsStraem(), especially for zero-lenth // input and zero-lenth match cases String[][] input = new String[][] { { " ", "Abc Efg Hij" }, // normal non-zero-match { " ", " Abc Efg Hij" }, // leading empty str for non-zero-match { " ", "Abc Efg Hij" }, // non-zero-match in the middle { "(?=\\p{Lu})", "AbcEfgHij" }, // no leading empty str for zero-match { "(?=\\p{Lu})", "AbcEfg" }, { "(?=\\p{Lu})", "Abc" }, { " ", "" }, // zero-length input { ".*", "" }, // some tests from PatternStreamTest.java { "4", "awgqwefg1fefw4vssv1vvv1" }, { "\u00a3a", "afbfq\u00a3abgwgb\u00a3awngnwggw\u00a3a\u00a3ahjrnhneerh" }, { "1", "awgqwefg1fefw4vssv1vvv1" }, { "1", "a\u4ebafg1fefw\u4eba4\u9f9cvssv\u9f9c1v\u672c\u672cvv" }, { "\u56da", "1\u56da23\u56da456\u56da7890" }, { "\u56da", "1\u56da23\u9f9c\u672c\u672c\u56da456\u56da\u9f9c\u672c7890" }, { "\u56da", "" }, { "[ \t,:.]","This is,testing: with\tdifferent separators." }, //multiple septs { "o", "boo:and:foo" }, { "o", "booooo:and:fooooo" }, { "o", "fooooo:" }, }; String[][] expected = new String[][] { { "Abc", "Efg", "Hij" }, { "", "Abc", "Efg", "Hij" }, { "Abc", "", "Efg", "Hij" }, { "Abc", "Efg", "Hij" }, { "Abc", "Efg" }, { "Abc" }, { "" }, { "" }, { "awgqwefg1fefw", "vssv1vvv1" }, { "afbfq", "bgwgb", "wngnwggw", "", "hjrnhneerh" }, { "awgqwefg", "fefw4vssv", "vvv" }, { "a\u4ebafg", "fefw\u4eba4\u9f9cvssv\u9f9c", "v\u672c\u672cvv" }, { "1", "23", "456", "7890" }, { "1", "23\u9f9c\u672c\u672c", "456", "\u9f9c\u672c7890" }, { "" }, { "This", "is", "testing", "", "with", "different", "separators" }, { "b", "", ":and:f" }, { "b", "", "", "", "", ":and:f" }, { "f", "", "", "", "", ":" }, }; for (int i = 0; i < input.length; i++) { pattern = Pattern.compile(input[i][0]); if (!Arrays.equals(pattern.split(input[i][1]), expected[i])) { failCount++; } if (input[i][1].length() > 0 && // splitAsStream() return empty resulting // array for zero-length input for now !Arrays.equals(pattern.splitAsStream(input[i][1]).toArray(), expected[i])) { failCount++; } } report("Split"); } private static void negationTest() { Pattern pattern = Pattern.compile("[\\[@^]+"); Matcher matcher = pattern.matcher("@@@@[[[[^^^^"); if (!matcher.find()) failCount++; if (!matcher.group(0).equals("@@@@[[[[^^^^")) failCount++; pattern = Pattern.compile("[@\\[^]+"); matcher = pattern.matcher("@@@@[[[[^^^^"); if (!matcher.find()) failCount++; if (!matcher.group(0).equals("@@@@[[[[^^^^")) failCount++; pattern = Pattern.compile("[@\\[^@]+"); matcher = pattern.matcher("@@@@[[[[^^^^"); if (!matcher.find()) failCount++; if (!matcher.group(0).equals("@@@@[[[[^^^^")) failCount++; pattern = Pattern.compile("\\)"); matcher = pattern.matcher("xxx)xxx"); if (!matcher.find()) failCount++; report("Negation"); } private static void ampersandTest() { Pattern pattern = Pattern.compile("[&@]+"); check(pattern, "@@@@&&&&", true); pattern = Pattern.compile("[@&]+"); check(pattern, "@@@@&&&&", true); pattern = Pattern.compile("[@\\&]+"); check(pattern, "@@@@&&&&", true); report("Ampersand"); } private static void octalTest() throws Exception { Pattern pattern = Pattern.compile("\\u0007"); Matcher matcher = pattern.matcher("\u0007"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("\\07"); matcher = pattern.matcher("\u0007"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("\\007"); matcher = pattern.matcher("\u0007"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("\\0007"); matcher = pattern.matcher("\u0007"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("\\040"); matcher = pattern.matcher("\u0020"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("\\0403"); matcher = pattern.matcher("\u00203"); if (!matcher.matches()) failCount++; pattern = Pattern.compile("\\0103"); matcher = pattern.matcher("\u0043"); if (!matcher.matches()) failCount++; report("Octal"); } private static void longPatternTest() throws Exception { try { Pattern pattern = Pattern.compile( "a 32-character-long pattern xxxx"); pattern = Pattern.compile("a 33-character-long pattern xxxxx"); pattern = Pattern.compile("a thirty four character long regex"); StringBuffer patternToBe = new StringBuffer(101); for (int i=0; i<100; i++) patternToBe.append((char)(97 + i%26)); pattern = Pattern.compile(patternToBe.toString()); } catch (PatternSyntaxException e) { failCount++; } // Supplementary character test try { Pattern pattern = Pattern.compile( toSupplementaries("a 32-character-long pattern xxxx")); pattern = Pattern.compile(toSupplementaries("a 33-character-long pattern xxxxx")); pattern = Pattern.compile(toSupplementaries("a thirty four character long regex")); StringBuffer patternToBe = new StringBuffer(101*2); for (int i=0; i<100; i++) patternToBe.append(Character.toChars(Character.MIN_SUPPLEMENTARY_CODE_POINT + 97 + i%26)); pattern = Pattern.compile(patternToBe.toString()); } catch (PatternSyntaxException e) { failCount++; } report("LongPattern"); } private static void group0Test() throws Exception { Pattern pattern = Pattern.compile("(tes)ting"); Matcher matcher = pattern.matcher("testing"); check(matcher, "testing"); matcher.reset("testing"); if (matcher.lookingAt()) { if (!matcher.group(0).equals("testing")) failCount++; } else { failCount++; } matcher.reset("testing"); if (matcher.matches()) { if (!matcher.group(0).equals("testing")) failCount++; } else { failCount++; } pattern = Pattern.compile("(tes)ting"); matcher = pattern.matcher("testing"); if (matcher.lookingAt()) { if (!matcher.group(0).equals("testing")) failCount++; } else { failCount++; } pattern = Pattern.compile("^(tes)ting"); matcher = pattern.matcher("testing"); if (matcher.matches()) { if (!matcher.group(0).equals("testing")) failCount++; } else { failCount++; } // Supplementary character test pattern = Pattern.compile(toSupplementaries("(tes)ting")); matcher = pattern.matcher(toSupplementaries("testing")); check(matcher, toSupplementaries("testing")); matcher.reset(toSupplementaries("testing")); if (matcher.lookingAt()) { if (!matcher.group(0).equals(toSupplementaries("testing"))) failCount++; } else { failCount++; } matcher.reset(toSupplementaries("testing")); if (matcher.matches()) { if (!matcher.group(0).equals(toSupplementaries("testing"))) failCount++; } else { failCount++; } pattern = Pattern.compile(toSupplementaries("(tes)ting")); matcher = pattern.matcher(toSupplementaries("testing")); if (matcher.lookingAt()) { if (!matcher.group(0).equals(toSupplementaries("testing"))) failCount++; } else { failCount++; } pattern = Pattern.compile(toSupplementaries("^(tes)ting")); matcher = pattern.matcher(toSupplementaries("testing")); if (matcher.matches()) { if (!matcher.group(0).equals(toSupplementaries("testing"))) failCount++; } else { failCount++; } report("Group0"); } private static void findIntTest() throws Exception { Pattern p = Pattern.compile("blah"); Matcher m = p.matcher("zzzzblahzzzzzblah"); boolean result = m.find(2); if (!result) failCount++; p = Pattern.compile("$"); m = p.matcher("1234567890"); result = m.find(10); if (!result) failCount++; try { result = m.find(11); failCount++; } catch (IndexOutOfBoundsException e) { // correct result } // Supplementary character test p = Pattern.compile(toSupplementaries("blah")); m = p.matcher(toSupplementaries("zzzzblahzzzzzblah")); result = m.find(2); if (!result) failCount++; report("FindInt"); } private static void emptyPatternTest() throws Exception { Pattern p = Pattern.compile(""); Matcher m = p.matcher("foo"); // Should find empty pattern at beginning of input boolean result = m.find(); if (result != true) failCount++; if (m.start() != 0) failCount++; // Should not match entire input if input is not empty m.reset(); result = m.matches(); if (result == true) failCount++; try { m.start(0); failCount++; } catch (IllegalStateException e) { // Correct result } // Should match entire input if input is empty m.reset(""); result = m.matches(); if (result != true) failCount++; result = Pattern.matches("", ""); if (result != true) failCount++; result = Pattern.matches("", "foo"); if (result == true) failCount++; report("EmptyPattern"); } private static void charClassTest() throws Exception { Pattern pattern = Pattern.compile("blah[ab]]blech"); check(pattern, "blahb]blech", true); pattern = Pattern.compile("[abc[def]]"); check(pattern, "b", true); // Supplementary character tests pattern = Pattern.compile(toSupplementaries("blah[ab]]blech")); check(pattern, toSupplementaries("blahb]blech"), true); pattern = Pattern.compile(toSupplementaries("[abc[def]]")); check(pattern, toSupplementaries("b"), true); try { // u00ff when UNICODE_CASE pattern = Pattern.compile("[ab\u00ffcd]", Pattern.CASE_INSENSITIVE| Pattern.UNICODE_CASE); check(pattern, "ab\u00ffcd", true); check(pattern, "Ab\u0178Cd", true); // u00b5 when UNICODE_CASE pattern = Pattern.compile("[ab\u00b5cd]", Pattern.CASE_INSENSITIVE| Pattern.UNICODE_CASE); check(pattern, "ab\u00b5cd", true); check(pattern, "Ab\u039cCd", true); } catch (Exception e) { failCount++; } /* Special cases (1)LatinSmallLetterLongS u+017f (2)LatinSmallLetterDotlessI u+0131 (3)LatineCapitalLetterIWithDotAbove u+0130 (4)KelvinSign u+212a (5)AngstromSign u+212b */ int flags = Pattern.UNICODE_CASE | Pattern.CASE_INSENSITIVE; pattern = Pattern.compile("[sik\u00c5]+", flags); if (!pattern.matcher("\u017f\u0130\u0131\u212a\u212b").matches()) failCount++; report("CharClass"); } private static void caretTest() throws Exception { Pattern pattern = Pattern.compile("\\w*"); Matcher matcher = pattern.matcher("a#bc#def##g"); check(matcher, "a"); check(matcher, ""); check(matcher, "bc"); check(matcher, ""); check(matcher, "def"); check(matcher, ""); check(matcher, ""); check(matcher, "g"); check(matcher, ""); if (matcher.find()) failCount++; pattern = Pattern.compile("^\\w*"); matcher = pattern.matcher("a#bc#def##g"); check(matcher, "a"); if (matcher.find()) failCount++; pattern = Pattern.compile("\\w"); matcher = pattern.matcher("abc##x"); check(matcher, "a"); check(matcher, "b"); check(matcher, "c"); check(matcher, "x"); if (matcher.find()) failCount++; pattern = Pattern.compile("^\\w"); matcher = pattern.matcher("abc##x"); check(matcher, "a"); if (matcher.find()) failCount++; pattern = Pattern.compile("\\A\\p{Alpha}{3}"); matcher = pattern.matcher("abcdef-ghi\njklmno"); check(matcher, "abc"); if (matcher.find()) failCount++; pattern = Pattern.compile("^\\p{Alpha}{3}", Pattern.MULTILINE); matcher = pattern.matcher("abcdef-ghi\njklmno"); check(matcher, "abc"); check(matcher, "jkl"); if (matcher.find()) failCount++; pattern = Pattern.compile("^", Pattern.MULTILINE); matcher = pattern.matcher("this is some text"); String result = matcher.replaceAll("X"); if (!result.equals("Xthis is some text")) failCount++; pattern = Pattern.compile("^"); matcher = pattern.matcher("this is some text"); result = matcher.replaceAll("X"); if (!result.equals("Xthis is some text")) failCount++; pattern = Pattern.compile("^", Pattern.MULTILINE | Pattern.UNIX_LINES); matcher = pattern.matcher("this is some text\n"); result = matcher.replaceAll("X"); if (!result.equals("Xthis is some text\n")) failCount++; report("Caret"); } private static void groupCaptureTest() throws Exception { // Independent group Pattern pattern = Pattern.compile("x+(?>y+)z+"); Matcher matcher = pattern.matcher("xxxyyyzzz"); matcher.find(); try { String blah = matcher.group(1); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Good result } // Pure group pattern = Pattern.compile("x+(?:y+)z+"); matcher = pattern.matcher("xxxyyyzzz"); matcher.find(); try { String blah = matcher.group(1); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Good result } // Supplementary character tests // Independent group pattern = Pattern.compile(toSupplementaries("x+(?>y+)z+")); matcher = pattern.matcher(toSupplementaries("xxxyyyzzz")); matcher.find(); try { String blah = matcher.group(1); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Good result } // Pure group pattern = Pattern.compile(toSupplementaries("x+(?:y+)z+")); matcher = pattern.matcher(toSupplementaries("xxxyyyzzz")); matcher.find(); try { String blah = matcher.group(1); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Good result } report("GroupCapture"); } private static void backRefTest() throws Exception { Pattern pattern = Pattern.compile("(a*)bc\\1"); check(pattern, "zzzaabcazzz", true); pattern = Pattern.compile("(a*)bc\\1"); check(pattern, "zzzaabcaazzz", true); pattern = Pattern.compile("(abc)(def)\\1"); check(pattern, "abcdefabc", true); pattern = Pattern.compile("(abc)(def)\\3"); check(pattern, "abcdefabc", false); try { for (int i = 1; i < 10; i++) { // Make sure backref 1-9 are always accepted pattern = Pattern.compile("abcdef\\" + i); // and fail to match if the target group does not exit check(pattern, "abcdef", false); } } catch(PatternSyntaxException e) { failCount++; } pattern = Pattern.compile("(a)(b)(c)(d)(e)(f)(g)(h)(i)(j)\\11"); check(pattern, "abcdefghija", false); check(pattern, "abcdefghija1", true); pattern = Pattern.compile("(a)(b)(c)(d)(e)(f)(g)(h)(i)(j)(k)\\11"); check(pattern, "abcdefghijkk", true); pattern = Pattern.compile("(a)bcdefghij\\11"); check(pattern, "abcdefghija1", true); // Supplementary character tests pattern = Pattern.compile(toSupplementaries("(a*)bc\\1")); check(pattern, toSupplementaries("zzzaabcazzz"), true); pattern = Pattern.compile(toSupplementaries("(a*)bc\\1")); check(pattern, toSupplementaries("zzzaabcaazzz"), true); pattern = Pattern.compile(toSupplementaries("(abc)(def)\\1")); check(pattern, toSupplementaries("abcdefabc"), true); pattern = Pattern.compile(toSupplementaries("(abc)(def)\\3")); check(pattern, toSupplementaries("abcdefabc"), false); pattern = Pattern.compile(toSupplementaries("(a)(b)(c)(d)(e)(f)(g)(h)(i)(j)\\11")); check(pattern, toSupplementaries("abcdefghija"), false); check(pattern, toSupplementaries("abcdefghija1"), true); pattern = Pattern.compile(toSupplementaries("(a)(b)(c)(d)(e)(f)(g)(h)(i)(j)(k)\\11")); check(pattern, toSupplementaries("abcdefghijkk"), true); report("BackRef"); } /** * Unicode Technical Report #18, section 2.6 End of Line * There is no empty line to be matched in the sequence \u000D\u000A * but there is an empty line in the sequence \u000A\u000D. */ private static void anchorTest() throws Exception { Pattern p = Pattern.compile("^.*$", Pattern.MULTILINE); Matcher m = p.matcher("blah1\r\nblah2"); m.find(); m.find(); if (!m.group().equals("blah2")) failCount++; m.reset("blah1\n\rblah2"); m.find(); m.find(); m.find(); if (!m.group().equals("blah2")) failCount++; // Test behavior of $ with \r\n at end of input p = Pattern.compile(".+$"); m = p.matcher("blah1\r\n"); if (!m.find()) failCount++; if (!m.group().equals("blah1")) failCount++; if (m.find()) failCount++; // Test behavior of $ with \r\n at end of input in multiline p = Pattern.compile(".+$", Pattern.MULTILINE); m = p.matcher("blah1\r\n"); if (!m.find()) failCount++; if (m.find()) failCount++; // Test for $ recognition of \u0085 for bug 4527731 p = Pattern.compile(".+$", Pattern.MULTILINE); m = p.matcher("blah1\u0085"); if (!m.find()) failCount++; // Supplementary character test p = Pattern.compile("^.*$", Pattern.MULTILINE); m = p.matcher(toSupplementaries("blah1\r\nblah2")); m.find(); m.find(); if (!m.group().equals(toSupplementaries("blah2"))) failCount++; m.reset(toSupplementaries("blah1\n\rblah2")); m.find(); m.find(); m.find(); if (!m.group().equals(toSupplementaries("blah2"))) failCount++; // Test behavior of $ with \r\n at end of input p = Pattern.compile(".+$"); m = p.matcher(toSupplementaries("blah1\r\n")); if (!m.find()) failCount++; if (!m.group().equals(toSupplementaries("blah1"))) failCount++; if (m.find()) failCount++; // Test behavior of $ with \r\n at end of input in multiline p = Pattern.compile(".+$", Pattern.MULTILINE); m = p.matcher(toSupplementaries("blah1\r\n")); if (!m.find()) failCount++; if (m.find()) failCount++; // Test for $ recognition of \u0085 for bug 4527731 p = Pattern.compile(".+$", Pattern.MULTILINE); m = p.matcher(toSupplementaries("blah1\u0085")); if (!m.find()) failCount++; report("Anchors"); } /** * A basic sanity test of Matcher.lookingAt(). */ private static void lookingAtTest() throws Exception { Pattern p = Pattern.compile("(ab)(c*)"); Matcher m = p.matcher("abccczzzabcczzzabccc"); if (!m.lookingAt()) failCount++; if (!m.group().equals(m.group(0))) failCount++; m = p.matcher("zzzabccczzzabcczzzabccczzz"); if (m.lookingAt()) failCount++; // Supplementary character test p = Pattern.compile(toSupplementaries("(ab)(c*)")); m = p.matcher(toSupplementaries("abccczzzabcczzzabccc")); if (!m.lookingAt()) failCount++; if (!m.group().equals(m.group(0))) failCount++; m = p.matcher(toSupplementaries("zzzabccczzzabcczzzabccczzz")); if (m.lookingAt()) failCount++; report("Looking At"); } /** * A basic sanity test of Matcher.matches(). */ private static void matchesTest() throws Exception { // matches() Pattern p = Pattern.compile("ulb(c*)"); Matcher m = p.matcher("ulbcccccc"); if (!m.matches()) failCount++; // find() but not matches() m.reset("zzzulbcccccc"); if (m.matches()) failCount++; // lookingAt() but not matches() m.reset("ulbccccccdef"); if (m.matches()) failCount++; // matches() p = Pattern.compile("a|ad"); m = p.matcher("ad"); if (!m.matches()) failCount++; // Supplementary character test // matches() p = Pattern.compile(toSupplementaries("ulb(c*)")); m = p.matcher(toSupplementaries("ulbcccccc")); if (!m.matches()) failCount++; // find() but not matches() m.reset(toSupplementaries("zzzulbcccccc")); if (m.matches()) failCount++; // lookingAt() but not matches() m.reset(toSupplementaries("ulbccccccdef")); if (m.matches()) failCount++; // matches() p = Pattern.compile(toSupplementaries("a|ad")); m = p.matcher(toSupplementaries("ad")); if (!m.matches()) failCount++; report("Matches"); } /** * A basic sanity test of Pattern.matches(). */ private static void patternMatchesTest() throws Exception { // matches() if (!Pattern.matches(toSupplementaries("ulb(c*)"), toSupplementaries("ulbcccccc"))) failCount++; // find() but not matches() if (Pattern.matches(toSupplementaries("ulb(c*)"), toSupplementaries("zzzulbcccccc"))) failCount++; // lookingAt() but not matches() if (Pattern.matches(toSupplementaries("ulb(c*)"), toSupplementaries("ulbccccccdef"))) failCount++; // Supplementary character test // matches() if (!Pattern.matches(toSupplementaries("ulb(c*)"), toSupplementaries("ulbcccccc"))) failCount++; // find() but not matches() if (Pattern.matches(toSupplementaries("ulb(c*)"), toSupplementaries("zzzulbcccccc"))) failCount++; // lookingAt() but not matches() if (Pattern.matches(toSupplementaries("ulb(c*)"), toSupplementaries("ulbccccccdef"))) failCount++; report("Pattern Matches"); } /** * Canonical equivalence testing. Tests the ability of the engine * to match sequences that are not explicitly specified in the * pattern when they are considered equivalent by the Unicode Standard. */ private static void ceTest() throws Exception { // Decomposed char outside char classes Pattern p = Pattern.compile("testa\u030a", Pattern.CANON_EQ); Matcher m = p.matcher("test\u00e5"); if (!m.matches()) failCount++; m.reset("testa\u030a"); if (!m.matches()) failCount++; // Composed char outside char classes p = Pattern.compile("test\u00e5", Pattern.CANON_EQ); m = p.matcher("test\u00e5"); if (!m.matches()) failCount++; m.reset("testa\u030a"); if (!m.find()) failCount++; // Decomposed char inside a char class p = Pattern.compile("test[abca\u030a]", Pattern.CANON_EQ); m = p.matcher("test\u00e5"); if (!m.find()) failCount++; m.reset("testa\u030a"); if (!m.find()) failCount++; // Composed char inside a char class p = Pattern.compile("test[abc\u00e5def\u00e0]", Pattern.CANON_EQ); m = p.matcher("test\u00e5"); if (!m.find()) failCount++; m.reset("testa\u0300"); if (!m.find()) failCount++; m.reset("testa\u030a"); if (!m.find()) failCount++; // Marks that cannot legally change order and be equivalent p = Pattern.compile("testa\u0308\u0300", Pattern.CANON_EQ); check(p, "testa\u0308\u0300", true); check(p, "testa\u0300\u0308", false); // Marks that can legally change order and be equivalent p = Pattern.compile("testa\u0308\u0323", Pattern.CANON_EQ); check(p, "testa\u0308\u0323", true); check(p, "testa\u0323\u0308", true); // Test all equivalences of the sequence a\u0308\u0323\u0300 p = Pattern.compile("testa\u0308\u0323\u0300", Pattern.CANON_EQ); check(p, "testa\u0308\u0323\u0300", true); check(p, "testa\u0323\u0308\u0300", true); check(p, "testa\u0308\u0300\u0323", true); check(p, "test\u00e4\u0323\u0300", true); check(p, "test\u00e4\u0300\u0323", true); Object[][] data = new Object[][] { // JDK-4867170 { "[\u1f80-\u1f82]", "ab\u1f80cd", "f", true }, { "[\u1f80-\u1f82]", "ab\u1f81cd", "f", true }, { "[\u1f80-\u1f82]", "ab\u1f82cd", "f", true }, { "[\u1f80-\u1f82]", "ab\u03b1\u0314\u0345cd", "f", true }, { "[\u1f80-\u1f82]", "ab\u03b1\u0345\u0314cd", "f", true }, { "[\u1f80-\u1f82]", "ab\u1f01\u0345cd", "f", true }, { "[\u1f80-\u1f82]", "ab\u1f00\u0345cd", "f", true }, { "\\p{IsGreek}", "ab\u1f80cd", "f", true }, { "\\p{IsGreek}", "ab\u1f81cd", "f", true }, { "\\p{IsGreek}", "ab\u1f82cd", "f", true }, { "\\p{IsGreek}", "ab\u03b1\u0314\u0345cd", "f", true }, { "\\p{IsGreek}", "ab\u1f01\u0345cd", "f", true }, // backtracking, force to match "\u1f80", instead of \u1f82" { "ab\\p{IsGreek}\u0300cd", "ab\u03b1\u0313\u0345\u0300cd", "m", true }, { "[\\p{IsGreek}]", "\u03b1\u0314\u0345", "m", true }, { "\\p{IsGreek}", "\u03b1\u0314\u0345", "m", true }, { "[^\u1f80-\u1f82]","\u1f81", "m", false }, { "[^\u1f80-\u1f82]","\u03b1\u0314\u0345", "m", false }, { "[^\u1f01\u0345]", "\u1f81", "f", false }, { "[^\u1f81]+", "\u1f80\u1f82", "f", true }, { "[\u1f80]", "ab\u1f80cd", "f", true }, { "\u1f80", "ab\u1f80cd", "f", true }, { "\u1f00\u0345\u0300", "\u1f82", "m", true }, { "\u1f80", "-\u1f00\u0345\u0300-", "f", true }, { "\u1f82", "\u1f00\u0345\u0300", "m", true }, { "\u1f82", "\u1f80\u0300", "m", true }, // JDK-7080302 # compile failed { "a(\u0041\u0301\u0328)", "a\u0041\u0301\u0328", "m", true}, // JDK-6728861, same cause as above one { "\u00e9\u00e9n", "e\u0301e\u0301n", "m", true}, // JDK-6995635 { "(\u00e9)", "e\u0301", "m", true }, // JDK-6736245 // intereting special case, nfc(u2add+u0338) -> u2add+u0338) NOT u2adc { "\u2ADC", "\u2ADC", "m", true}, // NFC { "\u2ADC", "\u2ADD\u0338", "m", true}, // NFD // 4916384. // Decomposed hangul (jamos) works inside clazz { "[\u1100\u1161]", "\u1100\u1161", "m", true}, { "[\u1100\u1161]", "\uac00", "m", true}, { "[\uac00]", "\u1100\u1161", "m", true}, { "[\uac00]", "\uac00", "m", true}, // Decomposed hangul (jamos) { "\u1100\u1161", "\u1100\u1161", "m", true}, { "\u1100\u1161", "\uac00", "m", true}, // Composed hangul { "\uac00", "\u1100\u1161", "m", true }, { "\uac00", "\uac00", "m", true }, /* Need a NFDSlice to nfd the source to solve this issue u+1d1c0 -> nfd: -> nfc: u+1d1bc -> nfd: -> nfc: -> nfd: -> nfc: // Decomposed supplementary outside char classes // { "test\ud834\uddbc\ud834\udd6f", "test\ud834\uddc0", "m", true }, // Composed supplementary outside char classes // { "test\ud834\uddc0", "test\ud834\uddbc\ud834\udd6f", "m", true }, */ { "test\ud834\uddbc\ud834\udd6f", "test\ud834\uddbc\ud834\udd6f", "m", true }, { "test\ud834\uddc0", "test\ud834\uddbc\ud834\udd6f", "m", true }, { "test\ud834\uddc0", "test\ud834\uddc0", "m", true }, { "test\ud834\uddbc\ud834\udd6f", "test\ud834\uddc0", "m", true }, }; int failCount = 0; for (Object[] d : data) { String pn = (String)d[0]; String tt = (String)d[1]; boolean isFind = "f".equals(((String)d[2])); boolean expected = (boolean)d[3]; boolean ret = isFind ? Pattern.compile(pn, Pattern.CANON_EQ).matcher(tt).find() : Pattern.compile(pn, Pattern.CANON_EQ).matcher(tt).matches(); if (ret != expected) { failCount++; continue; } } report("Canonical Equivalence"); } /** * A basic sanity test of Matcher.replaceAll(). */ private static void globalSubstitute() throws Exception { // Global substitution with a literal Pattern p = Pattern.compile("(ab)(c*)"); Matcher m = p.matcher("abccczzzabcczzzabccc"); if (!m.replaceAll("test").equals("testzzztestzzztest")) failCount++; m.reset("zzzabccczzzabcczzzabccczzz"); if (!m.replaceAll("test").equals("zzztestzzztestzzztestzzz")) failCount++; // Global substitution with groups m.reset("zzzabccczzzabcczzzabccczzz"); String result = m.replaceAll("$1"); if (!result.equals("zzzabzzzabzzzabzzz")) failCount++; // Supplementary character test // Global substitution with a literal p = Pattern.compile(toSupplementaries("(ab)(c*)")); m = p.matcher(toSupplementaries("abccczzzabcczzzabccc")); if (!m.replaceAll(toSupplementaries("test")). equals(toSupplementaries("testzzztestzzztest"))) failCount++; m.reset(toSupplementaries("zzzabccczzzabcczzzabccczzz")); if (!m.replaceAll(toSupplementaries("test")). equals(toSupplementaries("zzztestzzztestzzztestzzz"))) failCount++; // Global substitution with groups m.reset(toSupplementaries("zzzabccczzzabcczzzabccczzz")); result = m.replaceAll("$1"); if (!result.equals(toSupplementaries("zzzabzzzabzzzabzzz"))) failCount++; report("Global Substitution"); } /** * Tests the usage of Matcher.appendReplacement() with literal * and group substitutions. */ private static void stringbufferSubstitute() throws Exception { // SB substitution with literal String blah = "zzzblahzzz"; Pattern p = Pattern.compile("blah"); Matcher m = p.matcher(blah); StringBuffer result = new StringBuffer(); try { m.appendReplacement(result, "blech"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "blech"); if (!result.toString().equals("zzzblech")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzblechzzz")) failCount++; // SB substitution with groups blah = "zzzabcdzzz"; p = Pattern.compile("(ab)(cd)*"); m = p.matcher(blah); result = new StringBuffer(); try { m.appendReplacement(result, "$1"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1"); if (!result.toString().equals("zzzab")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzabzzz")) failCount++; // SB substitution with 3 groups blah = "zzzabcdcdefzzz"; p = Pattern.compile("(ab)(cd)*(ef)"); m = p.matcher(blah); result = new StringBuffer(); try { m.appendReplacement(result, "$1w$2w$3"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1w$2w$3"); if (!result.toString().equals("zzzabwcdwef")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzabwcdwefzzz")) failCount++; // SB substitution with groups and three matches // skipping middle match blah = "zzzabcdzzzabcddzzzabcdzzz"; p = Pattern.compile("(ab)(cd*)"); m = p.matcher(blah); result = new StringBuffer(); try { m.appendReplacement(result, "$1"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1"); if (!result.toString().equals("zzzab")) failCount++; m.find(); m.find(); m.appendReplacement(result, "$2"); if (!result.toString().equals("zzzabzzzabcddzzzcd")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzabzzzabcddzzzcdzzz")) failCount++; // Check to make sure escaped $ is ignored blah = "zzzabcdcdefzzz"; p = Pattern.compile("(ab)(cd)*(ef)"); m = p.matcher(blah); result = new StringBuffer(); m.find(); m.appendReplacement(result, "$1w\\$2w$3"); if (!result.toString().equals("zzzabw$2wef")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzabw$2wefzzz")) failCount++; // Check to make sure a reference to nonexistent group causes error blah = "zzzabcdcdefzzz"; p = Pattern.compile("(ab)(cd)*(ef)"); m = p.matcher(blah); result = new StringBuffer(); m.find(); try { m.appendReplacement(result, "$1w$5w$3"); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Correct result } // Check double digit group references blah = "zzz123456789101112zzz"; p = Pattern.compile("(1)(2)(3)(4)(5)(6)(7)(8)(9)(10)(11)"); m = p.matcher(blah); result = new StringBuffer(); m.find(); m.appendReplacement(result, "$1w$11w$3"); if (!result.toString().equals("zzz1w11w3")) failCount++; // Check to make sure it backs off $15 to $1 if only three groups blah = "zzzabcdcdefzzz"; p = Pattern.compile("(ab)(cd)*(ef)"); m = p.matcher(blah); result = new StringBuffer(); m.find(); m.appendReplacement(result, "$1w$15w$3"); if (!result.toString().equals("zzzabwab5wef")) failCount++; // Supplementary character test // SB substitution with literal blah = toSupplementaries("zzzblahzzz"); p = Pattern.compile(toSupplementaries("blah")); m = p.matcher(blah); result = new StringBuffer(); try { m.appendReplacement(result, toSupplementaries("blech")); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, toSupplementaries("blech")); if (!result.toString().equals(toSupplementaries("zzzblech"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzblechzzz"))) failCount++; // SB substitution with groups blah = toSupplementaries("zzzabcdzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*")); m = p.matcher(blah); result = new StringBuffer(); try { m.appendReplacement(result, "$1"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1"); if (!result.toString().equals(toSupplementaries("zzzab"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzabzzz"))) failCount++; // SB substitution with 3 groups blah = toSupplementaries("zzzabcdcdefzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*(ef)")); m = p.matcher(blah); result = new StringBuffer(); try { m.appendReplacement(result, toSupplementaries("$1w$2w$3")); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, toSupplementaries("$1w$2w$3")); if (!result.toString().equals(toSupplementaries("zzzabwcdwef"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzabwcdwefzzz"))) failCount++; // SB substitution with groups and three matches // skipping middle match blah = toSupplementaries("zzzabcdzzzabcddzzzabcdzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd*)")); m = p.matcher(blah); result = new StringBuffer(); try { m.appendReplacement(result, "$1"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1"); if (!result.toString().equals(toSupplementaries("zzzab"))) failCount++; m.find(); m.find(); m.appendReplacement(result, "$2"); if (!result.toString().equals(toSupplementaries("zzzabzzzabcddzzzcd"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzabzzzabcddzzzcdzzz"))) failCount++; // Check to make sure escaped $ is ignored blah = toSupplementaries("zzzabcdcdefzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*(ef)")); m = p.matcher(blah); result = new StringBuffer(); m.find(); m.appendReplacement(result, toSupplementaries("$1w\\$2w$3")); if (!result.toString().equals(toSupplementaries("zzzabw$2wef"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzabw$2wefzzz"))) failCount++; // Check to make sure a reference to nonexistent group causes error blah = toSupplementaries("zzzabcdcdefzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*(ef)")); m = p.matcher(blah); result = new StringBuffer(); m.find(); try { m.appendReplacement(result, toSupplementaries("$1w$5w$3")); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Correct result } // Check double digit group references blah = toSupplementaries("zzz123456789101112zzz"); p = Pattern.compile("(1)(2)(3)(4)(5)(6)(7)(8)(9)(10)(11)"); m = p.matcher(blah); result = new StringBuffer(); m.find(); m.appendReplacement(result, toSupplementaries("$1w$11w$3")); if (!result.toString().equals(toSupplementaries("zzz1w11w3"))) failCount++; // Check to make sure it backs off $15 to $1 if only three groups blah = toSupplementaries("zzzabcdcdefzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*(ef)")); m = p.matcher(blah); result = new StringBuffer(); m.find(); m.appendReplacement(result, toSupplementaries("$1w$15w$3")); if (!result.toString().equals(toSupplementaries("zzzabwab5wef"))) failCount++; // Check nothing has been appended into the output buffer if // the replacement string triggers IllegalArgumentException. p = Pattern.compile("(abc)"); m = p.matcher("abcd"); result = new StringBuffer(); m.find(); try { m.appendReplacement(result, ("xyz$g")); failCount++; } catch (IllegalArgumentException iae) { if (result.length() != 0) failCount++; } report("SB Substitution"); } /** * Tests the usage of Matcher.appendReplacement() with literal * and group substitutions. */ private static void stringbuilderSubstitute() throws Exception { // SB substitution with literal String blah = "zzzblahzzz"; Pattern p = Pattern.compile("blah"); Matcher m = p.matcher(blah); StringBuilder result = new StringBuilder(); try { m.appendReplacement(result, "blech"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "blech"); if (!result.toString().equals("zzzblech")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzblechzzz")) failCount++; // SB substitution with groups blah = "zzzabcdzzz"; p = Pattern.compile("(ab)(cd)*"); m = p.matcher(blah); result = new StringBuilder(); try { m.appendReplacement(result, "$1"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1"); if (!result.toString().equals("zzzab")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzabzzz")) failCount++; // SB substitution with 3 groups blah = "zzzabcdcdefzzz"; p = Pattern.compile("(ab)(cd)*(ef)"); m = p.matcher(blah); result = new StringBuilder(); try { m.appendReplacement(result, "$1w$2w$3"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1w$2w$3"); if (!result.toString().equals("zzzabwcdwef")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzabwcdwefzzz")) failCount++; // SB substitution with groups and three matches // skipping middle match blah = "zzzabcdzzzabcddzzzabcdzzz"; p = Pattern.compile("(ab)(cd*)"); m = p.matcher(blah); result = new StringBuilder(); try { m.appendReplacement(result, "$1"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1"); if (!result.toString().equals("zzzab")) failCount++; m.find(); m.find(); m.appendReplacement(result, "$2"); if (!result.toString().equals("zzzabzzzabcddzzzcd")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzabzzzabcddzzzcdzzz")) failCount++; // Check to make sure escaped $ is ignored blah = "zzzabcdcdefzzz"; p = Pattern.compile("(ab)(cd)*(ef)"); m = p.matcher(blah); result = new StringBuilder(); m.find(); m.appendReplacement(result, "$1w\\$2w$3"); if (!result.toString().equals("zzzabw$2wef")) failCount++; m.appendTail(result); if (!result.toString().equals("zzzabw$2wefzzz")) failCount++; // Check to make sure a reference to nonexistent group causes error blah = "zzzabcdcdefzzz"; p = Pattern.compile("(ab)(cd)*(ef)"); m = p.matcher(blah); result = new StringBuilder(); m.find(); try { m.appendReplacement(result, "$1w$5w$3"); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Correct result } // Check double digit group references blah = "zzz123456789101112zzz"; p = Pattern.compile("(1)(2)(3)(4)(5)(6)(7)(8)(9)(10)(11)"); m = p.matcher(blah); result = new StringBuilder(); m.find(); m.appendReplacement(result, "$1w$11w$3"); if (!result.toString().equals("zzz1w11w3")) failCount++; // Check to make sure it backs off $15 to $1 if only three groups blah = "zzzabcdcdefzzz"; p = Pattern.compile("(ab)(cd)*(ef)"); m = p.matcher(blah); result = new StringBuilder(); m.find(); m.appendReplacement(result, "$1w$15w$3"); if (!result.toString().equals("zzzabwab5wef")) failCount++; // Supplementary character test // SB substitution with literal blah = toSupplementaries("zzzblahzzz"); p = Pattern.compile(toSupplementaries("blah")); m = p.matcher(blah); result = new StringBuilder(); try { m.appendReplacement(result, toSupplementaries("blech")); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, toSupplementaries("blech")); if (!result.toString().equals(toSupplementaries("zzzblech"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzblechzzz"))) failCount++; // SB substitution with groups blah = toSupplementaries("zzzabcdzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*")); m = p.matcher(blah); result = new StringBuilder(); try { m.appendReplacement(result, "$1"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1"); if (!result.toString().equals(toSupplementaries("zzzab"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzabzzz"))) failCount++; // SB substitution with 3 groups blah = toSupplementaries("zzzabcdcdefzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*(ef)")); m = p.matcher(blah); result = new StringBuilder(); try { m.appendReplacement(result, toSupplementaries("$1w$2w$3")); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, toSupplementaries("$1w$2w$3")); if (!result.toString().equals(toSupplementaries("zzzabwcdwef"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzabwcdwefzzz"))) failCount++; // SB substitution with groups and three matches // skipping middle match blah = toSupplementaries("zzzabcdzzzabcddzzzabcdzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd*)")); m = p.matcher(blah); result = new StringBuilder(); try { m.appendReplacement(result, "$1"); failCount++; } catch (IllegalStateException e) { } m.find(); m.appendReplacement(result, "$1"); if (!result.toString().equals(toSupplementaries("zzzab"))) failCount++; m.find(); m.find(); m.appendReplacement(result, "$2"); if (!result.toString().equals(toSupplementaries("zzzabzzzabcddzzzcd"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzabzzzabcddzzzcdzzz"))) failCount++; // Check to make sure escaped $ is ignored blah = toSupplementaries("zzzabcdcdefzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*(ef)")); m = p.matcher(blah); result = new StringBuilder(); m.find(); m.appendReplacement(result, toSupplementaries("$1w\\$2w$3")); if (!result.toString().equals(toSupplementaries("zzzabw$2wef"))) failCount++; m.appendTail(result); if (!result.toString().equals(toSupplementaries("zzzabw$2wefzzz"))) failCount++; // Check to make sure a reference to nonexistent group causes error blah = toSupplementaries("zzzabcdcdefzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*(ef)")); m = p.matcher(blah); result = new StringBuilder(); m.find(); try { m.appendReplacement(result, toSupplementaries("$1w$5w$3")); failCount++; } catch (IndexOutOfBoundsException ioobe) { // Correct result } // Check double digit group references blah = toSupplementaries("zzz123456789101112zzz"); p = Pattern.compile("(1)(2)(3)(4)(5)(6)(7)(8)(9)(10)(11)"); m = p.matcher(blah); result = new StringBuilder(); m.find(); m.appendReplacement(result, toSupplementaries("$1w$11w$3")); if (!result.toString().equals(toSupplementaries("zzz1w11w3"))) failCount++; // Check to make sure it backs off $15 to $1 if only three groups blah = toSupplementaries("zzzabcdcdefzzz"); p = Pattern.compile(toSupplementaries("(ab)(cd)*(ef)")); m = p.matcher(blah); result = new StringBuilder(); m.find(); m.appendReplacement(result, toSupplementaries("$1w$15w$3")); if (!result.toString().equals(toSupplementaries("zzzabwab5wef"))) failCount++; // Check nothing has been appended into the output buffer if // the replacement string triggers IllegalArgumentException. p = Pattern.compile("(abc)"); m = p.matcher("abcd"); result = new StringBuilder(); m.find(); try { m.appendReplacement(result, ("xyz$g")); failCount++; } catch (IllegalArgumentException iae) { if (result.length() != 0) failCount++; } report("SB Substitution 2"); } /* * 5 groups of characters are created to make a substitution string. * A base string will be created including random lead chars, the * substitution string, and random trailing chars. * A pattern containing the 5 groups is searched for and replaced with: * random group + random string + random group. * The results are checked for correctness. */ private static void substitutionBasher() { for (int runs = 0; runs<1000; runs++) { // Create a base string to work in int leadingChars = generator.nextInt(10); StringBuffer baseBuffer = new StringBuffer(100); String leadingString = getRandomAlphaString(leadingChars); baseBuffer.append(leadingString); // Create 5 groups of random number of random chars // Create the string to substitute // Create the pattern string to search for StringBuffer bufferToSub = new StringBuffer(25); StringBuffer bufferToPat = new StringBuffer(50); String[] groups = new String[5]; for(int i=0; i<5; i++) { int aGroupSize = generator.nextInt(5)+1; groups[i] = getRandomAlphaString(aGroupSize); bufferToSub.append(groups[i]); bufferToPat.append('('); bufferToPat.append(groups[i]); bufferToPat.append(')'); } String stringToSub = bufferToSub.toString(); String pattern = bufferToPat.toString(); // Place sub string into working string at random index baseBuffer.append(stringToSub); // Append random chars to end int trailingChars = generator.nextInt(10); String trailingString = getRandomAlphaString(trailingChars); baseBuffer.append(trailingString); String baseString = baseBuffer.toString(); // Create test pattern and matcher Pattern p = Pattern.compile(pattern); Matcher m = p.matcher(baseString); // Reject candidate if pattern happens to start early m.find(); if (m.start() < leadingChars) continue; // Reject candidate if more than one match if (m.find()) continue; // Construct a replacement string with : // random group + random string + random group StringBuffer bufferToRep = new StringBuffer(); int groupIndex1 = generator.nextInt(5); bufferToRep.append("$" + (groupIndex1 + 1)); String randomMidString = getRandomAlphaString(5); bufferToRep.append(randomMidString); int groupIndex2 = generator.nextInt(5); bufferToRep.append("$" + (groupIndex2 + 1)); String replacement = bufferToRep.toString(); // Do the replacement String result = m.replaceAll(replacement); // Construct expected result StringBuffer bufferToRes = new StringBuffer(); bufferToRes.append(leadingString); bufferToRes.append(groups[groupIndex1]); bufferToRes.append(randomMidString); bufferToRes.append(groups[groupIndex2]); bufferToRes.append(trailingString); String expectedResult = bufferToRes.toString(); // Check results if (!result.equals(expectedResult)) failCount++; } report("Substitution Basher"); } /* * 5 groups of characters are created to make a substitution string. * A base string will be created including random lead chars, the * substitution string, and random trailing chars. * A pattern containing the 5 groups is searched for and replaced with: * random group + random string + random group. * The results are checked for correctness. */ private static void substitutionBasher2() { for (int runs = 0; runs<1000; runs++) { // Create a base string to work in int leadingChars = generator.nextInt(10); StringBuilder baseBuffer = new StringBuilder(100); String leadingString = getRandomAlphaString(leadingChars); baseBuffer.append(leadingString); // Create 5 groups of random number of random chars // Create the string to substitute // Create the pattern string to search for StringBuilder bufferToSub = new StringBuilder(25); StringBuilder bufferToPat = new StringBuilder(50); String[] groups = new String[5]; for(int i=0; i<5; i++) { int aGroupSize = generator.nextInt(5)+1; groups[i] = getRandomAlphaString(aGroupSize); bufferToSub.append(groups[i]); bufferToPat.append('('); bufferToPat.append(groups[i]); bufferToPat.append(')'); } String stringToSub = bufferToSub.toString(); String pattern = bufferToPat.toString(); // Place sub string into working string at random index baseBuffer.append(stringToSub); // Append random chars to end int trailingChars = generator.nextInt(10); String trailingString = getRandomAlphaString(trailingChars); baseBuffer.append(trailingString); String baseString = baseBuffer.toString(); // Create test pattern and matcher Pattern p = Pattern.compile(pattern); Matcher m = p.matcher(baseString); // Reject candidate if pattern happens to start early m.find(); if (m.start() < leadingChars) continue; // Reject candidate if more than one match if (m.find()) continue; // Construct a replacement string with : // random group + random string + random group StringBuilder bufferToRep = new StringBuilder(); int groupIndex1 = generator.nextInt(5); bufferToRep.append("$" + (groupIndex1 + 1)); String randomMidString = getRandomAlphaString(5); bufferToRep.append(randomMidString); int groupIndex2 = generator.nextInt(5); bufferToRep.append("$" + (groupIndex2 + 1)); String replacement = bufferToRep.toString(); // Do the replacement String result = m.replaceAll(replacement); // Construct expected result StringBuilder bufferToRes = new StringBuilder(); bufferToRes.append(leadingString); bufferToRes.append(groups[groupIndex1]); bufferToRes.append(randomMidString); bufferToRes.append(groups[groupIndex2]); bufferToRes.append(trailingString); String expectedResult = bufferToRes.toString(); // Check results if (!result.equals(expectedResult)) { failCount++; } } report("Substitution Basher 2"); } /** * Checks the handling of some escape sequences that the Pattern * class should process instead of the java compiler. These are * not in the file because the escapes should be be processed * by the Pattern class when the regex is compiled. */ private static void escapes() throws Exception { Pattern p = Pattern.compile("\\043"); Matcher m = p.matcher("#"); if (!m.find()) failCount++; p = Pattern.compile("\\x23"); m = p.matcher("#"); if (!m.find()) failCount++; p = Pattern.compile("\\u0023"); m = p.matcher("#"); if (!m.find()) failCount++; report("Escape sequences"); } /** * Checks the handling of blank input situations. These * tests are incompatible with my test file format. */ private static void blankInput() throws Exception { Pattern p = Pattern.compile("abc", Pattern.CASE_INSENSITIVE); Matcher m = p.matcher(""); if (m.find()) failCount++; p = Pattern.compile("a*", Pattern.CASE_INSENSITIVE); m = p.matcher(""); if (!m.find()) failCount++; p = Pattern.compile("abc"); m = p.matcher(""); if (m.find()) failCount++; p = Pattern.compile("a*"); m = p.matcher(""); if (!m.find()) failCount++; report("Blank input"); } /** * Tests the Boyer-Moore pattern matching of a character sequence * on randomly generated patterns. */ private static void bm() throws Exception { doBnM('a'); report("Boyer Moore (ASCII)"); doBnM(Character.MIN_SUPPLEMENTARY_CODE_POINT - 10); report("Boyer Moore (Supplementary)"); } private static void doBnM(int baseCharacter) throws Exception { int achar=0; for (int i=0; i<100; i++) { // Create a short pattern to search for int patternLength = generator.nextInt(7) + 4; StringBuffer patternBuffer = new StringBuffer(patternLength); String pattern; retry: for (;;) { for (int x=0; xy+)z+"), "xxxyyyzzz", "gname", "yyy"); check(Pattern.compile("x+(?y+)z+"), "xxxyyyzzz", "gname8", "yyy"); //backref Pattern pattern = Pattern.compile("(a*)bc\\1"); check(pattern, "zzzaabcazzz", true); // found "abca" check(Pattern.compile("(?a*)bc\\k"), "zzzaabcaazzz", true); check(Pattern.compile("(?abc)(def)\\k"), "abcdefabc", true); check(Pattern.compile("(a)(b)(c)(d)(e)(f)(g)(h)(i)(j)(?k)\\k"), "abcdefghijkk", true); // Supplementary character tests check(Pattern.compile("(?" + toSupplementaries("a*)bc") + "\\k"), toSupplementaries("zzzaabcazzz"), true); check(Pattern.compile("(?" + toSupplementaries("a*)bc") + "\\k"), toSupplementaries("zzzaabcaazzz"), true); check(Pattern.compile("(?" + toSupplementaries("abc)(def)") + "\\k"), toSupplementaries("abcdefabc"), true); check(Pattern.compile(toSupplementaries("(a)(b)(c)(d)(e)(f)(g)(h)(i)(j)") + "(?" + toSupplementaries("k)") + "\\k"), toSupplementaries("abcdefghijkk"), true); check(Pattern.compile("x+(?y+)z+\\k"), "xxxyyyzzzyyy", "gname", "yyy"); //replaceFirst/All checkReplaceFirst("(?ab)(c*)", "abccczzzabcczzzabccc", "${gn}", "abzzzabcczzzabccc"); checkReplaceAll("(?ab)(c*)", "abccczzzabcczzzabccc", "${gn}", "abzzzabzzzab"); checkReplaceFirst("(?ab)(c*)", "zzzabccczzzabcczzzabccczzz", "${gn}", "zzzabzzzabcczzzabccczzz"); checkReplaceAll("(?ab)(c*)", "zzzabccczzzabcczzzabccczzz", "${gn}", "zzzabzzzabzzzabzzz"); checkReplaceFirst("(?ab)(?c*)", "zzzabccczzzabcczzzabccczzz", "${gn2}", "zzzccczzzabcczzzabccczzz"); checkReplaceAll("(?ab)(?c*)", "zzzabccczzzabcczzzabccczzz", "${gn2}", "zzzccczzzcczzzccczzz"); //toSupplementaries("(ab)(c*)")); checkReplaceFirst("(?" + toSupplementaries("ab") + ")(?" + toSupplementaries("c") + "*)", toSupplementaries("abccczzzabcczzzabccc"), "${gn1}", toSupplementaries("abzzzabcczzzabccc")); checkReplaceAll("(?" + toSupplementaries("ab") + ")(?" + toSupplementaries("c") + "*)", toSupplementaries("abccczzzabcczzzabccc"), "${gn1}", toSupplementaries("abzzzabzzzab")); checkReplaceFirst("(?" + toSupplementaries("ab") + ")(?" + toSupplementaries("c") + "*)", toSupplementaries("abccczzzabcczzzabccc"), "${gn2}", toSupplementaries("ccczzzabcczzzabccc")); checkReplaceAll("(?" + toSupplementaries("ab") + ")(?" + toSupplementaries("c") + "*)", toSupplementaries("abccczzzabcczzzabccc"), "${gn2}", toSupplementaries("ccczzzcczzzccc")); checkReplaceFirst("(?Dog)AndCat", "zzzDogAndCatzzzDogAndCatzzz", "${dog}", "zzzDogzzzDogAndCatzzz"); checkReplaceAll("(?Dog)AndCat", "zzzDogAndCatzzzDogAndCatzzz", "${dog}", "zzzDogzzzDogzzz"); // backref in Matcher & String if (!"abcdefghij".replaceFirst("cd(?ef)gh", "${gn}").equals("abefij") || !"abbbcbdbefgh".replaceAll("(?[a-e])b", "${gn}").equals("abcdefgh")) failCount++; // negative checkExpectedFail("(?abc)(def)"); checkExpectedFail("(?abc)(def)"); checkExpectedFail("(?<6groupnamestartswithdigit>abc)(def)"); checkExpectedFail("(?abc)(def)\\k"); checkExpectedFail("(?abc)(?def)\\k"); checkExpectedIAE(Pattern.compile("(?abc)(def)").matcher("abcdef"), "gnameX"); checkExpectedNPE(Pattern.compile("(?abc)(def)").matcher("abcdef")); report("NamedGroupCapture"); } // This is for bug 6919132 private static void nonBmpClassComplementTest() throws Exception { Pattern p = Pattern.compile("\\P{Lu}"); Matcher m = p.matcher(new String(new int[] {0x1d400}, 0, 1)); if (m.find() && m.start() == 1) failCount++; // from a unicode category p = Pattern.compile("\\P{Lu}"); m = p.matcher(new String(new int[] {0x1d400}, 0, 1)); if (m.find()) failCount++; if (!m.hitEnd()) failCount++; // block p = Pattern.compile("\\P{InMathematicalAlphanumericSymbols}"); m = p.matcher(new String(new int[] {0x1d400}, 0, 1)); if (m.find() && m.start() == 1) failCount++; p = Pattern.compile("\\P{sc=GRANTHA}"); m = p.matcher(new String(new int[] {0x11350}, 0, 1)); if (m.find() && m.start() == 1) failCount++; report("NonBmpClassComplement"); } private static void unicodePropertiesTest() throws Exception { // different forms if (!Pattern.compile("\\p{IsLu}").matcher("A").matches() || !Pattern.compile("\\p{Lu}").matcher("A").matches() || !Pattern.compile("\\p{gc=Lu}").matcher("A").matches() || !Pattern.compile("\\p{general_category=Lu}").matcher("A").matches() || !Pattern.compile("\\p{IsLatin}").matcher("B").matches() || !Pattern.compile("\\p{sc=Latin}").matcher("B").matches() || !Pattern.compile("\\p{script=Latin}").matcher("B").matches() || !Pattern.compile("\\p{InBasicLatin}").matcher("c").matches() || !Pattern.compile("\\p{blk=BasicLatin}").matcher("c").matches() || !Pattern.compile("\\p{block=BasicLatin}").matcher("c").matches()) failCount++; Matcher common = Pattern.compile("\\p{script=Common}").matcher(""); Matcher unknown = Pattern.compile("\\p{IsUnknown}").matcher(""); Matcher lastSM = common; Character.UnicodeScript lastScript = Character.UnicodeScript.of(0); Matcher latin = Pattern.compile("\\p{block=basic_latin}").matcher(""); Matcher greek = Pattern.compile("\\p{InGreek}").matcher(""); Matcher lastBM = latin; Character.UnicodeBlock lastBlock = Character.UnicodeBlock.of(0); for (int cp = 1; cp < Character.MAX_CODE_POINT; cp++) { if (cp >= 0x30000 && (cp & 0x70) == 0){ continue; // only pick couple code points, they are the same } // Unicode Script Character.UnicodeScript script = Character.UnicodeScript.of(cp); Matcher m; String str = new String(Character.toChars(cp)); if (script == lastScript) { m = lastSM; m.reset(str); } else { m = Pattern.compile("\\p{Is" + script.name() + "}").matcher(str); } if (!m.matches()) { failCount++; } Matcher other = (script == Character.UnicodeScript.COMMON)? unknown : common; other.reset(str); if (other.matches()) { failCount++; } lastSM = m; lastScript = script; // Unicode Block Character.UnicodeBlock block = Character.UnicodeBlock.of(cp); if (block == null) { //System.out.printf("Not a Block: cp=%x%n", cp); continue; } if (block == lastBlock) { m = lastBM; m.reset(str); } else { m = Pattern.compile("\\p{block=" + block.toString() + "}").matcher(str); } if (!m.matches()) { failCount++; } other = (block == Character.UnicodeBlock.BASIC_LATIN)? greek : latin; other.reset(str); if (other.matches()) { failCount++; } lastBM = m; lastBlock = block; } report("unicodeProperties"); } private static void unicodeHexNotationTest() throws Exception { // negative checkExpectedFail("\\x{-23}"); checkExpectedFail("\\x{110000}"); checkExpectedFail("\\x{}"); checkExpectedFail("\\x{AB[ef]"); // codepoint check("^\\x{1033c}$", "\uD800\uDF3C", true); check("^\\xF0\\x90\\x8C\\xBC$", "\uD800\uDF3C", false); check("^\\x{D800}\\x{DF3c}+$", "\uD800\uDF3C", false); check("^\\xF0\\x90\\x8C\\xBC$", "\uD800\uDF3C", false); // in class check("^[\\x{D800}\\x{DF3c}]+$", "\uD800\uDF3C", false); check("^[\\xF0\\x90\\x8C\\xBC]+$", "\uD800\uDF3C", false); check("^[\\x{D800}\\x{DF3C}]+$", "\uD800\uDF3C", false); check("^[\\x{DF3C}\\x{D800}]+$", "\uD800\uDF3C", false); check("^[\\x{D800}\\x{DF3C}]+$", "\uDF3C\uD800", true); check("^[\\x{DF3C}\\x{D800}]+$", "\uDF3C\uD800", true); for (int cp = 0; cp <= 0x10FFFF; cp++) { String s = "A" + new String(Character.toChars(cp)) + "B"; String hexUTF16 = (cp <= 0xFFFF)? String.format("\\u%04x", cp) : String.format("\\u%04x\\u%04x", (int) Character.toChars(cp)[0], (int) Character.toChars(cp)[1]); String hexCodePoint = "\\x{" + Integer.toHexString(cp) + "}"; if (!Pattern.matches("A" + hexUTF16 + "B", s)) failCount++; if (!Pattern.matches("A[" + hexUTF16 + "]B", s)) failCount++; if (!Pattern.matches("A" + hexCodePoint + "B", s)) failCount++; if (!Pattern.matches("A[" + hexCodePoint + "]B", s)) failCount++; } report("unicodeHexNotation"); } private static void unicodeClassesTest() throws Exception { Matcher lower = Pattern.compile("\\p{Lower}").matcher(""); Matcher upper = Pattern.compile("\\p{Upper}").matcher(""); Matcher ASCII = Pattern.compile("\\p{ASCII}").matcher(""); Matcher alpha = Pattern.compile("\\p{Alpha}").matcher(""); Matcher digit = Pattern.compile("\\p{Digit}").matcher(""); Matcher alnum = Pattern.compile("\\p{Alnum}").matcher(""); Matcher punct = Pattern.compile("\\p{Punct}").matcher(""); Matcher graph = Pattern.compile("\\p{Graph}").matcher(""); Matcher print = Pattern.compile("\\p{Print}").matcher(""); Matcher blank = Pattern.compile("\\p{Blank}").matcher(""); Matcher cntrl = Pattern.compile("\\p{Cntrl}").matcher(""); Matcher xdigit = Pattern.compile("\\p{XDigit}").matcher(""); Matcher space = Pattern.compile("\\p{Space}").matcher(""); Matcher bound = Pattern.compile("\\b").matcher(""); Matcher word = Pattern.compile("\\w++").matcher(""); // UNICODE_CHARACTER_CLASS Matcher lowerU = Pattern.compile("\\p{Lower}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher upperU = Pattern.compile("\\p{Upper}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher ASCIIU = Pattern.compile("\\p{ASCII}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher alphaU = Pattern.compile("\\p{Alpha}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher digitU = Pattern.compile("\\p{Digit}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher alnumU = Pattern.compile("\\p{Alnum}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher punctU = Pattern.compile("\\p{Punct}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher graphU = Pattern.compile("\\p{Graph}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher printU = Pattern.compile("\\p{Print}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher blankU = Pattern.compile("\\p{Blank}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher cntrlU = Pattern.compile("\\p{Cntrl}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher xdigitU = Pattern.compile("\\p{XDigit}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher spaceU = Pattern.compile("\\p{Space}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher boundU = Pattern.compile("\\b", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher wordU = Pattern.compile("\\w", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); // embedded flag (?U) Matcher lowerEU = Pattern.compile("(?U)\\p{Lower}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher graphEU = Pattern.compile("(?U)\\p{Graph}", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher wordEU = Pattern.compile("(?U)\\w", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher bwb = Pattern.compile("\\b\\w\\b").matcher(""); Matcher bwbU = Pattern.compile("\\b\\w++\\b", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); Matcher bwbEU = Pattern.compile("(?U)\\b\\w++\\b", Pattern.UNICODE_CHARACTER_CLASS).matcher(""); // properties Matcher lowerP = Pattern.compile("\\p{IsLowerCase}").matcher(""); Matcher upperP = Pattern.compile("\\p{IsUpperCase}").matcher(""); Matcher titleP = Pattern.compile("\\p{IsTitleCase}").matcher(""); Matcher letterP = Pattern.compile("\\p{IsLetter}").matcher(""); Matcher alphaP = Pattern.compile("\\p{IsAlphabetic}").matcher(""); Matcher ideogP = Pattern.compile("\\p{IsIdeographic}").matcher(""); Matcher cntrlP = Pattern.compile("\\p{IsControl}").matcher(""); Matcher spaceP = Pattern.compile("\\p{IsWhiteSpace}").matcher(""); Matcher definedP = Pattern.compile("\\p{IsAssigned}").matcher(""); Matcher nonCCPP = Pattern.compile("\\p{IsNoncharacterCodePoint}").matcher(""); Matcher joinCrtl = Pattern.compile("\\p{IsJoinControl}").matcher(""); // javaMethod Matcher lowerJ = Pattern.compile("\\p{javaLowerCase}").matcher(""); Matcher upperJ = Pattern.compile("\\p{javaUpperCase}").matcher(""); Matcher alphaJ = Pattern.compile("\\p{javaAlphabetic}").matcher(""); Matcher ideogJ = Pattern.compile("\\p{javaIdeographic}").matcher(""); // GC/C Matcher gcC = Pattern.compile("\\p{C}").matcher(""); for (int cp = 1; cp < 0x30000; cp++) { String str = new String(Character.toChars(cp)); int type = Character.getType(cp); if (// lower POSIX_ASCII.isLower(cp) != lower.reset(str).matches() || Character.isLowerCase(cp) != lowerU.reset(str).matches() || Character.isLowerCase(cp) != lowerP.reset(str).matches() || Character.isLowerCase(cp) != lowerEU.reset(str).matches()|| Character.isLowerCase(cp) != lowerJ.reset(str).matches()|| // upper POSIX_ASCII.isUpper(cp) != upper.reset(str).matches() || POSIX_Unicode.isUpper(cp) != upperU.reset(str).matches() || Character.isUpperCase(cp) != upperP.reset(str).matches() || Character.isUpperCase(cp) != upperJ.reset(str).matches() || // alpha POSIX_ASCII.isAlpha(cp) != alpha.reset(str).matches() || POSIX_Unicode.isAlpha(cp) != alphaU.reset(str).matches() || Character.isAlphabetic(cp)!= alphaP.reset(str).matches() || Character.isAlphabetic(cp)!= alphaJ.reset(str).matches() || // digit POSIX_ASCII.isDigit(cp) != digit.reset(str).matches() || Character.isDigit(cp) != digitU.reset(str).matches() || // alnum POSIX_ASCII.isAlnum(cp) != alnum.reset(str).matches() || POSIX_Unicode.isAlnum(cp) != alnumU.reset(str).matches() || // punct POSIX_ASCII.isPunct(cp) != punct.reset(str).matches() || POSIX_Unicode.isPunct(cp) != punctU.reset(str).matches() || // graph POSIX_ASCII.isGraph(cp) != graph.reset(str).matches() || POSIX_Unicode.isGraph(cp) != graphU.reset(str).matches() || POSIX_Unicode.isGraph(cp) != graphEU.reset(str).matches()|| // blank POSIX_ASCII.isType(cp, POSIX_ASCII.BLANK) != blank.reset(str).matches() || POSIX_Unicode.isBlank(cp) != blankU.reset(str).matches() || // print POSIX_ASCII.isPrint(cp) != print.reset(str).matches() || POSIX_Unicode.isPrint(cp) != printU.reset(str).matches() || // cntrl POSIX_ASCII.isCntrl(cp) != cntrl.reset(str).matches() || POSIX_Unicode.isCntrl(cp) != cntrlU.reset(str).matches() || (Character.CONTROL == type) != cntrlP.reset(str).matches() || // hexdigit POSIX_ASCII.isHexDigit(cp) != xdigit.reset(str).matches() || POSIX_Unicode.isHexDigit(cp) != xdigitU.reset(str).matches() || // space POSIX_ASCII.isSpace(cp) != space.reset(str).matches() || POSIX_Unicode.isSpace(cp) != spaceU.reset(str).matches() || POSIX_Unicode.isSpace(cp) != spaceP.reset(str).matches() || // word POSIX_ASCII.isWord(cp) != word.reset(str).matches() || POSIX_Unicode.isWord(cp) != wordU.reset(str).matches() || POSIX_Unicode.isWord(cp) != wordEU.reset(str).matches()|| // bwordb POSIX_ASCII.isWord(cp) != bwb.reset(str).matches() || POSIX_Unicode.isWord(cp) != bwbU.reset(str).matches() || // properties Character.isTitleCase(cp) != titleP.reset(str).matches() || Character.isLetter(cp) != letterP.reset(str).matches()|| Character.isIdeographic(cp) != ideogP.reset(str).matches() || Character.isIdeographic(cp) != ideogJ.reset(str).matches() || (Character.UNASSIGNED == type) == definedP.reset(str).matches() || POSIX_Unicode.isNoncharacterCodePoint(cp) != nonCCPP.reset(str).matches() || POSIX_Unicode.isJoinControl(cp) != joinCrtl.reset(str).matches() || // gc_C (Character.CONTROL == type || Character.FORMAT == type || Character.PRIVATE_USE == type || Character.SURROGATE == type || Character.UNASSIGNED == type) != gcC.reset(str).matches()) { failCount++; } } // bounds/word align twoFindIndexes(" \u0180sherman\u0400 ", bound, 1, 10); if (!bwbU.reset("\u0180sherman\u0400").matches()) failCount++; twoFindIndexes(" \u0180sh\u0345erman\u0400 ", bound, 1, 11); if (!bwbU.reset("\u0180sh\u0345erman\u0400").matches()) failCount++; twoFindIndexes(" \u0724\u0739\u0724 ", bound, 1, 4); if (!bwbU.reset("\u0724\u0739\u0724").matches()) failCount++; if (!bwbEU.reset("\u0724\u0739\u0724").matches()) failCount++; report("unicodePredefinedClasses"); } private static void unicodeCharacterNameTest() throws Exception { for (int cp = 0; cp < Character.MAX_CODE_POINT; cp++) { if (!Character.isValidCodePoint(cp) || Character.getType(cp) == Character.UNASSIGNED) continue; String str = new String(Character.toChars(cp)); // single String p = "\\N{" + Character.getName(cp) + "}"; if (!Pattern.compile(p).matcher(str).matches()) { failCount++; } // class[c] p = "[\\N{" + Character.getName(cp) + "}]"; if (!Pattern.compile(p).matcher(str).matches()) { failCount++; } } // range for (int i = 0; i < 10; i++) { int start = generator.nextInt(20); int end = start + generator.nextInt(200); String p = "[\\N{" + Character.getName(start) + "}-\\N{" + Character.getName(end) + "}]"; String str; for (int cp = start; cp < end; cp++) { str = new String(Character.toChars(cp)); if (!Pattern.compile(p).matcher(str).matches()) { failCount++; } } str = new String(Character.toChars(end + 10)); if (Pattern.compile(p).matcher(str).matches()) { failCount++; } } // slice for (int i = 0; i < 10; i++) { int n = generator.nextInt(256); int[] buf = new int[n]; StringBuffer sb = new StringBuffer(1024); for (int j = 0; j < n; j++) { int cp = generator.nextInt(1000); if (!Character.isValidCodePoint(cp) || Character.getType(cp) == Character.UNASSIGNED) cp = 0x4e00; // just use 4e00 sb.append("\\N{" + Character.getName(cp) + "}"); buf[j] = cp; } String p = sb.toString(); String str = new String(buf, 0, buf.length); if (!Pattern.compile(p).matcher(str).matches()) { failCount++; } } report("unicodeCharacterName"); } private static void horizontalAndVerticalWSTest() throws Exception { String hws = new String (new char[] { 0x09, 0x20, 0xa0, 0x1680, 0x180e, 0x2000, 0x2001, 0x2002, 0x2003, 0x2004, 0x2005, 0x2006, 0x2007, 0x2008, 0x2009, 0x200a, 0x202f, 0x205f, 0x3000 }); String vws = new String (new char[] { 0x0a, 0x0b, 0x0c, 0x0d, 0x85, 0x2028, 0x2029 }); if (!Pattern.compile("\\h+").matcher(hws).matches() || !Pattern.compile("[\\h]+").matcher(hws).matches()) failCount++; if (Pattern.compile("\\H").matcher(hws).find() || Pattern.compile("[\\H]").matcher(hws).find()) failCount++; if (!Pattern.compile("\\v+").matcher(vws).matches() || !Pattern.compile("[\\v]+").matcher(vws).matches()) failCount++; if (Pattern.compile("\\V").matcher(vws).find() || Pattern.compile("[\\V]").matcher(vws).find()) failCount++; String prefix = "abcd"; String suffix = "efgh"; String ng = "A"; for (int i = 0; i < hws.length(); i++) { String c = String.valueOf(hws.charAt(i)); Matcher m = Pattern.compile("\\h").matcher(prefix + c + suffix); if (!m.find() || !c.equals(m.group())) failCount++; m = Pattern.compile("[\\h]").matcher(prefix + c + suffix); if (!m.find() || !c.equals(m.group())) failCount++; m = Pattern.compile("\\H").matcher(hws.substring(0, i) + ng + hws.substring(i)); if (!m.find() || !ng.equals(m.group())) failCount++; m = Pattern.compile("[\\H]").matcher(hws.substring(0, i) + ng + hws.substring(i)); if (!m.find() || !ng.equals(m.group())) failCount++; } for (int i = 0; i < vws.length(); i++) { String c = String.valueOf(vws.charAt(i)); Matcher m = Pattern.compile("\\v").matcher(prefix + c + suffix); if (!m.find() || !c.equals(m.group())) failCount++; m = Pattern.compile("[\\v]").matcher(prefix + c + suffix); if (!m.find() || !c.equals(m.group())) failCount++; m = Pattern.compile("\\V").matcher(vws.substring(0, i) + ng + vws.substring(i)); if (!m.find() || !ng.equals(m.group())) failCount++; m = Pattern.compile("[\\V]").matcher(vws.substring(0, i) + ng + vws.substring(i)); if (!m.find() || !ng.equals(m.group())) failCount++; } // \v in range is interpreted as 0x0B. This is the undocumented behavior if (!Pattern.compile("[\\v-\\v]").matcher(String.valueOf((char)0x0B)).matches()) failCount++; report("horizontalAndVerticalWSTest"); } private static void linebreakTest() throws Exception { String linebreaks = new String (new char[] { 0x0A, 0x0B, 0x0C, 0x0D, 0x85, 0x2028, 0x2029 }); String crnl = "\r\n"; if (!(Pattern.compile("\\R+").matcher(linebreaks).matches() && Pattern.compile("\\R").matcher(crnl).matches() && Pattern.compile("\\Rabc").matcher(crnl + "abc").matches() && Pattern.compile("\\Rabc").matcher("\rabc").matches() && Pattern.compile("\\R\\R").matcher(crnl).matches() && // backtracking Pattern.compile("\\R\\n").matcher(crnl).matches()) && // backtracking !Pattern.compile("((? p = Pattern.compile("[a-z]+").asPredicate(); if (p.test("")) { failCount++; } if (!p.test("word")) { failCount++; } if (p.test("1234")) { failCount++; } if (!p.test("word1234")) { failCount++; } report("Pattern.asPredicate"); } // This test is for 8184692 private static void patternAsMatchPredicate() throws Exception { Predicate p = Pattern.compile("[a-z]+").asMatchPredicate(); if (p.test("")) { failCount++; } if (!p.test("word")) { failCount++; } if (p.test("1234word")) { failCount++; } if (p.test("1234")) { failCount++; } report("Pattern.asMatchPredicate"); } // This test is for 8035975 private static void invalidFlags() throws Exception { for (int flag = 1; flag != 0; flag <<= 1) { switch (flag) { case Pattern.CASE_INSENSITIVE: case Pattern.MULTILINE: case Pattern.DOTALL: case Pattern.UNICODE_CASE: case Pattern.CANON_EQ: case Pattern.UNIX_LINES: case Pattern.LITERAL: case Pattern.UNICODE_CHARACTER_CLASS: case Pattern.COMMENTS: // valid flag, continue break; default: try { Pattern.compile(".", flag); failCount++; } catch (IllegalArgumentException expected) { } } } report("Invalid compile flags"); } // This test is for 8158482 private static void embeddedFlags() throws Exception { try { Pattern.compile("(?i).(?-i)."); Pattern.compile("(?m).(?-m)."); Pattern.compile("(?s).(?-s)."); Pattern.compile("(?d).(?-d)."); Pattern.compile("(?u).(?-u)."); Pattern.compile("(?c).(?-c)."); Pattern.compile("(?x).(?-x)."); Pattern.compile("(?U).(?-U)."); Pattern.compile("(?imsducxU).(?-imsducxU)."); } catch (PatternSyntaxException x) { failCount++; } report("Embedded flags"); } private static void grapheme() throws Exception { Files.lines(UCDFiles.GRAPHEME_BREAK_TEST) .filter( ln -> ln.length() != 0 && !ln.startsWith("#") ) .forEach( ln -> { ln = ln.replaceAll("\\s+|\\([a-zA-Z]+\\)|\\[[a-zA-Z]]+\\]|#.*", ""); // System.out.println(str); String[] strs = ln.split("\u00f7|\u00d7"); StringBuilder src = new StringBuilder(); ArrayList graphemes = new ArrayList<>(); StringBuilder buf = new StringBuilder(); int offBk = 0; for (String str : strs) { if (str.length() == 0) // first empty str continue; int cp = Integer.parseInt(str, 16); src.appendCodePoint(cp); buf.appendCodePoint(cp); offBk += (str.length() + 1); if (ln.charAt(offBk) == '\u00f7') { // DIV graphemes.add(buf.toString()); buf = new StringBuilder(); } } Pattern p = Pattern.compile("\\X"); Matcher m = p.matcher(src.toString()); Scanner s = new Scanner(src.toString()).useDelimiter("\\b{g}"); for (String g : graphemes) { // System.out.printf(" grapheme:=[%s]%n", g); // (1) test \\X directly if (!m.find() || !m.group().equals(g)) { System.out.println("Failed \\X [" + ln + "] : " + g); failCount++; } // (2) test \\b{g} + \\X via Scanner boolean hasNext = s.hasNext(p); // if (!s.hasNext() || !s.next().equals(next)) { if (!s.hasNext(p) || !s.next(p).equals(g)) { System.out.println("Failed b{g} [" + ln + "] : " + g); failCount++; } } }); // some sanity checks if (!Pattern.compile("\\X{10}").matcher("abcdefghij").matches() || !Pattern.compile("\\b{g}(?:\\X\\b{g}){5}\\b{g}").matcher("abcde").matches() || !Pattern.compile("(?:\\X\\b{g}){2}").matcher("\ud800\udc00\ud801\udc02").matches()) failCount++; // make sure "\b{n}" still works if (!Pattern.compile("\\b{1}hello\\b{1} \\b{1}world\\b{1}").matcher("hello world").matches()) failCount++; report("Unicode extended grapheme cluster"); } // hangup/timeout if go into exponential backtracking private static void expoBacktracking() throws Exception { Object[][] patternMatchers = { // 6328855 { "(.*\n*)*", "this little fine string lets\r\njava.lang.String.matches\r\ncrash\r\n(We don't know why but adding \r* to the regex makes it work again)", false }, // 6192895 { " *([a-zA-Z0-9/\\-\\?:\\(\\)\\.,'\\+\\{\\}]+ *)+", "Hello World this is a test this is a test this is a test A", true }, { " *([a-zA-Z0-9/\\-\\?:\\(\\)\\.,'\\+\\{\\}]+ *)+", "Hello World this is a test this is a test this is a test \u4e00 ", false }, { " *([a-z0-9]+ *)+", "hello world this is a test this is a test this is a test A", false }, // 4771934 [FIXED] #5013651? { "^(\\w+([\\.-]?\\w+)*@\\w+([\\.-]?\\w+)*(\\.\\w{2,4})+[,;]?)+$", "abc@efg.abc,efg@abc.abc,abc@xyz.mno;abc@sdfsd.com", true }, // 4866249 [FIXED] { "<\\s*" + "(meta|META)" + "(\\s|[^>])+" + "(CHARSET|charset)=" + "(\\s|[^>])+>", "", true }, { "^(\\w+([\\.-]?\\w+)*@\\w+([\\.-]?\\w+)*(\\.\\w{2,4})+[,;]?)+$", "abc@efg.abc,efg@abc.abc,abc@xyz.mno;sdfsd.com", false }, // 6345469 { "((<[^>]+>)?(((\\s)?)*(\\ )?)*((\\s)?)*)+", "   < br/>   < / p>

 

", true }, // --> matched { "((<[^>]+>)?(((\\s)?)*(\\ )?)*((\\s)?)*)+", "   < br/>   < / p>

  p

", false }, // 5026912 { "^\\s*" + "(\\w|\\d|[\\xC0-\\xFF]|/)+" + "\\s+|$", "156580451111112225588087755221111111566969655555555", false}, // 6988218 { "^([+-]?((0[xX](\\p{XDigit}+))|(((\\p{Digit}+)(\\.)?((\\p{Digit}+)?)([eE][+-]?(\\p{Digit}+))?)|(\\.((\\p{Digit}+))([eE][+-]?(\\p{Digit}+))?)))|[n|N]?'([^']*(?:'')*[^']*)*')", "'%)) order by ANGEBOT.ID", false}, // find // 6693451 { "^(\\s*foo\\s*)*$", "foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo", true }, { "^(\\s*foo\\s*)*$", "foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo foo fo", false }, // 7006761 { "(([0-9A-Z]+)([_]?+)*)*", "FOOOOO_BAAAR_FOOOOOOOOO_BA_", true}, { "(([0-9A-Z]+)([_]?+)*)*", "FOOOOO_BAAAR_FOOOOOOOOO_BA_ ", false}, // 8140212 { "(?.*)\\{(?\\w+):(?\\w+(\\.?\\w+(\\(((?(('[^']*')|((/|\\w)+))(,(('[^']*')|((/|\\w)+)))*))?\\))?)*)\\}(?.*)", "{CeGlobal:getSodCutoff.getGui.getAmqp.getSimpleModeEnabled()", false }, { "^(a+)+$", "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa", true}, { "^(a+)+$", "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa!", false}, { "(x+)*y", "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxy", true }, { "(x+)*y", "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxz", false}, { "(x+x+)+y", "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxy", true}, { "(x+x+)+y", "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxz", false}, { "(([0-9A-Z]+)([_]?+)*)*", "--------------------------------------", false}, /* not fixed //8132141 ---> second level exponential backtracking { "(h|h|ih(((i|a|c|c|a|i|i|j|b|a|i|b|a|a|j))+h)ahbfhba|c|i)*", "hchcchicihcchciiicichhcichcihcchiihichiciiiihhcchicchhcihchcihiihciichhccciccichcichiihcchcihhicchcciicchcccihiiihhihihihichicihhcciccchihhhcchichchciihiicihciihcccciciccicciiiiiiiiicihhhiiiihchccchchhhhiiihchihcccchhhiiiiiiiicicichicihcciciihichhhhchihciiihhiccccccciciihhichiccchhicchicihihccichicciihcichccihhiciccccccccichhhhihihhcchchihihiihhihihihicichihiiiihhhhihhhchhichiicihhiiiiihchccccchichci" }, */ }; for (Object[] pm : patternMatchers) { String p = (String)pm[0]; String s = (String)pm[1]; boolean r = (Boolean)pm[2]; if (r != Pattern.compile(p).matcher(s).matches()) { failCount++; } } } private static void invalidGroupName() { // Invalid start of a group name for (String groupName : List.of("", ".", "0", "\u0040", "\u005b", "\u0060", "\u007b", "\u0416")) { for (String pat : List.of("(?<" + groupName + ">)", "\\k<" + groupName + ">")) { try { Pattern.compile(pat); failCount++; } catch (PatternSyntaxException e) { if (!e.getMessage().startsWith( "capturing group name does not start with a" + " Latin letter")) { failCount++; } } } } // Invalid char in a group name for (String groupName : List.of("a.", "b\u0040", "c\u005b", "d\u0060", "e\u007b", "f\u0416")) { for (String pat : List.of("(?<" + groupName + ">)", "\\k<" + groupName + ">")) { try { Pattern.compile(pat); failCount++; } catch (PatternSyntaxException e) { if (!e.getMessage().startsWith( "named capturing group is missing trailing '>'")) { failCount++; } } } } report("Invalid capturing group names"); } private static void illegalRepetitionRange() { // huge integers > (2^31 - 1) String n = BigInteger.valueOf(1L << 32) .toString(); String m = BigInteger.valueOf(1L << 31) .add(new BigInteger(80, generator)) .toString(); for (String rep : List.of("", "x", ".", ",", "-1", "2,1", n, n + ",", "0," + n, n + "," + m, m, m + ",", "0," + m)) { String pat = ".{" + rep + "}"; try { Pattern.compile(pat); failCount++; System.out.println("Expected to fail. Pattern: " + pat); } catch (PatternSyntaxException e) { if (!e.getMessage().startsWith("Illegal repetition")) { failCount++; System.out.println("Unexpected error message: " + e.getMessage()); } } catch (Throwable t) { failCount++; System.out.println("Unexpected exception: " + t); } } report("illegalRepetitionRange"); } }