8023524: Mechanism to dump generated lambda classes / log lambda code generation
Co-authored-by: Brian Goetz <brian.goetz@oracle.com> Reviewed-by: plevart, mchung, forax, jjb
This commit is contained in:
parent
242f0dd3c2
commit
1f02e9968e
@ -27,12 +27,15 @@ package java.lang.invoke;
|
|||||||
|
|
||||||
import jdk.internal.org.objectweb.asm.*;
|
import jdk.internal.org.objectweb.asm.*;
|
||||||
import sun.misc.Unsafe;
|
import sun.misc.Unsafe;
|
||||||
|
import sun.security.action.GetPropertyAction;
|
||||||
|
|
||||||
|
import java.io.FilePermission;
|
||||||
import java.lang.reflect.Constructor;
|
import java.lang.reflect.Constructor;
|
||||||
import java.security.AccessController;
|
import java.security.AccessController;
|
||||||
import java.security.PrivilegedAction;
|
import java.security.PrivilegedAction;
|
||||||
import java.security.ProtectionDomain;
|
import java.security.ProtectionDomain;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
import java.util.PropertyPermission;
|
||||||
|
|
||||||
import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
||||||
|
|
||||||
@ -66,12 +69,23 @@ import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
|||||||
// Used to ensure that each spun class name is unique
|
// Used to ensure that each spun class name is unique
|
||||||
private static final AtomicInteger counter = new AtomicInteger(0);
|
private static final AtomicInteger counter = new AtomicInteger(0);
|
||||||
|
|
||||||
|
// For dumping generated classes to disk, for debugging purposes
|
||||||
|
private static final ProxyClassesDumper dumper;
|
||||||
|
|
||||||
|
static {
|
||||||
|
final String key = "jdk.internal.lambda.dumpProxyClasses";
|
||||||
|
String path = AccessController.doPrivileged(
|
||||||
|
new GetPropertyAction(key), null,
|
||||||
|
new PropertyPermission(key , "read"));
|
||||||
|
dumper = (null == path) ? null : ProxyClassesDumper.getInstance(path);
|
||||||
|
}
|
||||||
|
|
||||||
// See context values in AbstractValidatingLambdaMetafactory
|
// See context values in AbstractValidatingLambdaMetafactory
|
||||||
private final String implMethodClassName; // Name of type containing implementation "CC"
|
private final String implMethodClassName; // Name of type containing implementation "CC"
|
||||||
private final String implMethodName; // Name of implementation method "impl"
|
private final String implMethodName; // Name of implementation method "impl"
|
||||||
private final String implMethodDesc; // Type descriptor for implementation methods "(I)Ljava/lang/String;"
|
private final String implMethodDesc; // Type descriptor for implementation methods "(I)Ljava/lang/String;"
|
||||||
private final Type[] implMethodArgumentTypes; // ASM types for implementaion method parameters
|
private final Type[] implMethodArgumentTypes; // ASM types for implementation method parameters
|
||||||
private final Type implMethodReturnType; // ASM type for implementaion method return type "Ljava/lang/String;"
|
private final Type implMethodReturnType; // ASM type for implementation method return type "Ljava/lang/String;"
|
||||||
private final MethodType constructorType; // Generated class constructor type "(CC)void"
|
private final MethodType constructorType; // Generated class constructor type "(CC)void"
|
||||||
private final String constructorDesc; // Type descriptor for constructor "(LCC;)V"
|
private final String constructorDesc; // Type descriptor for constructor "(LCC;)V"
|
||||||
private final ClassWriter cw; // ASM class writer
|
private final ClassWriter cw; // ASM class writer
|
||||||
@ -259,29 +273,31 @@ import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
|||||||
|
|
||||||
final byte[] classBytes = cw.toByteArray();
|
final byte[] classBytes = cw.toByteArray();
|
||||||
|
|
||||||
/*** Uncomment to dump the generated file
|
// If requested, dump out to a file for debugging purposes
|
||||||
System.out.printf("Loaded: %s (%d bytes) %n", lambdaClassName,
|
if (dumper != null) {
|
||||||
classBytes.length);
|
AccessController.doPrivileged(new PrivilegedAction<Void>() {
|
||||||
try (FileOutputStream fos = new FileOutputStream(lambdaClassName
|
@Override
|
||||||
.replace('/', '.') + ".class")) {
|
public Void run() {
|
||||||
fos.write(classBytes);
|
dumper.dumpClass(lambdaClassName, classBytes);
|
||||||
} catch (IOException ex) {
|
return null;
|
||||||
PlatformLogger.getLogger(InnerClassLambdaMetafactory.class
|
}
|
||||||
.getName()).severe(ex.getMessage(), ex);
|
}, null,
|
||||||
}
|
new FilePermission("<<ALL FILES>>", "read, write"),
|
||||||
***/
|
// createDirectories may need it
|
||||||
|
new PropertyPermission("user.dir", "read"));
|
||||||
|
}
|
||||||
|
|
||||||
ClassLoader loader = targetClass.getClassLoader();
|
ClassLoader loader = targetClass.getClassLoader();
|
||||||
ProtectionDomain pd = (loader == null)
|
ProtectionDomain pd = (loader == null)
|
||||||
? null
|
? null
|
||||||
: AccessController.doPrivileged(
|
: AccessController.doPrivileged(
|
||||||
new PrivilegedAction<ProtectionDomain>() {
|
new PrivilegedAction<ProtectionDomain>() {
|
||||||
@Override
|
@Override
|
||||||
public ProtectionDomain run() {
|
public ProtectionDomain run() {
|
||||||
return targetClass.getProtectionDomain();
|
return targetClass.getProtectionDomain();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
return UNSAFE.defineClass(lambdaClassName,
|
return UNSAFE.defineClass(lambdaClassName,
|
||||||
classBytes, 0, classBytes.length,
|
classBytes, 0, classBytes.length,
|
||||||
|
147
jdk/src/share/classes/java/lang/invoke/ProxyClassesDumper.java
Normal file
147
jdk/src/share/classes/java/lang/invoke/ProxyClassesDumper.java
Normal file
@ -0,0 +1,147 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 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 java.lang.invoke;
|
||||||
|
|
||||||
|
import sun.util.logging.PlatformLogger;
|
||||||
|
|
||||||
|
import java.io.FilePermission;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.InvalidPathException;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
import java.security.AccessController;
|
||||||
|
import java.security.PrivilegedAction;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper class used by InnerClassLambdaMetafactory to log generated classes
|
||||||
|
*
|
||||||
|
* @implNote
|
||||||
|
* <p> Because this class is called by LambdaMetafactory, make use
|
||||||
|
* of lambda lead to recursive calls cause stack overflow.
|
||||||
|
*/
|
||||||
|
final class ProxyClassesDumper {
|
||||||
|
private static final char[] HEX = {
|
||||||
|
'0', '1', '2', '3', '4', '5', '6', '7',
|
||||||
|
'8', '9', 'A', 'B', 'C', 'D', 'E', 'F'
|
||||||
|
};
|
||||||
|
private static final char[] BAD_CHARS = {
|
||||||
|
'\\', ':', '*', '?', '"', '<', '>', '|'
|
||||||
|
};
|
||||||
|
private static final String[] REPLACEMENT = {
|
||||||
|
"%5C", "%3A", "%2A", "%3F", "%22", "%3C", "%3E", "%7C"
|
||||||
|
};
|
||||||
|
|
||||||
|
private final Path dumpDir;
|
||||||
|
|
||||||
|
public static ProxyClassesDumper getInstance(String path) {
|
||||||
|
if (null == path) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
path = path.trim();
|
||||||
|
final Path dir = Paths.get(path.length() == 0 ? "." : path);
|
||||||
|
AccessController.doPrivileged(new PrivilegedAction<Void>() {
|
||||||
|
@Override
|
||||||
|
public Void run() {
|
||||||
|
validateDumpDir(dir);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}, null, new FilePermission("<<ALL FILES>>", "read, write"));
|
||||||
|
return new ProxyClassesDumper(dir);
|
||||||
|
} catch (InvalidPathException ex) {
|
||||||
|
PlatformLogger.getLogger(ProxyClassesDumper.class.getName())
|
||||||
|
.warning("Path " + path + " is not valid - dumping disabled", ex);
|
||||||
|
} catch (IllegalArgumentException iae) {
|
||||||
|
PlatformLogger.getLogger(ProxyClassesDumper.class.getName())
|
||||||
|
.warning(iae.getMessage() + " - dumping disabled");
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private ProxyClassesDumper(Path path) {
|
||||||
|
dumpDir = Objects.requireNonNull(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void validateDumpDir(Path path) {
|
||||||
|
if (!Files.exists(path)) {
|
||||||
|
throw new IllegalArgumentException("Directory " + path + " does not exist");
|
||||||
|
} else if (!Files.isDirectory(path)) {
|
||||||
|
throw new IllegalArgumentException("Path " + path + " is not a directory");
|
||||||
|
} else if (!Files.isWritable(path)) {
|
||||||
|
throw new IllegalArgumentException("Directory " + path + " is not writable");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String encodeForFilename(String className) {
|
||||||
|
final int len = className.length();
|
||||||
|
StringBuilder sb = new StringBuilder(len);
|
||||||
|
|
||||||
|
for (int i = 0; i < len; i++) {
|
||||||
|
char c = className.charAt(i);
|
||||||
|
// control characters
|
||||||
|
if (c <= 31) {
|
||||||
|
sb.append('%');
|
||||||
|
sb.append(HEX[c >> 4 & 0x0F]);
|
||||||
|
sb.append(HEX[c & 0x0F]);
|
||||||
|
} else {
|
||||||
|
int j = 0;
|
||||||
|
for (; j < BAD_CHARS.length; j++) {
|
||||||
|
if (c == BAD_CHARS[j]) {
|
||||||
|
sb.append(REPLACEMENT[j]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (j >= BAD_CHARS.length) {
|
||||||
|
sb.append(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void dumpClass(String className, final byte[] classBytes) {
|
||||||
|
Path file;
|
||||||
|
try {
|
||||||
|
file = dumpDir.resolve(encodeForFilename(className) + ".class");
|
||||||
|
} catch (InvalidPathException ex) {
|
||||||
|
PlatformLogger.getLogger(ProxyClassesDumper.class.getName())
|
||||||
|
.warning("Invalid path for class " + className);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
Path dir = file.getParent();
|
||||||
|
Files.createDirectories(dir);
|
||||||
|
Files.write(file, classBytes);
|
||||||
|
} catch (Exception ignore) {
|
||||||
|
PlatformLogger.getLogger(ProxyClassesDumper.class.getName())
|
||||||
|
.warning("Exception writing to path at " + file.toString());
|
||||||
|
// simply don't care if this operation failed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
208
jdk/test/java/lang/invoke/lambda/LogGeneratedClassesTest.java
Normal file
208
jdk/test/java/lang/invoke/lambda/LogGeneratedClassesTest.java
Normal file
@ -0,0 +1,208 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 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.
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
* @bug 8023524
|
||||||
|
* @summary tests logging generated classes for lambda
|
||||||
|
* @library /java/nio/file
|
||||||
|
* @run testng LogGeneratedClassesTest
|
||||||
|
*/
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.LinkOption;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
import java.nio.file.attribute.PosixFileAttributeView;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
|
import org.testng.annotations.AfterClass;
|
||||||
|
import org.testng.annotations.BeforeClass;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
import org.testng.SkipException;
|
||||||
|
|
||||||
|
import static java.nio.file.attribute.PosixFilePermissions.*;
|
||||||
|
import static org.testng.Assert.assertEquals;
|
||||||
|
import static org.testng.Assert.assertFalse;
|
||||||
|
import static org.testng.Assert.assertTrue;
|
||||||
|
|
||||||
|
public class LogGeneratedClassesTest extends LUtils {
|
||||||
|
String longFQCN;
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public void setup() throws IOException {
|
||||||
|
final List<String> scratch = new ArrayList<>();
|
||||||
|
scratch.clear();
|
||||||
|
scratch.add("package com.example;");
|
||||||
|
scratch.add("public class TestLambda {");
|
||||||
|
scratch.add(" interface I {");
|
||||||
|
scratch.add(" int foo();");
|
||||||
|
scratch.add(" }");
|
||||||
|
scratch.add(" public static void main(String[] args) {");
|
||||||
|
scratch.add(" I lam = () -> 10;");
|
||||||
|
scratch.add(" Runnable r = () -> {");
|
||||||
|
scratch.add(" System.out.println(\"Runnable\");");
|
||||||
|
scratch.add(" };");
|
||||||
|
scratch.add(" r.run();");
|
||||||
|
scratch.add(" System.out.println(\"Finish\");");
|
||||||
|
scratch.add(" }");
|
||||||
|
scratch.add("}");
|
||||||
|
|
||||||
|
File test = new File("TestLambda.java");
|
||||||
|
createFile(test, scratch);
|
||||||
|
compile("-d", ".", test.getName());
|
||||||
|
|
||||||
|
scratch.remove(0);
|
||||||
|
scratch.remove(0);
|
||||||
|
scratch.add(0, "public class LongPackageName {");
|
||||||
|
StringBuilder sb = new StringBuilder("com.example.");
|
||||||
|
// longer than 255 which exceed max length of most filesystems
|
||||||
|
for (int i = 0; i < 30; i++) {
|
||||||
|
sb.append("nonsense.");
|
||||||
|
}
|
||||||
|
sb.append("enough");
|
||||||
|
longFQCN = sb.toString() + ".LongPackageName";
|
||||||
|
sb.append(";");
|
||||||
|
sb.insert(0, "package ");
|
||||||
|
scratch.add(0, sb.toString());
|
||||||
|
test = new File("LongPackageName.java");
|
||||||
|
createFile(test, scratch);
|
||||||
|
compile("-d", ".", test.getName());
|
||||||
|
|
||||||
|
// create target
|
||||||
|
Files.createDirectory(Paths.get("dump"));
|
||||||
|
Files.createDirectories(Paths.get("dumpLong/com/example/nonsense"));
|
||||||
|
Files.createFile(Paths.get("dumpLong/com/example/nonsense/nonsense"));
|
||||||
|
Files.createFile(Paths.get("file"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterClass
|
||||||
|
public void cleanup() throws IOException {
|
||||||
|
Files.delete(Paths.get("TestLambda.java"));
|
||||||
|
Files.delete(Paths.get("LongPackageName.java"));
|
||||||
|
Files.delete(Paths.get("file"));
|
||||||
|
TestUtil.removeAll(Paths.get("com"));
|
||||||
|
TestUtil.removeAll(Paths.get("dump"));
|
||||||
|
TestUtil.removeAll(Paths.get("dumpLong"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testNotLogging() {
|
||||||
|
TestResult tr = doExec(JAVA_CMD.getAbsolutePath(),
|
||||||
|
"-cp", ".",
|
||||||
|
"-Djava.security.manager",
|
||||||
|
"com.example.TestLambda");
|
||||||
|
tr.assertZero("Should still return 0");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testLogging() throws IOException {
|
||||||
|
assertTrue(Files.exists(Paths.get("dump")));
|
||||||
|
TestResult tr = doExec(JAVA_CMD.getAbsolutePath(),
|
||||||
|
"-cp", ".",
|
||||||
|
"-Djdk.internal.lambda.dumpProxyClasses=dump",
|
||||||
|
"-Djava.security.manager",
|
||||||
|
"com.example.TestLambda");
|
||||||
|
// dump/com/example + 2 class files
|
||||||
|
assertEquals(Files.walk(Paths.get("dump")).count(), 5, "Two lambda captured");
|
||||||
|
tr.assertZero("Should still return 0");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDumpDirNotExist() throws IOException {
|
||||||
|
assertFalse(Files.exists(Paths.get("notExist")));
|
||||||
|
TestResult tr = doExec(JAVA_CMD.getAbsolutePath(),
|
||||||
|
"-cp", ".",
|
||||||
|
"-Djdk.internal.lambda.dumpProxyClasses=notExist",
|
||||||
|
"-Djava.security.manager",
|
||||||
|
"com.example.TestLambda");
|
||||||
|
assertEquals(tr.testOutput.stream()
|
||||||
|
.filter(s -> s.startsWith("WARNING"))
|
||||||
|
.peek(s -> assertTrue(s.contains("does not exist")))
|
||||||
|
.count(),
|
||||||
|
1, "only show error once");
|
||||||
|
tr.assertZero("Should still return 0");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDumpDirIsFile() throws IOException {
|
||||||
|
assertTrue(Files.isRegularFile(Paths.get("file")));
|
||||||
|
TestResult tr = doExec(JAVA_CMD.getAbsolutePath(),
|
||||||
|
"-cp", ".",
|
||||||
|
"-Djdk.internal.lambda.dumpProxyClasses=file",
|
||||||
|
"-Djava.security.manager",
|
||||||
|
"com.example.TestLambda");
|
||||||
|
assertEquals(tr.testOutput.stream()
|
||||||
|
.filter(s -> s.startsWith("WARNING"))
|
||||||
|
.peek(s -> assertTrue(s.contains("not a directory")))
|
||||||
|
.count(),
|
||||||
|
1, "only show error once");
|
||||||
|
tr.assertZero("Should still return 0");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDumpDirNotWritable() throws IOException {
|
||||||
|
if (! Files.getFileStore(Paths.get("."))
|
||||||
|
.supportsFileAttributeView(PosixFileAttributeView.class)) {
|
||||||
|
// No easy way to setup readonly directory
|
||||||
|
throw new SkipException("Posix not supported");
|
||||||
|
}
|
||||||
|
|
||||||
|
Files.createDirectory(Paths.get("readOnly"),
|
||||||
|
asFileAttribute(fromString("r-xr-xr-x")));
|
||||||
|
|
||||||
|
TestResult tr = doExec(JAVA_CMD.getAbsolutePath(),
|
||||||
|
"-cp", ".",
|
||||||
|
"-Djdk.internal.lambda.dumpProxyClasses=readOnly",
|
||||||
|
"-Djava.security.manager",
|
||||||
|
"com.example.TestLambda");
|
||||||
|
assertEquals(tr.testOutput.stream()
|
||||||
|
.filter(s -> s.startsWith("WARNING"))
|
||||||
|
.peek(s -> assertTrue(s.contains("not writable")))
|
||||||
|
.count(),
|
||||||
|
1, "only show error once");
|
||||||
|
tr.assertZero("Should still return 0");
|
||||||
|
|
||||||
|
TestUtil.removeAll(Paths.get("readOnly"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testLoggingException() throws IOException {
|
||||||
|
assertTrue(Files.exists(Paths.get("dumpLong")));
|
||||||
|
TestResult tr = doExec(JAVA_CMD.getAbsolutePath(),
|
||||||
|
"-cp", ".",
|
||||||
|
"-Djdk.internal.lambda.dumpProxyClasses=dumpLong",
|
||||||
|
"-Djava.security.manager",
|
||||||
|
longFQCN);
|
||||||
|
assertEquals(tr.testOutput.stream()
|
||||||
|
.filter(s -> s.startsWith("WARNING: Exception"))
|
||||||
|
.count(),
|
||||||
|
2, "show error each capture");
|
||||||
|
// dumpLong/com/example/nosense/nosense
|
||||||
|
assertEquals(Files.walk(Paths.get("dumpLong")).count(), 5, "Two lambda captured failed to log");
|
||||||
|
tr.assertZero("Should still return 0");
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user