|
@@ -1,12 +1,20 @@
|
|
|
package com.gxzc.zen.common.util
|
|
|
|
|
|
-import com.gxzc.zen.common.dto.ZenMultipartFileDTO
|
|
|
+import com.github.benmanes.caffeine.cache.Caffeine
|
|
|
+import com.gxzc.zen.common.dto.ZenFileMetadata
|
|
|
+import com.gxzc.zen.common.dto.ZenFileResponse
|
|
|
+import com.gxzc.zen.common.exception.ZenException
|
|
|
+import com.gxzc.zen.common.exception.ZenExceptionEnum
|
|
|
import com.gxzc.zen.common.properties.UploadProperties
|
|
|
+import org.apache.commons.io.FilenameUtils
|
|
|
import org.slf4j.LoggerFactory
|
|
|
+import org.springframework.cache.caffeine.CaffeineCache
|
|
|
+import org.springframework.web.multipart.MultipartFile
|
|
|
import java.io.*
|
|
|
import java.nio.file.Files
|
|
|
import java.nio.file.Paths
|
|
|
import java.util.concurrent.ConcurrentHashMap
|
|
|
+import java.util.concurrent.TimeUnit
|
|
|
|
|
|
/**
|
|
|
* 上传 工具类
|
|
@@ -25,70 +33,190 @@ object UploadUtil {
|
|
|
return field
|
|
|
}
|
|
|
|
|
|
- private val uploadedMap = ConcurrentHashMap<String, Int>() // 分片上传数
|
|
|
+ private val uploadedStatusMap = ConcurrentHashMap<String, String>() // 文件上传状态缓存
|
|
|
+ private val batchCountMap = CaffeineCache("batchCountMap", Caffeine.newBuilder().expireAfterWrite(1L, TimeUnit.HOURS).build(), true) // 文件批量上传批次文件数量记录
|
|
|
+
|
|
|
+ private val FILE_SEPARATOR = System.getProperty("file.separator") // 适配操作系统的文件路径
|
|
|
+
|
|
|
+ object STATUS {
|
|
|
+ const val CHECKING = "checking" // 检查中
|
|
|
+ const val MERGING = "merging" // 合并中
|
|
|
+ const val UPLOADED = "uploaded" // 单文件上传完毕
|
|
|
+ const val BATCH_UPLOADED = "batchUploaded" // 单文件上传完毕
|
|
|
+ const val UPLOADING = "uploading" // 上传中
|
|
|
+ }
|
|
|
|
|
|
/**
|
|
|
* 正常上传<br>
|
|
|
- * 保留源 文件夹结构/文件名
|
|
|
+ * 文件夹结构/文件名
|
|
|
*/
|
|
|
- fun upload(fileDTO: ZenMultipartFileDTO): Boolean {
|
|
|
+ fun upload(fileMetadata: ZenFileMetadata, file: MultipartFile): ZenFileResponse {
|
|
|
val tmpPath = uploadProperties!!.tmpPath!!
|
|
|
val dataPath = uploadProperties!!.dataPath!!
|
|
|
- if (validateRequest(fileDTO)) {
|
|
|
- val chunkFilename = getChunkFilename(tmpPath, fileDTO.chunkNumber, fileDTO.identifier)
|
|
|
- val directory = File(tmpPath)
|
|
|
- if (!directory.exists()) {
|
|
|
- Files.createDirectories(Paths.get(tmpPath))
|
|
|
- }
|
|
|
- // 文件较小 直接transfer
|
|
|
- fileDTO.file!!.transferTo(File(chunkFilename))
|
|
|
- // 检查分块完整性
|
|
|
- if (checkChunks(fileDTO.identifier!!, fileDTO.totalChunks!!)) {
|
|
|
- // 合并
|
|
|
- mergeChunks(tmpPath, dataPath, fileDTO)
|
|
|
+ val chunkSize = uploadProperties!!.chunkSize!!
|
|
|
+ var retFile: File? = null
|
|
|
+ if (validateRequest(fileMetadata, file)) {
|
|
|
+ // 文件分片 如果分片小于chunkSize && totalChunk = 1,直接转存
|
|
|
+ if (fileMetadata.totalChunks == 1 && fileMetadata.chunkSize!! <= chunkSize) {
|
|
|
+ val filename = fileMetadata.filename!!
|
|
|
+ val relativePath = fileMetadata.relativePath!!
|
|
|
+
|
|
|
+ val destPath = FilenameUtils.normalize("$dataPath$FILE_SEPARATOR${getDestFilePath(filename, relativePath, fileMetadata.repath)}")
|
|
|
+ val destDir = Paths.get(destPath)
|
|
|
+ if (Files.notExists(destDir)) {
|
|
|
+ Files.createDirectories(destDir)
|
|
|
+ }
|
|
|
+ val outputPath = FilenameUtils.normalize("$destPath$FILE_SEPARATOR${getDestFileName(filename, fileMetadata.rename)}")
|
|
|
+ retFile = File(outputPath)
|
|
|
+ file.transferTo(retFile)
|
|
|
+ retFile.setLastModified(fileMetadata.lastModified!!)
|
|
|
+ } else {
|
|
|
+ val chunkFilename = getChunkFilename(tmpPath, fileMetadata.chunkNumber, fileMetadata.md5)
|
|
|
+ val directory = Paths.get(tmpPath)
|
|
|
+ if (Files.notExists(directory)) {
|
|
|
+ Files.createDirectories(directory)
|
|
|
+ }
|
|
|
+ file.transferTo(File(chunkFilename))
|
|
|
+ // 检查分块完整性
|
|
|
+ if (checkChunks(tmpPath, fileMetadata)) {
|
|
|
+ // 合并
|
|
|
+ retFile = try {
|
|
|
+ mergeChunks(tmpPath, dataPath, fileMetadata)
|
|
|
+ } catch (e: Throwable) {
|
|
|
+ logger.error("merge file chunks exception, cause ", e)
|
|
|
+ uploadedStatusMap.remove(fileMetadata.md5!!)
|
|
|
+ null
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
- return true
|
|
|
} else {
|
|
|
- return false
|
|
|
+ throw ZenException(ZenExceptionEnum.FILE_METADATA_VALIDATE_ERROR)
|
|
|
+ }
|
|
|
+ var status: String = STATUS.UPLOADING
|
|
|
+ if (retFile != null) {
|
|
|
+ status = setBatch(fileMetadata)
|
|
|
+ }
|
|
|
+ return ZenFileResponse().apply {
|
|
|
+ this.status = status
|
|
|
+ this.file = retFile
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ private fun setBatch(fileMetadata: ZenFileMetadata): String {
|
|
|
+ val batchId = fileMetadata.batchId!!
|
|
|
+ val totalNumber = fileMetadata.totalNumber!!
|
|
|
+ synchronized(batchCountMap) {
|
|
|
+ var count = batchCountMap[batchId]?.get() as? Int ?: 0
|
|
|
+ return if (++count >= totalNumber) {
|
|
|
+ batchCountMap.evict(batchId)
|
|
|
+ STATUS.BATCH_UPLOADED
|
|
|
+ } else {
|
|
|
+ batchCountMap.put(batchId, count)
|
|
|
+ STATUS.UPLOADED
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- fun uploadGet(fileDTO: ZenMultipartFileDTO): Boolean {
|
|
|
+ /**
|
|
|
+ * 检测文件是否存在 实现文件秒传
|
|
|
+ */
|
|
|
+ fun checkUpload(fileMetadata: ZenFileMetadata): ZenFileResponse {
|
|
|
+ val ret = ZenFileResponse()
|
|
|
+ if (validateRequest(fileMetadata, null)) {
|
|
|
+ if (fileExists(fileMetadata)) {
|
|
|
+ ret.uploadedChunks = mutableListOf()
|
|
|
+ for (i in 1..fileMetadata.totalChunks!!) {
|
|
|
+ ret.uploadedChunks?.add(i)
|
|
|
+ }
|
|
|
+ ret.status = setBatch(fileMetadata)
|
|
|
+ } else {
|
|
|
+ // 检查分片存在情况
|
|
|
+ for (i in 1..fileMetadata.totalChunks!!) {
|
|
|
+ if (chunkExists(i, fileMetadata.md5!!)) {
|
|
|
+ if (ret.uploadedChunks == null) {
|
|
|
+ ret.uploadedChunks = mutableListOf()
|
|
|
+ }
|
|
|
+ ret.uploadedChunks?.add(i)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (ret.uploadedChunks != null && ret.uploadedChunks!!.size == fileMetadata.totalChunks) {
|
|
|
+ ret.status = STATUS.CHECKING // 秒传
|
|
|
+ // 所有分块传完 移除其中一个分块,再传一次而后 merge
|
|
|
+ ret.uploadedChunks!!.removeAt(0)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return ret
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 检查文件是否存在 (真实文件)
|
|
|
+ */
|
|
|
+ private fun fileExists(fileMetadata: ZenFileMetadata): Boolean {
|
|
|
+ val dataPath = uploadProperties!!.dataPath!!
|
|
|
+ // 获取filename 真实文件
|
|
|
+ val filename = fileMetadata.filename!!
|
|
|
+ val relativePath = fileMetadata.relativePath!!
|
|
|
+ val md5 = fileMetadata.md5!!
|
|
|
+ val path = FilenameUtils.normalize("$dataPath$FILE_SEPARATOR${getDestFilePath(filename, relativePath, fileMetadata.repath)}$FILE_SEPARATOR${getDestFileName(filename, fileMetadata.rename)}")
|
|
|
+ return Files.exists(Paths.get(path)) && FileUtil.md5HeadTail(path, uploadProperties!!.chunkSize!!.toInt()) == md5 // # 防篡改
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 检查分片是否存在 (tmp目录)
|
|
|
+ */
|
|
|
+ private fun chunkExists(chunkNumber: Int?, md5: String): Boolean {
|
|
|
val tmpPath = uploadProperties!!.tmpPath!!
|
|
|
- return if (validateRequest(fileDTO)) {
|
|
|
- val existsFile = File(getChunkFilename(tmpPath, fileDTO.chunkNumber, fileDTO.identifier))
|
|
|
- existsFile.exists()
|
|
|
- } else {
|
|
|
- false
|
|
|
+ return Files.exists(Paths.get(getChunkFilename(tmpPath, chunkNumber, md5)))
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 检查分块是否上传完毕
|
|
|
+ */
|
|
|
+ private fun checkChunks(tmpPath: String, fileMetadata: ZenFileMetadata): Boolean {
|
|
|
+ val totalChunks = fileMetadata.totalChunks!!
|
|
|
+ val md5 = fileMetadata.md5!!
|
|
|
+ for (i in 1..totalChunks) {
|
|
|
+ if (Files.notExists(Paths.get(getChunkFilename(tmpPath, i, md5)))) {
|
|
|
+ return false
|
|
|
+ }
|
|
|
}
|
|
|
+ return true
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 合并所有分块
|
|
|
* 支持 文件夹合并
|
|
|
*/
|
|
|
- private fun mergeChunks(sourceRootPath: String, destRootPath: String, fileDTO: ZenMultipartFileDTO) {
|
|
|
+ private fun mergeChunks(sourceRootPath: String, destRootPath: String, fileMetadata: ZenFileMetadata): File? {
|
|
|
// 源文件的文件夹信息 然后组合拼接
|
|
|
- val filename = fileDTO.filename!!
|
|
|
- val relativePath = fileDTO.relativePath!!
|
|
|
- val totalChunks = fileDTO.totalChunks!!
|
|
|
- logger.debug("start merging chunks for [$filename]")
|
|
|
+ val filename = fileMetadata.filename!!
|
|
|
+ val relativePath = fileMetadata.relativePath!!
|
|
|
+ val totalChunks = fileMetadata.totalChunks!!
|
|
|
+ val md5 = fileMetadata.md5!!
|
|
|
|
|
|
- val folderRelativePath = relativePath.replace(filename, "")
|
|
|
- // 示例 "sql/xx.sql" -> "sql/"
|
|
|
- // "xx.sql" -> ""
|
|
|
+ if (uploadedStatusMap[md5] == STATUS.MERGING) {
|
|
|
+ return null
|
|
|
+ }
|
|
|
+
|
|
|
+ uploadedStatusMap[md5] = STATUS.MERGING
|
|
|
+
|
|
|
+ logger.debug("start merging chunks for [$filename]")
|
|
|
|
|
|
- val destDir = File("$destRootPath/$folderRelativePath")
|
|
|
- if (!destDir.exists()) {
|
|
|
- Files.createDirectories(Paths.get("$destRootPath/$folderRelativePath"))
|
|
|
+ val destPath = FilenameUtils.normalize("$destRootPath$FILE_SEPARATOR${getDestFilePath(filename, relativePath, fileMetadata.repath)}")
|
|
|
+ val destDir = Paths.get(destPath)
|
|
|
+ if (Files.notExists(destDir)) {
|
|
|
+ Files.createDirectories(destDir)
|
|
|
}
|
|
|
// 目标文件流 通过文件名来拼接
|
|
|
- val destOutputStream = BufferedOutputStream(FileOutputStream("$destRootPath/$folderRelativePath$filename"))
|
|
|
+ val outputPath = FilenameUtils.normalize("$destPath$FILE_SEPARATOR${getDestFileName(filename, fileMetadata.rename)}")
|
|
|
+ val outputFile = File(outputPath)
|
|
|
+ val destOutputStream = BufferedOutputStream(FileOutputStream(outputFile))
|
|
|
|
|
|
- val buffer = ByteArray(1024)
|
|
|
+ val buffer = ByteArray(4096)
|
|
|
var readBytesLength: Int
|
|
|
for (i: Int in 1..totalChunks) {
|
|
|
- val sourceFile = File(getChunkFilename(sourceRootPath, i, fileDTO.identifier))
|
|
|
+ val sourceFile = File(getChunkFilename(sourceRootPath, i, md5))
|
|
|
val sourceInputStream = BufferedInputStream(FileInputStream(sourceFile))
|
|
|
|
|
|
readBytesLength = sourceInputStream.read(buffer)
|
|
@@ -105,51 +233,58 @@ object UploadUtil {
|
|
|
destOutputStream.close()
|
|
|
|
|
|
logger.debug("merging successful.")
|
|
|
+
|
|
|
+ // 修改文件 修改时间
|
|
|
+ outputFile.setLastModified(fileMetadata.lastModified!!)
|
|
|
+
|
|
|
+ uploadedStatusMap.remove(md5)
|
|
|
+
|
|
|
+ return outputFile
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * 检查分块是否上传完毕
|
|
|
+ * 获取 分片文件名
|
|
|
*/
|
|
|
- private fun checkChunks(identifier: String, totalChunks: Int): Boolean {
|
|
|
- val cleanIdentifier = cleanIdentifier(identifier)!!
|
|
|
- var chunksNow = uploadedMap.getOrDefault(cleanIdentifier, 0)
|
|
|
- return if (totalChunks == ++chunksNow) {
|
|
|
- uploadedMap.remove(cleanIdentifier)
|
|
|
- true
|
|
|
- } else {
|
|
|
- uploadedMap[cleanIdentifier] = chunksNow
|
|
|
- false
|
|
|
- }
|
|
|
+ private fun getChunkFilename(path: String, chunkNumber: Int?, identifier: String?): String {
|
|
|
+ return "$path/upload-$identifier.$chunkNumber"
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * 移除多余的identifier字符
|
|
|
+ * 生成 输出的文件名
|
|
|
*/
|
|
|
- private fun cleanIdentifier(identifier: String?): String? {
|
|
|
- return identifier?.replace(Regex("[^0-9A-Za-z_-]"), "")
|
|
|
+ private fun getDestFileName(filename: String, rename: String?): String {
|
|
|
+ return if (rename == null || rename.isEmpty()) {
|
|
|
+ filename
|
|
|
+ } else {
|
|
|
+ rename
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * 获取 分片文件名
|
|
|
+ * 生成 输出的文件夹结构
|
|
|
*/
|
|
|
- private fun getChunkFilename(path: String, chunkNumber: Int?, identifier: String?): String {
|
|
|
- return "$path/upload-${cleanIdentifier(identifier)}.$chunkNumber"
|
|
|
+ private fun getDestFilePath(filename: String, relativePath: String, repath: String?): String {
|
|
|
+ return if (repath.isNullOrEmpty()) {
|
|
|
+ relativePath.replace(filename, "")
|
|
|
+ } else {
|
|
|
+ repath!!
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* validate request multipart chunks
|
|
|
*/
|
|
|
- private fun validateRequest(fileDTO: ZenMultipartFileDTO): Boolean {
|
|
|
- val identifier = cleanIdentifier(fileDTO.identifier)
|
|
|
- val chunkNumber = fileDTO.chunkNumber
|
|
|
- val chunkSize = fileDTO.chunkSize
|
|
|
- val totalSize = fileDTO.totalSize
|
|
|
- val filename = fileDTO.filename
|
|
|
+ private fun validateRequest(fileMetadata: ZenFileMetadata, file: MultipartFile?): Boolean {
|
|
|
+ val md5 = fileMetadata.md5
|
|
|
+ val chunkNumber = fileMetadata.chunkNumber
|
|
|
+ val chunkSize = fileMetadata.chunkSize
|
|
|
+ val totalSize = fileMetadata.totalSize
|
|
|
+ val filename = fileMetadata.filename
|
|
|
|
|
|
if (chunkNumber == null || chunkNumber <= 0 ||
|
|
|
chunkSize == null || chunkSize <= 0 ||
|
|
|
totalSize == null || totalSize <= 0 ||
|
|
|
- identifier == null || identifier.isEmpty() ||
|
|
|
+ md5 == null || md5.isEmpty() ||
|
|
|
filename == null || filename.isEmpty()) {
|
|
|
return false
|
|
|
}
|
|
@@ -170,8 +305,6 @@ object UploadUtil {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-
|
|
|
- val file = fileDTO.file
|
|
|
if (file != null) {
|
|
|
// The chunk in the POST request isn't the correct size
|
|
|
if (chunkNumber < numberOfChunks && file.size != chunkSize) {
|