8020802: Need an ability to create jar files that are invariant to the pack200 packing/unpacking
Reviewed-by: alanb, ksrini
This commit is contained in:
parent
7108f683fc
commit
28f4ef62dd
@ -31,6 +31,7 @@ import java.nio.file.Files;
|
|||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.zip.*;
|
import java.util.zip.*;
|
||||||
import java.util.jar.*;
|
import java.util.jar.*;
|
||||||
|
import java.util.jar.Pack200.*;
|
||||||
import java.util.jar.Manifest;
|
import java.util.jar.Manifest;
|
||||||
import java.text.MessageFormat;
|
import java.text.MessageFormat;
|
||||||
import sun.misc.JarIndex;
|
import sun.misc.JarIndex;
|
||||||
@ -72,8 +73,9 @@ class Main {
|
|||||||
* flag0: no zip compression (store only)
|
* flag0: no zip compression (store only)
|
||||||
* Mflag: DO NOT generate a manifest file (just ZIP)
|
* Mflag: DO NOT generate a manifest file (just ZIP)
|
||||||
* iflag: generate jar index
|
* iflag: generate jar index
|
||||||
|
* nflag: Perform jar normalization at the end
|
||||||
*/
|
*/
|
||||||
boolean cflag, uflag, xflag, tflag, vflag, flag0, Mflag, iflag;
|
boolean cflag, uflag, xflag, tflag, vflag, flag0, Mflag, iflag, nflag;
|
||||||
|
|
||||||
static final String MANIFEST_DIR = "META-INF/";
|
static final String MANIFEST_DIR = "META-INF/";
|
||||||
static final String VERSION = "1.0";
|
static final String VERSION = "1.0";
|
||||||
@ -197,12 +199,56 @@ class Main {
|
|||||||
vflag = false;
|
vflag = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
File tmpfile = null;
|
||||||
|
final OutputStream finalout = out;
|
||||||
|
final String tmpbase = (fname == null)
|
||||||
|
? "tmpjar"
|
||||||
|
: fname.substring(fname.indexOf(File.separatorChar) + 1);
|
||||||
|
if (nflag) {
|
||||||
|
tmpfile = createTemporaryFile(tmpbase, ".jar");
|
||||||
|
out = new FileOutputStream(tmpfile);
|
||||||
|
}
|
||||||
expand(null, files, false);
|
expand(null, files, false);
|
||||||
create(new BufferedOutputStream(out, 4096), manifest);
|
create(new BufferedOutputStream(out, 4096), manifest);
|
||||||
if (in != null) {
|
if (in != null) {
|
||||||
in.close();
|
in.close();
|
||||||
}
|
}
|
||||||
out.close();
|
out.close();
|
||||||
|
if(nflag) {
|
||||||
|
JarFile jarFile = null;
|
||||||
|
File packFile = null;
|
||||||
|
JarOutputStream jos = null;
|
||||||
|
try {
|
||||||
|
Packer packer = Pack200.newPacker();
|
||||||
|
Map<String, String> p = packer.properties();
|
||||||
|
p.put(Packer.EFFORT, "1"); // Minimal effort to conserve CPU
|
||||||
|
jarFile = new JarFile(tmpfile.getCanonicalPath());
|
||||||
|
packFile = createTemporaryFile(tmpbase, ".pack");
|
||||||
|
out = new FileOutputStream(packFile);
|
||||||
|
packer.pack(jarFile, out);
|
||||||
|
jos = new JarOutputStream(finalout);
|
||||||
|
Unpacker unpacker = Pack200.newUnpacker();
|
||||||
|
unpacker.unpack(packFile, jos);
|
||||||
|
} catch (IOException ioe) {
|
||||||
|
fatalError(ioe);
|
||||||
|
} finally {
|
||||||
|
if (jarFile != null) {
|
||||||
|
jarFile.close();
|
||||||
|
}
|
||||||
|
if (out != null) {
|
||||||
|
out.close();
|
||||||
|
}
|
||||||
|
if (jos != null) {
|
||||||
|
jos.close();
|
||||||
|
}
|
||||||
|
if (tmpfile != null && tmpfile.exists()) {
|
||||||
|
tmpfile.delete();
|
||||||
|
}
|
||||||
|
if (packFile != null && packFile.exists()) {
|
||||||
|
packFile.delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
} else if (uflag) {
|
} else if (uflag) {
|
||||||
File inputFile = null, tmpFile = null;
|
File inputFile = null, tmpFile = null;
|
||||||
FileInputStream in;
|
FileInputStream in;
|
||||||
@ -358,6 +404,9 @@ class Main {
|
|||||||
rootjar = args[count++];
|
rootjar = args[count++];
|
||||||
iflag = true;
|
iflag = true;
|
||||||
break;
|
break;
|
||||||
|
case 'n':
|
||||||
|
nflag = true;
|
||||||
|
break;
|
||||||
case 'e':
|
case 'e':
|
||||||
ename = args[count++];
|
ename = args[count++];
|
||||||
break;
|
break;
|
||||||
@ -1215,4 +1264,34 @@ class Main {
|
|||||||
e.setCrc(crc.getValue());
|
e.setCrc(crc.getValue());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to create temporary file in the system-provided temporary folder, if failed attempts
|
||||||
|
* to create it in the same folder as the file in parameter (if any)
|
||||||
|
*/
|
||||||
|
private File createTemporaryFile(String tmpbase, String suffix) {
|
||||||
|
File tmpfile = null;
|
||||||
|
|
||||||
|
try {
|
||||||
|
tmpfile = File.createTempFile(tmpbase, suffix);
|
||||||
|
} catch (IOException | SecurityException e) {
|
||||||
|
// Unable to create file due to permission violation or security exception
|
||||||
|
}
|
||||||
|
if (tmpfile == null) {
|
||||||
|
// Were unable to create temporary file, fall back to temporary file in the same folder
|
||||||
|
if (fname != null) {
|
||||||
|
try {
|
||||||
|
File tmpfolder = new File(fname).getAbsoluteFile().getParentFile();
|
||||||
|
tmpfile = File.createTempFile(fname, ".tmp" + suffix, tmpfolder);
|
||||||
|
} catch (IOException ioe) {
|
||||||
|
// Last option failed - fall gracefully
|
||||||
|
fatalError(ioe);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// No options left - we can not compress to stdout without access to the temporary folder
|
||||||
|
fatalError(new IOException(getMsg("error.create.tempfile")));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return tmpfile;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -44,6 +44,8 @@ error.create.dir=\
|
|||||||
{0} : could not create directory
|
{0} : could not create directory
|
||||||
error.incorrect.length=\
|
error.incorrect.length=\
|
||||||
incorrect length while processing: {0}
|
incorrect length while processing: {0}
|
||||||
|
error.create.tempfile=\
|
||||||
|
Could not create a temporary file
|
||||||
out.added.manifest=\
|
out.added.manifest=\
|
||||||
added manifest
|
added manifest
|
||||||
out.update.manifest=\
|
out.update.manifest=\
|
||||||
@ -66,7 +68,7 @@ out.size=\
|
|||||||
(in = {0}) (out= {1})
|
(in = {0}) (out= {1})
|
||||||
|
|
||||||
usage=\
|
usage=\
|
||||||
Usage: jar {ctxui}[vfm0Me] [jar-file] [manifest-file] [entry-point] [-C dir] files ...\n\
|
Usage: jar {ctxui}[vfmn0Me] [jar-file] [manifest-file] [entry-point] [-C dir] files ...\n\
|
||||||
Options:\n\
|
Options:\n\
|
||||||
\ \ -c create new archive\n\
|
\ \ -c create new archive\n\
|
||||||
\ \ -t list table of contents for archive\n\
|
\ \ -t list table of contents for archive\n\
|
||||||
@ -75,6 +77,7 @@ Options:\n\
|
|||||||
\ \ -v generate verbose output on standard output\n\
|
\ \ -v generate verbose output on standard output\n\
|
||||||
\ \ -f specify archive file name\n\
|
\ \ -f specify archive file name\n\
|
||||||
\ \ -m include manifest information from specified manifest file\n\
|
\ \ -m include manifest information from specified manifest file\n\
|
||||||
|
\ \ -n perform Pack200 normalization after creating a new archive\n\
|
||||||
\ \ -e specify application entry point for stand-alone application \n\
|
\ \ -e specify application entry point for stand-alone application \n\
|
||||||
\ \ bundled into an executable jar file\n\
|
\ \ bundled into an executable jar file\n\
|
||||||
\ \ -0 store only; use no ZIP compression\n\
|
\ \ -0 store only; use no ZIP compression\n\
|
||||||
|
147
jdk/test/tools/jar/normalize/TestNormal.java
Normal file
147
jdk/test/tools/jar/normalize/TestNormal.java
Normal file
@ -0,0 +1,147 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2013, 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
|
||||||
|
* @run main/timeout=600 TestNormal
|
||||||
|
* @bug 8020802
|
||||||
|
* @summary Need an ability to create jar files that are invariant to the pack200 packing/unpacking
|
||||||
|
* @author Alexander Zuev
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.*;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Properties;
|
||||||
|
import java.util.jar.JarEntry;
|
||||||
|
import java.util.jar.JarFile;
|
||||||
|
|
||||||
|
public class TestNormal {
|
||||||
|
private static String FS = File.separator;
|
||||||
|
|
||||||
|
public static void main(String args[]) throws Exception {
|
||||||
|
Properties p = System.getProperties();
|
||||||
|
String java_home = p.getProperty("test.jdk");
|
||||||
|
String dtjar = java_home + File.separator + "lib"
|
||||||
|
+ File.separator + "dt.jar";
|
||||||
|
|
||||||
|
File folder = new File("dt");
|
||||||
|
if (folder.exists()) {
|
||||||
|
delete(folder);
|
||||||
|
}
|
||||||
|
folder.mkdir();
|
||||||
|
|
||||||
|
try {
|
||||||
|
extractJar(new JarFile(dtjar), folder);
|
||||||
|
execJavaCommand(java_home, "jar cnf normalized.jar -C dt .");
|
||||||
|
execJavaCommand(java_home, "jar cf original.jar -C dt .");
|
||||||
|
execJavaCommand(java_home, "pack200 -r repacked.jar original.jar");
|
||||||
|
compareJars(new JarFile("normalized.jar"), new JarFile("repacked.jar"));
|
||||||
|
} finally {
|
||||||
|
String[] cleanupList = {"dt", "normalized.jar", "original.jar", "repacked.jar"};
|
||||||
|
for (String s : cleanupList) {
|
||||||
|
delete(new File(s));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void execJavaCommand(String java_home, String cmd) throws Exception {
|
||||||
|
Process proc = Runtime.getRuntime().exec(java_home + FS + "bin" + FS + cmd);
|
||||||
|
String s;
|
||||||
|
BufferedReader stdInput =
|
||||||
|
new BufferedReader(new InputStreamReader(proc.getInputStream()));
|
||||||
|
BufferedReader stdError =
|
||||||
|
new BufferedReader(new InputStreamReader(proc.getErrorStream()));
|
||||||
|
while ((s = stdInput.readLine()) != null) {
|
||||||
|
System.out.println(s);
|
||||||
|
}
|
||||||
|
while ((s = stdError.readLine()) != null) {
|
||||||
|
System.err.println(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void compareJars(JarFile jf1, JarFile jf2) throws Exception {
|
||||||
|
try {
|
||||||
|
if (jf1.size() != jf2.size()) {
|
||||||
|
throw new Exception("Jars " + jf1.getName() + " and " + jf2.getName()
|
||||||
|
+ " have different number of entries");
|
||||||
|
}
|
||||||
|
for (JarEntry elem1 : Collections.list(jf1.entries())) {
|
||||||
|
JarEntry elem2 = jf2.getJarEntry(elem1.getName());
|
||||||
|
if (elem2 == null) {
|
||||||
|
throw new Exception("Element " + elem1.getName() + " is missing from " + jf2.getName());
|
||||||
|
}
|
||||||
|
if (!elem1.isDirectory() && elem1.getCrc() != elem2.getCrc()) {
|
||||||
|
throw new Exception("The crc of " + elem1.getName() + " is different.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
jf1.close();
|
||||||
|
jf2.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void extractJar(JarFile jf, File where) throws Exception {
|
||||||
|
for (JarEntry file : Collections.list(jf.entries())) {
|
||||||
|
File out = new File(where, file.getName());
|
||||||
|
if (file.isDirectory()) {
|
||||||
|
out.mkdirs();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
File parent = out.getParentFile();
|
||||||
|
if (parent != null && !parent.exists()) {
|
||||||
|
parent.mkdirs();
|
||||||
|
}
|
||||||
|
InputStream is = null;
|
||||||
|
OutputStream os = null;
|
||||||
|
try {
|
||||||
|
is = jf.getInputStream(file);
|
||||||
|
os = new FileOutputStream(out);
|
||||||
|
while (is.available() > 0) {
|
||||||
|
os.write(is.read());
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
if (is != null) {
|
||||||
|
is.close();
|
||||||
|
}
|
||||||
|
if (os != null) {
|
||||||
|
os.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void delete(File f) throws IOException {
|
||||||
|
if (!f.exists()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (f.isDirectory()) {
|
||||||
|
for (File c : f.listFiles()) {
|
||||||
|
delete(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!f.delete()) {
|
||||||
|
throw new FileNotFoundException("Failed to delete file: " + f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user