Skip to content

Concept Download

Linyuzai edited this page Jan 7, 2024 · 54 revisions

概述

主要用于简单快速的实现一个下载功能

你只需要提供一个文件路径,或者一个http地址,甚至是你的自定义业务对象,其他的事情都由它帮你完成

  1. 有多个文件需要压缩?
  2. 压缩文件的缓存?
  3. 需要下载网络资源?
  4. 网络资源的并发?
  5. 网络资源的缓存?
  6. 需要自己处理输入输出流?

上述问题都不需要管,省时省力省心

示例说明

@Download(source = "classpath:/download/README.txt")
@GetMapping("/classpath")
public void classpath() {
}

@Download
@GetMapping("/file")
public File file() {
    return new File("/Users/Shared/README.txt");
}

@Download
@GetMapping("/http")
public String http() {
    return "http://127.0.0.1:8080/concept-download/image.jpg";
}

借助@Download注解,你可以把被下载的资源写在source参数中或者作为方法的返回值return

两者没有任何区别,只是返回值支持动态的对象

兼容webmvcwebflux

需要注意webflux中需要返回DownloadMono并且方法返回值类型固定为Mono<Void>,因为webflux会校验返回值的类型

@Download(source = "classpath:/download/README.txt")
@GetMapping("/classpath")
public Mono<Void> classpath() {
    return DownloadMono.empty();
}

@Download
@GetMapping("/file")
public Mono<Void> file() {
    return DownloadMono.value(new File("/Users/Shared/README.txt"));
}

@Download
@GetMapping("/http")
public Mono<Void> http() {
    return DownloadMono.value("http://127.0.0.1:8080/concept-download/image.jpg");
}

集成

implementation 'com.github.linyuzai:concept-download-spring-boot-starter:1.2.2'

implementation 'org.springframework.boot:spring-boot-starter-aop'
<dependency>
  <groupId>com.github.linyuzai</groupId>
  <artifactId>concept-download-spring-boot-starter</artifactId>
  <version>1.2.2</version>
</dependency>

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

引入spring-boot-starter包就可以直接使用@Download注解啦

支持http请求的资源类型

默认内置了HttpURLConnection处理,并且也提供了OkHttp的方式

使用OkHttp需要手动依赖如下模块

implementation 'com.github.linyuzai:concept-download-source-okhttp:1.2.0'
<dependency>
  <groupId>com.github.linyuzai</groupId>
  <artifactId>concept-download-source-okhttp</artifactId>
  <version>1.2.0</version>
</dependency>

Kotlin协程实现网络资源的并发请求

注意仅支持webmvc不支持webflux

需要手动依赖如下模块

implementation 'com.github.linyuzai:concept-download-load-coroutines:1.2.0'
<dependency>
  <groupId>com.github.linyuzai</groupId>
  <artifactId>concept-download-load-coroutines</artifactId>
  <version>1.2.0</version>
</dependency>

并且手动注入

@Configuration
public class ConceptDownloadConfig {

    @Bean
    public CoroutinesSourceLoader coroutinesSourceLoader() {
        return new CoroutinesSourceLoader();
    }
}

模块说明

模块 说明
concept-download-core 核心功能
concept-download-source-okhttp 支持OkHttp加载网络资源
concept-download-load-coroutines 支持Kotlin协程并发加载
concept-download-spring-boot-starter Spring Boot自动配置

@Download 注解说明

参数 说明
source 需要下载的内容,但是优先级低于返回值
如果方法返回值不为null则会使用返回值作为下载的内容
inline 如果为true,可以直接在浏览器预览
需要配合contentType,如图片或视频,默认false
视频文件目前存在一些问题,还在测试阶段
filename 指定下载时浏览器上显示的名称
如果不指定则会获取下载内容的名称,如文件则使用文件名
contentType 如果未指定,会尝试获取
如果尝试获取失败,则默认application/octet-stream
application/x-zip-compressed
compressFormat 压缩格式,默认zip
forceCompress 强制压缩
如果为true,不管下载的文件有几个都会压缩
如果为false,有多个文件时压缩,只有一个文件时不压缩
默认false
charset 如果下载包含中文的文本文件出现乱码,可以尝试指定编码
headers 统一的响应头,每2个为一组
extra 额外的数据,当需要自行编写额外流程业务时可能会用到

