jdk-24/langtools/test/tools/jdeps/VerboseFormat/JdepsDependencyClosure.java

497 lines
21 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2015, 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.
*/
import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.Stream;
/**
* @test
* @bug 8080608
* @summary Test that jdeps verbose output has a summary line when dependencies
* are found within the same archive. For each testcase, compare the
* result obtained from jdeps with the expected result.
* @modules jdk.jdeps/com.sun.tools.jdeps
* @build use.indirect.DontUseUnsafe2
* @build use.indirect.UseUnsafeIndirectly
* @build use.indirect2.DontUseUnsafe3
* @build use.indirect2.UseUnsafeIndirectly2
* @build use.unsafe.DontUseUnsafe
* @build use.unsafe.UseClassWithUnsafe
* @build use.unsafe.UseUnsafeClass
* @build use.unsafe.UseUnsafeClass2
* @run main JdepsDependencyClosure --test:0
* @run main JdepsDependencyClosure --test:1
* @run main JdepsDependencyClosure --test:2
* @run main JdepsDependencyClosure --test:3
*/
public class JdepsDependencyClosure {
static boolean VERBOSE = false;
static boolean COMPARE_TEXT = true;
static final String JDEPS_SUMMARY_TEXT_FORMAT = "%s -> %s%n";
static final String JDEPS_VERBOSE_TEXT_FORMAT = " %-50s -> %-50s %s%n";
/**
* Helper class used to store arguments to pass to
* {@code JdepsDependencyClosure.test} as well as expected
* results.
*/
static class TestCaseData {
final Map<String, Set<String>> expectedDependencies;
final String expectedText;
final String[] args;
final boolean closure;
TestCaseData(Map<String, Set<String>> expectedDependencies,
String expectedText,
boolean closure,
String[] args) {
this.expectedDependencies = expectedDependencies;
this.expectedText = expectedText;
this.closure = closure;
this.args = args;
}
public void test() {
if (expectedDependencies != null) {
String format = closure
? "Running (closure): jdeps %s %s %s %s"
: "Running: jdeps %s %s %s %s";
System.out.println(String.format(format, (Object[])args));
}
JdepsDependencyClosure.test(args, expectedDependencies, expectedText, closure);
}
/**
* Make a new test case data to invoke jdeps and test its output.
* @param pattern The pattern that will passed through to jdeps -e
* This is expected to match only one class.
* @param arcPath The archive to analyze. A jar or a class directory.
* @param classes For each reported archive dependency couple, the
* expected list of classes in the source that will
* be reported as having a dependency on the class
* in the target that matches the given pattern.
* @param dependencies For each archive dependency couple, a singleton list
* containing the name of the class in the target that
* matches the pattern. It is expected that the pattern
* will match only one class in the target.
* If the pattern matches several classes the
* expected text may no longer match the jdeps output.
* @param archives A list of archive dependency couple in the form
* {{sourceName1, sourcePath1, targetDescription1, targetPath1}
* {sourceName2, sourcePath2, targetDescription2, targetPath2}
* ... }
* For a JDK module - e.g. java.base, the targetDescription
* is usually something like "JDK internal API (java.base)"
* and the targetPath is usually the module name "java.base".
* @param closure Whether jdeps should be recursively invoked to build
* the closure.
* @return An instance of TestCaseData containing all the information
* needed to perform the jdeps invokation and test its output.
*/
public static TestCaseData make(String pattern, String arcPath, String[][] classes,
String[][] dependencies, String[][] archives, boolean closure) {
final String[] args = new String[] {
"-e", pattern, "-v", arcPath
};
Map<String, Set<String>> expected = new HashMap<>();
String expectedText = "";
for (int i=0; i<classes.length; i++) {
final int index = i;
expectedText += Stream.of(classes[i])
.map((cn) -> String.format(JDEPS_VERBOSE_TEXT_FORMAT, cn,
dependencies[index][0], archives[index][2]))
.reduce(String.format(JDEPS_SUMMARY_TEXT_FORMAT, archives[i][0],
archives[index][3]), (s1,s2) -> s1.concat(s2));
for (String cn : classes[index]) {
expected.putIfAbsent(cn, new HashSet<>());
expected.get(cn).add(dependencies[index][0]);
}
}
return new TestCaseData(expected, expectedText, closure, args);
}
public static TestCaseData valueOf(String[] args) {
if (args.length == 1 && args[0].startsWith("--test:")) {
// invoked from jtreg. build test case data for selected test.
int index = Integer.parseInt(args[0].substring("--test:".length()));
if (index >= dataSuppliers.size()) {
throw new RuntimeException("No such test case: " + index
+ " - available testcases are [0.."
+ (dataSuppliers.size()-1) + "]");
}
return dataSuppliers.get(index).get();
} else {
// invoked in standalone. just take the given argument
// and perform no validation on the output (except that it
// must start with a summary line)
return new TestCaseData(null, null, true, args);
}
}
}
static TestCaseData makeTestCaseOne() {
final String arcPath = System.getProperty("test.classes", "build/classes");
final String arcName = Paths.get(arcPath).getFileName().toString();
final String[][] classes = new String[][] {
{"use.indirect2.UseUnsafeIndirectly2", "use.unsafe.UseClassWithUnsafe"},
};
final String[][] dependencies = new String[][] {
{"use.unsafe.UseUnsafeClass"},
};
final String[][] archives = new String[][] {
{arcName, arcPath, arcName, arcPath},
};
return TestCaseData.make("use.unsafe.UseUnsafeClass", arcPath, classes,
dependencies, archives, false);
}
static TestCaseData makeTestCaseTwo() {
String arcPath = System.getProperty("test.classes", "build/classes");
String arcName = Paths.get(arcPath).getFileName().toString();
String[][] classes = new String[][] {
{"use.unsafe.UseUnsafeClass", "use.unsafe.UseUnsafeClass2"}
};
String[][] dependencies = new String[][] {
{"sun.misc.Unsafe"}
};
String[][] archive = new String[][] {
{arcName, arcPath, "JDK internal API (java.base)", "java.base"},
};
return TestCaseData.make("sun.misc.Unsafe", arcPath, classes,
dependencies, archive, false);
}
static TestCaseData makeTestCaseThree() {
final String arcPath = System.getProperty("test.classes", "build/classes");
final String arcName = Paths.get(arcPath).getFileName().toString();
final String[][] classes = new String[][] {
{"use.indirect2.UseUnsafeIndirectly2", "use.unsafe.UseClassWithUnsafe"},
{"use.indirect.UseUnsafeIndirectly"}
};
final String[][] dependencies = new String[][] {
{"use.unsafe.UseUnsafeClass"},
{"use.unsafe.UseClassWithUnsafe"}
};
final String[][] archives = new String[][] {
{arcName, arcPath, arcName, arcPath},
{arcName, arcPath, arcName, arcPath}
};
return TestCaseData.make("use.unsafe.UseUnsafeClass", arcPath, classes,
dependencies, archives, true);
}
static TestCaseData makeTestCaseFour() {
final String arcPath = System.getProperty("test.classes", "build/classes");
final String arcName = Paths.get(arcPath).getFileName().toString();
final String[][] classes = new String[][] {
{"use.unsafe.UseUnsafeClass", "use.unsafe.UseUnsafeClass2"},
{"use.indirect2.UseUnsafeIndirectly2", "use.unsafe.UseClassWithUnsafe"},
{"use.indirect.UseUnsafeIndirectly"}
};
final String[][] dependencies = new String[][] {
{"sun.misc.Unsafe"},
{"use.unsafe.UseUnsafeClass"},
{"use.unsafe.UseClassWithUnsafe"}
};
final String[][] archives = new String[][] {
{arcName, arcPath, "JDK internal API (java.base)", "java.base"},
{arcName, arcPath, arcName, arcPath},
{arcName, arcPath, arcName, arcPath}
};
return TestCaseData.make("sun.misc.Unsafe", arcPath, classes, dependencies,
archives, true);
}
static final List<Supplier<TestCaseData>> dataSuppliers = Arrays.asList(
JdepsDependencyClosure::makeTestCaseOne,
JdepsDependencyClosure::makeTestCaseTwo,
JdepsDependencyClosure::makeTestCaseThree,
JdepsDependencyClosure::makeTestCaseFour
);
/**
* The OutputStreamParser is used to parse the format of jdeps.
* It is thus dependent on that format.
*/
static class OutputStreamParser extends OutputStream {
// OutputStreamParser will populate this map:
//
// For each archive, a list of class in where dependencies where
// found...
final Map<String, Set<String>> deps;
final StringBuilder text = new StringBuilder();
StringBuilder[] lines = { new StringBuilder(), new StringBuilder() };
int line = 0;
int sepi = 0;
char[] sep;
public OutputStreamParser(Map<String, Set<String>> deps) {
this.deps = deps;
this.sep = System.getProperty("line.separator").toCharArray();
}
@Override
public void write(int b) throws IOException {
lines[line].append((char)b);
if (b == sep[sepi]) {
if (++sepi == sep.length) {
text.append(lines[line]);
if (lines[0].toString().startsWith(" ")) {
throw new RuntimeException("Bad formatting: "
+ "summary line missing for\n"+lines[0]);
}
// Usually the output looks like that:
// <archive-1> -> java.base
// <class-1> -> <dependency> <dependency description>
// <class-2> -> <dependency> <dependency description>
// ...
// <archive-2> -> java.base
// <class-3> -> <dependency> <dependency description>
// <class-4> -> <dependency> <dependency description>
// ...
//
// We want to keep the <archive> line in lines[0]
// and have the ith <class-i> line in lines[1]
if (line == 1) {
// we have either a <class> line or an <archive> line.
String line1 = lines[0].toString();
String line2 = lines[1].toString();
if (line2.startsWith(" ")) {
// we have a class line, record it.
parse(line1, line2);
// prepare for next <class> line.
lines[1] = new StringBuilder();
} else {
// We have an archive line: We are switching to the next archive.
// put the new <archive> line in lines[0], and prepare
// for reading the next <class> line
lines[0] = lines[1];
lines[1] = new StringBuilder();
}
} else {
// we just read the first <archive> line.
// prepare to read <class> lines.
line = 1;
}
sepi = 0;
}
} else {
sepi = 0;
}
}
// Takes a couple of lines, where line1 is an <archive> line and
// line 2 is a <class> line. Parses the line to extract the archive
// name and dependent class name, and record them in the map...
void parse(String line1, String line2) {
String archive = line1.substring(0, line1.indexOf(" -> "));
int l2ArrowIndex = line2.indexOf(" -> ");
String className = line2.substring(2, l2ArrowIndex).replace(" ", "");
String depdescr = line2.substring(l2ArrowIndex + 4);
String depclass = depdescr.substring(0, depdescr.indexOf(" "));
deps.computeIfAbsent(archive, (k) -> new HashSet<>());
deps.get(archive).add(className);
if (VERBOSE) {
System.out.println(archive+": "+className+" depends on "+depclass);
}
}
}
/**
* The main method.
*
* Can be run in two modes:
* <ul>
* <li>From jtreg: expects 1 argument in the form {@code --test:<test-nb>}</li>
* <li>From command line: expected syntax is {@code -e <pattern> -v jar [jars..]}</li>
* </ul>
* <p>When called from the command line this method will call jdeps recursively
* to build a closure of the dependencies on {@code <pattern>} and print a summary.
* <p>When called from jtreg - it will call jdeps either once only or
* recursively depending on the pattern.
* @param args either {@code --test:<test-nb>} or {@code -e <pattern> -v jar [jars..]}.
*/
public static void main(String[] args) {
runWithLocale(Locale.ENGLISH, TestCaseData.valueOf(args)::test);
}
private static void runWithLocale(Locale loc, Runnable run) {
final Locale defaultLocale = Locale.getDefault();
Locale.setDefault(loc);
try {
run.run();
} finally {
Locale.setDefault(defaultLocale);
}
}
public static void test(String[] args, Map<String, Set<String>> expected,
String expectedText, boolean closure) {
try {
doTest(args, expected, expectedText, closure);
} catch (Throwable t) {
try {
printDiagnostic(args, expectedText, t, closure);
} catch(Throwable tt) {
throw t;
}
throw t;
}
}
static class TextFormatException extends RuntimeException {
final String expected;
final String actual;
TextFormatException(String message, String expected, String actual) {
super(message);
this.expected = expected;
this.actual = actual;
}
}
public static void printDiagnostic(String[] args, String expectedText,
Throwable t, boolean closure) {
if (expectedText != null || t instanceof TextFormatException) {
System.err.println("===== TEST FAILED =======");
System.err.println("command: " + Stream.of(args)
.reduce("jdeps", (s1,s2) -> s1.concat(" ").concat(s2)));
System.err.println("===== Expected Output =======");
System.err.append(expectedText);
System.err.println("===== Command Output =======");
if (t instanceof TextFormatException) {
System.err.print(((TextFormatException)t).actual);
} else {
com.sun.tools.jdeps.Main.run(args, new PrintWriter(System.err));
if (closure) System.err.println("... (closure not available) ...");
}
System.err.println("=============================");
}
}
public static void doTest(String[] args, Map<String, Set<String>> expected,
String expectedText, boolean closure) {
if (args.length < 3 || !"-e".equals(args[0]) || !"-v".equals(args[2])) {
System.err.println("Syntax: -e <classname> -v [list of jars or directories]");
return;
}
Map<String, Map<String, Set<String>>> alldeps = new HashMap<>();
String depName = args[1];
List<String> search = new ArrayList<>();
search.add(depName);
Set<String> searched = new LinkedHashSet<>();
StringBuilder text = new StringBuilder();
while(!search.isEmpty()) {
args[1] = search.remove(0);
if (VERBOSE) {
System.out.println("Looking for " + args[1]);
}
searched.add(args[1]);
Map<String, Set<String>> deps =
alldeps.computeIfAbsent(args[1], (k) -> new HashMap<>());
OutputStreamParser parser = new OutputStreamParser(deps);
PrintWriter writer = new PrintWriter(parser);
com.sun.tools.jdeps.Main.run(args, writer);
if (VERBOSE) {
System.out.println("Found: " + deps.values().stream()
.flatMap(s -> s.stream()).collect(Collectors.toSet()));
}
if (expectedText != null) {
text.append(parser.text.toString());
}
search.addAll(deps.values().stream()
.flatMap(s -> s.stream())
.filter(k -> !searched.contains(k))
.collect(Collectors.toSet()));
if (!closure) break;
}
// Print summary...
final Set<String> classes = alldeps.values().stream()
.flatMap((m) -> m.values().stream())
.flatMap(s -> s.stream()).collect(Collectors.toSet());
Map<String, Set<String>> result = new HashMap<>();
for (String c : classes) {
Set<String> archives = new HashSet<>();
Set<String> dependencies = new HashSet<>();
for (String d : alldeps.keySet()) {
Map<String, Set<String>> m = alldeps.get(d);
for (String a : m.keySet()) {
Set<String> s = m.get(a);
if (s.contains(c)) {
archives.add(a);
dependencies.add(d);
}
}
}
result.put(c, dependencies);
System.out.println(c + " " + archives + " depends on " + dependencies);
}
// If we're in jtreg, then check result (expectedText != null)
if (expectedText != null && COMPARE_TEXT) {
//text.append(String.format("%n"));
if (text.toString().equals(expectedText)) {
System.out.println("SUCCESS - got expected text");
} else {
throw new TextFormatException("jdeps output is not as expected",
expectedText, text.toString());
}
}
if (expected != null) {
if (expected.equals(result)) {
System.out.println("SUCCESS - found expected dependencies");
} else if (expectedText == null) {
throw new RuntimeException("Bad dependencies: Expected " + expected
+ " but found " + result);
} else {
throw new TextFormatException("Bad dependencies: Expected "
+ expected
+ " but found " + result,
expectedText, text.toString());
}
}
}
}