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/). (...)