Spring ClasspathResource FileNotFoundException的原因

时间:2017-05-18 10:44:07

标签: java tomcat classpath embedded-resource

我的Web应用程序使用Spring 4.3.5。我们需要加载一个Freemarker模板,该模板存储在我们的一个JAR中。 jar内部的classpath。

让我尽量概括我的场景(稍后我会提供代码)

Jar结构

  • SRC
    • /com/acme/package/ComponentUsingResource.java
  • META-INF
    • 资源
      • template1.ftl.html
      • template1.ft2.html

jar包含在WAR应用程序内的WEB-INF / lib中。

从ComponentUsingResource.java我使用以下语句加载资源:new ClasspathResource("META-INF/templates/template1.ftl.html").getInputStream()

在我的笔记本电脑(JDK 8 121,Tomcat 8.0.43)上,在我的笔记本电脑上安装的Tomcat 8.0.36上,在我们的SIT环境(JDK 8> 100,Tomcat 8.0.39)上运行相同的WAR应用程序在旅途中,但它不适用于我们的客户站点(JDK 8 96,Tomcat 8.0.36)。加载ClasspathResource时因为FileNotFoundException而无法正常工作

当应用程序启动时,@Autowired依赖项链将进入基于资源的bean的初始化

    Arrays.asList("ftt-mail-natixclose-it", "ftt-mail-natixflussi-it", "ftt-mail-processreport-en", "ftt-mail-processreport-it",
                  "ftt-mail-regclosed-en", "ftt-mail-regclosed-it", "ftt-mail-regnotclosed-multi-en", "ftt-mail-regnotclosed-multi-it",
                  "ftt-mail-regnotclosed-single-en", "ftt-mail-regnotclosed-single-it", "ftt-mail-timestamps-en", "ftt-mail-timestamps-it",
                  "ftt-mail-missingdata-multi-en", "ftt-mail-missingdata-multi-it", "ftt-mail-missingdata-single-en",
                  "ftt-mail-missingdata-single-it", "ftt-mail-natixflussi-it")
          .parallelStream().forEach(templateName -> {

              ClassPathResource classpathResource = new ClassPathResource("META-INF/templates/" + templateName + ".ftl.html");
              try (Reader reader = new InputStreamReader(classpathResource.getInputStream(), Charset.forName("utf-8")))
              {
                  freemarker.template.Template tmpl = new freemarker.template.Template(templateName, reader, configuration);
                  cacheTemplate.put(templateName, tmpl);
              }
              catch (IOException e)
              {
                  throw new RuntimeException("Errror processing template " + templateName, e);
              }

          });

我知道在没有正确的Java同步的情况下从ParallelStream写入缓存是一个值得怀疑的做法,这将在以后解决。

问题在于,在所有这些资源中,代码无法找到单个资源:" ftt-mail-regclosed-en.ftl.html"。它除了在客户的SIT之外的任何地方加载资源。由于这是一个并行流,我可能会猜测这是唯一有问题的资源。

root例外是:java.io.FileNotFoundException: class path resource [META-INF/templates/ftt-mail-regclosed-en.ftl.html] cannot be opened because it does not exist

我无法提供完整的堆栈跟踪,因为我无法通过技术方式复制和粘贴碎片。

我已经三次检查了jar文件的内容,并且所有预期的资源都已到位。我问客户(请记住,WAR是完全相同的!)进行相同的检查,但不是解压缩它们,而是为我提供了jar文件的部分二进制转储的屏幕截图,我可以看到文件名。

我知道该应用程序已由Black Duck扫描程序处理软件漏洞。它经常报告可疑的依赖关系,但我(也不是客户)没有关于Black Duck剥离资源的证据或知识。我之所以这么说,是因为在我们的帖子中,有一位技术人员怀疑BD可能已从包中剥离出资源。我不太可能。

我可以采取哪些其他措施来调查此问题?仅在单个环境中导致此类FileNotFoundException的原因是什么?

请注意,它非常重要:早期版本的软件其他部分使用了相同的语法new ClasspathResource("META-INF/something")。在前一版本中,包含这种加载资源的方法,确实可以正常工作。这意味着它从包中嵌入的其他jar中加载了自己的类路径资源

1 个答案:

答案 0 :(得分:0)

这是一个潜在的答案。我还没有能够重现这个问题。

使用Spring调用new ClassPathResource时,您没有设置类加载器。我认为Spring使用了System类加载器(ClassLoader.getSystemClassLoader())。

错误。 Spring使用内部实用程序方法来获取绑定到线程的当前上下文的类加载器。由于我不知道的原因,可能发生了绑定到bean初始化方法的类加载器(恰好是由Apache Tomcat赞助的WebApplicationClassLoader)与系统类加载器不同。

事实上,Oracle系统类加载器无法从Jar文件加载资源。至少因为我已经调试并自己找到答案。

也许,我说可能,删除并行性而不是保存2秒的一次性启动处理,使Lambda表达式能够获取Web应用程序类加载器实例而不是Oracle JDK。 / p>

然而,这并不能解释为什么这只发生在客户现场而不是我们所有的环境中,包括PROD。

解决方案2是将getClass().getClassLoader()显式传递给ClassPathResource的构造函数