在客户端直接上传文件到OSS

客户端直传是指客户端直接上传文件到对象存储OSS。相对于服务端代理上传,客户端直传避免了业务服务器中转文件,提高了上传速度,节省了服务器资源。本文介绍客户端直传的方案优势、安全实现和实践参考。

为什么客户端直传

在典型的服务端和客户端架构下,常见的文件上传方式是服务端代理上传:客户端将文件上传到业务服务器,然后业务服务器将文件上传到OSS。在这个过程中,一份数据需要在网络上传输两次,会造成网络资源的浪费,增加服务端的资源开销。为了解决这一问题,您可以在客户端直连OSS来完成文件上传,无需经过业务服务器中转。

如何实现客户端直传

实现客户端直传需要解决以下两个大问题:

跨域访问

如果您的客户端是Web端或小程序,您需要解决跨域访问被限制的问题。浏览器以及小程序容器出于安全考虑,通常都会限制跨域访问,这一限制也会限制您的客户端代码直连OSS。您可以通过配置OSS Bucket的跨域访问规则,来允许指定域名下的Web应用或小程序直接访问OSS。更多信息,请参见跨域设置

安全授权

上传文件到OSS需要使用RAM用户的访问密钥(AccessKey)来完成签名认证,但是在客户端中使用长期有效的访问密钥,可能会导致访问密钥泄露,进而引起安全问题。为了解决这一问题,如下是一个方案实现安全上传:

服务端生成PutObject所需的签名****URL

对于简单上传文件的场景,您可以在服务端使用OSS SDK生成PutObject所需的签名URL,客户端可以凭借签名URL,不依赖OSS SDK直接上传文件。需要注意的是,此方案不适用于基于分片上传大文件、基于分片断点续传的场景。在服务端对每个分片生成签名URL,并将签名URL返回给客户端,会增加与服务端的交互次数和网络请求的复杂性。另外,客户端可能会修改分片的内容或顺序,导致最终合并的文件不正确。

服务端通过签名URL授权客户端上传文件到OSS的过程如下。

  1. 客户端向业务服务器请求签名URL。
  2. 业务服务器使用OSS SDK生成PUT类型的签名URL,然后将其返回给客户端。
  3. 客户端使用PUT类型的签名URL调用PutObject上传文件到OSS。
  4. OSS向客户端返回成功响应。

示例代码:

java 复制代码
    /**
     * 生成用于直传的 PUT 类型的预签名 URL。
     * 客户端必须在上传时设置与签名一致的 Content-Type。
     * 具体看 OSS 的官方文档,有详细的案例
     *
     * @param objectKey 目标对象键
     * @param contentType 上传内容类型(如 text/markdown, image/png)
     * @param expiresInSeconds 有效期秒数(建议 300-900)
     * @return 可直接用于 PUT 上传的预签名 URL
     */
    public String generatePresignedPutUrl(String objectKey, String contentType, int expiresInSeconds) {
        // 确保配置文件中,相关配置已正常配置,若未配置则抛出异常
        ensureConfigured();

        // 初始化 OSS 客户端实例
        OSS client = new OSSClientBuilder()
                .build(props.getEndpoint(), props.getAccessKeyId(), props.getAccessKeySecret());

        try {
            GeneratePresignedUrlRequest request = new GeneratePresignedUrlRequest(props.getBucket(), objectKey, HttpMethod.PUT);

            // System.currentTimeMillis() → 返回当前时间距离 1970-01-01 00:00:00 UTC 的毫秒数(一个 long 类型的值)
            // expiresInSeconds * 1000L → 把秒数转换成毫秒数(因为时间戳是用毫秒计量的,所以要 ×1000)
            // 两者相加 → 得到未来某个时刻的时间戳(毫秒)
            // new Date(那个毫秒值) → 创建一个代表"那个未来时刻"的 Date 对象
            Date expiration = new Date(System.currentTimeMillis() + expiresInSeconds * 1000L);
            // 设置过期时间(文档中说,单位是毫秒)
            request.setExpiration(expiration);

            // 客户端必须在上传时设置与签名一致的 Content-Type
            if (contentType != null && !contentType.isBlank()) {
                request.setContentType(contentType);
            }

            // 生成用于直传的 PUT 类型的预签名 URL
            URL url = client.generatePresignedUrl(request);
            // 返回
            return url.toString();
        } finally {
            // 最后释放资源
            client.shutdown();
        }
    }
相关推荐
明夜之约15 小时前
Spring Boot 自动装配源码
java·spring boot·后端
Leaton Lee15 小时前
Spring Boot分层架构详解:从Controller到Service再到Mapper的完整流程
java·spring boot·后端·架构
Micro麦可乐15 小时前
Spring Boot 实战:从零设计一个短链系统(含完整代码与数据库设计)
数据库·spring boot·后端·哈希算法·雪花算法·短链系统
Jinkxs15 小时前
Resilience4j- 与 Spring Boot 快速集成:自动配置与基础注解使用
java·spring boot·后端
毕设源码_郑学姐15 小时前
计算机毕业设计springboot网络相册设计与实现 基于Spring Boot框架的在线相册管理系统开发与应用 Spring Boot驱动的网络影集设计与实践
spring boot·后端·课程设计
辣机小司15 小时前
【踩坑记录:Spring Boot 配置文件读取值不一致?警惕 YAML 的“八进制陷阱”与 SnakeYAML 版本之谜】
java·spring boot·后端·yaml·踩坑记录
码农阿豪15 小时前
从零到一:Spring Boot快速接入金仓数据库实战
数据库·spring boot·后端
追逐时光者15 小时前
一个基于 .NET 与 Avalonia 构建、面向 TrinityCore 的开源 WoW 数据库编辑器
后端·.net
fangdengfu12316 小时前
ES分析系统各个服务日志占用量
java·前端·elasticsearch
追逐时光者16 小时前
精选 5 款基于 .NET 开源免费、功能强大的 Windows 系统优化工具
后端·.net