整体流程

整体架构

整个下载流程由DownloadHandlerDownloadHandlerChain实现链式处理

序号 步骤 说明
1. 全局配置 全局配置DownloadConfiguration支持配置文件和DownloadConfigurer配置
只在服务启动时配置一次
2. 切面拦截 拦截方法上的@Download注解
3. 参数构建 结合注解与全局配置DownloadConfiguration构建参数DownloadOptions
方法返回DownloadOptions.Rewriter可以重写参数
4. 创建上下文 上下文DownloadContext用于下载过程中的数据传递与共享
通过DownloadContextFactory创建
5. 初始化上下文 支持自定义DownloadContextInitializer实现初始化逻辑
6. 解析适配下载数据 基于DownloadHandler实现CreateSourceHandler
通过自定义SourceSourceFactory支持任意类型数据的扩展
7. 预加载数据 基于DownloadHandler实现LoadSourceHandler
可实现SourceLoader自定义加载流程
8. 压缩 基于DownloadHandler实现CompressSourceHandler
可实现SourceCompressor自定义压缩逻辑
9. 写入响应 基于DownloadHandler实现WriteResponseHandler
抽象DownloadResponseDownloadWriter执行写入操作
DownloadResponse用于兼容webmvcwebflux
可自定义DownloadWriter用于处理输入输出流
10. 销毁上下文 支持自定义DownloadContextDestroyer实现销毁逻辑

自定义流程扩展

可以自定义实现DownloadHandler并注入到Spring的容器中

/**
 * 下载处理器。
 */
public interface DownloadHandler extends OrderProvider {

    /**
     * 执行处理。
     *
     * @param context {@link DownloadContext}
     * @param chain   {@link DownloadHandlerChain}
     */
    Mono<Void> handle(DownloadContext context, DownloadHandlerChain chain);
}

全局配置

配置文件

concept:
  download:
    source:
      cache:
        enabled: true #网络资源缓存是否启用
        path: /source_cache #网络资源缓存路径,默认为 user.home/concept/download
        delete: false #下载结束后网络资源缓存是否删除
    compress:
      format: zip #压缩格式
      cache:
        enabled: true #压缩缓存是否启用
        path: /compress_cache #压缩缓存路径,默认为 user.home/concept/download
        delete: false #下载结束后压缩缓存是否删除
    response:
      headers: #额外的响应头
        header1 : 1
        header2 : 2
    logger:
      enabled: true #日志总开关
      standard:
        enabled: true #标准流程日志是否启用
      progress:
        enabled: true #进度计算日志是否启用,包括加载进度,压缩进度,写入响应进度
        duration: 500 #进度计算日志输出间隔,ms
        percentage: true #进度计算日志是否使用百分比输出
      time-spent:
        enabled: true #时间计算日志是否启用

代码配置

使用DownloadConfigurer修改配置

@Configuration
public class ConceptDownloadConfig implements DownloadConfigurer {

    @Override
    public void configure(DownloadConfiguration configuration) {
        //可以在这里覆盖配置文件的配置
    }
}

下载参数

下载时会结合全局配置和注解配置构造一个下载参数DownloadOptions

自定义下载参数

接口方法返回DownloadOptions.Rewriter即可重写下载参数

同时可以设置临时的事件监听器并只对该方法生效

@Download(source = "classpath:/download/README.txt")
@GetMapping("/rewrite")
public DownloadOptions.Rewriter rewrite() {
    return new DownloadOptions.Rewriter() {
        @Override
        public DownloadOptions rewrite(DownloadOptions options) {
            return options.toBuilder()
                    .eventListener(new DownloadEventListener() {
                            @Override
                            public void onEvent(Object event) {
                                
                            }
                        })
                    .build();
        }
    };
}

下载上下文

每次下载都会生成一个DownloadContext实例

在同一个下载流程中可以通过上下文传递和共享数据

初始化上下文

