Skip to content

Commit 43f7f47

Browse files
committedSep 19, 2022
8293499: Provide jmod --compress option
Reviewed-by: alanb, mchung, jpai, redestad
1 parent 26e08cf commit 43f7f47

File tree

4 files changed

+154
-7
lines changed

4 files changed

+154
-7
lines changed
 

‎src/jdk.jlink/share/classes/jdk/tools/jmod/JmodOutputStream.java

+4-3
Original file line numberDiff line numberDiff line change
@@ -55,16 +55,17 @@ class JmodOutputStream extends OutputStream implements AutoCloseable {
5555
* This method creates (or overrides, if exists) the JMOD file,
5656
* returning the output stream to write to the JMOD file.
5757
*/
58-
static JmodOutputStream newOutputStream(Path file, LocalDateTime date) throws IOException {
58+
static JmodOutputStream newOutputStream(Path file, LocalDateTime date, int compressLevel) throws IOException {
5959
OutputStream out = Files.newOutputStream(file);
6060
BufferedOutputStream bos = new BufferedOutputStream(out);
61-
return new JmodOutputStream(bos, date);
61+
return new JmodOutputStream(bos, date, compressLevel);
6262
}
6363

6464
private final ZipOutputStream zos;
6565
private final LocalDateTime date;
66-
private JmodOutputStream(OutputStream out, LocalDateTime date) {
66+
private JmodOutputStream(OutputStream out, LocalDateTime date, int compressLevel) {
6767
this.zos = new ZipOutputStream(out);
68+
this.zos.setLevel(compressLevel);
6869
this.date = date;
6970
try {
7071
JmodFile.writeMagicNumber(out);

‎src/jdk.jlink/share/classes/jdk/tools/jmod/JmodTask.java

+48-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -60,6 +60,7 @@
6060
import java.util.regex.PatternSyntaxException;
6161
import java.util.stream.Collectors;
6262
import java.util.stream.Stream;
63+
import java.util.zip.Deflater;
6364
import java.util.zip.ZipEntry;
6465
import java.util.zip.ZipException;
6566
import java.util.zip.ZipFile;
@@ -167,6 +168,7 @@ static class Options {
167168
List<PathMatcher> excludes;
168169
Path extractDir;
169170
LocalDateTime date;
171+
int compressLevel;
170172
}
171173

172174
// Valid --date range
@@ -438,7 +440,7 @@ private boolean create() throws IOException {
438440
Path target = options.jmodFile;
439441
Path tempTarget = jmodTempFilePath(target);
440442
try {
441-
try (JmodOutputStream jos = JmodOutputStream.newOutputStream(tempTarget, options.date)) {
443+
try (JmodOutputStream jos = JmodOutputStream.newOutputStream(tempTarget, options.date, options.compressLevel)) {
442444
jmod.write(jos);
443445
}
444446
Files.move(tempTarget, target);
@@ -1024,7 +1026,7 @@ private void updateJmodFile(Path target, Path tempTarget,
10241026
{
10251027

10261028
try (JmodFile jf = new JmodFile(target);
1027-
JmodOutputStream jos = JmodOutputStream.newOutputStream(tempTarget, options.date))
1029+
JmodOutputStream jos = JmodOutputStream.newOutputStream(tempTarget, options.date, options.compressLevel))
10281030
{
10291031
jf.stream().forEach(e -> {
10301032
try (InputStream in = jf.getInputStream(e.section(), e.name())) {
@@ -1179,6 +1181,33 @@ public LocalDateTime convert(String value) {
11791181
@Override public String valuePattern() { return "date"; }
11801182
}
11811183

1184+
static class CompLevelConverter implements ValueConverter<Integer> {
1185+
@Override
1186+
public Integer convert(String value) {
1187+
int idx = value.indexOf("-");
1188+
int lastIdx = value.lastIndexOf("-");
1189+
if (idx == -1 || idx != lastIdx) {
1190+
throw new CommandException("err.compress.incorrect", value);
1191+
}
1192+
if (!value.substring(0, idx).equals("zip")) {
1193+
throw new CommandException("err.compress.incorrect", value);
1194+
}
1195+
try {
1196+
int level = Integer.parseInt(value.substring(idx + 1));
1197+
if (level < 0 || level > 9) {
1198+
throw new CommandException("err.compress.incorrect", value);
1199+
}
1200+
return level;
1201+
} catch (NumberFormatException x) {
1202+
throw new CommandException("err.compress.incorrect", value);
1203+
}
1204+
}
1205+
1206+
@Override public Class<Integer> valueType() { return Integer.class; }
1207+
1208+
@Override public String valuePattern() { return "compress"; }
1209+
}
1210+
11821211
static class WarnIfResolvedReasonConverter
11831212
implements ValueConverter<ModuleResolution>
11841213
{
@@ -1419,6 +1448,11 @@ private void handleOptions(String[] args) {
14191448
.withRequiredArg()
14201449
.withValuesConvertedBy(new DateConverter());
14211450

1451+
OptionSpec<Integer> compress
1452+
= parser.accepts("compress", getMessage("main.opt.compress"))
1453+
.withRequiredArg()
1454+
.withValuesConvertedBy(new CompLevelConverter());
1455+
14221456
NonOptionArgumentSpec<String> nonOptions
14231457
= parser.nonOptions();
14241458

@@ -1488,6 +1522,17 @@ private void handleOptions(String[] args) {
14881522
throw new CommandException("err.modulepath.must.be.specified")
14891523
.showUsage(true);
14901524
}
1525+
if (opts.has(compress)) {
1526+
if (!options.mode.equals(Mode.CREATE)) {
1527+
throw new CommandException("err.compress.wrong.mode")
1528+
.showUsage(true);
1529+
}
1530+
options.compressLevel = getLastElement(opts.valuesOf(compress));
1531+
} else {
1532+
// Default to the default from zlib. Hard-coded here to avoid accidental
1533+
// compression level change if zlib ever changes the default.
1534+
options.compressLevel = 6;
1535+
}
14911536

14921537
if (options.mode.equals(Mode.HASH)) {
14931538
if (options.moduleFinder == null || options.modulesToHash == null)

‎src/jdk.jlink/share/classes/jdk/tools/jmod/resources/jmod.properties

+6
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,10 @@ main.opt.date=Date and time for the timestamps of entries, specified in ISO-8601
8282

8383
main.opt.cmdfile=Read options from the specified file
8484

85+
main.opt.compress=Compression to use when creating the JMOD archive.\
86+
\ Accepted values are: zip-[0-9], where zip-0 provides no compression, and zip-9\
87+
\ provides the best compression. Default is zip-6.
88+
8589
module.hashes.recorded=Hashes are recorded in module {0}
8690

8791
err.missing.mode=one of create, extract, list, describe, or hash must be specified
@@ -113,6 +117,8 @@ err.module.resolution.fail=Resolution failed: {0}
113117
err.no.moduleToHash=No hashes recorded: no module matching {0} found to record hashes
114118
err.invalid.date=--date {0} is not a valid ISO-8601 extended offset date-time with optional time-zone format: {1}
115119
err.date.out.of.range=--date {0} is out of the valid range 1980-01-01T00:00:02Z to 2099-12-31T23:59:59Z
120+
err.compress.incorrect=--compress value is invalid: {0}
121+
err.compress.wrong.mode=--compress is only accepted with create mode
116122
warn.invalid.arg=Invalid classname or pathname not exist: {0}
117123
warn.no.module.hashes=No hashes recorded: no module specified for hashing depends on {0}
118124
warn.ignore.entry=ignoring entry {0}, in section {1}

‎test/jdk/tools/jmod/JmodTest.java

+96-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -739,6 +739,101 @@ public void testTmpFileRemoved() throws IOException {
739739
});
740740
}
741741

742+
@Test
743+
public void testCompressionLevel() throws IOException {
744+
String cp = EXPLODED_DIR.resolve("foo").resolve("classes").toString();
745+
Path jmod = MODS_DIR.resolve("foo.jmod");
746+
FileUtils.deleteFileIfExistsWithRetry(jmod);
747+
748+
jmod("create",
749+
"--class-path", cp,
750+
"--compress", "zip-0",
751+
jmod.toString())
752+
.assertSuccess();
753+
754+
jmod("list",
755+
"--compress", "zip-0",
756+
jmod.toString())
757+
.assertFailure()
758+
.resultChecker(r -> {
759+
assertTrue(r.output.contains("--compress is only accepted with create mode"), "Error message printed");
760+
});
761+
762+
FileUtils.deleteFileIfExistsWithRetry(jmod);
763+
764+
jmod("create",
765+
"--class-path", cp,
766+
"--compress", "zip-9",
767+
jmod.toString())
768+
.assertSuccess();
769+
770+
FileUtils.deleteFileIfExistsWithRetry(jmod);
771+
772+
jmod("create",
773+
"--class-path", cp,
774+
"--compress", "zip--1",
775+
jmod.toString())
776+
.assertFailure()
777+
.resultChecker(r -> {
778+
assertTrue(r.output.contains("--compress value is invalid"), "Error message printed");
779+
});
780+
781+
FileUtils.deleteFileIfExistsWithRetry(jmod);
782+
783+
jmod("create",
784+
"--class-path", cp,
785+
"--compress", "zip-1-something",
786+
jmod.toString())
787+
.assertFailure()
788+
.resultChecker(r -> {
789+
assertTrue(r.output.contains("--compress value is invalid"), "Error message printed");
790+
});
791+
792+
FileUtils.deleteFileIfExistsWithRetry(jmod);
793+
794+
jmod("create",
795+
"--class-path", cp,
796+
"--compress", "zip-10",
797+
jmod.toString())
798+
.assertFailure()
799+
.resultChecker(r -> {
800+
assertTrue(r.output.contains("--compress value is invalid"), "Error message printed");
801+
});
802+
803+
FileUtils.deleteFileIfExistsWithRetry(jmod);
804+
805+
jmod("create",
806+
"--class-path", cp,
807+
"--compress", "zip-",
808+
jmod.toString())
809+
.assertFailure()
810+
.resultChecker(r -> {
811+
assertTrue(r.output.contains("--compress value is invalid"), "Error message printed");
812+
});
813+
814+
FileUtils.deleteFileIfExistsWithRetry(jmod);
815+
816+
jmod("create",
817+
"--class-path", cp,
818+
"--compress", "test",
819+
jmod.toString())
820+
.assertFailure()
821+
.resultChecker(r -> {
822+
assertTrue(r.output.contains("--compress value is invalid"), "Error message printed");
823+
});
824+
825+
FileUtils.deleteFileIfExistsWithRetry(jmod);
826+
827+
jmod("create",
828+
"--class-path", cp,
829+
"--compress", "test-0",
830+
jmod.toString())
831+
.assertFailure()
832+
.resultChecker(r -> {
833+
assertTrue(r.output.contains("--compress value is invalid"), "Error message printed");
834+
});
835+
}
836+
742837
// ---
743838

744839
static boolean compileModule(String name, Path dest) throws IOException {

0 commit comments

Comments
 (0)
Please sign in to comment.