/* * Copyright (c) 2015, 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 Test zip compressor * @author Jean-Francois Denise * @modules java.base/jdk.internal.jimage.decompressor * jdk.jlink/jdk.tools.jlink.internal * jdk.jlink/jdk.tools.jlink.internal.plugins * @run main CompressorPluginTest */ import java.net.URI; import java.nio.ByteOrder; import java.nio.file.FileSystem; import java.nio.file.FileSystemNotFoundException; import java.nio.file.FileSystems; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.ProviderNotFoundException; import java.util.Collections; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.Properties; import java.util.regex.Pattern; import java.util.stream.Collectors; import java.util.stream.Stream; import jdk.internal.jimage.decompressor.CompressedResourceHeader; import jdk.internal.jimage.decompressor.ResourceDecompressor; import jdk.internal.jimage.decompressor.ResourceDecompressorFactory; import jdk.internal.jimage.decompressor.StringSharingDecompressorFactory; import jdk.internal.jimage.decompressor.ZipDecompressorFactory; import jdk.tools.jlink.internal.ModulePoolImpl; import jdk.tools.jlink.internal.StringTable; import jdk.tools.jlink.internal.plugins.DefaultCompressPlugin; import jdk.tools.jlink.internal.plugins.StringSharingPlugin; import jdk.tools.jlink.internal.plugins.ZipPlugin; import jdk.tools.jlink.plugin.Plugin; import jdk.tools.jlink.plugin.ModuleEntry; import jdk.tools.jlink.plugin.ModulePool; public class CompressorPluginTest { private static int strID = 1; public static void main(String[] args) throws Exception { new CompressorPluginTest().test(); } public void test() throws Exception { FileSystem fs; try { fs = FileSystems.getFileSystem(URI.create("jrt:/")); } catch (ProviderNotFoundException | FileSystemNotFoundException e) { System.err.println("Not an image build, test skipped."); return; } Path javabase = fs.getPath("/modules/java.base"); checkCompress(gatherResources(javabase), new ZipPlugin(), null, new ResourceDecompressorFactory[]{ new ZipDecompressorFactory() }); ModulePool classes = gatherClasses(javabase); // compress = String sharing checkCompress(classes, new StringSharingPlugin(), null, new ResourceDecompressorFactory[]{ new StringSharingDecompressorFactory()}); // compress == ZIP + String sharing Properties options = new Properties(); options.setProperty(ZipPlugin.NAME, ""); checkCompress(classes, new DefaultCompressPlugin(), options, new ResourceDecompressorFactory[]{ new ZipDecompressorFactory(), new StringSharingDecompressorFactory() }); // compress == ZIP + String sharing + filter options.setProperty(DefaultCompressPlugin.FILTER, "**Exception.class"); checkCompress(classes, new DefaultCompressPlugin(), options, new ResourceDecompressorFactory[]{ new ZipDecompressorFactory(), new StringSharingDecompressorFactory() }, Collections.singletonList(".*Exception.class")); // compress level 1 == ZIP Properties options1 = new Properties(); options1.setProperty(DefaultCompressPlugin.NAME, "1"); checkCompress(classes, new DefaultCompressPlugin(), options1, new ResourceDecompressorFactory[]{ new ZipDecompressorFactory() }); // compress level 1 == ZIP options1.setProperty(DefaultCompressPlugin.FILTER, "**Exception.class"); checkCompress(classes, new DefaultCompressPlugin(), options1, new ResourceDecompressorFactory[]{ new ZipDecompressorFactory() }, Collections.singletonList(".*Exception.class")); // compress level 2 == ZIP + String sharing Properties options2 = new Properties(); options2.setProperty(DefaultCompressPlugin.NAME, "2"); checkCompress(classes, new DefaultCompressPlugin(), options2, new ResourceDecompressorFactory[]{ new ZipDecompressorFactory(), new StringSharingDecompressorFactory() }); // compress level 2 == ZIP + String sharing + filter options2.setProperty(DefaultCompressPlugin.FILTER, "**Exception.class"); checkCompress(classes, new DefaultCompressPlugin(), options2, new ResourceDecompressorFactory[]{ new ZipDecompressorFactory(), new StringSharingDecompressorFactory() }, Collections.singletonList(".*Exception.class")); // compress level 0 == String sharing Properties options0 = new Properties(); options0.setProperty(DefaultCompressPlugin.NAME, "0"); checkCompress(classes, new DefaultCompressPlugin(), options0, new ResourceDecompressorFactory[]{ new StringSharingDecompressorFactory() }); // compress level 0 == String sharing + filter options0.setProperty(DefaultCompressPlugin.FILTER, "**Exception.class"); checkCompress(classes, new DefaultCompressPlugin(), options0, new ResourceDecompressorFactory[]{ new StringSharingDecompressorFactory() }, Collections.singletonList(".*Exception.class")); } private ModulePool gatherResources(Path module) throws Exception { ModulePool pool = new ModulePoolImpl(ByteOrder.nativeOrder(), new StringTable() { @Override public int addString(String str) { return -1; } @Override public String getString(int id) { return null; } }); try (Stream stream = Files.walk(module)) { for (Iterator iterator = stream.iterator(); iterator.hasNext();) { Path p = iterator.next(); if (Files.isRegularFile(p)) { byte[] content = Files.readAllBytes(p); pool.add(ModuleEntry.create(p.toString(), content)); } } } return pool; } private ModulePool gatherClasses(Path module) throws Exception { ModulePool pool = new ModulePoolImpl(ByteOrder.nativeOrder(), new StringTable() { @Override public int addString(String str) { return -1; } @Override public String getString(int id) { return null; } }); try (Stream stream = Files.walk(module)) { for (Iterator iterator = stream.iterator(); iterator.hasNext();) { Path p = iterator.next(); if (Files.isRegularFile(p) && p.toString().endsWith(".class")) { byte[] content = Files.readAllBytes(p); pool.add(ModuleEntry.create(p.toString(), content)); } } } return pool; } private void checkCompress(ModulePool resources, Plugin prov, Properties config, ResourceDecompressorFactory[] factories) throws Exception { checkCompress(resources, prov, config, factories, Collections.emptyList()); } private void checkCompress(ModulePool resources, Plugin prov, Properties config, ResourceDecompressorFactory[] factories, List includes) throws Exception { long[] original = new long[1]; long[] compressed = new long[1]; resources.entries().forEach(resource -> { List includesPatterns = includes.stream() .map(Pattern::compile) .collect(Collectors.toList()); Map props = new HashMap<>(); if (config != null) { for (String p : config.stringPropertyNames()) { props.put(p, config.getProperty(p)); } } prov.configure(props); final Map strings = new HashMap<>(); ModulePoolImpl inputResources = new ModulePoolImpl(ByteOrder.nativeOrder(), new StringTable() { @Override public int addString(String str) { int id = strID; strID += 1; strings.put(id, str); return id; } @Override public String getString(int id) { return strings.get(id); } }); inputResources.add(resource); ModulePool compressedResources = applyCompressor(prov, inputResources, resource, includesPatterns); original[0] += resource.getLength(); compressed[0] += compressedResources.findEntry(resource.getPath()).get().getLength(); applyDecompressors(factories, inputResources, compressedResources, strings, includesPatterns); }); String compressors = Stream.of(factories) .map(Object::getClass) .map(Class::getSimpleName) .collect(Collectors.joining(", ")); String size = "Compressed size: " + compressed[0] + ", original size: " + original[0]; System.out.println("Used " + compressors + ". " + size); if (original[0] <= compressed[0]) { throw new AssertionError("java.base not compressed."); } } private ModulePool applyCompressor(Plugin plugin, ModulePoolImpl inputResources, ModuleEntry res, List includesPatterns) { ModulePool compressedModulePool = new ModulePoolImpl(ByteOrder.nativeOrder(), inputResources.getStringTable()); plugin.visit(inputResources, compressedModulePool); String path = res.getPath(); ModuleEntry compressed = compressedModulePool.findEntry(path).get(); CompressedResourceHeader header = CompressedResourceHeader.readFromResource(ByteOrder.nativeOrder(), compressed.getBytes()); if (isIncluded(includesPatterns, path)) { if (header == null) { throw new AssertionError("Path should be compressed: " + path); } if (header.getDecompressorNameOffset() == 0) { throw new AssertionError("Invalid plugin offset " + header.getDecompressorNameOffset()); } if (header.getResourceSize() <= 0) { throw new AssertionError("Invalid compressed size " + header.getResourceSize()); } } else if (header != null) { throw new AssertionError("Path should not be compressed: " + path); } return compressedModulePool; } private void applyDecompressors(ResourceDecompressorFactory[] decompressors, ModulePool inputResources, ModulePool compressedResources, Map strings, List includesPatterns) { compressedResources.entries().forEach(compressed -> { CompressedResourceHeader header = CompressedResourceHeader.readFromResource( ByteOrder.nativeOrder(), compressed.getBytes()); String path = compressed.getPath(); ModuleEntry orig = inputResources.findEntry(path).get(); if (!isIncluded(includesPatterns, path)) { return; } byte[] decompressed = compressed.getBytes(); for (ResourceDecompressorFactory factory : decompressors) { try { ResourceDecompressor decompressor = factory.newDecompressor(new Properties()); decompressed = decompressor.decompress( strings::get, decompressed, CompressedResourceHeader.getSize(), header.getUncompressedSize()); } catch (Exception exp) { throw new RuntimeException(exp); } } if (decompressed.length != orig.getLength()) { throw new AssertionError("Invalid uncompressed size " + header.getUncompressedSize()); } byte[] origContent = orig.getBytes(); for (int i = 0; i < decompressed.length; i++) { if (decompressed[i] != origContent[i]) { throw new AssertionError("Decompressed and original differ at index " + i); } } }); } private boolean isIncluded(List includesPatterns, String path) { return includesPatterns.isEmpty() || includesPatterns.stream().anyMatch((pattern) -> pattern.matcher(path).matches()); } }