2016-11-16 15:11:56 +00:00
|
|
|
/*
|
2018-02-19 19:02:43 +00:00
|
|
|
* Copyright (c) 2016, 2018, Oracle and/or its affiliates. All rights reserved.
|
2016-11-16 15:11:56 +00:00
|
|
|
* 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 to verify jimage 'extract' action
|
|
|
|
* @library /test/lib
|
|
|
|
* @modules jdk.jlink/jdk.tools.jimage
|
|
|
|
* @build jdk.test.lib.Asserts
|
2018-10-15 18:36:20 +00:00
|
|
|
* @run main/othervm/timeout=300 JImageExtractTest
|
2016-11-16 15:11:56 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
import java.io.IOException;
|
2019-05-31 02:57:06 +00:00
|
|
|
import java.io.UncheckedIOException;
|
2016-11-16 15:11:56 +00:00
|
|
|
import java.nio.file.Files;
|
|
|
|
import java.nio.file.Path;
|
|
|
|
import java.nio.file.Paths;
|
2018-07-05 19:10:12 +00:00
|
|
|
import java.nio.file.attribute.*;
|
2016-11-16 15:11:56 +00:00
|
|
|
import java.util.Arrays;
|
|
|
|
import java.util.HashSet;
|
2018-07-05 19:10:12 +00:00
|
|
|
import java.util.List;
|
2016-11-16 15:11:56 +00:00
|
|
|
import java.util.Set;
|
|
|
|
import java.util.stream.Collectors;
|
2019-05-31 02:57:06 +00:00
|
|
|
import java.util.spi.ToolProvider;
|
2016-11-16 15:11:56 +00:00
|
|
|
|
|
|
|
import static jdk.test.lib.Asserts.assertEquals;
|
|
|
|
import static jdk.test.lib.Asserts.assertTrue;
|
|
|
|
|
|
|
|
public class JImageExtractTest extends JImageCliTest {
|
2019-05-31 02:57:06 +00:00
|
|
|
private static final ToolProvider JLINK_TOOL = ToolProvider.findFirst("jlink")
|
|
|
|
.orElseThrow(() ->
|
|
|
|
new RuntimeException("jlink tool not found")
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
private String smallBootImagePath;
|
|
|
|
|
|
|
|
public JImageExtractTest() {
|
|
|
|
try {
|
|
|
|
Path tmp = Files.createTempDirectory(Paths.get("."), getClass().getName());
|
|
|
|
tmp = tmp.toAbsolutePath();
|
|
|
|
tmp.toFile().deleteOnExit();
|
|
|
|
Path smalljre = tmp.resolve("smalljdk");
|
|
|
|
if (JLINK_TOOL.run(System.out, System.err,
|
|
|
|
"--add-modules", "java.base",
|
|
|
|
"--add-modules", "jdk.zipfs",
|
|
|
|
"--output", smalljre.toString()) != 0) {
|
|
|
|
throw new RuntimeException("failed to create small boot image");
|
|
|
|
}
|
|
|
|
this.smallBootImagePath = smalljre.resolve("lib").resolve("modules").toString();
|
|
|
|
} catch (IOException ioExp) {
|
|
|
|
throw new UncheckedIOException(ioExp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String getImagePath() {
|
|
|
|
return smallBootImagePath;
|
|
|
|
}
|
|
|
|
|
2016-11-16 15:11:56 +00:00
|
|
|
public void testExtract() throws IOException {
|
2018-07-04 12:19:21 +00:00
|
|
|
Set<Path> notJImageModules = Files.walk(Paths.get("."),1).collect(Collectors.toSet());
|
2016-11-16 15:11:56 +00:00
|
|
|
jimage("extract", getImagePath())
|
|
|
|
.assertSuccess()
|
|
|
|
.resultChecker(r -> {
|
|
|
|
assertTrue(r.output.isEmpty(), "Output is not expected");
|
|
|
|
});
|
2018-07-04 12:19:21 +00:00
|
|
|
verifyExplodedImage(Paths.get("."), notJImageModules);
|
2016-11-16 15:11:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractHelp() {
|
|
|
|
for (String opt : Arrays.asList("-h", "--help")) {
|
|
|
|
jimage("extract", "--help")
|
|
|
|
.assertSuccess()
|
|
|
|
.resultChecker(r -> {
|
|
|
|
// extract - descriptive text
|
|
|
|
assertMatches("\\s+extract\\s+-\\s+.*", r.output);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractToDir() throws IOException {
|
|
|
|
Path tmp = Files.createTempDirectory(Paths.get("."), getClass().getName());
|
2018-07-04 12:19:21 +00:00
|
|
|
Set<Path> notJImageModules = Files.walk(tmp,1).collect(Collectors.toSet());
|
2016-11-16 15:11:56 +00:00
|
|
|
jimage("extract", "--dir", tmp.toString(), getImagePath())
|
|
|
|
.assertSuccess()
|
|
|
|
.resultChecker(r -> {
|
|
|
|
assertTrue(r.output.isEmpty(), "Output is not expected");
|
|
|
|
});
|
2018-07-04 12:19:21 +00:00
|
|
|
verifyExplodedImage(tmp, notJImageModules);
|
2016-11-16 15:11:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractNoImageSpecified() {
|
|
|
|
jimage("extract", "")
|
|
|
|
.assertFailure()
|
|
|
|
.assertShowsError();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractNotAnImage() throws IOException {
|
|
|
|
Path tmp = Files.createTempFile(Paths.get("."), getClass().getName(), "not_an_image");
|
|
|
|
jimage("extract", tmp.toString())
|
|
|
|
.assertFailure()
|
|
|
|
.assertShowsError();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractNotExistingImage() throws IOException {
|
|
|
|
Path tmp = Paths.get(".", "not_existing_image");
|
|
|
|
Files.deleteIfExists(tmp);
|
|
|
|
jimage("extract", tmp.toString())
|
|
|
|
.assertFailure()
|
|
|
|
.assertShowsError();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractToUnspecifiedDir() {
|
|
|
|
jimage("extract", "--dir", "--", getImagePath())
|
|
|
|
.assertFailure()
|
|
|
|
.assertShowsError();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractToNotExistingDir() throws IOException {
|
|
|
|
Path tmp = Files.createTempDirectory(Paths.get("."), getClass().getName());
|
2018-07-04 12:19:21 +00:00
|
|
|
Set<Path> notJImageModules = Files.walk(tmp,1).collect(Collectors.toSet());
|
2016-11-16 15:11:56 +00:00
|
|
|
Files.delete(tmp);
|
|
|
|
jimage("extract", "--dir", tmp.toString(), getImagePath())
|
|
|
|
.assertSuccess()
|
|
|
|
.resultChecker(r -> {
|
|
|
|
assertTrue(r.output.isEmpty(), "Output is not expected");
|
|
|
|
});
|
2018-07-04 12:19:21 +00:00
|
|
|
verifyExplodedImage(tmp, notJImageModules);
|
2016-11-16 15:11:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractFromDir() {
|
|
|
|
Path imagePath = Paths.get(getImagePath());
|
|
|
|
Path imageDirPath = imagePath.subpath(0, imagePath.getNameCount() - 1);
|
|
|
|
jimage("extract", imageDirPath.toString())
|
|
|
|
.assertFailure()
|
|
|
|
.assertShowsError();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractToDirBySymlink() throws IOException {
|
|
|
|
Path tmp = Files.createTempDirectory(Paths.get("."), getClass().getName());
|
2018-01-19 00:15:16 +00:00
|
|
|
Path symlink;
|
2016-11-16 15:11:56 +00:00
|
|
|
try {
|
2018-01-19 00:15:16 +00:00
|
|
|
symlink = Files.createSymbolicLink(Paths.get(".", "symlink"), tmp);
|
|
|
|
} catch (IOException|UnsupportedOperationException e) {
|
2016-11-16 15:11:56 +00:00
|
|
|
// symlinks are not supported
|
|
|
|
// nothing to test
|
2018-01-19 00:15:16 +00:00
|
|
|
return;
|
2016-11-16 15:11:56 +00:00
|
|
|
}
|
2018-07-04 12:19:21 +00:00
|
|
|
Set<Path> notJImageModules = Files.walk(tmp,1).collect(Collectors.toSet());
|
2018-01-19 00:15:16 +00:00
|
|
|
jimage("extract", "--dir", symlink.toString(), getImagePath())
|
|
|
|
.assertSuccess()
|
|
|
|
.resultChecker(r -> {
|
|
|
|
assertTrue(r.output.isEmpty(), "Output is not expected");
|
|
|
|
});
|
2018-07-04 12:19:21 +00:00
|
|
|
verifyExplodedImage(tmp, notJImageModules);
|
2016-11-16 15:11:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractToReadOnlyDir() throws IOException {
|
2018-07-05 19:10:12 +00:00
|
|
|
Path filePath = Files.createTempDirectory(Paths.get("."), getClass().getName());
|
|
|
|
Set<String> supportedAttr = filePath.getFileSystem().supportedFileAttributeViews();
|
|
|
|
if (supportedAttr.contains("posix")) {
|
|
|
|
Files.setPosixFilePermissions(filePath, PosixFilePermissions.fromString("r-xr--r--"));
|
|
|
|
} else if (supportedAttr.contains("acl")) {
|
|
|
|
System.out.println("Entered into acl block");
|
|
|
|
UserPrincipal fileOwner = Files.getOwner(filePath);
|
|
|
|
AclFileAttributeView view = Files.getFileAttributeView(filePath, AclFileAttributeView.class);
|
|
|
|
AclEntry entry = AclEntry.newBuilder()
|
|
|
|
.setType(AclEntryType.DENY)
|
|
|
|
.setPrincipal(fileOwner)
|
|
|
|
.setPermissions(AclEntryPermission.WRITE_DATA)
|
|
|
|
.setFlags(AclEntryFlag.FILE_INHERIT, AclEntryFlag.DIRECTORY_INHERIT)
|
|
|
|
.build();
|
|
|
|
List<AclEntry> acl = view.getAcl();
|
|
|
|
acl.add(0, entry);
|
|
|
|
view.setAcl(acl);
|
2018-07-04 12:19:21 +00:00
|
|
|
}
|
2018-07-05 19:10:12 +00:00
|
|
|
jimage("extract", "--dir", filePath.toString(), getImagePath())
|
2016-11-16 15:11:56 +00:00
|
|
|
.assertFailure()
|
|
|
|
.assertShowsError();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractToNotEmptyDir() throws IOException {
|
|
|
|
Path tmp = Files.createTempDirectory(Paths.get("."), getClass().getName());
|
|
|
|
Files.createFile(Paths.get(tmp.toString(), ".not_empty"));
|
|
|
|
jimage("extract", "--dir", tmp.toString(), getImagePath())
|
2018-02-19 19:02:43 +00:00
|
|
|
.assertSuccess()
|
|
|
|
.resultChecker(r -> {
|
|
|
|
assertTrue(r.output.isEmpty(), "Output is not expected");
|
|
|
|
});
|
2016-11-16 15:11:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void testExtractToFile() throws IOException {
|
|
|
|
Path tmp = Files.createTempFile(Paths.get("."), getClass().getName(), "not_a_dir");
|
|
|
|
jimage("extract", "--dir", tmp.toString(), getImagePath())
|
|
|
|
.assertFailure()
|
|
|
|
.assertShowsError();
|
|
|
|
}
|
|
|
|
|
2018-07-04 12:19:21 +00:00
|
|
|
private void verifyExplodedImage(Path imagePath, Set<Path> notJImageModules) throws IOException {
|
2016-11-16 15:11:56 +00:00
|
|
|
Set<Path> allModules = Files.walk(imagePath, 1).collect(Collectors.toSet());
|
|
|
|
assertTrue(allModules.stream().anyMatch(p -> "java.base".equals(p.getFileName().toString())),
|
|
|
|
"Exploded image contains java.base module.");
|
|
|
|
Set<Path> badModules = allModules.stream()
|
|
|
|
.filter(p -> !Files.exists(p.resolve("module-info.class")))
|
|
|
|
.collect(Collectors.toSet());
|
2018-07-04 12:19:21 +00:00
|
|
|
// filter bad modules which are not part of jimage
|
|
|
|
badModules.removeAll(notJImageModules);
|
|
|
|
assertEquals(badModules, new HashSet<Path>() {{}},
|
2016-11-16 15:11:56 +00:00
|
|
|
"There are no exploded modules with missing 'module-info.class'");
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void main(String[] args) throws Throwable {
|
|
|
|
new JImageExtractTest().runTests();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|