226 lines
7.9 KiB
Java

/**
* 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
* @bug 8157068
* @summary Patch java.base and user module with Hashes attribute tied with
* other module.
* @library /lib/testlibrary
* @modules jdk.compiler
* @build CompilerUtils
* @run testng PatchSystemModules
*/
import java.io.File;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Stream;
import jdk.testlibrary.FileUtils;
import jdk.testlibrary.JDKToolFinder;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.Test;
import static jdk.testlibrary.ProcessTools.executeCommand;
import static org.testng.Assert.*;
public class PatchSystemModules {
private static final String JAVA_HOME = System.getProperty("java.home");
private static final Path TEST_SRC = Paths.get(System.getProperty("test.src"));
private static final Path PATCH_SRC_DIR = TEST_SRC.resolve("src1");
private static final Path JMODS = Paths.get(JAVA_HOME, "jmods");
private static final Path MODS_DIR = Paths.get("mods");
private static final Path JARS_DIR = Paths.get("jars");
private static final Path PATCH_DIR = Paths.get("patches");
private static final Path IMAGE = Paths.get("image");
private static final String JAVA_BASE = "java.base";
private final String[] modules = new String[] { "m1", "m2" };
@BeforeTest
private void setup() throws Throwable {
Path src = TEST_SRC.resolve("src");
for (String name : modules) {
assertTrue(CompilerUtils.compile(src.resolve(name),
MODS_DIR,
"-modulesourcepath", src.toString()));
}
// compile patched source
assertTrue(CompilerUtils.compile(PATCH_SRC_DIR.resolve(JAVA_BASE),
PATCH_DIR.resolve(JAVA_BASE),
"-Xmodule:java.base"));
assertTrue(CompilerUtils.compile(PATCH_SRC_DIR.resolve("m2"),
PATCH_DIR.resolve("m2")));
// create an image with only m1 and m2
if (Files.exists(JMODS)) {
// create an image with m1,m2
createImage();
}
}
@Test
public void test() throws Throwable {
Path patchedJavaBase = PATCH_DIR.resolve(JAVA_BASE);
Path patchedM2 = PATCH_DIR.resolve("m2");
Path home = Paths.get(JAVA_HOME);
runTest(home,
"-mp", MODS_DIR.toString(),
"-m", "m1/p1.Main", "1");
runTest(home,
"-Xpatch:java.base=" + patchedJavaBase.toString(),
"-mp", MODS_DIR.toString(),
"-m", "m1/p1.Main", "1");
runTest(home,
"-Xpatch:m2=" + patchedM2.toString(),
"-mp", MODS_DIR.toString(),
"-m", "m1/p1.Main", "2");
}
@Test
public void testImage() throws Throwable {
if (Files.notExists(JMODS))
return;
Path patchedJavaBase = PATCH_DIR.resolve(JAVA_BASE);
Path patchedM2 = PATCH_DIR.resolve("m2");
runTest(IMAGE,
"-m", "m1/p1.Main", "1");
runTest(IMAGE,
"-Xpatch:java.base=" + patchedJavaBase.toString(),
"-m", "m1/p1.Main", "1");
runTest(IMAGE,
"-Xpatch:m2=" + patchedM2.toString(),
"-m", "m1/p1.Main", "2");
}
@Test
public void upgradeTiedModule() throws Throwable {
if (Files.notExists(JMODS))
return;
Path m1 = MODS_DIR.resolve("m1.jar");
// create another m1.jar
jar("--create",
"--file=" + m1.toString(),
"-C", MODS_DIR.resolve("m1").toString(), ".");
// Fail to upgrade m1.jar with mismatched hash
runTestWithExitCode(getJava(IMAGE),
"-upgrademodulepath", m1.toString(),
"-m", "m1/p1.Main");
runTestWithExitCode(getJava(IMAGE),
"-Xpatch:java.base=" + PATCH_DIR.resolve(JAVA_BASE).toString(),
"-upgrademodulepath", m1.toString(),
"-m", "m1/p1.Main", "1");
}
private void runTestWithExitCode(String... options) throws Throwable {
assertTrue(executeCommand(options)
.outputTo(System.out)
.errorTo(System.out)
.shouldContain("differs to expected hash")
.getExitValue() != 0);
}
private void runTest(Path image, String... opts) throws Throwable {
String[] options =
Stream.concat(Stream.of(getJava(image)),
Stream.of(opts))
.toArray(String[]::new);
ProcessBuilder pb = new ProcessBuilder(options);
int exitValue = executeCommand(pb)
.outputTo(System.out)
.errorTo(System.out)
.getExitValue();
assertTrue(exitValue == 0);
}
static void createImage() throws Throwable {
FileUtils.deleteFileTreeUnchecked(JARS_DIR);
FileUtils.deleteFileTreeUnchecked(IMAGE);
Files.createDirectories(JARS_DIR);
Path m1 = JARS_DIR.resolve("m1.jar");
Path m2 = JARS_DIR.resolve("m2.jar");
// hash m1 in m2's Hashes attribute
jar("--create",
"--file=" + m1.toString(),
"-C", MODS_DIR.resolve("m1").toString(), ".");
jar("--create",
"--file=" + m2.toString(),
"--modulepath", JARS_DIR.toString(),
"--hash-modules", "m1",
"-C", MODS_DIR.resolve("m2").toString(), ".");
String mpath = JARS_DIR.toString() + File.pathSeparator + JMODS.toString();
execTool("jlink", "--modulepath", mpath,
"--addmods", "m1",
"--output", IMAGE.toString());
}
static void jar(String... args) throws Throwable {
execTool("jar", args);
}
static void execTool(String tool, String... args) throws Throwable {
String path = JDKToolFinder.getJDKTool(tool);
List<String> commands = new ArrayList<>();
commands.add(path);
Stream.of(args).forEach(commands::add);
ProcessBuilder pb = new ProcessBuilder(commands);
int exitValue = executeCommand(pb)
.outputTo(System.out)
.errorTo(System.out)
.shouldNotContain("no module is recorded in hash")
.getExitValue();
assertTrue(exitValue == 0);
}
static String getJava(Path image) {
boolean isWindows = System.getProperty("os.name").toLowerCase().startsWith("win");
Path java = image.resolve("bin").resolve(isWindows ? "java.exe" : "java");
if (Files.notExists(java))
throw new RuntimeException(java + " not found");
return java.toAbsolutePath().toString();
}
}