JdepsDependencyClosure.java revision 2942:08092deced3f
1/*
2 * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 */
23import java.io.IOException;
24import java.io.OutputStream;
25import java.io.PrintWriter;
26import java.nio.file.Paths;
27import java.util.ArrayList;
28import java.util.Arrays;
29import java.util.HashMap;
30import java.util.HashSet;
31import java.util.LinkedHashSet;
32import java.util.List;
33import java.util.Locale;
34import java.util.Map;
35import java.util.Set;
36import java.util.function.Supplier;
37import java.util.stream.Collectors;
38import java.util.stream.Stream;
39
40/**
41 * @test
42 * @bug 8080608
43 * @summary Test that jdeps verbose output has a summary line when dependencies
44 *          are found within the same archive. For each testcase, compare the
45 *          result obtained from jdeps with the expected result.
46 * @modules jdk.jdeps/com.sun.tools.jdeps
47 * @build use.indirect.DontUseUnsafe2
48 * @build use.indirect.UseUnsafeIndirectly
49 * @build use.indirect2.DontUseUnsafe3
50 * @build use.indirect2.UseUnsafeIndirectly2
51 * @build use.unsafe.DontUseUnsafe
52 * @build use.unsafe.UseClassWithUnsafe
53 * @build use.unsafe.UseUnsafeClass
54 * @build use.unsafe.UseUnsafeClass2
55 * @run main JdepsDependencyClosure --test:0
56 * @run main JdepsDependencyClosure --test:1
57 * @run main JdepsDependencyClosure --test:2
58 * @run main JdepsDependencyClosure --test:3
59 */
60public class JdepsDependencyClosure {
61
62    static boolean VERBOSE = false;
63    static boolean COMPARE_TEXT = true;
64
65    static final String JDEPS_SUMMARY_TEXT_FORMAT = "%s -> %s%n";
66    static final String JDEPS_VERBOSE_TEXT_FORMAT = "   %-50s -> %-50s %s%n";
67
68    /**
69     * Helper class used to store arguments to pass to
70     * {@code JdepsDependencyClosure.test} as well as expected
71     * results.
72     */
73    static class TestCaseData {
74        final Map<String, Set<String>> expectedDependencies;
75        final String expectedText;
76        final String[] args;
77        final boolean closure;
78
79        TestCaseData(Map<String, Set<String>> expectedDependencies,
80                        String expectedText,
81                        boolean closure,
82                        String[] args) {
83            this.expectedDependencies = expectedDependencies;
84            this.expectedText = expectedText;
85            this.closure = closure;
86            this.args = args;
87        }
88
89        public void test() {
90            if (expectedDependencies != null) {
91                String format = closure
92                        ? "Running (closure): jdeps %s %s %s %s"
93                        : "Running: jdeps %s %s %s %s";
94                System.out.println(String.format(format, (Object[])args));
95            }
96            JdepsDependencyClosure.test(args, expectedDependencies, expectedText, closure);
97        }
98
99        /**
100         * Make a new test case data to invoke jdeps and test its output.
101         * @param pattern The pattern that will passed through to jdeps -e
102         *                This is expected to match only one class.
103         * @param arcPath The archive to analyze. A jar or a class directory.
104         * @param classes For each reported archive dependency couple, the
105         *                expected list of classes in the source that will
106         *                be reported as having a dependency on the class
107         *                in the target that matches the given pattern.
108         * @param dependencies For each archive dependency couple, a singleton list
109         *                containing the name of the class in the target that
110         *                matches the pattern. It is expected that the pattern
111         *                will match only one class in the target.
112         *                If the pattern matches several classes the
113         *                expected text may no longer match the jdeps output.
114         * @param archives A list of archive dependency couple in the form
115         *               {{sourceName1, sourcePath1, targetDescription1, targetPath1}
116         *                {sourceName2, sourcePath2, targetDescription2, targetPath2}
117         *                ... }
118         *               For a JDK module - e.g. java.base, the targetDescription
119         *               is usually something like "JDK internal API (java.base)"
120         *               and the targetPath is usually the module name "java.base".
121         * @param closure Whether jdeps should be recursively invoked to build
122         *                the closure.
123         * @return An instance of TestCaseData containing all the information
124         *         needed to perform the jdeps invokation and test its output.
125         */
126        public static TestCaseData make(String pattern, String arcPath, String[][] classes,
127                String[][] dependencies, String[][] archives, boolean closure) {
128            final String[] args = new String[] {
129                "-e", pattern, "-v", arcPath
130            };
131            Map<String, Set<String>> expected = new HashMap<>();
132            String expectedText = "";
133            for (int i=0; i<classes.length; i++) {
134                final int index = i;
135                expectedText += Stream.of(classes[i])
136                    .map((cn) -> String.format(JDEPS_VERBOSE_TEXT_FORMAT, cn,
137                            dependencies[index][0], archives[index][2]))
138                    .reduce(String.format(JDEPS_SUMMARY_TEXT_FORMAT, archives[i][0],
139                            archives[index][3]), (s1,s2) -> s1.concat(s2));
140                for (String cn : classes[index]) {
141                    expected.putIfAbsent(cn, new HashSet<>());
142                    expected.get(cn).add(dependencies[index][0]);
143                }
144            }
145            return new TestCaseData(expected, expectedText, closure, args);
146        }
147
148        public static TestCaseData valueOf(String[] args) {
149            if (args.length == 1 && args[0].startsWith("--test:")) {
150                // invoked from jtreg. build test case data for selected test.
151                int index = Integer.parseInt(args[0].substring("--test:".length()));
152                if (index >= dataSuppliers.size()) {
153                    throw new RuntimeException("No such test case: " + index
154                            + " - available testcases are [0.."
155                            + (dataSuppliers.size()-1) + "]");
156                }
157                return dataSuppliers.get(index).get();
158            } else {
159                // invoked in standalone. just take the given argument
160                // and perform no validation on the output (except that it
161                // must start with a summary line)
162                return new TestCaseData(null, null, true, args);
163            }
164        }
165
166    }
167
168    static TestCaseData makeTestCaseOne() {
169        final String arcPath = System.getProperty("test.classes", "build/classes");
170        final String arcName = Paths.get(arcPath).getFileName().toString();
171        final String[][] classes = new String[][] {
172            {"use.indirect2.UseUnsafeIndirectly2", "use.unsafe.UseClassWithUnsafe"},
173        };
174        final String[][] dependencies = new String[][] {
175            {"use.unsafe.UseUnsafeClass"},
176        };
177        final String[][] archives = new String[][] {
178            {arcName, arcPath, arcName, arcPath},
179        };
180        return TestCaseData.make("use.unsafe.UseUnsafeClass", arcPath, classes,
181                dependencies, archives, false);
182    }
183
184    static TestCaseData makeTestCaseTwo() {
185        String arcPath = System.getProperty("test.classes", "build/classes");
186        String arcName = Paths.get(arcPath).getFileName().toString();
187        String[][] classes = new String[][] {
188            {"use.unsafe.UseUnsafeClass", "use.unsafe.UseUnsafeClass2"}
189        };
190        String[][] dependencies = new String[][] {
191            {"sun.misc.Unsafe"}
192        };
193        String[][] archive = new String[][] {
194            {arcName, arcPath, "JDK internal API (java.base)", "java.base"},
195        };
196        return TestCaseData.make("sun.misc.Unsafe", arcPath, classes,
197                dependencies, archive, false);
198    }
199
200    static TestCaseData makeTestCaseThree() {
201        final String arcPath = System.getProperty("test.classes", "build/classes");
202        final String arcName = Paths.get(arcPath).getFileName().toString();
203        final String[][] classes = new String[][] {
204            {"use.indirect2.UseUnsafeIndirectly2", "use.unsafe.UseClassWithUnsafe"},
205            {"use.indirect.UseUnsafeIndirectly"}
206        };
207        final String[][] dependencies = new String[][] {
208            {"use.unsafe.UseUnsafeClass"},
209            {"use.unsafe.UseClassWithUnsafe"}
210        };
211        final String[][] archives = new String[][] {
212            {arcName, arcPath, arcName, arcPath},
213            {arcName, arcPath, arcName, arcPath}
214        };
215        return TestCaseData.make("use.unsafe.UseUnsafeClass", arcPath, classes,
216                dependencies, archives, true);
217    }
218
219
220    static TestCaseData makeTestCaseFour() {
221        final String arcPath = System.getProperty("test.classes", "build/classes");
222        final String arcName = Paths.get(arcPath).getFileName().toString();
223        final String[][] classes = new String[][] {
224            {"use.unsafe.UseUnsafeClass", "use.unsafe.UseUnsafeClass2"},
225            {"use.indirect2.UseUnsafeIndirectly2", "use.unsafe.UseClassWithUnsafe"},
226            {"use.indirect.UseUnsafeIndirectly"}
227        };
228        final String[][] dependencies = new String[][] {
229            {"sun.misc.Unsafe"},
230            {"use.unsafe.UseUnsafeClass"},
231            {"use.unsafe.UseClassWithUnsafe"}
232        };
233        final String[][] archives = new String[][] {
234            {arcName, arcPath, "JDK internal API (java.base)", "java.base"},
235            {arcName, arcPath, arcName, arcPath},
236            {arcName, arcPath, arcName, arcPath}
237        };
238        return TestCaseData.make("sun.misc.Unsafe", arcPath, classes, dependencies,
239                archives, true);
240    }
241
242    static final List<Supplier<TestCaseData>> dataSuppliers = Arrays.asList(
243        JdepsDependencyClosure::makeTestCaseOne,
244        JdepsDependencyClosure::makeTestCaseTwo,
245        JdepsDependencyClosure::makeTestCaseThree,
246        JdepsDependencyClosure::makeTestCaseFour
247    );
248
249
250
251    /**
252     * The OutputStreamParser is used to parse the format of jdeps.
253     * It is thus dependent on that format.
254     */
255    static class OutputStreamParser extends OutputStream {
256        // OutputStreamParser will populate this map:
257        //
258        // For each archive, a list of class in where dependencies where
259        //     found...
260        final Map<String, Set<String>> deps;
261        final StringBuilder text = new StringBuilder();
262
263        StringBuilder[] lines = { new StringBuilder(), new StringBuilder() };
264        int line = 0;
265        int sepi = 0;
266        char[] sep;
267
268        public OutputStreamParser(Map<String, Set<String>> deps) {
269            this.deps = deps;
270            this.sep = System.getProperty("line.separator").toCharArray();
271        }
272
273        @Override
274        public void write(int b) throws IOException {
275            lines[line].append((char)b);
276            if (b == sep[sepi]) {
277                if (++sepi == sep.length) {
278                    text.append(lines[line]);
279                    if (lines[0].toString().startsWith("  ")) {
280                        throw new RuntimeException("Bad formatting: "
281                                + "summary line missing for\n"+lines[0]);
282                    }
283                    // Usually the output looks like that:
284                    // <archive-1> -> java.base
285                    //   <class-1>      -> <dependency> <dependency description>
286                    //   <class-2>      -> <dependency> <dependency description>
287                    //   ...
288                    // <archive-2> -> java.base
289                    //   <class-3>      -> <dependency> <dependency description>
290                    //   <class-4>      -> <dependency> <dependency description>
291                    //   ...
292                    //
293                    // We want to keep the <archive> line in lines[0]
294                    // and have the ith <class-i> line in lines[1]
295                    if (line == 1) {
296                        // we have either a <class> line or an <archive> line.
297                        String line1 = lines[0].toString();
298                        String line2 = lines[1].toString();
299                        if (line2.startsWith("  ")) {
300                            // we have a class line, record it.
301                            parse(line1, line2);
302                            // prepare for next <class> line.
303                            lines[1] = new StringBuilder();
304                        } else {
305                            // We have an archive line: We are switching to the next archive.
306                            // put the new <archive> line in lines[0], and prepare
307                            // for reading the next <class> line
308                            lines[0] = lines[1];
309                            lines[1] = new StringBuilder();
310                         }
311                    } else {
312                        // we just read the first <archive> line.
313                        // prepare to read <class> lines.
314                        line = 1;
315                    }
316                    sepi = 0;
317                }
318            } else {
319                sepi = 0;
320            }
321        }
322
323        // Takes a couple of lines, where line1 is an <archive> line and
324        // line 2 is a <class> line. Parses the line to extract the archive
325        // name and dependent class name, and record them in the map...
326        void parse(String line1, String line2) {
327            String archive = line1.substring(0, line1.indexOf(" -> "));
328            int l2ArrowIndex = line2.indexOf(" -> ");
329            String className = line2.substring(2, l2ArrowIndex).replace(" ", "");
330            String depdescr = line2.substring(l2ArrowIndex + 4);
331            String depclass = depdescr.substring(0, depdescr.indexOf(" "));
332            deps.computeIfAbsent(archive, (k) -> new HashSet<>());
333            deps.get(archive).add(className);
334            if (VERBOSE) {
335                System.out.println(archive+": "+className+" depends on "+depclass);
336            }
337        }
338
339    }
340
341    /**
342     * The main method.
343     *
344     * Can be run in two modes:
345     * <ul>
346     * <li>From jtreg: expects 1 argument in the form {@code --test:<test-nb>}</li>
347     * <li>From command line: expected syntax is {@code -e <pattern> -v jar [jars..]}</li>
348     * </ul>
349     * <p>When called from the command line this method will call jdeps recursively
350     * to build a closure of the dependencies on {@code <pattern>} and print a summary.
351     * <p>When called from jtreg - it will call jdeps either once only or
352     * recursively depending on the pattern.
353     * @param args either {@code --test:<test-nb>} or {@code -e <pattern> -v jar [jars..]}.
354     */
355    public static void main(String[] args) {
356        runWithLocale(Locale.ENGLISH, TestCaseData.valueOf(args)::test);
357    }
358
359    private static void runWithLocale(Locale loc, Runnable run) {
360        final Locale defaultLocale = Locale.getDefault();
361        Locale.setDefault(loc);
362        try {
363            run.run();
364        } finally {
365            Locale.setDefault(defaultLocale);
366        }
367    }
368
369
370    public static void test(String[] args, Map<String, Set<String>> expected,
371            String expectedText, boolean closure) {
372        try {
373            doTest(args, expected, expectedText, closure);
374        } catch (Throwable t) {
375            try {
376                printDiagnostic(args, expectedText, t, closure);
377            } catch(Throwable tt) {
378                throw t;
379            }
380            throw t;
381        }
382    }
383
384    static class TextFormatException extends RuntimeException {
385        final String expected;
386        final String actual;
387        TextFormatException(String message, String expected, String actual) {
388            super(message);
389            this.expected = expected;
390            this.actual = actual;
391        }
392    }
393
394    public static void printDiagnostic(String[] args, String expectedText,
395            Throwable t, boolean closure) {
396        if (expectedText != null || t instanceof TextFormatException) {
397            System.err.println("=====   TEST FAILED   =======");
398            System.err.println("command: " + Stream.of(args)
399                    .reduce("jdeps", (s1,s2) -> s1.concat(" ").concat(s2)));
400            System.err.println("===== Expected Output =======");
401            System.err.append(expectedText);
402            System.err.println("===== Command  Output =======");
403            if (t instanceof TextFormatException) {
404                System.err.print(((TextFormatException)t).actual);
405            } else {
406                com.sun.tools.jdeps.Main.run(args, new PrintWriter(System.err));
407                if (closure) System.err.println("... (closure not available) ...");
408            }
409            System.err.println("=============================");
410        }
411    }
412
413    public static void doTest(String[] args, Map<String, Set<String>> expected,
414            String expectedText, boolean closure) {
415        if (args.length < 3 || !"-e".equals(args[0]) || !"-v".equals(args[2])) {
416            System.err.println("Syntax: -e <classname> -v [list of jars or directories]");
417            return;
418        }
419        Map<String, Map<String, Set<String>>> alldeps = new HashMap<>();
420        String depName = args[1];
421        List<String> search = new ArrayList<>();
422        search.add(depName);
423        Set<String> searched = new LinkedHashSet<>();
424        StringBuilder text = new StringBuilder();
425        while(!search.isEmpty()) {
426            args[1] = search.remove(0);
427            if (VERBOSE) {
428                System.out.println("Looking for " + args[1]);
429            }
430            searched.add(args[1]);
431            Map<String, Set<String>> deps =
432                    alldeps.computeIfAbsent(args[1], (k) -> new HashMap<>());
433            OutputStreamParser parser = new OutputStreamParser(deps);
434            PrintWriter writer = new PrintWriter(parser);
435            com.sun.tools.jdeps.Main.run(args, writer);
436            if (VERBOSE) {
437                System.out.println("Found: " + deps.values().stream()
438                        .flatMap(s -> s.stream()).collect(Collectors.toSet()));
439            }
440            if (expectedText != null) {
441                text.append(parser.text.toString());
442            }
443            search.addAll(deps.values().stream()
444                    .flatMap(s -> s.stream())
445                    .filter(k -> !searched.contains(k))
446                    .collect(Collectors.toSet()));
447            if (!closure) break;
448        }
449
450        // Print summary...
451        final Set<String> classes = alldeps.values().stream()
452                .flatMap((m) -> m.values().stream())
453                .flatMap(s -> s.stream()).collect(Collectors.toSet());
454        Map<String, Set<String>> result = new HashMap<>();
455        for (String c : classes) {
456            Set<String> archives = new HashSet<>();
457            Set<String> dependencies = new HashSet<>();
458            for (String d : alldeps.keySet()) {
459                Map<String, Set<String>> m = alldeps.get(d);
460                for (String a : m.keySet()) {
461                    Set<String> s = m.get(a);
462                    if (s.contains(c)) {
463                        archives.add(a);
464                        dependencies.add(d);
465                    }
466                }
467            }
468            result.put(c, dependencies);
469            System.out.println(c + " " + archives + " depends on " + dependencies);
470        }
471
472        // If we're in jtreg, then check result (expectedText != null)
473        if (expectedText != null && COMPARE_TEXT) {
474            //text.append(String.format("%n"));
475            if (text.toString().equals(expectedText)) {
476                System.out.println("SUCCESS - got expected text");
477            } else {
478                throw new TextFormatException("jdeps output is not as expected",
479                                expectedText, text.toString());
480            }
481        }
482        if (expected != null) {
483            if (expected.equals(result)) {
484                System.out.println("SUCCESS - found expected dependencies");
485            } else if (expectedText == null) {
486                throw new RuntimeException("Bad dependencies: Expected " + expected
487                        + " but found " + result);
488            } else {
489                throw new TextFormatException("Bad dependencies: Expected "
490                        + expected
491                        + " but found " + result,
492                        expectedText, text.toString());
493            }
494        }
495    }
496}
497