Tio Boot DocsTio Boot Docs
Home
  • java-db
  • api-table
  • Enjoy
  • Tio Boot Admin
  • ai_agent
  • translator
  • knowlege_base
  • ai-search
  • 案例
Abount
  • Github
  • Gitee
Home
  • java-db
  • api-table
  • Enjoy
  • Tio Boot Admin
  • ai_agent
  • translator
  • knowlege_base
  • ai-search
  • 案例
Abount
  • Github
  • Gitee
  • 01_tio-boot 简介

    • tio-boot:新一代高性能 Java Web 开发框架
    • tio-boot 入门示例
    • Tio-Boot 配置 : 现代化的配置方案
    • tio-boot 整合 Logback
    • tio-boot 整合 hotswap-classloader 实现热加载
    • 自行编译 tio-boot
    • 最新版本
    • 开发规范
  • 02_部署

    • 使用 Maven Profile 实现分环境打包 tio-boot 项目
    • Maven 项目配置详解:依赖与 Profiles 配置
    • tio-boot 打包成 FastJar
    • 使用 GraalVM 构建 tio-boot Native 程序
    • 使用 Docker 部署 tio-boot
    • 部署到 Fly.io
    • 部署到 AWS Lambda
    • 到阿里云云函数
    • 使用 Deploy 工具部署
    • 使用Systemctl启动项目
    • 使用 Jenkins 部署 Tio-Boot 项目
    • 使用 Nginx 反向代理 Tio-Boot
    • 使用 Supervisor 管理 Java 应用
    • 已过时
    • 胖包与瘦包的打包与部署
  • 03_配置

    • 配置参数
    • 服务器监听器
    • 内置缓存系统 AbsCache
    • 使用 Redis 作为内部 Cache
    • 静态文件处理器
    • 基于域名的静态资源隔离
    • DecodeExceptionHandler
  • 04_原理

    • 生命周期
    • 请求处理流程
    • 重要的类
  • 05_json

    • Json
    • 接受 JSON 和响应 JSON
    • 响应实体类
  • 06_web

    • 概述
    • 文件上传
    • 接收请求参数
    • 接收日期参数
    • 接收数组参数
    • 返回字符串
    • 返回文本数据
    • 返回网页
    • 请求和响应字节
    • 文件下载
    • 返回视频文件并支持断点续传
    • http Session
    • Cookie
    • HttpRequest
    • HttpResponse
    • Resps
    • RespBodyVo
    • 拦截器
    • 全局异常处理器
    • 异步
    • 动态 返回 CSS 实现
    • 返回图片
    • Transfer-Encoding: chunked 实时音频播放
    • Server-Sent Events (SSE)
    • 接口访问统计
    • 接口请求和响应数据记录
    • 自定义 Handler 转发请求
    • 使用 HttpForwardHandler 转发所有请求
    • 跨域
    • 添加 Controller
    • 常用工具类
    • HTTP Basic 认证
    • Http响应加密
    • WebJars
    • JProtobuf
  • 07_validate

    • 数据紧校验规范
    • 参数校验
  • 08_websocket

    • 使用 tio-boot 搭建 WebSocket 服务
    • WebSocket 聊天室项目示例
  • 09_java-db

    • java‑db
    • 操作数据库入门示例
    • SQL 模板
    • 数据源配置与使用
    • ActiveRecord
    • Model
    • 生成器与 Model
    • Db 工具类
    • 批量操作
    • 数据库事务处理
    • Cache 缓存
    • Dialect 多数据库支持
    • 表关联操作
    • 复合主键
    • Oracle 支持
    • Enjoy SQL 模板
    • Java-DB 整合 Enjoy 模板最佳实践
    • 多数据源支持
    • 独立使用 ActiveRecord
    • 调用存储过程
    • java-db 整合 Guava 的 Striped 锁优化
    • 生成 SQL
    • 通过实体类操作数据库
    • java-db 读写分离
    • Spring Boot 整合 Java-DB
    • like 查询
    • 常用操作示例
    • Druid 监控集成指南
    • SQL 统计
  • 10_api-table

    • ApiTable 概述
    • 使用 ApiTable 连接 SQLite
    • 使用 ApiTable 连接 Mysql
    • 使用 ApiTable 连接 Postgres
    • 使用 ApiTable 连接 TDEngine
    • 使用 api-table 连接 oracle
    • 使用 api-table 连接 mysql and tdengine 多数据源
    • EasyExcel 导出
    • EasyExcel 导入
    • TQL(Table SQL)前端输入规范
    • ApiTable 实现增删改查
    • 数组类型
    • 单独使用 ApiTable
  • 11_aop

    • JFinal-aop
    • Aop 工具类
    • 配置
    • 配置
    • 独立使用 JFinal Aop
    • @AImport
    • 原理解析
  • 12_cache

    • Caffine
    • Jedis-redis
    • hutool RedisDS
    • Redisson
    • Caffeine and redis
    • CacheUtils 工具类
    • 使用 CacheUtils 整合 caffeine 和 redis 实现的两级缓存
    • 使用 java-db 整合 ehcache
    • 使用 java-db 整合 redis
    • Java DB Redis 相关 Api
    • redis 使用示例
  • 13_认证和权限

    • hutool-JWT
    • FixedTokenInterceptor
    • 使用内置 TokenManager 实现登录
    • 用户系统
    • 重置密码
    • 匿名登录
    • Google 登录
    • 权限校验注解
    • Sa-Token
    • sa-token 登录注册
    • StpUtil.isLogin() 源码解析
    • 短信登录
    • 移动端微信登录实现指南
    • 移动端重置密码
  • 14_i18n

    • i18n
  • 15_enjoy

    • tio-boot 整合 Enjoy 模版引擎文档
    • 引擎配置
    • 表达式
    • 指令
    • 注释
    • 原样输出
    • Shared Method 扩展
    • Shared Object 扩展
    • Extension Method 扩展
    • Spring boot 整合
    • 独立使用 Enjoy
    • tio-boot enjoy 自定义指令 localeDate
    • PromptEngine
    • Enjoy 入门示例-擎渲染大模型请求体
    • Enjoy 使用示例
  • 16_定时任务

    • Quartz 定时任务集成指南
    • 分布式定时任务 xxl-jb
    • cron4j 使用指南
  • 17_tests

    • TioBootTest 类
  • 18_tio

    • TioBootServer
    • tio-core
    • 内置 TCP 处理器
    • 独立启动 UDPServer
    • 使用内置 UDPServer
    • t-io 消息处理流程
    • tio-运行原理详解
    • TioConfig
    • ChannelContext
    • Tio 工具类
    • 业务数据绑定
    • 业务数据解绑
    • 发送数据
    • 关闭连接
    • Packet
    • 监控: 心跳
    • 监控: 客户端的流量数据
    • 监控: 单条 TCP 连接的流量数据
    • 监控: 端口的流量数据
    • 单条通道统计: ChannelStat
    • 所有通道统计: GroupStat
    • 资源共享
    • 成员排序
    • SSL
    • DecodeRunnable
    • 使用 AsynchronousSocketChannel 响应数据
    • 拉黑 IP
    • 深入解析 Tio 源码:构建高性能 Java 网络应用
  • 19_aio

    • ByteBuffer
    • AIO HTTP 服务器
    • 自定义和线程池和池化 ByteBuffer
    • AioHttpServer 应用示例 IP 属地查询
    • 手写 AIO Http 服务器
  • 20_netty

    • Netty TCP Server
    • Netty Web Socket Server
    • 使用 protoc 生成 Java 包文件
    • Netty WebSocket Server 二进制数据传输
    • Netty 组件详解
  • 21_netty-boot

    • Netty-Boot
    • 原理解析
    • 整合 Hot Reload
    • 整合 数据库
    • 整合 Redis
    • 整合 Elasticsearch
    • 整合 Dubbo
    • Listener
    • 文件上传
    • 拦截器
    • Spring Boot 整合 Netty-Boot
    • SSL 配置指南
    • ChannelInitializer
    • Reserve
  • 22_MQ

    • Mica-mqtt
    • EMQX
    • Disruptor
  • 23_tio-utils

    • tio-utils
    • HttpUtils
    • Notification
    • 邮箱
    • JSON
    • 读取文件
    • Base64
    • 上传和下载
    • Http
    • Telegram
    • RsaUtils
    • EnvUtils 使用文档
    • 系统监控
    • 毫秒并发 ID (MCID) 生成方案
  • 24_tio-http-server

    • 使用 Tio-Http-Server 搭建简单的 HTTP 服务
    • tio-boot 添加 HttpRequestHandler
    • 在 Android 上使用 tio-boot 运行 HTTP 服务
    • tio-http-server-native
    • handler 常用操作
  • 25_tio-websocket

    • WebSocket 服务器
    • WebSocket Client
  • 26_tio-im

    • 通讯协议文档
    • ChatPacket.proto 文档
    • java protobuf
    • 数据表设计
    • 创建工程
    • 登录
    • 历史消息
    • 发消息
  • 27_mybatis

    • Tio-Boot 整合 MyBatis
    • 使用配置类方式整合 MyBatis
    • 整合数据源
    • 使用 mybatis-plus 整合 tdengine
    • 整合 mybatis-plus
  • 28_mongodb

    • tio-boot 使用 mongo-java-driver 操作 mongodb
  • 29_elastic-search

    • Elasticsearch
    • JavaDB 整合 ElasticSearch
    • Elastic 工具类使用指南
    • Elastic-search 注意事项
    • ES 课程示例文档
  • 30_magic-script

    • tio-boot 与 magic-script 集成指南
  • 31_groovy

    • tio-boot 整合 Groovy
  • 32_firebase

    • 整合 google firebase
    • Firebase Storage
    • Firebase Authentication
    • 使用 Firebase Admin SDK 进行匿名用户管理与自定义状态标记
    • 导出用户
    • 注册回调
    • 登录注册
  • 33_文件存储

    • 文件上传数据表
    • 本地存储
    • 使用 AWS S3 存储文件并整合到 Tio-Boot 项目中
    • 存储文件到 腾讯 COS
  • 34_spider

    • jsoup
    • 爬取 z-lib.io 数据
    • 整合 WebMagic
    • WebMagic 示例:爬取学校课程数据
    • Playwright
    • Flexmark (Markdown 处理器)
    • tio-boot 整合 Playwright
    • 缓存网页数据
  • 36_integration_thirty_party

    • tio-boot 整合 okhttp
    • 整合 GrpahQL
    • 集成 Mailjet
    • 整合 ip2region
    • 整合 GeoLite 离线库
    • 整合 Lark 机器人指南
    • 集成 Lark Mail 实现邮件发送
    • Thymeleaf
    • Swagger
    • Clerk 验证
  • 37_dubbo

    • 概述
    • dubbo 2.6.0
    • dubbo 2.6.0 调用过程
    • dubbo 3.2.0
  • 38_spring

    • Spring Boot Web 整合 Tio Boot
    • spring-boot-starter-webflux 整合 tio-boot
    • Tio Boot 整合 Spring Boot Starter
    • Tio Boot 整合 Spring Boot Starter Data Redis 指南
  • 39_spring-cloud

    • tio-boot spring-cloud
  • 40_mysql

    • 使用 Docker 运行 MySQL
    • /zh/42_mysql/02.html
  • 41_postgresql

    • PostgreSQL 安装
    • PostgreSQL 主键自增
    • PostgreSQL 日期类型
    • Postgresql 金融类型
    • PostgreSQL 数组类型
    • PostgreSQL 全文检索
    • PostgreSQL 查询优化
    • 获取字段类型
    • PostgreSQL 向量
    • PostgreSQL 优化向量查询
    • PostgreSQL 其他
  • 43_oceanbase

    • 快速体验 OceanBase 社区版
    • 快速上手 OceanBase 数据库单机部署与管理
    • 诊断集群性能
    • 优化 SQL 性能指南
    • /zh/43_oceanbase/05.html
  • 50_media

    • JAVE 提取视频中的声音
    • Jave 提取视频中的图片
    • /zh/50_media/03.html
  • 51_asr

    • Whisper-JNI
  • 54_native-media

    • java-native-media
    • JNI 入门示例
    • mp3 拆分
    • mp4 转 mp3
    • 使用 libmp3lame 实现高质量 MP3 编码
    • Linux 编译
    • macOS 编译
    • 从 JAR 包中加载本地库文件
    • 支持的音频和视频格式
    • 任意格式转为 mp3
    • 通用格式转换
    • 通用格式拆分
    • 视频合并
    • VideoToHLS
    • split_video_to_hls 支持其他语言
    • 持久化 HLS 会话
  • 55_telegram4j

    • 数据库设计
    • /zh/55_telegram4j/02.html
    • 基于 MTProto 协议开发 Telegram 翻译机器人
    • 过滤旧消息
    • 保存机器人消息
    • 定时推送
    • 增加命令菜单
    • 使用 telegram-Client
    • 使用自定义 StoreLayout
    • 延迟测试
    • Reactor 错误处理
    • Telegram4J 常见错误处理指南
  • 56_telegram-bots

    • TelegramBots 入门指南
    • 使用工具库 telegram-bot-base 开发翻译机器人
  • 60_LLM

    • 简介
    • AI 问答
    • /zh/60_LLM/03.html
    • /zh/60_LLM/04.html
    • 增强检索(RAG)
    • 结构化数据检索
    • 搜索+AI
    • 集成第三方 API
    • 后置处理
    • 推荐问题生成
    • 连接代码执行器
    • 避免 GPT 混乱
    • /zh/60_LLM/13.html
  • 61_ai_agent

    • 数据库设计
    • 示例问题管理
    • 会话管理
    • 历史记录
    • 对接 Perplexity API
    • 意图识别与生成提示词
    • 智能问答模块设计与实现
    • 文件上传与解析文档
    • 翻译
    • 名人搜索功能实现
    • Ai studio gemini youbue 问答使用说明
    • 自建 YouTube 字幕问答系统
    • 自建 获取 youtube 字幕服务
    • 通用搜索
    • /zh/61_ai_agent/15.html
    • 16
    • 17
    • 18
    • 在 tio-boot 应用中整合 ai-agent
    • 16
  • 62_translator

    • 简介
  • 63_knowlege_base

    • 数据库设计
    • 用户登录实现
    • 模型管理
    • 知识库管理
    • 文档拆分
    • 片段向量
    • 命中测试
    • 文档管理
    • 片段管理
    • 问题管理
    • 应用管理
    • 向量检索
    • 推理问答
    • 问答模块
    • 统计分析
    • 用户管理
    • api 管理
    • 存储文件到 S3
    • 文档解析优化
    • 片段汇总
    • 段落分块与检索
    • 多文档解析
    • 对话日志
    • 检索性能优化
    • Milvus
    • 文档解析方案和费用对比
    • 离线运行向量模型
  • 64_ai-search

    • ai-search 项目简介
    • ai-search 数据库文档
    • ai-search SearxNG 搜索引擎
    • ai-search Jina Reader API
    • ai-search Jina Search API
    • ai-search 搜索、重排与读取内容
    • ai-search PDF 文件处理
    • ai-search 推理问答
    • Google Custom Search JSON API
    • ai-search 意图识别
    • ai-search 问题重写
    • ai-search 系统 API 接口 WebSocket 版本
    • ai-search 搜索代码实现 WebSocket 版本
    • ai-search 生成建议问
    • ai-search 生成问题标题
    • ai-search 历史记录
    • Discover API
    • 翻译
    • Tavily Search API 文档
    • 对接 Tavily Search
    • 火山引擎 DeepSeek
    • 对接 火山引擎 DeepSeek
    • ai-search 搜索代码实现 SSE 版本
    • jar 包部署
    • Docker 部署
    • 爬取一个静态网站的所有数据
    • 网页数据预处理
    • 网页数据检索与问答流程整合
  • 65_java-linux

    • Java 执行 python 代码
    • 通过大模型执行 Python 代码
    • MCP 协议
    • Cline 提示词
    • Cline 提示词-中文版本
  • 66_manim

    • Teach me anything - 基于大语言的知识点讲解视频生成系统
    • Manim 开发环境搭建
    • 生成场景提示词
    • 生成代码
    • 完整脚本示例
    • 语音合成系统
    • Fish.audio TTS 接口说明文档与 Java 客户端封装
    • 整合 fishaudio 到 java-uni-ai-server 项目
    • 执行 Python (Manim) 代码
    • 使用 SSE 流式传输生成进度的实现文档
    • 整合全流程完整文档
    • HLS 动态推流技术文档
    • manim 分场景生成代码
    • 分场景运行代码及流式播放支持
    • 分场景业务端完整实现流程
    • Maiim布局管理器
    • 仅仅生成场景代码
    • 使用 modal 运行 manim 代码
    • Python 使用 Modal GPU 加速渲染
    • Modal 平台 GPU 环境下运行 Manim
    • Modal Manim OpenGL 安装与使用
    • 优化 GPU 加速
    • 生成视频封面流程
    • Java 调用 manim 命令 执行代码 生成封面
    • Manim 图像生成服务客户端文档
    • manim render help
    • 显示 中文公式
    • manimgl
    • EGL
    • /zh/66_manim/30.html
    • /zh/66_manim/31.html
    • /zh/66_manim/32.html
    • /zh/66_manim/33.html
  • 68_java-llm-proxy

    • 使用tio-boot搭建openai 代理服务
  • 70_tio-boot-admin

    • 入门指南
    • 初始化数据
    • token 存储
    • 与前端集成
    • 文件上传
    • 网络请求
    • 图片管理
    • /zh/70_tio-boot-admin/08.html
    • Word 管理
    • PDF 管理
    • 文章管理
    • 富文本编辑器
  • 71_tio-boot

    • /zh/71_tio-boot/01.html
    • Swagger 整合到 Tio-Boot 中的指南
    • HTTP/1.1 Pipelining 性能测试报告
  • 80_性能测试

    • 压力测试 - tio-http-serer
    • 压力测试 - tio-boot
    • 压力测试 - tio-boot-native
    • 压力测试 - netty-boot
    • 性能测试对比
    • TechEmpower FrameworkBenchmarks
    • 压力测试 - tio-boot 12 C 32G
  • 99_案例

    • 封装 IP 查询服务
    • tio-boot 案例 - 全局异常捕获与企业微信群通知
    • tio-boot 案例 - 文件上传和下载
    • tio-boot 案例 - 整合 ant design pro 增删改查
    • tio-boot 案例 - 流失响应
    • tio-boot 案例 - 增强检索
    • tio-boot 案例 - 整合 function call
    • tio-boot 案例 - 定时任务 监控 PostgreSQL、Redis 和 Elasticsearch
    • Tio-Boot 案例:使用 SQLite 整合到登录注册系统
    • tio-boot 案例 - 执行 shell 命令

