2016-06-08 88 views
0

要在我的spring-boot应用中启用https,我做了以下操作。在弹簧引导中使用可执行jar命令时发生Resource FileNotFoundException

@Bean 
@Inject 
public EmbeddedServletContainerCustomizer containerCustomizer() throws FileNotFoundException 
{ 
    final String absoluteKeystoreFile = ZenoTuringServiceApp.class.getClassLoader().getResource("test.jks").getFile(); 

    return (ConfigurableEmbeddedServletContainer factory) -> { 
     TomcatEmbeddedServletContainerFactory containerFactory = (TomcatEmbeddedServletContainerFactory) factory; 
     containerFactory.addConnectorCustomizers((TomcatConnectorCustomizer) (Connector connector) -> { 
      connector.setSecure(true); 
      connector.setScheme("https"); 
      connector.setAttribute("keystoreFile", absoluteKeystoreFile); 
      connector.setAttribute("keystorePass", "test"); 
      connector.setAttribute("keystoreType", "JKS"); 
      connector.setAttribute("clientAuth", "false"); 
      connector.setAttribute("sslProtocol", "TLS"); 
      connector.setAttribute("SSLEnabled", true); 
     }); 
    }; 
} 

如果我把它用MVN弹簧启动时运行:运行它按预期工作。

但是,当我运行使用可执行jar与java -jar target/xxx-xxx-service-0.1.17-SNAPSHOT.jar我得到FileNotFoundException。

Caused by: org.apache.catalina.LifecycleException: service.getName(): "Tomcat"; Protocol handler start failed 
     at org.apache.catalina.connector.Connector.startInternal(Connector.java:993) 
     at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150) 
     ... 18 common frames omitted 

Caused by: java.io.FileNotFoundException: /private/var/folders/wl/xxx77_523z44yjdgx2y7xxxc217h/T/tomcat.8061417798873093914.8091/file:/Users/xxx/Work/Himalay/xxx/xxx-xxx-service/target/xxx-xxx-service-0.1.17-SNAPSHOT.jar!/test.jks (No such file or directory) 
     at java.io.FileInputStream.open0(Native Method) 
     at java.io.FileInputStream.open(FileInputStream.java:195) 
     at java.io.FileInputStream.<init>(FileInputStream.java:138) 
     at org.apache.tomcat.util.net.jsse.JSSESocketFactory.getStore(JSSESocketFactory.java:433) 
     at org.apache.tomcat.util.net.jsse.JSSESocketFactory.getKeystore(JSSESocketFactory.java:339) 
     at org.apache.tomcat.util.net.jsse.JSSESocketFactory.getKeyManagers(JSSESocketFactory.java:597) 
+1

不能引用资源(JAR内)作为'File'。你**必须**使用InputStream。所以'keystoreFile'应该可能位于你的JAR之外,因为显然Tomcat希望有一个到'File'的路径。 – Tunaki

+0

由于您在Tomcat之外执行此操作,因此您没有包含“web”罐。您需要包含来自Tomcat的Java EE库。 – Grayson

+2

这可能会有所帮助https://kamwo.me/java-load-file-from-classpath-in-spring-boot/ – austin

回答

0

上述评论和这个代码帮我..

// support loading the JKS from the classpath (to get around Tomcat limitation) 
private static File getTuringKeyStoreFile() throws IOException { 
    ClassPathResource resource = new ClassPathResource("test.jks"); 

    // Tomcat won't allow reading File from classpath so read as InputStream into temp File 
    File jks = File.createTempFile("ssl_keystore", ".jks"); 
    InputStream inputStream = resource.getInputStream(); 
    try { 
     FileUtils.copyInputStreamToFile(inputStream, jks); 
    } finally { 
     IOUtils.closeQuietly(inputStream); 
    } 

    return jks; 
} 

参考:https://github.com/robinhowlett/everything-ssl