上下文创建时可以通过DownloadContextInitializer实现自定义初始化逻辑

/**
 * {@link DownloadContext} 初始化器,{@link DownloadContext} 初始化时会回调。
 */
public interface DownloadContextInitializer extends OrderProvider {

    /**
     * 初始化。
     *
     * @param context {@link DownloadContext}
     */
    void initialize(DownloadContext context);
}

销毁上下文

上下文销毁时可以通过DownloadContextDestroyer实现自定义销毁逻辑

/**
 * {@link DownloadContext} 销毁器,{@link DownloadContext} 销毁时会回调。
 */
public interface DownloadContextDestroyer extends OrderProvider {

    /**
     * 销毁。
     *
     * @param context {@link DownloadContext}
     */
    void destroy(DownloadContext context);
}

自定义上下文

通过实现DownloadContextFactoryDownloadContextAbstractDownloadContext并注入到Spring容器中

/**
 * {@link DownloadContext} 工厂。
 */
public interface DownloadContextFactory {

    /**
     * 创建一个 {@link DownloadContext}。
     *
     * @param options {@link DownloadOptions}
     * @return {@link DownloadContext}
     */
    DownloadContext create(DownloadOptions options);
}

下载源创建

所有的下载对象最终都会通过Source体现,作为原始的下载数据的抽象提供统一的接口

支持的下载类型

类型 说明 依赖
FileSource 支持File对象
file:,user.home:,user-home:,user_home:前缀的字符串
ClassPathSource 支持ClassPathResource对象
classpath:前缀的字符串
TextSource 支持任意的String对象作为文本文件
HttpSource 支持http地址,基于HttpURLConnection
WebClientSource 支持http地址,基于WebClient,在webflux中使用
OkHttpSource 支持http地址,基于OkHttp source-okhttp
PublisherSource 支持Publisher对象
MultipleSource 支持arrayCollection对象

同时支持上述类型任意组合的数组或集合

@Download(filename = "压缩包.zip")
@GetMapping("/list")
public List<Object> list() {
    List<Object> list = new ArrayList<>();
    list.add(new File("/Users/Shared/README.txt"));
    list.add(new ClassPathResource("/download/image.jpg"));
    list.add("http://127.0.0.1:8080/concept-download/video.mp4");
    return list;
}

支持反射的方式表示下载源

对于已经存在的数据模型,可以通过注解的方式将一些属性覆盖到对应的Source

@Data
@SourceModel
@AllArgsConstructor
public static class BusinessModel {

    @SourceName
    private String name;

    @SourceObject
    private String url;
}

@Download
@GetMapping("/business-model")
public List<BusinessModel> businessModel() {
    List<BusinessModel> businessModels = new ArrayList<>();
    businessModels.add(new BusinessModel("BusinessModel.txt", "http://127.0.0.1:8080/concept-download/text.txt"));
    businessModels.add(new BusinessModel("BusinessModel.jpg", "http://127.0.0.1:8080/concept-download/image.jpg"));
    businessModels.add(new BusinessModel("BusinessModel.mp4", "http://127.0.0.1:8080/concept-download/video.mp4"));
    return businessModels;
}
注解说明
注解 说明
@SourceModel 标注在类上
表明是一个Source
@SourceObject 标注在具体下载对象上
@SourceName 指定名称
@SourceCharset 指定编码
@SourceLength 指定长度,即字节数
@SourceAsyncLoad 指定是否异步加载
@SourceCacheEnabled 指定是否启用缓存
@SourceCacheExisted 缓存是否存在
@SourceCachePath 缓存目录

除了@SourceModel必须标注在类上

其他注解都可以标注在字段或get方法上

所有注解子类优先于父类

如果Source本身没有对应属性的set方法或者属性字段,则注解无法生效

反射字段的数据类型

Source中的编码为Charset类型

如果我们的数据模型中对应的类型是String

那么在该属性上标注@SourceCharset将会导致反射异常

所以引入了ValueConvertor来处理类型的转换

当然StringCharset已经提供实现

/**
 * 将 {@link String} 转为 {@link Charset} 的 {@link ValueConvertor}。
 */
