langtools/test/tools/jdeps/modules/ModuleTest.java
author jjg
Wed, 10 Aug 2016 15:47:46 -0700
changeset 40308 274367a99f98
parent 38530 8e89d567748c
child 42407 f3702cff2933
permissions -rw-r--r--
8136930: Simplify use of module-system options by custom launchers 8160489: Multiple -Xpatch lines ignored by javac 8156998: javac should support new option -XinheritRuntimeEnvironment Reviewed-by: jlahoda, ksrini

/*
 * Copyright (c) 2016, 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
 * @summary Tests jdeps -m and --module-path options on named modules and unnamed modules
 * @library ../lib
 * @build CompilerUtils JdepsUtil
 * @modules jdk.jdeps/com.sun.tools.jdeps
 * @run testng ModuleTest
 */

import java.io.File;
import java.io.IOException;

import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.*;
import java.util.stream.Collectors;


import com.sun.tools.jdeps.DepsAnalyzer;
import com.sun.tools.jdeps.Graph;
import org.testng.annotations.DataProvider;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.Test;

import static org.testng.Assert.assertTrue;

public class ModuleTest {
    private static final String TEST_SRC = System.getProperty("test.src");
    private static final String TEST_CLASSES = System.getProperty("test.classes");

    private static final Path SRC_DIR = Paths.get(TEST_SRC, "src");
    private static final Path MODS_DIR = Paths.get("mods");
    private static final Path UNNAMED_DIR = Paths.get("unnamed");

    // the names of the modules in this test
    private static final String UNSUPPORTED = "unsupported";
    private static String[] modules = new String[] {"m1", "m2", "m3", "m4", UNSUPPORTED};
    /**
     * Compiles all modules used by the test
     */
    @BeforeTest
    public void compileAll() throws Exception {
        CompilerUtils.cleanDir(MODS_DIR);
        CompilerUtils.cleanDir(UNNAMED_DIR);

        assertTrue(CompilerUtils.compileModule(SRC_DIR, MODS_DIR, UNSUPPORTED,
                                               "--add-exports", "java.base/jdk.internal.perf=" + UNSUPPORTED));
        // m4 is not referenced
        Arrays.asList("m1", "m2", "m3", "m4")
              .forEach(mn -> assertTrue(CompilerUtils.compileModule(SRC_DIR, MODS_DIR, mn)));

        assertTrue(CompilerUtils.compile(SRC_DIR.resolve("m3"), UNNAMED_DIR, "-p", MODS_DIR.toString()));
        Files.delete(UNNAMED_DIR.resolve("module-info.class"));
    }

    @DataProvider(name = "modules")
    public Object[][] expected() {
        return new Object[][]{
                { "m3", new ModuleMetaData("m3").requiresPublic("java.sql")
                            .requiresPublic("m2")
                            .requires("java.logging")
                            .requiresPublic("m1")
                            .reference("p3", "java.lang", "java.base")
                            .reference("p3", "java.sql", "java.sql")
                            .reference("p3", "java.util.logging", "java.logging")
                            .reference("p3", "p1", "m1")
                            .reference("p3", "p2", "m2")
                            .qualified("p3", "p2.internal", "m2")
                },
                { "m2", new ModuleMetaData("m2").requiresPublic("m1")
                            .reference("p2", "java.lang", "java.base")
                            .reference("p2", "p1", "m1")
                            .reference("p2.internal", "java.lang", "java.base")
                            .reference("p2.internal", "java.io", "java.base")
                },
                { "m1", new ModuleMetaData("m1").requires("unsupported")
                            .reference("p1", "java.lang", "java.base")
                            .reference("p1.internal", "java.lang", "java.base")
                            .reference("p1.internal", "p1", "m1")
                            .reference("p1.internal", "q", "unsupported")
                },
                { "unsupported", new ModuleMetaData("unsupported")
                            .reference("q", "java.lang", "java.base")
                            .jdkInternal("q", "jdk.internal.perf", "java.base")
                },
        };
    }

    @Test(dataProvider = "modules")
    public void modularTest(String name, ModuleMetaData data) throws IOException {
        // jdeps --module-path mods -m <name>
        runTest(data, MODS_DIR.toString(), Set.of(name));

        // jdeps --module-path libs/m1.jar:.... -m <name>
        String mp = Arrays.stream(modules)
                .filter(mn -> !mn.equals(name))
                .map(mn -> MODS_DIR.resolve(mn).toString())
                .collect(Collectors.joining(File.pathSeparator));
        runTest(data, mp, Collections.emptySet(), MODS_DIR.resolve(name));
    }

    @DataProvider(name = "unnamed")
    public Object[][] unnamed() {
        return new Object[][]{
                { "unnamed", new ModuleMetaData("unnamed", false)
                            .depends("java.sql")
                            .depends("java.logging")
                            .depends("m1")
                            .depends("m2")
                            .reference("p3", "java.lang", "java.base")
                            .reference("p3", "java.sql", "java.sql")
                            .reference("p3", "java.util.logging", "java.logging")
                            .reference("p3", "p1", "m1")
                            .reference("p3", "p2", "m2")
                            .internal("p3", "p2.internal", "m2")
                },
        };
    }

    @Test(dataProvider = "unnamed")
    public void unnamedTest(String name, ModuleMetaData data) throws IOException {
        runTest(data, MODS_DIR.toString(), Set.of("m1", "m2"), UNNAMED_DIR);
    }

    private void runTest(ModuleMetaData data, String modulepath,
                         Set<String> roots, Path... paths)
        throws IOException
    {
        // jdeps --module-path <modulepath> -m root paths
        String cmd = String.format("jdeps --module-path %s --add-modules %s %s%n",
            MODS_DIR, roots.stream().collect(Collectors.joining(",")), paths);

        try (JdepsUtil.Command jdeps = JdepsUtil.newCommand(cmd)) {
            jdeps.appModulePath(modulepath)
                .addmods(roots);
            Arrays.stream(paths).forEach(jdeps::addRoot);

            // run the analyzer
            DepsAnalyzer analyzer = jdeps.getDepsAnalyzer();
            assertTrue(analyzer.run());

            // analyze result
            Graph<DepsAnalyzer.Node> g1 = analyzer.moduleGraph();
            g1.nodes().stream()
                .filter(u -> u.name.equals(data.moduleName))
                .forEach(u -> data.checkRequires(u.name, g1.adjacentNodes(u)));

            Graph<DepsAnalyzer.Node> g2 = analyzer.dependenceGraph();
            g2.nodes().stream()
                .filter(u -> u.name.equals(data.moduleName))
                .forEach(u -> data.checkDependences(u.name, g2.adjacentNodes(u)));

            jdeps.dumpOutput(System.err);
        }
    }
}