JarInputStream 为现有清单文件返回 null
JarInputStream retuns null for existing Manifest file
我在使用 JarInputStream 获取(完全)运行的 .jar 文件的清单时遇到问题,我得到空指针。尝试了一些不同的文件,没有抛出异常。注意:此文件来自代码生成器。具有以下结构(希望这个形状能帮助你理解它):
|[+]test.jar
|[+]META-INF
|MANIFEST.MF
|[+]mypackage
|MyClass.class
|text.txt
以上是 class 的一部分(注意:这个 class 不在 .jar 中)
/*...code...**/
File jar = new File("resources/test.jar");
FileInputStream fis = new FileInputStream(jar);
JarInputStream jis = new JarInputStream(fis);
Manifest jarManifest = jis.getManifest();
Attributes jarManifestAttributes = jarManifest.getMainAttributes(); //line 25
String mainClass = jarManifestAttributes.getValue("Main-Class");
/*...code...**/
当运行上述代码时,抛出以下异常:
Exception in thread "main" java.lang.NullPointerException
at crypter.LoadJar.main(LoadJar.java:25)
所以 jarManifest 有空值。
EDIT1
MANIFEST.MF
Manifest-Version: 1.0
Main-Class: mypackage.MyClass
Permissions: all-permissions
Name: text.txt
Name: mypackage/MyClass.class
EDIT2
jar -tvf reverse_tcp2.jar
37 Sat Jan 07 11:31:14 EET 2017 text.txt
0 Sat Jan 07 11:31:14 EET 2017 mypackage/
8513 Sat Jan 07 11:31:14 EET 2017 mypackage/MyClass.class
0 Sat Jan 07 11:31:14 EET 2017 META-INF/
145 Sat Jan 07 11:31:14 EET 2017 META-INF/MANIFEST.MF
JarInputStream
中的一个错误是 MANIFEST.MF
必须在 ZIP 文件的 前几条 中,如下所示:
$ jar -tvf example-correct-archive.jar
623 Thu Jun 13 12:13:42 CEST 2019 META-INF/MANIFEST.MF
0 Thu Jun 13 12:13:42 CEST 2019 META-INF/
...
在您的情况下,在这些条目之前还有 3 个其他文件。
解决方法
a) 使用 jar
工具重新排序 JAR 文件中的条目:
- 解压
test.jar
文件到目录 foo
- 将现有
MANIFEST.MF
文件移动到目录 foo
上一级并删除 META-INF
目录
- 在
foo
目录中 运行 命令:jar cvfm ../test.jar ../MANIFEST.MF .
- 在
foo
之外,您将获得具有正确条目顺序的固定 JAR 文件
b) 使用 JarFile
而不是 JarInputStream
更多信息
见JDK-8031748 Clarify jar entry orders in a jar file:
From the beginning, the jar file has an “undocumented” assumption that
the MANIFEST.MF file and signature-related files (block and SF) should
appear at the beginning (except for directory entries, say,
META-INF/).
(...)
我在使用 JarInputStream 获取(完全)运行的 .jar 文件的清单时遇到问题,我得到空指针。尝试了一些不同的文件,没有抛出异常。注意:此文件来自代码生成器。具有以下结构(希望这个形状能帮助你理解它):
|[+]test.jar
|[+]META-INF
|MANIFEST.MF
|[+]mypackage
|MyClass.class
|text.txt
以上是 class 的一部分(注意:这个 class 不在 .jar 中)
/*...code...**/
File jar = new File("resources/test.jar");
FileInputStream fis = new FileInputStream(jar);
JarInputStream jis = new JarInputStream(fis);
Manifest jarManifest = jis.getManifest();
Attributes jarManifestAttributes = jarManifest.getMainAttributes(); //line 25
String mainClass = jarManifestAttributes.getValue("Main-Class");
/*...code...**/
当运行上述代码时,抛出以下异常:
Exception in thread "main" java.lang.NullPointerException
at crypter.LoadJar.main(LoadJar.java:25)
所以 jarManifest 有空值。
EDIT1 MANIFEST.MF
Manifest-Version: 1.0
Main-Class: mypackage.MyClass
Permissions: all-permissions
Name: text.txt
Name: mypackage/MyClass.class
EDIT2
jar -tvf reverse_tcp2.jar
37 Sat Jan 07 11:31:14 EET 2017 text.txt
0 Sat Jan 07 11:31:14 EET 2017 mypackage/
8513 Sat Jan 07 11:31:14 EET 2017 mypackage/MyClass.class
0 Sat Jan 07 11:31:14 EET 2017 META-INF/
145 Sat Jan 07 11:31:14 EET 2017 META-INF/MANIFEST.MF
JarInputStream
中的一个错误是 MANIFEST.MF
必须在 ZIP 文件的 前几条 中,如下所示:
$ jar -tvf example-correct-archive.jar
623 Thu Jun 13 12:13:42 CEST 2019 META-INF/MANIFEST.MF
0 Thu Jun 13 12:13:42 CEST 2019 META-INF/
...
在您的情况下,在这些条目之前还有 3 个其他文件。
解决方法
a) 使用 jar
工具重新排序 JAR 文件中的条目:
- 解压
test.jar
文件到目录foo
- 将现有
MANIFEST.MF
文件移动到目录foo
上一级并删除META-INF
目录 - 在
foo
目录中 运行 命令:jar cvfm ../test.jar ../MANIFEST.MF .
- 在
foo
之外,您将获得具有正确条目顺序的固定 JAR 文件
b) 使用 JarFile
而不是 JarInputStream
更多信息
见JDK-8031748 Clarify jar entry orders in a jar file:
From the beginning, the jar file has an “undocumented” assumption that the MANIFEST.MF file and signature-related files (block and SF) should appear at the beginning (except for directory entries, say, META-INF/). (...)