2012-11-17 19:01:03 +00:00
|
|
|
/*
|
2014-04-01 10:00:43 -07:00
|
|
|
* Copyright (c) 2012, 2014, Oracle and/or its affiliates. All rights reserved.
|
2012-11-17 19:01:03 +00:00
|
|
|
* 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
|
2013-01-23 20:57:40 +00:00
|
|
|
* @bug 8003280 8004102 8006694
|
2012-11-17 19:01:03 +00:00
|
|
|
* @summary Add lambda tests
|
|
|
|
* perform several automated checks in lambda conversion, esp. around accessibility
|
2013-01-23 20:57:40 +00:00
|
|
|
* temporarily workaround combo tests are causing time out in several platforms
|
2012-11-17 19:01:03 +00:00
|
|
|
* @author Maurizio Cimadamore
|
2013-01-08 13:47:57 +00:00
|
|
|
* @library ../lib
|
|
|
|
* @build JavacTestingAbstractThreadedTest
|
2014-04-01 10:00:43 -07:00
|
|
|
* @run main/timeout=600/othervm FunctionalInterfaceConversionTest
|
2012-11-17 19:01:03 +00:00
|
|
|
*/
|
|
|
|
|
2013-01-23 20:57:40 +00:00
|
|
|
// use /othervm to avoid jtreg timeout issues (CODETOOLS-7900047)
|
|
|
|
// see JDK-8006746
|
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
import java.io.IOException;
|
2012-11-17 19:01:03 +00:00
|
|
|
import java.net.URI;
|
|
|
|
import java.util.Arrays;
|
|
|
|
import javax.tools.Diagnostic;
|
|
|
|
import javax.tools.JavaCompiler;
|
|
|
|
import javax.tools.JavaFileObject;
|
|
|
|
import javax.tools.SimpleJavaFileObject;
|
|
|
|
import javax.tools.ToolProvider;
|
2013-01-08 13:47:57 +00:00
|
|
|
import com.sun.source.util.JavacTask;
|
2012-11-17 19:01:03 +00:00
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
public class FunctionalInterfaceConversionTest
|
|
|
|
extends JavacTestingAbstractThreadedTest
|
|
|
|
implements Runnable {
|
2012-11-17 19:01:03 +00:00
|
|
|
|
|
|
|
enum PackageKind {
|
|
|
|
NO_PKG(""),
|
|
|
|
PKG_A("a");
|
|
|
|
|
|
|
|
String pkg;
|
|
|
|
|
|
|
|
PackageKind(String pkg) {
|
|
|
|
this.pkg = pkg;
|
|
|
|
}
|
|
|
|
|
|
|
|
String getPkgDecl() {
|
|
|
|
return this == NO_PKG ?
|
|
|
|
"" :
|
|
|
|
"package " + pkg + ";";
|
|
|
|
}
|
|
|
|
|
|
|
|
String getImportStat() {
|
|
|
|
return this == NO_PKG ?
|
|
|
|
"" :
|
|
|
|
"import " + pkg + ".*;";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum SamKind {
|
|
|
|
CLASS("public class Sam { }"),
|
|
|
|
ABSTACT_CLASS("public abstract class Sam { }"),
|
|
|
|
ANNOTATION("public @interface Sam { }"),
|
|
|
|
ENUM("public enum Sam { }"),
|
|
|
|
INTERFACE("public interface Sam { \n #METH; \n }");
|
|
|
|
|
|
|
|
String sam_str;
|
|
|
|
|
|
|
|
SamKind(String sam_str) {
|
|
|
|
this.sam_str = sam_str;
|
|
|
|
}
|
|
|
|
|
|
|
|
String getSam(String methStr) {
|
|
|
|
return sam_str.replaceAll("#METH", methStr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum ModifierKind {
|
|
|
|
PUBLIC("public"),
|
|
|
|
PACKAGE("");
|
|
|
|
|
|
|
|
String modifier_str;
|
|
|
|
|
|
|
|
ModifierKind(String modifier_str) {
|
|
|
|
this.modifier_str = modifier_str;
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean stricterThan(ModifierKind that) {
|
|
|
|
return this.ordinal() > that.ordinal();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum TypeKind {
|
|
|
|
EXCEPTION("Exception"),
|
|
|
|
PKG_CLASS("PackageClass");
|
|
|
|
|
|
|
|
String typeStr;
|
|
|
|
|
|
|
|
private TypeKind(String typeStr) {
|
|
|
|
this.typeStr = typeStr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-30 15:14:25 +00:00
|
|
|
enum ExprKind {
|
|
|
|
LAMBDA("x -> null"),
|
|
|
|
MREF("this::m");
|
|
|
|
|
|
|
|
String exprStr;
|
|
|
|
|
|
|
|
private ExprKind(String exprStr) {
|
|
|
|
this.exprStr = exprStr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-17 19:01:03 +00:00
|
|
|
enum MethodKind {
|
|
|
|
NONE(""),
|
2012-11-30 15:14:25 +00:00
|
|
|
NON_GENERIC("public abstract #R m(#ARG s) throws #T;"),
|
|
|
|
GENERIC("public abstract <X> #R m(#ARG s) throws #T;");
|
2012-11-17 19:01:03 +00:00
|
|
|
|
|
|
|
String methodTemplate;
|
|
|
|
|
|
|
|
private MethodKind(String methodTemplate) {
|
|
|
|
this.methodTemplate = methodTemplate;
|
|
|
|
}
|
|
|
|
|
|
|
|
String getMethod(TypeKind retType, TypeKind argType, TypeKind thrownType) {
|
|
|
|
return methodTemplate.replaceAll("#R", retType.typeStr).
|
|
|
|
replaceAll("#ARG", argType.typeStr).
|
|
|
|
replaceAll("#T", thrownType.typeStr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
|
|
for (PackageKind samPkg : PackageKind.values()) {
|
|
|
|
for (ModifierKind modKind : ModifierKind.values()) {
|
|
|
|
for (SamKind samKind : SamKind.values()) {
|
2012-11-30 15:14:25 +00:00
|
|
|
for (MethodKind samMeth : MethodKind.values()) {
|
|
|
|
for (MethodKind clientMeth : MethodKind.values()) {
|
|
|
|
for (TypeKind retType : TypeKind.values()) {
|
|
|
|
for (TypeKind argType : TypeKind.values()) {
|
|
|
|
for (TypeKind thrownType : TypeKind.values()) {
|
|
|
|
for (ExprKind exprKind : ExprKind.values()) {
|
2013-01-08 13:47:57 +00:00
|
|
|
pool.execute(
|
|
|
|
new FunctionalInterfaceConversionTest(
|
|
|
|
samPkg, modKind, samKind,
|
|
|
|
samMeth, clientMeth, retType,
|
|
|
|
argType, thrownType, exprKind));
|
2012-11-30 15:14:25 +00:00
|
|
|
}
|
|
|
|
}
|
2012-11-17 19:01:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-01-08 13:47:57 +00:00
|
|
|
|
|
|
|
checkAfterExec(false);
|
2012-11-17 19:01:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PackageKind samPkg;
|
|
|
|
ModifierKind modKind;
|
|
|
|
SamKind samKind;
|
2012-11-30 15:14:25 +00:00
|
|
|
MethodKind samMeth;
|
|
|
|
MethodKind clientMeth;
|
2012-11-17 19:01:03 +00:00
|
|
|
TypeKind retType;
|
|
|
|
TypeKind argType;
|
|
|
|
TypeKind thrownType;
|
2012-11-30 15:14:25 +00:00
|
|
|
ExprKind exprKind;
|
|
|
|
DiagnosticChecker dc;
|
2012-11-17 19:01:03 +00:00
|
|
|
|
|
|
|
SourceFile samSourceFile = new SourceFile("Sam.java", "#P \n #C") {
|
2013-01-08 13:47:57 +00:00
|
|
|
@Override
|
2012-11-17 19:01:03 +00:00
|
|
|
public String toString() {
|
|
|
|
return template.replaceAll("#P", samPkg.getPkgDecl()).
|
2013-01-08 13:47:57 +00:00
|
|
|
replaceAll("#C", samKind.getSam(
|
|
|
|
samMeth.getMethod(retType, argType, thrownType)));
|
2012-11-17 19:01:03 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
SourceFile pkgClassSourceFile =
|
|
|
|
new SourceFile("PackageClass.java",
|
|
|
|
"#P\n #M class PackageClass extends Exception { }") {
|
|
|
|
@Override
|
2012-11-17 19:01:03 +00:00
|
|
|
public String toString() {
|
|
|
|
return template.replaceAll("#P", samPkg.getPkgDecl()).
|
|
|
|
replaceAll("#M", modKind.modifier_str);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
SourceFile clientSourceFile =
|
|
|
|
new SourceFile("Client.java",
|
|
|
|
"#I\n abstract class Client { \n" +
|
|
|
|
" Sam s = #E;\n" +
|
|
|
|
" #M \n }") {
|
|
|
|
@Override
|
2012-11-17 19:01:03 +00:00
|
|
|
public String toString() {
|
2012-11-30 15:14:25 +00:00
|
|
|
return template.replaceAll("#I", samPkg.getImportStat())
|
|
|
|
.replaceAll("#E", exprKind.exprStr)
|
|
|
|
.replaceAll("#M", clientMeth.getMethod(retType, argType, thrownType));
|
2012-11-17 19:01:03 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
FunctionalInterfaceConversionTest(PackageKind samPkg, ModifierKind modKind,
|
|
|
|
SamKind samKind, MethodKind samMeth, MethodKind clientMeth,
|
|
|
|
TypeKind retType, TypeKind argType, TypeKind thrownType,
|
|
|
|
ExprKind exprKind) {
|
2012-11-17 19:01:03 +00:00
|
|
|
this.samPkg = samPkg;
|
|
|
|
this.modKind = modKind;
|
|
|
|
this.samKind = samKind;
|
2012-11-30 15:14:25 +00:00
|
|
|
this.samMeth = samMeth;
|
|
|
|
this.clientMeth = clientMeth;
|
2012-11-17 19:01:03 +00:00
|
|
|
this.retType = retType;
|
|
|
|
this.argType = argType;
|
|
|
|
this.thrownType = thrownType;
|
2012-11-30 15:14:25 +00:00
|
|
|
this.exprKind = exprKind;
|
|
|
|
this.dc = new DiagnosticChecker();
|
2012-11-17 19:01:03 +00:00
|
|
|
}
|
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
@Override
|
|
|
|
public void run() {
|
|
|
|
final JavaCompiler tool = ToolProvider.getSystemJavaCompiler();
|
|
|
|
|
|
|
|
JavacTask ct = (JavacTask)tool.getTask(null, fm.get(), dc, null, null,
|
|
|
|
Arrays.asList(samSourceFile, pkgClassSourceFile, clientSourceFile));
|
|
|
|
try {
|
|
|
|
ct.analyze();
|
|
|
|
} catch (IOException ex) {
|
|
|
|
throw new AssertionError("Test failing with cause", ex.getCause());
|
|
|
|
}
|
2012-11-17 19:01:03 +00:00
|
|
|
if (dc.errorFound == checkSamConversion()) {
|
2013-01-08 13:47:57 +00:00
|
|
|
throw new AssertionError(samSourceFile + "\n\n" +
|
|
|
|
pkgClassSourceFile + "\n\n" + clientSourceFile);
|
2012-11-17 19:01:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean checkSamConversion() {
|
|
|
|
if (samKind != SamKind.INTERFACE) {
|
|
|
|
//sam type must be an interface
|
|
|
|
return false;
|
2012-11-30 15:14:25 +00:00
|
|
|
} else if (samMeth == MethodKind.NONE) {
|
|
|
|
//interface must have at least a method
|
|
|
|
return false;
|
|
|
|
} else if (exprKind == ExprKind.LAMBDA &&
|
|
|
|
samMeth != MethodKind.NON_GENERIC) {
|
|
|
|
//target method for lambda must be non-generic
|
|
|
|
return false;
|
|
|
|
} else if (exprKind == ExprKind.MREF &&
|
|
|
|
clientMeth == MethodKind.NONE) {
|
2012-11-17 19:01:03 +00:00
|
|
|
return false;
|
|
|
|
} else if (samPkg != PackageKind.NO_PKG &&
|
|
|
|
modKind != ModifierKind.PUBLIC &&
|
|
|
|
(retType == TypeKind.PKG_CLASS ||
|
|
|
|
argType == TypeKind.PKG_CLASS ||
|
|
|
|
thrownType == TypeKind.PKG_CLASS)) {
|
|
|
|
//target must not contain inaccessible types
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
abstract class SourceFile extends SimpleJavaFileObject {
|
|
|
|
|
|
|
|
protected String template;
|
|
|
|
|
|
|
|
public SourceFile(String filename, String template) {
|
|
|
|
super(URI.create("myfo:/" + filename), JavaFileObject.Kind.SOURCE);
|
|
|
|
this.template = template;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public CharSequence getCharContent(boolean ignoreEncodingErrors) {
|
|
|
|
return toString();
|
|
|
|
}
|
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
@Override
|
2012-11-17 19:01:03 +00:00
|
|
|
public abstract String toString();
|
|
|
|
}
|
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
static class DiagnosticChecker
|
|
|
|
implements javax.tools.DiagnosticListener<JavaFileObject> {
|
2012-11-17 19:01:03 +00:00
|
|
|
|
|
|
|
boolean errorFound = false;
|
|
|
|
|
2013-01-08 13:47:57 +00:00
|
|
|
@Override
|
2012-11-17 19:01:03 +00:00
|
|
|
public void report(Diagnostic<? extends JavaFileObject> diagnostic) {
|
|
|
|
if (diagnostic.getKind() == Diagnostic.Kind.ERROR) {
|
|
|
|
errorFound = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|