SpringBoot項目 文件上傳臨時目標被刪除異常處理

目錄

1、業務背景

2、三種解決方案

 2.1、直接修改CentOS清理臨時目則錄規

2.2、通過SpringBoot啓動配置註解(@Configuration) 指定自有上傳文件目錄

2.3、原理類似第二種方案,但是在SpringBoot的配置之中設定Profile信息

3、成果展現

4、總結

5、參考文章


1、業務背景

        我們使用了SpringCloud 進行項目開發,其中一個主要服務(涉及到圖片上傳)的SpringBoot微服務在測試環境之中。因爲此項目已經上線,很長一段時未針對此項目間做相關布更改和打包發。由於最近此項目業務甲方需要新增部分功能。但是測試在上傳課程時候,需要上傳課程封面,發現上傳課程封面的圖片上傳接口報錯500啦。本人在後端日誌目錄之中也無法查找到報錯信息。僅僅只有前後端分離的前端調用接口的時候返回一個如下錯誤提示

Could not parse multipart servlet request; 
nested exception is java.io.IOException: 
The temporary upload location [/tmp/tomcat/ocalhost/ROOT] is not valid

最後我根據錯誤提示搜索一下,結果發現是tomcat的臨時目錄被刪除了。最後找到一篇文章說得比較清楚有如下幾點:

(1)、SpringBoot項目啓動後,系統默認會在 /tmp 目錄下自動創建如下三個目錄

  •  hsperfdata_root,
  •  tomcat.************.8080,(結尾是項目的端後)
  •  tomcat-docbase.*********.8080

(2)、Multipart(form-data)的方式處理請求時,默認就是在第二個目錄下創建臨時文件的

(3)、CentOS7 定時清理臨時文件目錄

/tmp目錄的清理規則主要取決於/usr/lib/tmpfiles.d/tmp.conf文件的設定,默認的配置內容爲:

# Clear tmp directories separately, to make them easier to override
v /tmp 1777 root root 10d           #   清理/tmp下10天前的目錄和文件
v /var/tmp 1777 root root 30d       #   清理/var/tmp下30天前的目錄和文件

依據以上幾條情況,可以看得出我們上傳文件的臨時目錄,在CentOS7之中,會每10天進行定時清理掉。於是就出現了文章開始出所說的出現上傳文件的臨時目錄不存在,於是上傳問題報錯500的問題啦。我的同事說了一句啓動某個微服務還特定說了一句啓動網關服務就可以了。話有說回來,啓動任何一個SpringBoot的微服務都可以實現上傳(因爲啓動任何一個本機的微服務將生成對應的臨時目錄)。最後啓動了一個微服務,結果確實可以上傳啦。我不希望只要問題解決了就,淺嘗輒止的解決問題就了事啦。最後根據搜索到的文章和分析,個人認爲有如下三種解決方案

2、三種解決方案

 2.1、直接修改CentOS清理臨時目則錄規

         直接暴力指定不清楚所有臨時目錄,精細化管理針對上傳文件tomcat目錄不進行清除。

        /tmp目錄的清理規則主要取決於/usr/lib/tmpfiles.d/tmp.conf文件的設定:

   我們可以配置這個文件,比如你不想讓系統自動清理/tmp下以tomcat開頭的目錄,那麼增加下面這條內容到配置文件中即可:

x /tmp/tomcat.*

2.2、通過SpringBoot啓動配置註解(@Configuration) 指定自有上傳文件目錄

         改變臨時文件的存儲路徑,指定自定義非CentOS7的系統默認臨時目錄,這樣就可以避免系統在定時清除臨時目錄的情況。實現代碼如下

@Configuration
public class MultipartConfig {

    /**
     * 文件上傳臨時路徑
     */
    @Bean
    MultipartConfigElement multipartConfigElement() {
        MultipartConfigFactory factory = new MultipartConfigFactory();
        String location = System.getProperty("user.dir") + "/data/upload/tmp";
        File tmpFile = new File(location);
        if (!tmpFile.exists()) {
            tmpFile.mkdirs();
        }
        factory.setLocation(location);
        return factory.createMultipartConfig();
    }
}

2.3、原理類似第二種方案,但是在SpringBoot的配置之中設定Profile信息

在propertites/yaml文件中配置: spring.http.multipart.location= 你的緩存文件路徑

spring.mvc.static-path-pattern=/upload/**
spring.http.multipart.max-file-size=10MB
#指定上傳文件臨時目錄
spring.http.multipart.location=/opt/data/upload

3、成果展現

4、總結

       我們在遇見問題了,可能僅僅看見表面上問題已經解決了。但是作爲技術,應該深究是什麼原因導致的。在此也很感謝互聯網,讓很多大神能夠分享自己發現的問題和解決方案。同時也說明了其中的原理,這樣有助於從根源解決問題。

5、參考文章

    SpringBoot項目的上傳文件報錯 The temporary upload location ***is not valid

    SpringBoot項目的The temporary upload location ***is not valid 問題

    CentOS7的/tmp目錄自動清理規則

    

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章