public class StringToCharsetValueConvertor implements ValueConvertor<String, Charset> {

    @Override
    public Charset convert(String value) {
        return Charset.forName(value);
    }
}

支持自定义ValueConvertor实现

/**
 * 值转换器。
 *
 * @param <Original> 原始类型
 * @param <Target>   目标类型
 */
public interface ValueConvertor<Original, Target> {

    /**
     * 转换。
     *
     * @param value 原始值
     * @return 目标值
     */
    Target convert(Original value);
}

并通过ValueConversion注册

ValueConversion.getInstance().register(ValueConvertor);

自定义支持任意类型的下载数据

可以自定义实现SourceFactoryPrefixSourceFactorySourceAbstractSourceAbstractLoadableSource并注入到Spring容器中

/**
 * {@link Source} 工厂。
 */
public interface SourceFactory extends OrderProvider {

    /**
     * 是否支持需要下载的原始数据对象。
     *
     * @param source  需要下载的原始数据对象
     * @param context {@link DownloadContext}
     * @return 如果支持则返回 true
     */
    boolean support(Object source, DownloadContext context);

    /**
     * 创建。
     *
     * @param source  需要下载的原始数据对象
     * @param context {@link DownloadContext}
     * @return 创建的 {@link Source}
     */
    Source create(Object source, DownloadContext context);
}

网络资源加载

针对一些网络资源,如HTTP,支持并发的加载,通过SourceLoader来实现

内置的资源加载器

类型 说明 兼容 依赖
DefaultSourceLoader 按顺序加载,适用于本地文件,默认 webmvc webflux
SchedulerSourceLoader 依赖线程池加载,适合网络资源 webmvc
CoroutinesSourceLoader 依赖协程加载,适合网络资源 webmvc load-coroutines

每个Source都可以单独指定asyncLoad属性来控制是否需要异步加载,目前HttpSourceOkHttpSource默认为true,其他默认都为false

注意webflux本身就是非阻塞的,具有天然的并发优势,并且不兼容其他两种阻塞式的SourceLoader

通过手动注入来切换不同的加载方式

自定义加载方式

可以自定义实现SourceLoaderConcurrentSourceLoader并注入到Spring容器中

/**
 * {@link Source} 加载器。
 *
 * @see DefaultSourceLoader
 * @see SchedulerSourceLoader
 */
public interface SourceLoader {

    /**
     * 执行加载。
     *
     * @param source  {@link Source}
     * @param context {@link DownloadContext}
     * @return 加载后的 {@link Source}
     */
    Mono<Source> load(Source source, DownloadContext context);
}

网络资源缓存

配置文件配置

concept:
  download:
    source:
      cache:
        enabled: true #是否启用
        path: / #缓存目录
        delete: false #下载结束后是否删除

代码全局配置

@Configuration
public class ConceptDownloadConfig implements DownloadConfigurer {

    @Override
    public void configure(DownloadConfiguration configuration) {
        configuration.getSource().getCache().setEnabled(true);
        configuration.getSource().getCache().setPath("缓存路径");
        configuration.getSource().getCache().setDelete(false);
    }

注解配置单个方法

@Download(filename = "压缩包.zip")
@SourceCache(group = "source")
@GetMapping("/source-cache")
public String[] sourceCache() {
    return new String[]{
          "http://127.0.0.1:8080/concept-download/text.txt",
          "http://127.0.0.1:8080/concept-download/image.jpg",
          "http://127.0.0.1:8080/concept-download/video.mp4"
    };
}

使用@SourceCache注解配合@Download实现下载资源的缓存处理,优先级高于上面两种方式

@SourceCache注解说明
参数 说明
enabled 是否启用缓存
group 分组,会在缓存目录下额外创建一个对应的目录作为实际的缓存目录
考虑到不同功能出现相同名称的文件等冲突问题
默认空,不创建,及直接使用配置的缓存目录
delete 下载结束后是否删除缓存文件

资源压缩

默认情况下,如果是单个资源则不会压缩,如果是多个资源或者是一整个文件夹则会压缩处理

可以使用@Download(forceCompress = true)强制压缩

目前只实现了基于ZipOutputStreamZipSourceCompressor

自定义压缩

可以自定义实现SourceCompressorAbstractSourceCompressor并注入到Spring容器中

/**
 * {@link Source} 压缩器。
 *
 * @see ZipSourceCompressor
 */
public interface SourceCompressor extends OrderProvider {

