827e5e3226
8225052: javax.lang.model support for records 8225053: Preview APIs support for records 8225055: Javadoc for records 8226314: com.sun.source support for records 8227113: Specification for java.lang.Record 8233526: JVM support for records Implement records in the compiler and the JVM, including serialization, reflection and APIs support Co-authored-by: Brian Goetz <brian.goetz@oracle.com> Co-authored-by: Maurizio Cimadamore <maurizio.cimadamore@oracle.com> Co-authored-by: Harold Seigel <harold.seigel@oracle.com> Co-authored-by: Joe Darcy <joe.darcy@oracle.com> Co-authored-by: Jonathan Gibbons <jonathan.gibbons@oracle.com> Co-authored-by: Chris Hegarty <chris.hegarty@oracle.com> Co-authored-by: Jan Lahoda <jan.lahoda@oracle.com> Reviewed-by: mcimadamore, briangoetz, alanb, darcy, chegar, jrose, jlahoda, coleenp, dholmes, lfoltan, mchung, sadayapalam, hannesw, sspitsyn
90 lines
3.5 KiB
Java
90 lines
3.5 KiB
Java
/*
|
|
* Copyright (c) 2013, 2019, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
package tools.javac.combo;
|
|
|
|
import java.util.Map;
|
|
import java.util.function.Function;
|
|
import java.util.regex.Matcher;
|
|
import java.util.regex.Pattern;
|
|
|
|
/**
|
|
* A template into which tags of the form {@code #\{KEY\}} or
|
|
* {@code #\{KEY.SUBKEY\}} can be expanded.
|
|
*/
|
|
public interface Template {
|
|
static final Pattern KEY_PATTERN = Pattern.compile("#\\{([A-Z_][A-Z0-9_]*(?:\\[\\d+\\])?)(?:\\.([A-Z0-9_]*))?\\}");
|
|
|
|
String expand(String selector);
|
|
|
|
/* Looks for expandable keys. An expandable key can take the form:
|
|
* #{MAJOR}
|
|
* #{MAJOR.}
|
|
* #{MAJOR.MINOR}
|
|
* where MAJOR can be IDENTIFIER or IDENTIFIER[NUMERIC_INDEX]
|
|
* and MINOR can be an identifier.
|
|
*
|
|
* The ability to have an empty minor is provided on the
|
|
* assumption that some tests that can be written with this
|
|
* will find it useful to make a distinction akin to
|
|
* distinguishing F from F(), where F is a function pointer,
|
|
* and also cases of #{FOO.#{BAR}}, where BAR expands to an
|
|
* empty string.
|
|
*
|
|
* However, this being a general-purpose framework, the exact
|
|
* use is left up to the test writers.
|
|
*/
|
|
public static String expandTemplate(String template,
|
|
Map<String, Template> vars) {
|
|
return expandTemplate(template, vars::get);
|
|
}
|
|
|
|
private static String expandTemplate(String template, Function<String, Template> resolver) {
|
|
CharSequence in = template;
|
|
StringBuffer out = new StringBuffer();
|
|
while (true) {
|
|
boolean more = false;
|
|
Matcher m = KEY_PATTERN.matcher(in);
|
|
while (m.find()) {
|
|
String major = m.group(1);
|
|
String minor = m.group(2);
|
|
Template key = resolver.apply(major);
|
|
if (key == null)
|
|
throw new IllegalStateException("Unknown major key " + major);
|
|
|
|
String replacement = key.expand(minor == null ? "" : minor);
|
|
more |= KEY_PATTERN.matcher(replacement).find();
|
|
m.appendReplacement(out, replacement);
|
|
}
|
|
m.appendTail(out);
|
|
if (!more)
|
|
return out.toString();
|
|
else {
|
|
in = out;
|
|
out = new StringBuffer();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|