阿里云OSS使用预签名URL上传文件,状态码返回403,报SignatureDoesNotMatch异常

1. 背景

  • 前端使用 Vue.js,后端使用 Springboot 和 aliyun-sdk-oss
  • 使用阿里云 OSS 使用预签名URL上传文件,异常返回 SignatureDoesNotMatch 错误码 0002-00000201。
  • 异常信息如下:
xml 复制代码
<?xml version="1.0" encoding="UTF-8"?>
<Error>
    <Code>SignatureDoesNotMatch</Code>
    <Message>The request signature we calculated does not match the signature you provided. Check your key and signing method.</Message>
    
    <!-- 省略部分内容 -->

    <EC>0002-00000201</EC>
    <RecommendDoc>https://api.aliyun.com/troubleshoot?q=0002-00000201</RecommendDoc>
</Error>

2. 排查

  1. 根据 api.aliyun.com/troubleshoo... 逐一确认,Endpoint,AccessKey ID,AccessKey Secret 等参数均无误,排除配置参数问题
  2. 查看文档 Java 示例。先运行 Java 示例先生成预签名 url,再运行 Java 示例上传文件,上传成功。将预签名 url 放到前端调试,上传失败,同样异常返回 SignatureDoesNotMatch 错误码 0002-00000201。基本判断问题在前端。
  3. 查看文档 browser.js 示例,意外发现线索。基本确认问题在 Content-Type 请求头上。

如果您使用 Browser.js 上传文件时遇到 403 签名不匹配错误,通常是因为浏览器会自动添加 Content-Type 请求头,而生成预签名 URL 时未指定该请求头,导致签名验证失败。为解决此问题,您需要在生成预签名 URL 时指定 Content-Type 请求头。

3. 处理

  1. 参考文档,在后端代码添加 StorageClass 和 ContentType
java 复制代码
public String generatePresignedUploadUrl(String fileName) {
 // 指定生成的预签名URL过期时间,单位为毫秒。本示例以设置过期时间为1小时为例。
        Date expiration = new Date(new Date().getTime() + 3600 * 1000L);
        GeneratePresignedUrlRequest request = new GeneratePresignedUrlRequest(ossConfig.getBucketName(), fileName, HttpMethod.PUT);
        request.setExpiration(expiration);
        
        // 添加请求头
        Map<String, String> headers = new HashMap<>();
        // 指定StorageClass。
        headers.put(OSSHeaders.STORAGE_CLASS, StorageClass.Standard.toString());
        // 指定ContentType。
        headers.put(OSSHeaders.CONTENT_TYPE, "application/octet-stream");
        request.setHeaders(headers);
        
        URL url = ossClient.generatePresignedUrl(request);
        return url.toString();
}
  1. 参考文档,在前端代码添加相同的 StorageClass 和 ContentType
typescript 复制代码
      // 2. 使用预签名URL上传文件 - 使用PUT请求
      const uploadResponse = await fetch(presignedUrl, {
        method: 'PUT',
        body: file,
        headers: {
          'Content-Type': 'application/octet-stream',
          'x-oss-storage-class': 'Standard',
        },
      })
  1. 测试验证,上传成功。
相关推荐
李日灐27 分钟前
C++进阶必备:红黑树从 0 到 1: 手撕底层,带你搞懂平衡二叉树的平衡逻辑与黑高检验
开发语言·数据结构·c++·后端·面试·红黑树·自平衡二叉搜索树
qq_297574671 小时前
【实战】POI 实现 Excel 多级表头导出(含合并单元格完整方案)
java·spring boot·后端·excel
郝学胜-神的一滴1 小时前
超越Spring的Summer(一): PackageScanner 类实现原理详解
java·服务器·开发语言·后端·spring·软件构建
Tony Bai1 小时前
“Go 2,请不要发生!”:如果 Go 变成了“缝合怪”,你还会爱它吗?
开发语言·后端·golang
Victor3562 小时前
Hibernate(91)如何在数据库回归测试中使用Hibernate?
后端
Victor3562 小时前
MongoDB(1)什么是MongoDB?
后端
Victor3568 小时前
https://editor.csdn.net/md/?articleId=139321571&spm=1011.2415.3001.9698
后端
Victor3568 小时前
Hibernate(89)如何在压力测试中使用Hibernate?
后端
灰子学技术10 小时前
go response.Body.close()导致连接异常处理
开发语言·后端·golang
Gogo81611 小时前
BigInt 与 Number 的爱恨情仇,为何大佬都劝你“能用 Number 就别用 BigInt”?
后端