    /**
     * 获得压缩格式。
     *
     * @return 压缩格式
     */
    String getFormat();

    /**
     * 判断是否支持对应的压缩格式。
     *
     * @param format  压缩格式
     * @param context {@link DownloadContext}
     * @return 如果支持则返回 true
     */
    default boolean support(String format, DownloadContext context) {
        return format.equalsIgnoreCase(getFormat());
    }

    /**
     * 如果支持对应的格式就会调用该方法执行压缩。
     *
     * @param source  {@link Source}
     * @param writer  {@link DownloadWriter}
     * @param context {@link DownloadContext}
     * @return {@link Compression}
     */
    Compression compress(Source source, DownloadWriter writer, DownloadContext context);
}

压缩缓存

配置文件配置

concept:
  download:
    compress:
      cache:
        enabled: true #是否启用
        path: / #缓存目录
        delete: false #下载结束后是否删除

代码全局配置

@Configuration
public class ConceptDownloadConfig implements DownloadConfigurer {

    @Override
    public void configure(DownloadConfiguration configuration) {
        configuration.getCompress().getCache().setEnabled(true);
        configuration.getCompress().getCache().setPath("缓存路径");
        configuration.getCompress().getCache().setDelete(false);
    }

注解配置单个方法

@Download(filename = "压缩包.zip")
@CompressCache(group = "compress", delete = true)
@GetMapping("/compress-cache")
public String[] compressCache() {
    return new String[]{
          "http://127.0.0.1:8080/concept-download/text.txt",
          "http://127.0.0.1:8080/concept-download/image.jpg",
          "http://127.0.0.1:8080/concept-download/video.mp4"
    };
}

使用@CompressCache注解配合@Download实现压缩文件的缓存处理,优先级高于上面两种方式

@CompressCache注解说明
参数 说明
enabled 是否启用缓存
group 分组,会在缓存目录下额外创建一个对应的目录作为实际的缓存目录
考虑到不同功能出现相同名称的文件等冲突问题
默认空,不创建,及直接使用配置的缓存目录
name 压缩文件名称
单下载源会使用该下载源的名称
多下载源会使用第一个有名称的下载源的名称
否则使用CacheNameGenerator生成,默认使用时间戳
delete 下载结束后是否删除缓存文件

响应写入

这部分是对输入输出流的具体操作实现

默认实现BufferedDownloadWriter来操作字节流或字符流

自定义写入器

可以自定义实现DownloadWriter并注入到Spring容器中

/**
 * 具体操作 {@link InputStream} 和 {@link OutputStream} 的写入器。
 */
public interface DownloadWriter extends OrderProvider {

    /**
     * 该写入器是否支持写入。
     *
     * @param resource {@link Resource}
     * @param range    {@link Range}
     * @param context  {@link DownloadContext}
     * @return 如果支持则返回 true
     */
    boolean support(Resource resource, Range range, DownloadContext context);

    /**
     * 执行写入。
     *
     * @param is      {@link InputStream}
     * @param os      {@link OutputStream}
     * @param range   {@link Range}
     * @param charset {@link Charset}
     * @param length  总大小,可能为 null
     */
    default void write(InputStream is, OutputStream os, Range range, Charset charset, Long length) {
        write(is, os, range, charset, length, null);
    }

    /**
     * 执行写入。
     *
     * @param is       {@link InputStream}
     * @param os       {@link OutputStream}
     * @param range    {@link Range}
     * @param charset  {@link Charset}
     * @param length   总大小,可能为 null
     * @param callback 回调当前进度和增长的大小
     */
    void write(InputStream is, OutputStream os, Range range, Charset charset, Long length, Callback callback);

    /**
     * 进度回调。
     */
    interface Callback {

