1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
|
/*
* Copyright (c) 2010, 2013, 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. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* 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.
*/
package jdk.nashorn.internal.runtime.regexp;
import java.util.Collections;
import java.util.Set;
import java.util.WeakHashMap;
import jdk.nashorn.internal.runtime.ParserException;
import jdk.nashorn.internal.runtime.options.Options;
/**
* Factory class for regular expressions. This class creates instances of {@link JdkRegExp}.
* An alternative factory can be installed using the {@code nashorn.regexp.impl} system property.
*/
public class RegExpFactory {
private final static RegExpFactory instance;
private final static String JDK = "jdk";
private final static String JONI = "joni";
/** Weak cache of already validated regexps - when reparsing, we don't, for example
* need to recompile (reverify) all regexps that have previously been parsed by this
* RegExpFactory in a previous compilation. This saves significant time in e.g. avatar
* startup */
private static final Set<String> VALID_CACHE_SET =
Collections.newSetFromMap(
Collections.synchronizedMap(
new WeakHashMap<String, Boolean>()));
static {
final String impl = Options.getStringProperty("nashorn.regexp.impl", JONI);
switch (impl) {
case JONI:
instance = new JoniRegExp.Factory();
break;
case JDK:
instance = new RegExpFactory();
break;
default:
instance = null;
throw new InternalError("Unsupported RegExp factory: " + impl);
}
}
/**
* Creates a Regular expression from the given {@code pattern} and {@code flags} strings.
*
* @param pattern RegExp pattern string
* @param flags RegExp flags string
* @return new RegExp
* @throws ParserException if flags is invalid or pattern string has syntax error.
*/
public RegExp compile(final String pattern, final String flags) throws ParserException {
return new JdkRegExp(pattern, flags);
}
/**
* Compile a regexp with the given {@code source} and {@code flags}.
*
* @param pattern RegExp pattern string
* @param flags flag string
* @return new RegExp
* @throws ParserException if invalid source or flags
*/
public static RegExp create(final String pattern, final String flags) {
return instance.compile(pattern, flags);
}
/**
* Validate a regexp with the given {@code source} and {@code flags}.
*
* @param pattern RegExp pattern string
* @param flags flag string
*
* @throws ParserException if invalid source or flags
*/
// @SuppressWarnings({"unused"})
public static void validate(final String pattern, final String flags) throws ParserException {
if (VALID_CACHE_SET.add(pattern + flags)) {
instance.compile(pattern, flags);
}
}
/**
* Returns true if the instance uses the JDK's {@code java.util.regex} package.
*
* @return true if instance uses JDK regex package
*/
public static boolean usesJavaUtilRegex() {
return instance != null && instance.getClass() == RegExpFactory.class;
}
}
|