视频下载增加水印说明文档

  • 1. VideoWaterHandler 类
    • 代码说明
  • 2. VideoWaterUtils 类
    • 代码说明
  • 3. 测试与使用说明
  • 4. 总结

该功能提供了一个 HTTP 接口用于视频下载,同时在视频中增加右下角的水印。主要流程如下:

  1. 客户端通过 URL 发起请求,示例如下:

    http://localhost/video/download/water?path=/data/07/videos/1080p30/CombinedScene.mp4&text=videotutor.io&filename=什么三角函数.mp4
    

    请求参数说明:

    • path:视频文件存储的相对或绝对路径(本例中拼接了 “.” 作为根目录)。
    • text:水印文字内容,将在视频右下角显示。
    • filename:下载时客户端保存的视频文件名(通过设置 Content-Disposition 头实现)。
  2. 如果请求中包含水印文本,将先检查目标视频是否已经添加过相同文本的水印。

    • 使用 MD5 值生成唯一标识。
    • 若不存在,则调用 VideoWaterUtils.addWatermark 方法通过 ffmpeg 执行水印添加操作,生成新的视频文件。
  3. 返回视频时支持 HTTP 的 Range 请求(即断点续传),如果包含 Range 信息,则只返回对应字节范围的内容。

    • 同时设置响应头如 Accept-Ranges、Content-Range 和 Content-Disposition(当 filename 参数不为空时)。
    • 注意视频文件本身已经是压缩格式,故禁用 gzip 压缩以避免解码错误。

