javascript - 使用 JQuery 将文件上传到带有预签名 url 的 S3 时出错

标签 javascript jquery file-upload amazon-s3 form-data

我们正在尝试使用 jQuery ajax 和预签名 url 将文件上传到 S3。我们在服务器上生成预签名的 url。目前我们正在尝试使用 FormData上传文件。

          var uploadData = new FormData(),
            files = $(this.input).prop('files'),
            file = files[0];

          uploadData.append('file', file);

          $.ajax({
            url: '{presigned url string}',
            type: 'PUT',
            data: uploadData,
            cache: false,
            processData: false,
            contentType: false,
            success: function(response) {
              console.log('S3 upload success!');
            },
            error: function(response) {
              console.log('Error with S3 upload: ' + response.statusText);
            }
          });

这从 AWS 返回一个 SignatureDoesNotMatch 错误:

<Error><Code>SignatureDoesNotMatch</Code><Message>The request signature we 
calculated does not match the signature you provided. Check your key and 
signing method.</Message><StringToSignBytes>50 55 54 0a 0a 6d 75 6c 74 69 70 61 72 74 2f 66 6f 72 6d 2d 64 61 74 61 3b 20 62 6f 75 6e 64 61 72 79 3d 2d 2d 2d 2d 57 65 62 4b 69 74 46 6f 72 6d 42 6f 75 6e 64 61 72 79 67 57 6f 4a 73 42 56 79 41 4c 57 71 51 6b 73 69 0a 31 34 30 39 37 37 37 32 39 33 0a 2f 64 69 2d 6b 79 72 73 74 65 6e 2d 64 65 61 6c 73 2f 76 4e 79 4b 4e 55 4c 37 51 68 4f 30 45 4b 38 52 58 44 70 32 59 77 25 32 46 63 35 37 65 64 37 62 39 2d 64 63 61 62 2d 34 63 30 62 2d 62 36 63 30 2d 36 31 66 30 36 62 32 30 37 34 66 31 2d 74 65 73 74 2e 74 78 74</StringToSignBytes>
<RequestId>ED7C581570F547DB</RequestId><HostId>ZT6LsFYCbo1L0gYNcUwtdCWF6SNnyuUyKiL60ntJEZugx3cnDN/yH5KBjgEiBv5c</HostId><SignatureProvided>N2d7oNMVHvI6yxAXujNy8O5cF24=</SignatureProvided>
<StringToSign>PUT

multipart/form-data; boundary=----WebKitFormBoundarygWoJsBVyALWqQksi
1409777293
/test-bucket/vNyKNUL7QhO0EK8RXDp2Yw%2Fc57ed7b9-dcab-4c0b-b6c0-61f06b2074f1-test.txt</StringToSign><AWSAccessKeyId>FAKEACCESSKEY</AWSAccessKeyId></Error>  

我知道预签名 url 有效,因为我可以通过 CURL 正确上传文件:

curl -v --upload-file {filename} {presigned url}

最佳答案

我们自己发现了问题。这是我们用来生成预签名 url 的代码:

val dt: DateTime = new DateTime()
val expiration = DateTime.now.plusMillis(timeout.toInt)
val presignedUrlRequest: GeneratePresignedUrlRequest =
  new GeneratePresignedUrlRequest(awsBucket, key, HttpMethod.PUT)
presignedUrlRequest.setExpiration(expiration.toDate())
// WE ADDED THIS LINE
presignedUrlRequest.setContentType("multipart/form-data")

s3client.generatePresignedUrl(presignedUrlRequest).toString()

我们必须在 presignedUrlRequest 中设置内容类型并添加
headers: {'Content-Type': 'multipart/form-data'} 到我们的 ajax 调用。

我们最初使用 CURL 生成 url 的方式是因为我们没有在 presignedUrlRequest 中设置内容类型,而 curl 也没有设置内容类型。

关于javascript - 使用 JQuery 将文件上传到带有预签名 url 的 S3 时出错,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25632235/

相关文章:

javascript - 来自 jquery 的 mvc 日期时间选择器设置值未绑定(bind)到模型

python - 在 python 中通过 ftplib 上传比手动上传花费的时间要长得多

javascript - 以编程方式将文件上传到 ownCloud 服务器

javascript - D3 折线图未正确显示工具提示和数据点

javascript - 重现前端错误

JavaScript <canvas> 绘制缩放图像

post - PHP : upload file from one server to another server

javascript - 模板内的函数被多次调用(Angular JS)

jquery - 从 HTML 表格复制到剪贴板的用户选择引入了 Tabs

javascript - 如何通过 jQuery/Javascript 获取更新的表单输入值?