SpringBoot内嵌Tomcat临时目录问题及解决
SpringBoot内嵌Tomcat临时目录问题
听说后面上线可能tomcat临时文件夹会被Linux删除,会报找不到错误,现在赶紧记录一下,已被不时之需。
存在文件上传的SpringBoot项目,在Linux系统部署之后,会在系统的tmp目录下生成一个带tomcat 及 随机字符串的临时目录。
该目录有可能被linux系统在一定时间后自动清除掉,导致再次上传文件的时候,系统就会报错。
意思是tomcat的临时目录会被tmpwatch删除掉,甚至可能删除掉class文件,导致错误的发生
1.背景
线上保障,上线运行了几天的SpringBoot应用,突然遇到问题:
/tmp/tomcatXXX/work/Tomcat/localhost/XXX is not valid。
应用不会存在/tmp/tomcatXXX/work/Tomcat/localhost/ROOT目录。经查询,是tomcat在文件上传时,会先对文件进行复制到临时目录,就是该目录。
之前的应用运行是正常的,现在出现这个情况,显然是创建好的目录被删除了。对,就是这个特殊的/tmp目录Linux存在清除策略。
清除策略的配置文件路径如下:
/usr/lib/tmpfiles.d/tmp.conf
打开
# This file is part of systemd. # # systemd is free software; you can redistribute it and/or modify it # under the terms of the GNU Lesser General Public License as published by # the Free Software Foundation; either version 2.1 of the License, or # (at your option) any later version. # See tmpfiles.d(5) for details # Clear tmp directories separately, to make them easier to override v /tmp 1777 root root 10d v /var/tmp 1777 root root 30d # Exclude namespace mountpoints created with PrivateTmp=yes x /tmp/systemd-private-%b-* X /tmp/systemd-private-%b-*/tmp x /var/tmp/systemd-private-%b-* X /var/tmp/systemd-private-%b-*/tmp
发现会清除10天内没被访问过的文件。但是到了这里,有个疑问就是,昨天可以的也就是该目录是被访问过,今天怎么会被清除咧?
这个本人确实当时很疑惑,然后对应用的假设为:
/tmp/tomcat.4344543554352.8080/work/Tomcat/localhost/test,发现该目录下为空。也就是临时文件会被tomcat清理掉,但是test目录的创建时间确实是在10天前。
到了这里就明白了,虽然test目录下文件每天都会有更新,但是**不会影响test目录的访问时间**,并且该文件被删掉了。/tmp目录的清理机制发现test空目录是10天前,就直接清理了(**test为空目录**)。应用再去访问就报错了。
2.方案
原因搞清楚了,解决方案自然很明了,大致有3种:
- 1.从Linux层面修改 /tmp目录的清理策略,比较简单,略过
- 2.指定新的系统临时文件路径
-Djava.io.tmpdir=/var/tmp
- 3. 配置中修改tomcat的临时目录
server:
tomcat:
basedir: /var/tmp/3.代码中配置tomcat临时目录
@Configuration
public class MultipartConfig {
@Bean
MultipartConfigElement multipartConfigElement() {
MultipartConfigFactory factory = new MultipartConfigFactory();
String location = System.getProperty("user.dir") + "/data/tmp";
File tmpFile = new File(location);
if (!tmpFile.exists()) {
tmpFile.mkdirs();
}
factory.setLocation(location);
return factory.createMultipartConfig();
}
}4.tomcat在临时目录不存在先创建
这个方案稍微麻烦些,就多啰嗦下。
其实该方式在spring-boot2.1.4版本进行了修订:在临时目录不存在就创建临时目录。
在该类spring-boot/src/main/java/org/springframework/boot/context/embedded/tomcat/TomcatEmbeddedServletContainerFactory.java中添加了几行代码:
catch (NoSuchMethodError ex) {
// Tomcat is < 8.0.30. Continue
}
//新增代码开始
try {
context.setCreateUploadTargets(true);
}
catch (NoSuchMethodError ex) {
// Tomcat is < 8.5.39. Continue.
}
//新增代码结束
SkipPatternJarScanner.apply(context, this.tldSkipPatterns);总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
相关文章
myeclipse创建servlet_动力节点Java学院整理
这篇文章主要为大家详细介绍了myeclipse创建servlet的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下2017-07-07
Mybatis Plus 中的LambdaQueryWrapper示例详解
这篇文章主要介绍了Mybatis Plus 中的LambdaQueryWrapper,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2022-03-03
Spring IOC容器基于XML外部属性文件的Bean管理
这篇文章主要为大家介绍了Spring IOC容器Bean管理XML外部属性文件,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪2022-05-05
IDEA2020.2创建springboot项目卡死在reading maven project的问题
这篇文章主要介绍了关于2020.2IDEA用spring Initializr创建maven的springboot项目卡死在reading maven project的问题描述及解决方法,感兴趣的朋友跟随小编一起看看吧2020-09-09
将SpringBoot的Jar注册成Windows服务的实现方法
当前项目有个地图编辑器,后端用的是SpringBoot框架,外网刚好有一台空闲的Windows服务器就直接拿来用了,将Java程序部署成Windows服务可以用WinSW (Windows Service Wrapper)来实现,文中有详细的操作步骤,需要的朋友可以参考下2023-11-11


最新评论