下面是完整的代码及说明。


1. VideoWaterHandler 类

该类为 HTTP 请求处理类,主要负责接收客户端请求,根据请求参数对视频进行处理并返回响应结果。代码详细流程如下:

  • 参数获取与校验
    获取请求参数 path、text 和 filename。如果 path 为空则直接返回提示信息,同时判断指定的视频文件是否存在,不存在则返回 404 状态码。

  • 获取文件后缀与内容类型
    根据文件名解析出后缀,并使用工具类获取对应的 MIME 类型,使浏览器能够识别并正确处理返回数据。

  • 视频水印处理
    如果请求中包含 text 参数,则生成该文本的 MD5 值来构造输出文件名,使用 VideoWaterUtils.addWatermark 方法调用外部 ffmpeg 命令实现水印添加。

  • 分段读取支持断点续传
    如果请求头中存在 Range 参数,则使用 RandomAccessFile 按照指定的字节范围读取文件内容,并设置 Content-Range 及 Accept-Ranges 响应头,状态码置为 206(Partial Content)。否则直接读出全部文件内容。

  • 设置下载文件名
    无论在 Range 分支还是完整文件分支,如果传入了 filename 参数,都设置 Content-Disposition 头,令浏览器按照指定文件名保存文件。

  • 禁用 gzip 压缩
    由于视频文件已经是压缩格式,启用 gzip 压缩可能导致浏览器解码异常,因此调用 response.setHasGzipped(true)。

