All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.mengweijin.quickboot.util.DownLoadUtils Maven / Gradle / Ivy

package com.github.mengweijin.quickboot.util;

import com.github.mengweijin.quickboot.exception.QuickBootException;
import lombok.extern.slf4j.Slf4j;
import org.apache.catalina.connector.ClientAbortException;
import org.springframework.http.HttpHeaders;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.function.Function;

/**
 * @author mengweijin
 */
@Slf4j
public class DownLoadUtils {

    /**
     * 文件下载。
     * @param fileId 文件 ID。根据文件 id 获取文件对象。
     *               比如数据库中存储的文件表中的 id。或者一个在服务器上的文件绝对路径。
     *               或者其他情况,根据应用存放文件的方案来实现具体逻辑。return File
     *               (一般不要实现的是 fileId 为服务器上的绝对全路径这样搞,
     *               会被恶意用户下载到服务器上所有的文件!)。
     * @param request HttpServletRequest
     * @param response HttpServletResponse
     * @param function 函数式接口,需要开发者自定义逻辑。根据 fileId 返回一个 File 对象。
     */
    public static void download(String fileId, HttpServletRequest request, HttpServletResponse response, Function function) {
        download(function.apply(fileId), request, response);
    }

    /**
     * 文件下载,断点续传
     * @param fileId 文件 ID
     * @param request HttpServletRequest
     * @param response HttpServletResponse
     * @param function 函数式接口,需要开发者自定义逻辑。根据 fileId 返回一个 File 对象。
     */
    public static void chunkDownload(String fileId, HttpServletRequest request, HttpServletResponse response, Function function) {
        chunkDownload(function.apply(fileId), request, response);
    }

    public static void download(File file, HttpServletRequest request, HttpServletResponse response) {
        try {
            response.setCharacterEncoding(StandardCharsets.UTF_8.name());
            response.setContentType("multipart/form-data");
            response.setHeader(HttpHeaders.CONTENT_DISPOSITION,
                    "attachment;fileName=" + setFileName(request, file.getName()));
            Files.copy(Paths.get(file.toURI()), response.getOutputStream());
        } catch (ClientAbortException e) {
            //捕获此异常表示用户停止下载
            log.warn("User cancel download.");
        } catch (IOException e) {
            throw new QuickBootException(e);
        }
    }

    /**
     * 文件支持分块下载和断点续传
     *
     * @param file     文件
     * @param request  请求
     * @param response 响应
     */
    public static void chunkDownload(File file, HttpServletRequest request, HttpServletResponse response) {
        String range = request.getHeader(HttpHeaders.RANGE);
        log.debug("current request rang:" + range);
        //开始下载位置
        long startByte = 0;
        //结束下载位置
        long endByte = file.length() - 1;
        log.debug("File start location:{},File end location:{},File length:{}", startByte, endByte, file.length());

        //有range的话
        if (range != null && range.contains("bytes=") && range.contains(Const.DASH)) {
            range = range.substring(range.lastIndexOf(Const.EQUAL) + 1).trim();
            String[] ranges = range.split(Const.DASH);
            try {
                //判断range的类型, 如文件长度为10字节
                if (ranges.length == 1) {
                    //类型一:bytes=-5
                    if (range.startsWith(Const.DASH)) {
                        endByte = Long.parseLong(ranges[0]);
                    }
                    //类型二:bytes=5-
                    else if (range.endsWith(Const.DASH)) {
                        startByte = Long.parseLong(ranges[0]);
                    }
                }
                //类型三:bytes=3-8
                else if (ranges.length == 2) {
                    startByte = Long.parseLong(ranges[0]);
                    endByte = Long.parseLong(ranges[1]);
                }

            } catch (NumberFormatException e) {
                startByte = 0;
                endByte = file.length() - 1;
                log.error("Range Occur Error,Message:{}",e.getLocalizedMessage());
            }
        }

        //要下载的长度
        long contentLength = endByte - startByte + 1;
        //文件名
        String fileName = file.getName();
        //文件类型
        String contentType = request.getServletContext().getMimeType(fileName);

        //解决下载文件时文件名乱码问题
        byte[] fileNameBytes = fileName.getBytes(StandardCharsets.UTF_8);
        fileName = new String(fileNameBytes, 0, fileNameBytes.length, StandardCharsets.ISO_8859_1);

        //各种响应头设置
        //http状态码要为206:表示获取部分内容
        response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT);
        response.setContentType(contentType);
        //支持断点续传,获取部分字节内容:
        response.setHeader(HttpHeaders.ACCEPT_RANGES, "bytes");
        response.setHeader(HttpHeaders.CONTENT_TYPE, contentType);
        //inline表示浏览器直接使用,attachment表示下载,fileName表示下载的文件名
        response.setHeader(HttpHeaders.CONTENT_DISPOSITION, "inline;filename=" + fileName);
        response.setHeader(HttpHeaders.CONTENT_LENGTH, String.valueOf(contentLength));
        // Content-Range,格式为:[要下载的开始位置]-[结束位置]/[文件总大小]
        response.setHeader(HttpHeaders.CONTENT_RANGE, "bytes " + startByte + "-" + endByte + "/" + file.length());

        //已传送数据大小
        long transmitted = 0;
        try(RandomAccessFile randomAccessFile = new RandomAccessFile(file, "r");
            BufferedOutputStream outputStream = new BufferedOutputStream(response.getOutputStream())) {

            byte[] buff = new byte[4096];
            int len = 0;
            randomAccessFile.seek(startByte);
            //判断是否到了最后不足4096(buff的length)个byte这个逻辑((transmitted + len) <= contentLength)要放前面!
            //不然会会先读取randomAccessFile,造成后面读取位置出错,找了一天才发现问题所在
            while ((transmitted + len) <= contentLength && (len = randomAccessFile.read(buff)) != -1) {
                outputStream.write(buff, 0, len);
                transmitted += len;
            }
            //处理不足buff.length部分
            if (transmitted < contentLength) {
                len = randomAccessFile.read(buff, 0, (int) (contentLength - transmitted));
                outputStream.write(buff, 0, len);
                transmitted += len;
            }

            outputStream.flush();
            response.flushBuffer();
            log.debug("Download completed:" + startByte + "-" + endByte + ":" + transmitted);
        } catch (ClientAbortException e) {
            //捕获此异常表示用户停止下载
            log.warn("User stop download:" + startByte + "-" + endByte + ":" + transmitted);
        } catch (IOException e) {
            log.error("User download IO Exception,Message:{}", e.getLocalizedMessage());
        }
    }

    private static String setFileName(HttpServletRequest request, String fileName) throws UnsupportedEncodingException {
        final String agent = request.getHeader("USER-AGENT");
        String encodeFileName = fileName;
        if (agent.contains("MSIE")) {
            // IE浏览器
            encodeFileName = URLEncoder.encode(encodeFileName, StandardCharsets.UTF_8.name());
            encodeFileName = encodeFileName.replace("+", " ");
        } else if (agent.contains("Firefox")) {
            // 火狐浏览器
            encodeFileName = new String(fileName.getBytes(StandardCharsets.UTF_8), StandardCharsets.ISO_8859_1);
        } else if (agent.contains("Chrome")) {
            // google浏览器
            encodeFileName = URLEncoder.encode(encodeFileName, StandardCharsets.UTF_8.name());
            encodeFileName = encodeFileName.replace("+", " ");
        } else {
            // 其它浏览器
            encodeFileName = URLEncoder.encode(encodeFileName, StandardCharsets.UTF_8.name());
        }
        return encodeFileName;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy