b337f63361
Reviewed-by: rriggs
5294 lines
196 KiB
Java
5294 lines
196 KiB
Java
/*
|
|
* Copyright (c) 1999, 2021, 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 8228352 8230829 8236034 8235812
|
|
* 8216332 8214245 8237599 8241055 8247546 8258259 8037397
|
|
*
|
|
* @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.nio.file.Path;
|
|
import java.nio.file.Paths;
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
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 java.util.stream.IntStream;
|
|
import java.util.stream.Stream;
|
|
|
|
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();
|
|
surrogatePairWithCanonEq();
|
|
lineBreakWithQuantifier();
|
|
caseInsensitivePMatch();
|
|
surrogatePairOverlapRegion();
|
|
droppedClassesWithIntersection();
|
|
|
|
|
|
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<length; i++) {
|
|
char randChar = (char)(97 + generator.nextInt(26));
|
|
buf.append(randChar);
|
|
}
|
|
return buf.toString();
|
|
}
|
|
|
|
private static void check(Matcher m, String expected) {
|
|
m.find();
|
|
if (!m.group().equals(expected))
|
|
failCount++;
|
|
}
|
|
|
|
private static void check(Matcher m, String result, boolean expected) {
|
|
m.find();
|
|
if (m.group().equals(result) != expected)
|
|
failCount++;
|
|
}
|
|
|
|
private static void check(Pattern p, String s, boolean expected) {
|
|
if (p.matcher(s).find() != expected)
|
|
failCount++;
|
|
}
|
|
|
|
private static void check(String p, String s, boolean expected) {
|
|
Matcher matcher = Pattern.compile(p).matcher(s);
|
|
if (matcher.find() != expected)
|
|
failCount++;
|
|
}
|
|
|
|
private static void check(String p, char c, boolean expected) {
|
|
String propertyPattern = expected ? "\\p" + p : "\\P" + p;
|
|
Pattern pattern = Pattern.compile(propertyPattern);
|
|
char[] ca = new char[1]; ca[0] = c;
|
|
Matcher matcher = pattern.matcher(new String(ca));
|
|
if (!matcher.find())
|
|
failCount++;
|
|
}
|
|
|
|
private static void check(String p, int codePoint, boolean expected) {
|
|
String propertyPattern = expected ? "\\p" + p : "\\P" + p;
|
|
Pattern pattern = Pattern.compile(propertyPattern);
|
|
char[] ca = Character.toChars(codePoint);
|
|
Matcher matcher = pattern.matcher(new String(ca));
|
|
if (!matcher.find())
|
|
failCount++;
|
|
}
|
|
|
|
private static void check(String p, int flag, String input, String s,
|
|
boolean expected)
|
|
{
|
|
Pattern pattern = Pattern.compile(p, flag);
|
|
Matcher matcher = pattern.matcher(input);
|
|
if (expected)
|
|
check(matcher, s, expected);
|
|
else
|
|
check(pattern, input, false);
|
|
}
|
|
|
|
private static void report(String testName) {
|
|
int spacesToAdd = 30 - testName.length();
|
|
StringBuffer paddedNameBuffer = new StringBuffer(testName);
|
|
for (int i=0; i<spacesToAdd; i++)
|
|
paddedNameBuffer.append(" ");
|
|
String paddedName = paddedNameBuffer.toString();
|
|
System.err.println(paddedName + ": " +
|
|
(failCount==0 ? "Passed":"Failed("+failCount+")"));
|
|
if (failCount > 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<MatchResult, String>)null));
|
|
check(() -> m.replaceFirst((String)null));
|
|
check(() -> m.replaceFirst((Function<MatchResult, String>) 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<String> result = new ArrayList<String>();
|
|
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("(?<!abc )\\bfoo", "abc foo", new String[0]);
|
|
check("(?<!abc \\b)foo", "abc foo", new String[0]);
|
|
|
|
//Negative
|
|
check("(?<!%.{0,5})foo\\d",
|
|
"%foo1\n%bar foo2\n%bar foo3\n%blahblah foo4\nfoo5",
|
|
new String[] {"foo4", "foo5"});
|
|
|
|
//Positive greedy
|
|
check("(?<=%b{1,4})foo", "%bbbbfoo", new String[] {"foo"});
|
|
|
|
//Positive reluctant
|
|
check("(?<=%b{1,4}?)foo", "%bbbbfoo", new String[] {"foo"});
|
|
|
|
//supplementary
|
|
check("(?<=%b{1,4})fo\ud800\udc00o", "%bbbbfo\ud800\udc00o",
|
|
new String[] {"fo\ud800\udc00o"});
|
|
check("(?<=%b{1,4}?)fo\ud800\udc00o", "%bbbbfo\ud800\udc00o",
|
|
new String[] {"fo\ud800\udc00o"});
|
|
check("(?<!%b{1,4})fo\ud800\udc00o", "%afo\ud800\udc00o",
|
|
new String[] {"fo\ud800\udc00o"});
|
|
check("(?<!%b{1,4}?)fo\ud800\udc00o", "%afo\ud800\udc00o",
|
|
new String[] {"fo\ud800\udc00o"});
|
|
report("Lookbehind");
|
|
}
|
|
|
|
// This test is for 4938995
|
|
// Check to see if weak region boundaries are transparent to
|
|
// lookahead and lookbehind constructs
|
|
private static void boundsTest() throws Exception {
|
|
String fullMessage = "catdogcat";
|
|
Pattern pattern = Pattern.compile("(?<=cat)dog(?=cat)");
|
|
Matcher matcher = pattern.matcher("catdogca");
|
|
matcher.useTransparentBounds(true);
|
|
if (matcher.find())
|
|
failCount++;
|
|
matcher.reset("atdogcat");
|
|
if (matcher.find())
|
|
failCount++;
|
|
matcher.reset(fullMessage);
|
|
if (!matcher.find())
|
|
failCount++;
|
|
matcher.reset(fullMessage);
|
|
matcher.region(0,9);
|
|
if (!matcher.find())
|
|
failCount++;
|
|
matcher.reset(fullMessage);
|
|
matcher.region(0,6);
|
|
if (!matcher.find())
|
|
failCount++;
|
|
matcher.reset(fullMessage);
|
|
matcher.region(3,6);
|
|
if (!matcher.find())
|
|
failCount++;
|
|
matcher.useTransparentBounds(false);
|
|
if (matcher.find())
|
|
failCount++;
|
|
|
|
// Negative lookahead/lookbehind
|
|
pattern = Pattern.compile("(?<!cat)dog(?!cat)");
|
|
matcher = pattern.matcher("dogcat");
|
|
matcher.useTransparentBounds(true);
|
|
matcher.region(0,3);
|
|
if (matcher.find())
|
|
failCount++;
|
|
matcher.reset("catdog");
|
|
matcher.region(3,6);
|
|
if (matcher.find())
|
|
failCount++;
|
|
matcher.useTransparentBounds(false);
|
|
matcher.reset("dogcat");
|
|
matcher.region(0,3);
|
|
if (!matcher.find())
|
|
failCount++;
|
|
matcher.reset("catdog");
|
|
matcher.region(3,6);
|
|
if (!matcher.find())
|
|
failCount++;
|
|
|
|
report("Region bounds transparency");
|
|
}
|
|
|
|
// This test is for 4945394
|
|
private static void findFromTest() throws Exception {
|
|
String message = "This is 40 $0 message.";
|
|
Pattern pat = Pattern.compile("\\$0");
|
|
Matcher match = pat.matcher(message);
|
|
if (!match.find())
|
|
failCount++;
|
|
if (match.find())
|
|
failCount++;
|
|
if (match.find())
|
|
failCount++;
|
|
report("Check for alternating find");
|
|
}
|
|
|
|
// This test is for 4872664 and 4892980
|
|
private static void negatedCharClassTest() throws Exception {
|
|
Pattern pattern = Pattern.compile("[^>]");
|
|
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++;
|
|
|
|
// JDK-8230829
|
|
pattern = Pattern.compile("\\ud800\\udc61");
|
|
matcher = pattern.matcher("\ud800\udc61");
|
|
matcher.region(0, 1);
|
|
if (matcher.find()) {
|
|
failCount++;
|
|
System.out.println("Matched a surrogate pair" +
|
|
" that crosses border of region");
|
|
}
|
|
if (!matcher.hitEnd()) {
|
|
failCount++;
|
|
System.out.println("Expected to hit the end when" +
|
|
" matching a surrogate pair crossing region");
|
|
}
|
|
|
|
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: <u+1d1ba><u+1d165><u+1d16f> -> nfc: <u+1d1ba><u+1d165><u+1d16f>
|
|
u+1d1bc -> nfd: <u+1d1ba><u+1d165> -> nfc: <u+1d1ba><u+1d165>
|
|
<u+1d1bc><u+1d16f> -> nfd: <u+1d1ba><u+1d165><u+1d16f> -> nfc: <u+1d1ba><u+1d165><u+1d16f>
|
|
|
|
// 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; x<patternLength; x++) {
|
|
int ch = baseCharacter + generator.nextInt(26);
|
|
if (Character.isSupplementaryCodePoint(ch)) {
|
|
patternBuffer.append(Character.toChars(ch));
|
|
} else {
|
|
patternBuffer.append((char)ch);
|
|
}
|
|
}
|
|
pattern = patternBuffer.toString();
|
|
|
|
// Avoid patterns that start and end with the same substring
|
|
// See JDK-6854417
|
|
for (int x=1; x < pattern.length(); x++) {
|
|
if (pattern.startsWith(pattern.substring(x)))
|
|
continue retry;
|
|
}
|
|
break;
|
|
}
|
|
Pattern p = Pattern.compile(pattern);
|
|
|
|
// Create a buffer with random ASCII chars that does
|
|
// not match the sample
|
|
String toSearch = null;
|
|
StringBuffer s = null;
|
|
Matcher m = p.matcher("");
|
|
do {
|
|
s = new StringBuffer(100);
|
|
for (int x=0; x<100; x++) {
|
|
int ch = baseCharacter + generator.nextInt(26);
|
|
if (Character.isSupplementaryCodePoint(ch)) {
|
|
s.append(Character.toChars(ch));
|
|
} else {
|
|
s.append((char)ch);
|
|
}
|
|
}
|
|
toSearch = s.toString();
|
|
m.reset(toSearch);
|
|
} while (m.find());
|
|
|
|
// Insert the pattern at a random spot
|
|
int insertIndex = generator.nextInt(99);
|
|
if (Character.isLowSurrogate(s.charAt(insertIndex)))
|
|
insertIndex++;
|
|
s = s.insert(insertIndex, pattern);
|
|
toSearch = s.toString();
|
|
|
|
// Make sure that the pattern is found
|
|
m.reset(toSearch);
|
|
if (!m.find())
|
|
failCount++;
|
|
|
|
// Make sure that the match text is the pattern
|
|
if (!m.group().equals(pattern))
|
|
failCount++;
|
|
|
|
// Make sure match occured at insertion point
|
|
if (m.start() != insertIndex)
|
|
failCount++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Tests the matching of slices on randomly generated patterns.
|
|
* The Boyer-Moore optimization is not done on these patterns
|
|
* because it uses unicode case folding.
|
|
*/
|
|
private static void slice() throws Exception {
|
|
doSlice(Character.MAX_VALUE);
|
|
report("Slice");
|
|
|
|
doSlice(Character.MAX_CODE_POINT);
|
|
report("Slice (Supplementary)");
|
|
}
|
|
|
|
private static void doSlice(int maxCharacter) throws Exception {
|
|
Random generator = new Random();
|
|
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);
|
|
for (int x=0; x<patternLength; x++) {
|
|
int randomChar = 0;
|
|
while (!Character.isLetterOrDigit(randomChar))
|
|
randomChar = generator.nextInt(maxCharacter);
|
|
if (Character.isSupplementaryCodePoint(randomChar)) {
|
|
patternBuffer.append(Character.toChars(randomChar));
|
|
} else {
|
|
patternBuffer.append((char) randomChar);
|
|
}
|
|
}
|
|
String pattern = patternBuffer.toString();
|
|
Pattern p = Pattern.compile(pattern, Pattern.UNICODE_CASE);
|
|
|
|
// Create a buffer with random chars that does not match the sample
|
|
String toSearch = null;
|
|
StringBuffer s = null;
|
|
Matcher m = p.matcher("");
|
|
do {
|
|
s = new StringBuffer(100);
|
|
for (int x=0; x<100; x++) {
|
|
int randomChar = 0;
|
|
while (!Character.isLetterOrDigit(randomChar))
|
|
randomChar = generator.nextInt(maxCharacter);
|
|
if (Character.isSupplementaryCodePoint(randomChar)) {
|
|
s.append(Character.toChars(randomChar));
|
|
} else {
|
|
s.append((char) randomChar);
|
|
}
|
|
}
|
|
toSearch = s.toString();
|
|
m.reset(toSearch);
|
|
} while (m.find());
|
|
|
|
// Insert the pattern at a random spot
|
|
int insertIndex = generator.nextInt(99);
|
|
if (Character.isLowSurrogate(s.charAt(insertIndex)))
|
|
insertIndex++;
|
|
s = s.insert(insertIndex, pattern);
|
|
toSearch = s.toString();
|
|
|
|
// Make sure that the pattern is found
|
|
m.reset(toSearch);
|
|
if (!m.find())
|
|
failCount++;
|
|
|
|
// Make sure that the match text is the pattern
|
|
if (!m.group().equals(pattern))
|
|
failCount++;
|
|
|
|
// Make sure match occured at insertion point
|
|
if (m.start() != insertIndex)
|
|
failCount++;
|
|
}
|
|
}
|
|
|
|
private static void explainFailure(String pattern, String data,
|
|
String expected, String actual) {
|
|
System.err.println("----------------------------------------");
|
|
System.err.println("Pattern = "+pattern);
|
|
System.err.println("Data = "+data);
|
|
System.err.println("Expected = " + expected);
|
|
System.err.println("Actual = " + actual);
|
|
}
|
|
|
|
private static void explainFailure(String pattern, String data,
|
|
Throwable t) {
|
|
System.err.println("----------------------------------------");
|
|
System.err.println("Pattern = "+pattern);
|
|
System.err.println("Data = "+data);
|
|
t.printStackTrace(System.err);
|
|
}
|
|
|
|
// Testing examples from a file
|
|
|
|
/**
|
|
* Goes through the file "TestCases.txt" and creates many patterns
|
|
* described in the file, matching the patterns against input lines in
|
|
* the file, and comparing the results against the correct results
|
|
* also found in the file. The file format is described in comments
|
|
* at the head of the file.
|
|
*/
|
|
private static void processFile(String fileName) throws Exception {
|
|
File testCases = new File(System.getProperty("test.src", "."),
|
|
fileName);
|
|
FileInputStream in = new FileInputStream(testCases);
|
|
BufferedReader r = new BufferedReader(new InputStreamReader(in));
|
|
|
|
// Process next test case.
|
|
String aLine;
|
|
while((aLine = r.readLine()) != null) {
|
|
// Read a line for pattern
|
|
String patternString = grabLine(r);
|
|
Pattern p = null;
|
|
try {
|
|
p = compileTestPattern(patternString);
|
|
} catch (PatternSyntaxException e) {
|
|
String dataString = grabLine(r);
|
|
String expectedResult = grabLine(r);
|
|
if (expectedResult.startsWith("error"))
|
|
continue;
|
|
explainFailure(patternString, dataString, e);
|
|
failCount++;
|
|
continue;
|
|
}
|
|
|
|
// Read a line for input string
|
|
String dataString = grabLine(r);
|
|
Matcher m = p.matcher(dataString);
|
|
StringBuffer result = new StringBuffer();
|
|
|
|
// Check for IllegalStateExceptions before a match
|
|
failCount += preMatchInvariants(m);
|
|
|
|
boolean found = m.find();
|
|
|
|
if (found)
|
|
failCount += postTrueMatchInvariants(m);
|
|
else
|
|
failCount += postFalseMatchInvariants(m);
|
|
|
|
if (found) {
|
|
result.append("true ");
|
|
result.append(m.group(0) + " ");
|
|
} else {
|
|
result.append("false ");
|
|
}
|
|
|
|
result.append(m.groupCount());
|
|
|
|
if (found) {
|
|
for (int i=1; i<m.groupCount()+1; i++)
|
|
if (m.group(i) != null)
|
|
result.append(" " +m.group(i));
|
|
}
|
|
|
|
// Read a line for the expected result
|
|
String expectedResult = grabLine(r);
|
|
|
|
if (!result.toString().equals(expectedResult)) {
|
|
explainFailure(patternString, dataString, expectedResult, result.toString());
|
|
failCount++;
|
|
}
|
|
}
|
|
|
|
report(fileName);
|
|
}
|
|
|
|
private static int preMatchInvariants(Matcher m) {
|
|
int failCount = 0;
|
|
try {
|
|
m.start();
|
|
failCount++;
|
|
} catch (IllegalStateException ise) {}
|
|
try {
|
|
m.end();
|
|
failCount++;
|
|
} catch (IllegalStateException ise) {}
|
|
try {
|
|
m.group();
|
|
failCount++;
|
|
} catch (IllegalStateException ise) {}
|
|
return failCount;
|
|
}
|
|
|
|
private static int postFalseMatchInvariants(Matcher m) {
|
|
int failCount = 0;
|
|
try {
|
|
m.group();
|
|
failCount++;
|
|
} catch (IllegalStateException ise) {}
|
|
try {
|
|
m.start();
|
|
failCount++;
|
|
} catch (IllegalStateException ise) {}
|
|
try {
|
|
m.end();
|
|
failCount++;
|
|
} catch (IllegalStateException ise) {}
|
|
return failCount;
|
|
}
|
|
|
|
private static int postTrueMatchInvariants(Matcher m) {
|
|
int failCount = 0;
|
|
//assert(m.start() = m.start(0);
|
|
if (m.start() != m.start(0))
|
|
failCount++;
|
|
//assert(m.end() = m.end(0);
|
|
if (m.start() != m.start(0))
|
|
failCount++;
|
|
//assert(m.group() = m.group(0);
|
|
if (!m.group().equals(m.group(0)))
|
|
failCount++;
|
|
try {
|
|
m.group(50);
|
|
failCount++;
|
|
} catch (IndexOutOfBoundsException ise) {}
|
|
|
|
return failCount;
|
|
}
|
|
|
|
private static Pattern compileTestPattern(String patternString) {
|
|
if (!patternString.startsWith("'")) {
|
|
return Pattern.compile(patternString);
|
|
}
|
|
int break1 = patternString.lastIndexOf("'");
|
|
String flagString = patternString.substring(
|
|
break1+1, patternString.length());
|
|
patternString = patternString.substring(1, break1);
|
|
|
|
if (flagString.equals("i"))
|
|
return Pattern.compile(patternString, Pattern.CASE_INSENSITIVE);
|
|
|
|
if (flagString.equals("m"))
|
|
return Pattern.compile(patternString, Pattern.MULTILINE);
|
|
|
|
return Pattern.compile(patternString);
|
|
}
|
|
|
|
/**
|
|
* Reads a line from the input file. Keeps reading lines until a non
|
|
* empty non comment line is read. If the line contains a \n then
|
|
* these two characters are replaced by a newline char. If a \\uxxxx
|
|
* sequence is read then the sequence is replaced by the unicode char.
|
|
*/
|
|
private static String grabLine(BufferedReader r) throws Exception {
|
|
int index = 0;
|
|
String line = r.readLine();
|
|
while (line.startsWith("//") || line.length() < 1)
|
|
line = r.readLine();
|
|
while ((index = line.indexOf("\\n")) != -1) {
|
|
StringBuffer temp = new StringBuffer(line);
|
|
temp.replace(index, index+2, "\n");
|
|
line = temp.toString();
|
|
}
|
|
while ((index = line.indexOf("\\u")) != -1) {
|
|
StringBuffer temp = new StringBuffer(line);
|
|
String value = temp.substring(index+2, index+6);
|
|
char aChar = (char)Integer.parseInt(value, 16);
|
|
String unicodeChar = "" + aChar;
|
|
temp.replace(index, index+6, unicodeChar);
|
|
line = temp.toString();
|
|
}
|
|
|
|
return line;
|
|
}
|
|
|
|
private static void check(Pattern p, String s, String g, String expected) {
|
|
Matcher m = p.matcher(s);
|
|
m.find();
|
|
if (!m.group(g).equals(expected) ||
|
|
s.charAt(m.start(g)) != expected.charAt(0) ||
|
|
s.charAt(m.end(g) - 1) != expected.charAt(expected.length() - 1))
|
|
failCount++;
|
|
}
|
|
|
|
private static void checkReplaceFirst(String p, String s, String r, String expected)
|
|
{
|
|
if (!expected.equals(Pattern.compile(p)
|
|
.matcher(s)
|
|
.replaceFirst(r)))
|
|
failCount++;
|
|
}
|
|
|
|
private static void checkReplaceAll(String p, String s, String r, String expected)
|
|
{
|
|
if (!expected.equals(Pattern.compile(p)
|
|
.matcher(s)
|
|
.replaceAll(r)))
|
|
failCount++;
|
|
}
|
|
|
|
private static void checkExpectedFail(String p) {
|
|
try {
|
|
Pattern.compile(p);
|
|
} catch (PatternSyntaxException pse) {
|
|
//pse.printStackTrace();
|
|
return;
|
|
}
|
|
failCount++;
|
|
}
|
|
|
|
private static void checkExpectedIAE(Matcher m, String g) {
|
|
m.find();
|
|
try {
|
|
m.group(g);
|
|
} catch (IllegalArgumentException x) {
|
|
//iae.printStackTrace();
|
|
try {
|
|
m.start(g);
|
|
} catch (IllegalArgumentException xx) {
|
|
try {
|
|
m.start(g);
|
|
} catch (IllegalArgumentException xxx) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
failCount++;
|
|
}
|
|
|
|
private static void checkExpectedNPE(Matcher m) {
|
|
m.find();
|
|
try {
|
|
m.group(null);
|
|
} catch (NullPointerException x) {
|
|
try {
|
|
m.start(null);
|
|
} catch (NullPointerException xx) {
|
|
try {
|
|
m.end(null);
|
|
} catch (NullPointerException xxx) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
failCount++;
|
|
}
|
|
|
|
private static void namedGroupCaptureTest() throws Exception {
|
|
check(Pattern.compile("x+(?<gname>y+)z+"),
|
|
"xxxyyyzzz",
|
|
"gname",
|
|
"yyy");
|
|
|
|
check(Pattern.compile("x+(?<gname8>y+)z+"),
|
|
"xxxyyyzzz",
|
|
"gname8",
|
|
"yyy");
|
|
|
|
//backref
|
|
Pattern pattern = Pattern.compile("(a*)bc\\1");
|
|
check(pattern, "zzzaabcazzz", true); // found "abca"
|
|
|
|
check(Pattern.compile("(?<gname>a*)bc\\k<gname>"),
|
|
"zzzaabcaazzz", true);
|
|
|
|
check(Pattern.compile("(?<gname>abc)(def)\\k<gname>"),
|
|
"abcdefabc", true);
|
|
|
|
check(Pattern.compile("(a)(b)(c)(d)(e)(f)(g)(h)(i)(j)(?<gname>k)\\k<gname>"),
|
|
"abcdefghijkk", true);
|
|
|
|
// Supplementary character tests
|
|
check(Pattern.compile("(?<gname>" + toSupplementaries("a*)bc") + "\\k<gname>"),
|
|
toSupplementaries("zzzaabcazzz"), true);
|
|
|
|
check(Pattern.compile("(?<gname>" + toSupplementaries("a*)bc") + "\\k<gname>"),
|
|
toSupplementaries("zzzaabcaazzz"), true);
|
|
|
|
check(Pattern.compile("(?<gname>" + toSupplementaries("abc)(def)") + "\\k<gname>"),
|
|
toSupplementaries("abcdefabc"), true);
|
|
|
|
check(Pattern.compile(toSupplementaries("(a)(b)(c)(d)(e)(f)(g)(h)(i)(j)") +
|
|
"(?<gname>" +
|
|
toSupplementaries("k)") + "\\k<gname>"),
|
|
toSupplementaries("abcdefghijkk"), true);
|
|
|
|
check(Pattern.compile("x+(?<gname>y+)z+\\k<gname>"),
|
|
"xxxyyyzzzyyy",
|
|
"gname",
|
|
"yyy");
|
|
|
|
//replaceFirst/All
|
|
checkReplaceFirst("(?<gn>ab)(c*)",
|
|
"abccczzzabcczzzabccc",
|
|
"${gn}",
|
|
"abzzzabcczzzabccc");
|
|
|
|
checkReplaceAll("(?<gn>ab)(c*)",
|
|
"abccczzzabcczzzabccc",
|
|
"${gn}",
|
|
"abzzzabzzzab");
|
|
|
|
|
|
checkReplaceFirst("(?<gn>ab)(c*)",
|
|
"zzzabccczzzabcczzzabccczzz",
|
|
"${gn}",
|
|
"zzzabzzzabcczzzabccczzz");
|
|
|
|
checkReplaceAll("(?<gn>ab)(c*)",
|
|
"zzzabccczzzabcczzzabccczzz",
|
|
"${gn}",
|
|
"zzzabzzzabzzzabzzz");
|
|
|
|
checkReplaceFirst("(?<gn1>ab)(?<gn2>c*)",
|
|
"zzzabccczzzabcczzzabccczzz",
|
|
"${gn2}",
|
|
"zzzccczzzabcczzzabccczzz");
|
|
|
|
checkReplaceAll("(?<gn1>ab)(?<gn2>c*)",
|
|
"zzzabccczzzabcczzzabccczzz",
|
|
"${gn2}",
|
|
"zzzccczzzcczzzccczzz");
|
|
|
|
//toSupplementaries("(ab)(c*)"));
|
|
checkReplaceFirst("(?<gn1>" + toSupplementaries("ab") +
|
|
")(?<gn2>" + toSupplementaries("c") + "*)",
|
|
toSupplementaries("abccczzzabcczzzabccc"),
|
|
"${gn1}",
|
|
toSupplementaries("abzzzabcczzzabccc"));
|
|
|
|
|
|
checkReplaceAll("(?<gn1>" + toSupplementaries("ab") +
|
|
")(?<gn2>" + toSupplementaries("c") + "*)",
|
|
toSupplementaries("abccczzzabcczzzabccc"),
|
|
"${gn1}",
|
|
toSupplementaries("abzzzabzzzab"));
|
|
|
|
checkReplaceFirst("(?<gn1>" + toSupplementaries("ab") +
|
|
")(?<gn2>" + toSupplementaries("c") + "*)",
|
|
toSupplementaries("abccczzzabcczzzabccc"),
|
|
"${gn2}",
|
|
toSupplementaries("ccczzzabcczzzabccc"));
|
|
|
|
|
|
checkReplaceAll("(?<gn1>" + toSupplementaries("ab") +
|
|
")(?<gn2>" + toSupplementaries("c") + "*)",
|
|
toSupplementaries("abccczzzabcczzzabccc"),
|
|
"${gn2}",
|
|
toSupplementaries("ccczzzcczzzccc"));
|
|
|
|
checkReplaceFirst("(?<dog>Dog)AndCat",
|
|
"zzzDogAndCatzzzDogAndCatzzz",
|
|
"${dog}",
|
|
"zzzDogzzzDogAndCatzzz");
|
|
|
|
|
|
checkReplaceAll("(?<dog>Dog)AndCat",
|
|
"zzzDogAndCatzzzDogAndCatzzz",
|
|
"${dog}",
|
|
"zzzDogzzzDogzzz");
|
|
|
|
// backref in Matcher & String
|
|
if (!"abcdefghij".replaceFirst("cd(?<gn>ef)gh", "${gn}").equals("abefij") ||
|
|
!"abbbcbdbefgh".replaceAll("(?<gn>[a-e])b", "${gn}").equals("abcdefgh"))
|
|
failCount++;
|
|
|
|
// negative
|
|
checkExpectedFail("(?<groupnamehasnoascii.in>abc)(def)");
|
|
checkExpectedFail("(?<groupnamehasnoascii_in>abc)(def)");
|
|
checkExpectedFail("(?<6groupnamestartswithdigit>abc)(def)");
|
|
checkExpectedFail("(?<gname>abc)(def)\\k<gnameX>");
|
|
checkExpectedFail("(?<gname>abc)(?<gname>def)\\k<gnameX>");
|
|
checkExpectedIAE(Pattern.compile("(?<gname>abc)(def)").matcher("abcdef"),
|
|
"gnameX");
|
|
checkExpectedNPE(Pattern.compile("(?<gname>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("((?<!\\R)\\s)*").matcher(crnl).matches()) { // #8176029
|
|
failCount++;
|
|
}
|
|
report("linebreakTest");
|
|
}
|
|
|
|
// #7189363
|
|
private static void branchTest() throws Exception {
|
|
if (!Pattern.compile("(a)?bc|d").matcher("d").find() || // greedy
|
|
!Pattern.compile("(a)+bc|d").matcher("d").find() ||
|
|
!Pattern.compile("(a)*bc|d").matcher("d").find() ||
|
|
!Pattern.compile("(a)??bc|d").matcher("d").find() || // reluctant
|
|
!Pattern.compile("(a)+?bc|d").matcher("d").find() ||
|
|
!Pattern.compile("(a)*?bc|d").matcher("d").find() ||
|
|
!Pattern.compile("(a)?+bc|d").matcher("d").find() || // possessive
|
|
!Pattern.compile("(a)++bc|d").matcher("d").find() ||
|
|
!Pattern.compile("(a)*+bc|d").matcher("d").find() ||
|
|
!Pattern.compile("(a)?bc|d").matcher("d").matches() || // greedy
|
|
!Pattern.compile("(a)+bc|d").matcher("d").matches() ||
|
|
!Pattern.compile("(a)*bc|d").matcher("d").matches() ||
|
|
!Pattern.compile("(a)??bc|d").matcher("d").matches() || // reluctant
|
|
!Pattern.compile("(a)+?bc|d").matcher("d").matches() ||
|
|
!Pattern.compile("(a)*?bc|d").matcher("d").matches() ||
|
|
!Pattern.compile("(a)?+bc|d").matcher("d").matches() || // possessive
|
|
!Pattern.compile("(a)++bc|d").matcher("d").matches() ||
|
|
!Pattern.compile("(a)*+bc|d").matcher("d").matches() ||
|
|
!Pattern.compile("(a)?bc|de").matcher("de").find() || // others
|
|
!Pattern.compile("(a)??bc|de").matcher("de").find() ||
|
|
!Pattern.compile("(a)?bc|de").matcher("de").matches() ||
|
|
!Pattern.compile("(a)??bc|de").matcher("de").matches())
|
|
failCount++;
|
|
report("branchTest");
|
|
}
|
|
|
|
// This test is for 8007395
|
|
private static void groupCurlyNotFoundSuppTest() throws Exception {
|
|
String input = "test this as \ud83d\ude0d";
|
|
for (String pStr : new String[] { "test(.)+(@[a-zA-Z.]+)",
|
|
"test(.)*(@[a-zA-Z.]+)",
|
|
"test([^B])+(@[a-zA-Z.]+)",
|
|
"test([^B])*(@[a-zA-Z.]+)",
|
|
"test(\\P{IsControl})+(@[a-zA-Z.]+)",
|
|
"test(\\P{IsControl})*(@[a-zA-Z.]+)",
|
|
}) {
|
|
Matcher m = Pattern.compile(pStr, Pattern.CASE_INSENSITIVE)
|
|
.matcher(input);
|
|
try {
|
|
if (m.find()) {
|
|
failCount++;
|
|
}
|
|
} catch (Exception x) {
|
|
failCount++;
|
|
}
|
|
}
|
|
report("GroupCurly NotFoundSupp");
|
|
}
|
|
|
|
// This test is for 8023647
|
|
private static void groupCurlyBackoffTest() throws Exception {
|
|
if (!"abc1c".matches("(\\w)+1\\1") ||
|
|
"abc11".matches("(\\w)+1\\1")) {
|
|
failCount++;
|
|
}
|
|
report("GroupCurly backoff");
|
|
}
|
|
|
|
// This test is for 8012646
|
|
private static void patternAsPredicate() throws Exception {
|
|
Predicate<String> 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<String> 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 {
|
|
final int[] lineNumber = new int[1];
|
|
Stream.concat(Files.lines(UCDFiles.GRAPHEME_BREAK_TEST),
|
|
Files.lines(Paths.get(System.getProperty("test.src", "."), "GraphemeTestCases.txt")))
|
|
.forEach( ln -> {
|
|
lineNumber[0]++;
|
|
if (ln.length() == 0 || ln.startsWith("#")) {
|
|
return;
|
|
}
|
|
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<String> 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");
|
|
// (1) test \X directly
|
|
Matcher m = p.matcher(src.toString());
|
|
for (String g : graphemes) {
|
|
// System.out.printf(" grapheme:=[%s]%n", g);
|
|
String group = null;
|
|
if (!m.find() || !(group = m.group()).equals(g)) {
|
|
System.out.println("Failed pattern \\X [" + ln + "] : "
|
|
+ "expected: " + g + " - actual: " + group
|
|
+ "(line " + lineNumber[0] + ")");
|
|
failCount++;
|
|
}
|
|
}
|
|
if (m.find()) {
|
|
failCount++;
|
|
}
|
|
// test \b{g} without \X via Pattern
|
|
Pattern pbg = Pattern.compile("\\b{g}");
|
|
m = pbg.matcher(src.toString());
|
|
m.find();
|
|
int prev = m.end();
|
|
for (String g : graphemes) {
|
|
String group = null;
|
|
if (!m.find() || !(group = src.substring(prev, m.end())).equals(g)) {
|
|
System.out.println("Failed pattern \\b{g} [" + ln + "] : "
|
|
+ "expected: " + g + " - actual: " + group
|
|
+ "(line " + lineNumber[0] + ")");
|
|
failCount++;
|
|
}
|
|
if (!"".equals(m.group())) {
|
|
failCount++;
|
|
}
|
|
prev = m.end();
|
|
}
|
|
if (m.find()) {
|
|
failCount++;
|
|
}
|
|
// (2) test \b{g} + \X via Scanner
|
|
Scanner s = new Scanner(src.toString()).useDelimiter("\\b{g}");
|
|
for (String g : graphemes) {
|
|
String next = null;
|
|
if (!s.hasNext(p) || !(next = s.next(p)).equals(g)) {
|
|
System.out.println("Failed \\b{g} [" + ln + "] : "
|
|
+ "expected: " + g + " - actual: " + next
|
|
+ " (line " + lineNumber[0] + ")");
|
|
failCount++;
|
|
}
|
|
}
|
|
if (s.hasNext(p)) {
|
|
failCount++;
|
|
}
|
|
// test \b{g} without \X via Scanner
|
|
s = new Scanner(src.toString()).useDelimiter("\\b{g}");
|
|
for (String g : graphemes) {
|
|
String next = null;
|
|
if (!s.hasNext() || !(next = s.next()).equals(g)) {
|
|
System.out.println("Failed \\b{g} [" + ln + "] : "
|
|
+ "expected: " + g + " - actual: " + next
|
|
+ " (line " + lineNumber[0] + ")");
|
|
failCount++;
|
|
}
|
|
}
|
|
if (s.hasNext()) {
|
|
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|[^>])+>",
|
|
"<META http-equiv=\"Content-Type\" content=\"text/html; charset=ISO-8859-5\">",
|
|
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> <p> <html> <adfasfdasdf> </p>",
|
|
true }, // --> matched
|
|
{ "((<[^>]+>)?(((\\s)?)*(\\ )?)*((\\s)?)*)+",
|
|
" < br/> < / p> <p> <html> <adfasfdasdf> 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
|
|
{ "(?<before>.*)\\{(?<reflection>\\w+):(?<innerMethod>\\w+(\\.?\\w+(\\(((?<args>(('[^']*')|((/|\\w)+))(,(('[^']*')|((/|\\w)+)))*))?\\))?)*)\\}(?<after>.*)",
|
|
"{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");
|
|
}
|
|
|
|
private static void surrogatePairWithCanonEq() {
|
|
try {
|
|
Pattern.compile("\ud834\udd21", Pattern.CANON_EQ);
|
|
} catch (Throwable t) {
|
|
failCount++;
|
|
System.out.println("Unexpected exception: " + t);
|
|
}
|
|
report("surrogatePairWithCanonEq");
|
|
}
|
|
|
|
private static String s2x(String s) {
|
|
StringBuilder sb = new StringBuilder();
|
|
for (char ch : s.toCharArray()) {
|
|
sb.append(String.format("\\u%04x", (int)ch));
|
|
}
|
|
return sb.toString();
|
|
}
|
|
|
|
// This test is for 8235812, with cases excluded by 8258259
|
|
private static void lineBreakWithQuantifier() {
|
|
// key: pattern
|
|
// value: lengths of input that must match the pattern
|
|
Map<String, List<Integer>> cases = Map.ofEntries(
|
|
Map.entry("\\R?", List.of(0, 1)),
|
|
Map.entry("\\R*", List.of(0, 1, 2, 3)),
|
|
Map.entry("\\R+", List.of(1, 2, 3)),
|
|
Map.entry("\\R{0}", List.of(0)),
|
|
Map.entry("\\R{1}", List.of(1)),
|
|
// Map.entry("\\R{2}", List.of(2)), // 8258259
|
|
// Map.entry("\\R{3}", List.of(3)), // 8258259
|
|
Map.entry("\\R{0,}", List.of(0, 1, 2, 3)),
|
|
Map.entry("\\R{1,}", List.of(1, 2, 3)),
|
|
// Map.entry("\\R{2,}", List.of(2, 3)), // 8258259
|
|
// Map.entry("\\R{3,}", List.of(3)), // 8258259
|
|
Map.entry("\\R{0,0}", List.of(0)),
|
|
Map.entry("\\R{0,1}", List.of(0, 1)),
|
|
Map.entry("\\R{0,2}", List.of(0, 1, 2)),
|
|
Map.entry("\\R{0,3}", List.of(0, 1, 2, 3)),
|
|
Map.entry("\\R{1,1}", List.of(1)),
|
|
Map.entry("\\R{1,2}", List.of(1, 2)),
|
|
Map.entry("\\R{1,3}", List.of(1, 2, 3)),
|
|
// Map.entry("\\R{2,2}", List.of(2)), // 8258259
|
|
// Map.entry("\\R{2,3}", List.of(2, 3)), // 8258259
|
|
// Map.entry("\\R{3,3}", List.of(3)), // 8258259
|
|
Map.entry("\\R", List.of(1)),
|
|
Map.entry("\\R\\R", List.of(2)),
|
|
Map.entry("\\R\\R\\R", List.of(3))
|
|
);
|
|
|
|
// key: length of input
|
|
// value: all possible inputs of given length
|
|
Map<Integer, List<String>> inputs = new HashMap<>();
|
|
String[] Rs = { "\r\n", "\r", "\n",
|
|
"\u000B", "\u000C", "\u0085", "\u2028", "\u2029" };
|
|
StringBuilder sb = new StringBuilder();
|
|
for (int len = 0; len <= 3; ++len) {
|
|
int[] idx = new int[len + 1];
|
|
do {
|
|
sb.setLength(0);
|
|
for (int j = 0; j < len; ++j)
|
|
sb.append(Rs[idx[j]]);
|
|
inputs.computeIfAbsent(len, ArrayList::new).add(sb.toString());
|
|
idx[0]++;
|
|
for (int j = 0; j < len; ++j) {
|
|
if (idx[j] < Rs.length)
|
|
break;
|
|
idx[j] = 0;
|
|
idx[j+1]++;
|
|
}
|
|
} while (idx[len] == 0);
|
|
}
|
|
|
|
// exhaustive testing
|
|
for (String patStr : cases.keySet()) {
|
|
Pattern[] pats = patStr.endsWith("R")
|
|
? new Pattern[] { Pattern.compile(patStr) } // no quantifiers
|
|
: new Pattern[] { Pattern.compile(patStr), // greedy
|
|
Pattern.compile(patStr + "?") }; // reluctant
|
|
Matcher m = pats[0].matcher("");
|
|
for (Pattern p : pats) {
|
|
m.usePattern(p);
|
|
for (int len : cases.get(patStr)) {
|
|
for (String in : inputs.get(len)) {
|
|
if (!m.reset(in).matches()) {
|
|
failCount++;
|
|
System.err.println("Expected to match '" +
|
|
s2x(in) + "' =~ /" + p + "/");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
report("lineBreakWithQuantifier");
|
|
}
|
|
|
|
// This test is for 8214245
|
|
private static void caseInsensitivePMatch() {
|
|
for (String input : List.of("abcd", "AbCd", "ABCD")) {
|
|
for (String pattern : List.of("abcd", "aBcD", "[a-d]{4}",
|
|
"(?:a|b|c|d){4}", "\\p{Lower}{4}", "\\p{Ll}{4}",
|
|
"\\p{IsLl}{4}", "\\p{gc=Ll}{4}",
|
|
"\\p{general_category=Ll}{4}", "\\p{IsLowercase}{4}",
|
|
"\\p{javaLowerCase}{4}", "\\p{Upper}{4}", "\\p{Lu}{4}",
|
|
"\\p{IsLu}{4}", "\\p{gc=Lu}{4}", "\\p{general_category=Lu}{4}",
|
|
"\\p{IsUppercase}{4}", "\\p{javaUpperCase}{4}",
|
|
"\\p{Lt}{4}", "\\p{IsLt}{4}", "\\p{gc=Lt}{4}",
|
|
"\\p{general_category=Lt}{4}", "\\p{IsTitlecase}{4}",
|
|
"\\p{javaTitleCase}{4}", "[\\p{Lower}]{4}", "[\\p{Ll}]{4}",
|
|
"[\\p{IsLl}]{4}", "[\\p{gc=Ll}]{4}",
|
|
"[\\p{general_category=Ll}]{4}", "[\\p{IsLowercase}]{4}",
|
|
"[\\p{javaLowerCase}]{4}", "[\\p{Upper}]{4}", "[\\p{Lu}]{4}",
|
|
"[\\p{IsLu}]{4}", "[\\p{gc=Lu}]{4}",
|
|
"[\\p{general_category=Lu}]{4}", "[\\p{IsUppercase}]{4}",
|
|
"[\\p{javaUpperCase}]{4}", "[\\p{Lt}]{4}", "[\\p{IsLt}]{4}",
|
|
"[\\p{gc=Lt}]{4}", "[\\p{general_category=Lt}]{4}",
|
|
"[\\p{IsTitlecase}]{4}", "[\\p{javaTitleCase}]{4}"))
|
|
{
|
|
if (!Pattern.compile(pattern, Pattern.CASE_INSENSITIVE)
|
|
.matcher(input)
|
|
.matches())
|
|
{
|
|
failCount++;
|
|
System.err.println("Expected to match: " +
|
|
"'" + input + "' =~ /" + pattern + "/");
|
|
}
|
|
}
|
|
}
|
|
|
|
for (String input : List.of("\u01c7", "\u01c8", "\u01c9")) {
|
|
for (String pattern : List.of("\u01c7", "\u01c8", "\u01c9",
|
|
"[\u01c7\u01c8]", "[\u01c7\u01c9]", "[\u01c8\u01c9]",
|
|
"[\u01c7-\u01c8]", "[\u01c8-\u01c9]", "[\u01c7-\u01c9]",
|
|
"\\p{Lower}", "\\p{Ll}", "\\p{IsLl}", "\\p{gc=Ll}",
|
|
"\\p{general_category=Ll}", "\\p{IsLowercase}",
|
|
"\\p{javaLowerCase}", "\\p{Upper}", "\\p{Lu}",
|
|
"\\p{IsLu}", "\\p{gc=Lu}", "\\p{general_category=Lu}",
|
|
"\\p{IsUppercase}", "\\p{javaUpperCase}",
|
|
"\\p{Lt}", "\\p{IsLt}", "\\p{gc=Lt}",
|
|
"\\p{general_category=Lt}", "\\p{IsTitlecase}",
|
|
"\\p{javaTitleCase}", "[\\p{Lower}]", "[\\p{Ll}]",
|
|
"[\\p{IsLl}]", "[\\p{gc=Ll}]",
|
|
"[\\p{general_category=Ll}]", "[\\p{IsLowercase}]",
|
|
"[\\p{javaLowerCase}]", "[\\p{Upper}]", "[\\p{Lu}]",
|
|
"[\\p{IsLu}]", "[\\p{gc=Lu}]",
|
|
"[\\p{general_category=Lu}]", "[\\p{IsUppercase}]",
|
|
"[\\p{javaUpperCase}]", "[\\p{Lt}]", "[\\p{IsLt}]",
|
|
"[\\p{gc=Lt}]", "[\\p{general_category=Lt}]",
|
|
"[\\p{IsTitlecase}]", "[\\p{javaTitleCase}]"))
|
|
{
|
|
if (!Pattern.compile(pattern, Pattern.CASE_INSENSITIVE
|
|
| Pattern.UNICODE_CHARACTER_CLASS)
|
|
.matcher(input)
|
|
.matches())
|
|
{
|
|
failCount++;
|
|
System.err.println("Expected to match: " +
|
|
"'" + input + "' =~ /" + pattern + "/");
|
|
}
|
|
}
|
|
}
|
|
report("caseInsensitivePMatch");
|
|
}
|
|
|
|
// This test is for 8237599
|
|
private static void surrogatePairOverlapRegion() {
|
|
String input = "\ud801\udc37";
|
|
|
|
Pattern p = Pattern.compile(".+");
|
|
Matcher m = p.matcher(input);
|
|
m.region(0, 1);
|
|
|
|
boolean ok = m.find();
|
|
if (!ok || !m.group(0).equals(input.substring(0, 1)))
|
|
{
|
|
failCount++;
|
|
System.out.println("Input \"" + input + "\".substr(0, 1)" +
|
|
" expected to match pattern \"" + p + "\"");
|
|
if (ok) {
|
|
System.out.println("group(0): \"" + m.group(0) + "\"");
|
|
}
|
|
} else if (!m.hitEnd()) {
|
|
failCount++;
|
|
System.out.println("Expected m.hitEnd() == true");
|
|
}
|
|
|
|
p = Pattern.compile(".*(.)");
|
|
m = p.matcher(input);
|
|
m.region(1, 2);
|
|
|
|
ok = m.find();
|
|
if (!ok || !m.group(0).equals(input.substring(1, 2))
|
|
|| !m.group(1).equals(input.substring(1, 2)))
|
|
{
|
|
failCount++;
|
|
System.out.println("Input \"" + input + "\".substr(1, 2)" +
|
|
" expected to match pattern \"" + p + "\"");
|
|
if (ok) {
|
|
System.out.println("group(0): \"" + m.group(0) + "\"");
|
|
System.out.println("group(1): \"" + m.group(1) + "\"");
|
|
}
|
|
}
|
|
report("surrogatePairOverlapRegion");
|
|
}
|
|
|
|
//This test is for 8037397
|
|
private static void droppedClassesWithIntersection() {
|
|
String rx = "[A-Z&&[A-Z]0-9]";
|
|
String ry = "[A-Z&&[A-F][G-Z]0-9]";
|
|
|
|
Stream<Character> letterChars = IntStream.range('A', 'Z').mapToObj((i) -> (char) i);
|
|
Stream<Character> digitChars = IntStream.range('0', '9').mapToObj((i) -> (char) i);
|
|
|
|
boolean letterCharsMatch = letterChars.allMatch((ch) -> {
|
|
String chString = ch.toString();
|
|
return chString.matches(rx) && chString.matches(ry);
|
|
});
|
|
|
|
boolean digitCharsDontMatch = digitChars.noneMatch((ch) -> {
|
|
String chString = ch.toString();
|
|
return chString.matches(rx) && chString.matches(ry);
|
|
});
|
|
|
|
|
|
if (!letterCharsMatch) {
|
|
failCount++;
|
|
System.out.println("Compiling intersection pattern is dropping a character class in its matcher");
|
|
}
|
|
|
|
if (!digitCharsDontMatch) {
|
|
failCount++;
|
|
System.out.println("Compiling intersection pattern is matching digits where it should not");
|
|
}
|
|
|
|
}
|
|
}
|