完整代码如下:

package com.litongjava.linux.handler;

import java.io.File;
import java.io.IOException;
import java.io.RandomAccessFile;

import com.litongjava.media.utils.VideoWaterUtils;
import com.litongjava.tio.boot.http.TioRequestContext;
import com.litongjava.tio.http.common.HttpRequest;
import com.litongjava.tio.http.common.HttpResponse;
import com.litongjava.tio.http.common.ResponseHeaderKey;
import com.litongjava.tio.http.server.util.CORSUtils;
import com.litongjava.tio.http.server.util.Resps;
import com.litongjava.tio.utils.crypto.Md5Utils;
import com.litongjava.tio.utils.http.ContentTypeUtils;
import com.litongjava.tio.utils.hutool.FileUtil;
import com.litongjava.tio.utils.hutool.FilenameUtils;
import com.litongjava.tio.utils.hutool.StrUtil;

public class VideoWaterHandler {

  public HttpResponse index(HttpRequest request) {
    HttpResponse response = TioRequestContext.getResponse();
    CORSUtils.enableCORS(response);
    String path = request.getString("path");
    String text = request.getString("text");
    String filename = request.getString("filename");

    if (StrUtil.isBlank(path)) {
      return response.setString("path can not be empty");
    }

    String targetFile = "." + path;
    File file = new File(targetFile);
    if (!file.exists()) {
      response.setStatus(404);
      return response;
    }

    String suffix = FilenameUtils.getSuffix(path);
    String contentType = ContentTypeUtils.getContentType(suffix);

    if (StrUtil.isNotBlank(text)) {
      String md5 = Md5Utils.getMD5(text);
      String subPath = FilenameUtils.getSubPath(targetFile);
      String baseName = FilenameUtils.getBaseName(targetFile);
      String outputFile = subPath + File.separator + baseName + "_" + md5 + "." + suffix;
      file = new File(outputFile);
      if (!file.exists()) {
        try {
          VideoWaterUtils.addWatermark(targetFile, outputFile, 48, text);
          targetFile = outputFile;
        } catch (IOException e) {
          e.printStackTrace();
          return response.setString(e.getMessage());
        } catch (InterruptedException e) {
          e.printStackTrace();
          return response.setString(e.getMessage());
        }
      }
    }

    long fileLength = file.length();
    // 检查是否存在 Range 头信息
    String range = request.getHeader("Range");
    if (range != null && range.startsWith("bytes=")) {
      String rangeValue = range.substring("bytes=".length());
      String[] parts = rangeValue.split("-");
      try {
        long start = parts[0].isEmpty() ? 0 : Long.parseLong(parts[0]);
        long end = (parts.length > 1 && !parts[1].isEmpty()) ? Long.parseLong(parts[1]) : fileLength - 1;
        if (start > end || end >= fileLength) {
          response.setStatus(416);
          return response;
        }
        long contentLength = end - start + 1;
        byte[] data = new byte[(int) contentLength];

        try (RandomAccessFile raf = new RandomAccessFile(file, "r")) {
          raf.seek(start);
          raf.readFully(data);
        }
        // 设置响应头
        response.setStatus(206); // Partial Content
        response.setHeader("Content-Range", "bytes " + start + "-" + end + "/" + fileLength);
        response.setHeader("Accept-Ranges", "bytes");
        response.setHeader(ResponseHeaderKey.Content_Length, String.valueOf(contentLength));
        // 如果传入了 filename,则在响应头中指定下载文件名
        if (StrUtil.isNotBlank(filename)) {
          response.setHeader("Content-Disposition", "attachment; filename=\"" + filename + "\"");
        }
        Resps.bytesWithContentType(response, data, contentType);
      } catch (Exception e) {
        response.setStatus(416);
      }
    } else {
      // 如果没有 Range 头,则直接返回整个文件
      byte[] readBytes = FileUtil.readBytes(file);
      response.setHeader("Accept-Ranges", "bytes");
      if (StrUtil.isNotBlank(filename)) {
        response.setHeader("Content-Disposition", "attachment; filename=\"" + filename + "\"");
      }
      Resps.bytesWithContentType(response, readBytes, contentType);
    }
    // 视频文件(如 mp4)本身已经是压缩格式,再进行 gzip 压缩可能会破坏文件格式,导致浏览器无法正确解码。
    response.setHasGzipped(true);
    return response;
  }
}