        /**
         * 回调进度。
         *
         * @param current  当前值
         * @param increase 增长值
         */
        void onWrite(long current, long increase);
    }
}

事件

在下载的过程中会发布一系列事件

事件类型

事件 说明
AfterContextInitializedEvent 上下文初始化之后发布
AfterContextDestroyedEvent 上下文销毁之后发布
{Source}CreatedEvent 各种Source创建之后发布
AfterSourceCreatedEvent 所有Source创建之后发布
SourceCacheDeletedEvent Source缓存删除之后发布
SourceReleasedEvent Source资源释放之后发布
SourceAlreadyLoadedEvent 重复加载时发布
SourceLoadedCacheUsedEvent 加载使用缓存时发布
Load{HttpSource}Event 需要网络请求加载时发布
SourceLoadingProgressEvent 加载进度更新时发布
AfterSourceLoadedEvent 所有加载完成之后发布
SourceCompressedCacheUsedEvent 压缩使用缓存时发布
SourceCompressionFormatEvent 压缩格式确定时发布
SourceNoCompressionEvent 不压缩时发布
SourceMemoryCompressionEvent 使用内存压缩时发布
SourceFileCompressionEvent 使用文件压缩时发布
SourceCompressingProgressEvent 压缩进度更新时发布
AfterSourceCompressedEvent 压缩完成之后发布
CompressionCacheDeletedEvent 压缩缓存删除之后发布
CompressionReleasedEvent 压缩资源释放之后发布
ResponseWritingProgressEvent 响应写入进度更新时发布
AfterResponseWrittenEvent 响应写入之后发布

事件监听

可以通过实现DownloadEventListener来监听事件

同时支持Spring的事件监听

自定义事件发布者

可以自定义实现DownloadEventPublisher并注入Spring容器中,需要自己实现支持Spring的事件监听

/**
 * {@link DownloadEvent} 发布器。
 *
 * @see SimpleDownloadEventPublisher
 * @see ApplicationDownloadEventPublisher
 */
public interface DownloadEventPublisher {

    /**
     * 发布事件。
     *
     * @param event 事件
     */
    void publish(Object event);
}

控制台日志

日志通过特定的DownloadEventListener实现

日志类型

日志 说明
StandardDownloadLogger 标准流程日志,每个流程相关的事件都会打印
ProgressCalculationLogger 进度计算日志,包括加载进度,压缩进度,响应写入进度
TimeSpentCalculationLogger 时间计算日志

配置文件配置

concept:
  download:
    logger:
      enabled: true #日志总开关
      standard:
        enabled: true #标准流程日志是否启用
      progress:
        enabled: true #进度计算日志是否启用,包括加载进度,压缩进度,写入响应进度
        duration: 500 #进度计算日志输出间隔,ms
        percentage: true #进度计算日志是否使用百分比输出
      time-spent:
        enabled: true #时间计算日志是否启用

代码全局配置

@Configuration
public class ConceptDownloadConfig implements DownloadConfigurer {

    @Override
    public void configure(DownloadConfiguration configuration) {
        configuration.getLogger().setEnabled(true);
        configuration.getLogger().getStandard().setEnabled(true);
        configuration.getLogger().getProgress().setEnabled(true);
        configuration.getLogger().getProgress().setDuration(500);
        configuration.getLogger().getProgress().setPercentage(true);
        configuration.getLogger().getTimeSpent().setEnabled(true);
    }

缓存名称

默认将使用Source的名称或当前时间戳作为名称

自定义缓存名称

可以自定义实现CacheNameGeneratorAbstractCacheNameGenerator并注入到Spring容器中

Range支持

还在测试阶段,可以使用但可能存在问题

Bug

  • 当加载失败或压缩失败时,会遗留错误的缓存文件,预计下个版本修复

Todo

  • 考虑可以根据数据大小动态返回是否需要压缩或缓存的配置,具体看是否有需求

版本

列表

1.2.2
  • 修复设置自定义DownloadResponse时空指针异常

依赖

core source-okhttp load-coroutines
1.2.0 1.2.0 1.2.0
1.2.2 - -
Clone this wiki locally