• 首页 首页 icon
  • 工具库 工具库 icon
    • IP查询 IP查询 icon
  • 内容库 内容库 icon
    • 快讯库 快讯库 icon
    • 精品库 精品库 icon
    • 问答库 问答库 icon
  • 更多 更多 icon
    • 服务条款 服务条款 icon

Spring Boot内嵌Tomcat临时目录问题

武飞扬头像
奋豆来袭
帮助2

听说后面上线可能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

打开

  1.  
    # This file is part of systemd.
  2.  
    #
  3.  
    # systemd is free software; you can redistribute it and/or modify it
  4.  
    # under the terms of the GNU Lesser General Public License as published by
  5.  
    # the Free Software Foundation; either version 2.1 of the License, or
  6.  
    # (at your option) any later version.
  7.  
     
  8.  
    # See tmpfiles.d(5) for details
  9.  
     
  10.  
    # Clear tmp directories separately, to make them easier to override
  11.  
    v /tmp 1777 root root 10d
  12.  
    v /var/tmp 1777 root root 30d
  13.  
     
  14.  
    # Exclude namespace mountpoints created with PrivateTmp=yes
  15.  
    x /tmp/systemd-private-%b-*
  16.  
    X /tmp/systemd-private-%b-*/tmp
  17.  
    x /var/tmp/systemd-private-%b-*
  18.  
    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为空目录**)。应用再去访问就报错了。

3. 方案

原因搞清楚了,解决方案自然很明了,大致有3种:

1.从Linux层面修改 /tmp目录的清理策略,比较简单,略过

2.指定新的系统临时文件路径

-Djava.io.tmpdir=/var/tmp

3. 配置中修改tomcat的临时目录

server:
    tomcat:
       basedir: /var/tmp/

4.代码中配置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();
    }
}

5.tomcat在临时目录不存在先创建

这个方案稍微麻烦些,就多啰嗦下。其实该方式在spring-boot2.1.4版本进行了修订:在临时目录不存在就创建临时目录。

在该类spring-boot/src/main/java/org/springframework/boot/context/embedded/tomcat/TomcatEmbeddedServletContainerFactory.java中添加了几行代码:

  1.  
    catch (NoSuchMethodError ex) {
  2.  
    // Tomcat is < 8.0.30. Continue
  3.  
    }
  4.  
    //新增代码开始
  5.  
    try {
  6.  
    context.setCreateUploadTargets(true);
  7.  
    }
  8.  
    catch (NoSuchMethodError ex) {
  9.  
    // Tomcat is < 8.5.39. Continue.
  10.  
    }
  11.  
    //新增代码结束
  12.  
    SkipPatternJarScanner.apply(context, this.tldSkipPatterns);

这篇好文章是转载于:学新通技术网

  • 版权申明: 本站部分内容来自互联网,仅供学习及演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,请提供相关证据及您的身份证明,我们将在收到邮件后48小时内删除。
  • 本站站名: 学新通技术网
  • 本文地址: /boutique/detail/tanhfkkjaf
系列文章
更多 icon
同类精品
更多 icon
继续加载