代码说明

  • CORS 支持
    调用 CORSUtils.enableCORS(response) 后,使前端跨域请求得到支持。

  • 参数与文件检查
    使用 StrUtil.isBlank(path) 校验参数,并通过 new File(targetFile) 检查目标文件是否存在。

  • 水印处理逻辑
    当 text 不为空时,利用 MD5 值与文件基本名称组合生成新的输出文件名。如果对应的水印视频文件尚不存在,则调用 VideoWaterUtils.addWatermark 完成水印添加操作。若过程中产生异常,则返回错误信息。

  • Range 分支处理
    根据 Range 请求头解析出起始与结束字节,使用 RandomAccessFile 按指定范围读取字节,构造部分视频数据,设置相应响应头(例如 Content-Range 和 Content-Length),并返回状态码 206。

  • 完整内容返回
    若未检测到 Range 请求,则通过工具方法一次性读取全部文件字节数据,并返回。同时也设置了 Content-Disposition 用于指定下载文件名。

  • 禁止 gzip 压缩
    使用 response.setHasGzipped(true) 指定视频文件在传输过程中不做额外 gzip 压缩。


2. VideoWaterUtils 类

该工具类负责通过外部命令行工具 ffmpeg 为视频添加水印。关键实现点如下:

  • 操作系统判断与字体选择
    根据当前操作系统类型(Windows、macOS、Linux/Unix),选择对应的字体文件。不同操作系统中常见的字体文件路径不同,确保水印中文显示正常。

  • 构造 drawtext 过滤器
    使用 drawtext 过滤器向视频中添加文本水印,其参数解释如下:

    • fontfile:指定使用的字体文件路径。
    • text:水印文本(传入参数)。
    • x=w-tw-10:y=h-th-10:设置水印位置为视频右下角,并距离边缘 10 像素。
    • fontsize:文字字号。
    • fontcolor:字体颜色,此例设置为黄色。
  • 构造 ffmpeg 命令参数
    将以上参数拼接为 ffmpeg 命令的参数列表,命令中保留原音频数据(-codec:a copy),并将视频输出到指定文件中。

  • 输出重定向
    将 ffmpeg 的标准输出及错误输出分别重定向到 ffmpeg_stdout.log 和 ffmpeg_stderr.log 文件中,便于后续调试与日志查看。

  • 启动进程并等待完成
    使用 ProcessBuilder 启动 ffmpeg 进程并等待该进程结束,返回命令执行结果状态码。

完整代码如下:

package com.litongjava.media.utils;

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

public class VideoWaterUtils {

  /**
   * 使用 ffmpeg 给视频添加右下角水印,并将标准输出和错误输出分别写入日志文件。
   *
   * @param inputFile     输入视频文件路径
   * @param outputFile    输出视频文件路径
   * @param fontSize      水印文字的字号
   * @param watermarkText 水印文本
   * @throws IOException          当执行命令时发生 I/O 错误
   * @throws InterruptedException 当线程等待 ffmpeg 进程结束时被中断
   */
  public static int addWatermark(String inputFile, String outputFile, int fontSize, String watermarkText) throws IOException, InterruptedException {

    String osName = System.getProperty("os.name").toLowerCase();
    String fontFile;
    if (osName.contains("win")) {
      fontFile = "C\\:/Windows/Fonts/simhei.ttf";
    } else if (osName.contains("mac")) {
      fontFile = "/Library/Fonts/Arial Unicode.ttf";
    } else if (osName.contains("nix") || osName.contains("nux") || osName.contains("aix")) {
      fontFile = "/usr/share/fonts/truetype/noto/NotoSansCJK-Regular.ttc";
    } else {
      fontFile = "/usr/share/fonts/truetype/noto/NotoSansCJK-Regular.ttc";
    }

    // 构造 drawtext 过滤器参数
    // x=w-tw-10:y=h-th-10 表示让水印距离右下角各留10像素
    String template = "drawtext=fontfile='%s':text='%s':x=w-tw-10:y=h-th-10:fontsize=%d:fontcolor=yellow";
    String filterSpec = String.format(template, fontFile, watermarkText, fontSize);

    // 构造 ffmpeg 命令参数列表
    List<String> command = new ArrayList<>();
    command.add("ffmpeg");
    command.add("-i");
    command.add(inputFile);
    command.add("-vf");
    command.add(filterSpec);
    command.add("-codec:a");
    command.add("copy");
    command.add(outputFile);

    System.out.println("cmd:" + String.join(" ", command));

    ProcessBuilder pb = new ProcessBuilder(command);

    // 设置将标准输出和错误输出分别重定向到文件
    File stdoutFile = new File("ffmpeg_stdout.log");
    File stderrFile = new File("ffmpeg_stderr.log");
    pb.redirectOutput(ProcessBuilder.Redirect.to(stdoutFile));
    pb.redirectError(ProcessBuilder.Redirect.to(stderrFile));

    // 启动进程并等待完成
    Process process = pb.start();
    return process.waitFor();
  }
}

代码说明

  • 字体文件选择
    根据系统平台自动选择对应字体的路径,确保水印中文和特殊字符能够正常显示。

    注:对于 Windows 平台使用的路径格式为 C\\:/Windows/Fonts/simhei.ttf。

  • drawtext 参数构造
    通过 String.format 方式构造 ffmpeg 的 drawtext 滤镜参数,将用户传入的水印文本、字号、字体文件嵌入命令行参数中。

  • 日志输出重定向
    利用 ProcessBuilder 的 redirectOutput 和 redirectError 方法将 ffmpeg 的输出写入日志文件,这对于调试和排查问题非常有帮助。

  • 执行等待
    使用 process.waitFor() 阻塞等待外部进程执行完毕,并返回运行状态码。


3. 测试与使用说明

  • 前置条件

    • 服务器需预先安装 ffmpeg 命令行工具。
    • 系统中需要包含所选字体,确保对应路径有效。如果在特定平台上字体路径有所不同,请修改 VideoWaterUtils.addWatermark 中的字体路径。
  • 接口测试
    在浏览器或使用 HTTP 客户端(如 Postman)访问以下 URL 示例:

    http://localhost/video/download/water?path=/data/07/videos/1080p30/CombinedScene.mp4&text=videotutor.io&filename=什么三角函数.mp4
    
    • 若指定的 text 参数存在,则服务器会先生成一个添加水印的视频文件(命名规则为 {原文件名}_{水印文本MD5}.{后缀})。
    • 返回的 HTTP 响应将设置 Content-Disposition 头,从而使得浏览器下载时默认文件名为“什么三角函数.mp4”。
    • 同时支持 Range 请求,能够支持视频的断点续传播放。
  • 日志查看
    运行过程中 ffmpeg 的标准输出和错误输出均被重定向到当前目录下的 ffmpeg_stdout.log 与 ffmpeg_stderr.log 文件,便于调试视频处理的过程。


4. 总结

该实现结合了 HTTP 请求处理和基于 ffmpeg 的视频后处理逻辑,实现了视频水印添加以及支持断点续传下载。文档中详细介绍了代码逻辑、关键参数设置及相关说明,开发人员可以根据具体需求进行调整和扩展。

以上即为完整的“视频下载增加水印”的实现文档和代码示例。

Edit this page
Last Updated:
Contributors: Tong Li