PUTting到S3时ERR_CONNECTION_RESET

时间:2014-04-01 23:22:43

标签: ajax amazon-s3 xmlhttprequest

我通过ajax请求将文件输出到S3,大约50%的时间我收到ERR_CONNECTION_RESET错误。

我知道请求已正确签名 - 任何想法可能导致此问题?同样,这是我从多个位置和机器看到的间歇性问题。

以下是我用来将文件输入S3的相关coffeescript代码。它来自Micah Roberson和Rok Krulec在http://micahroberson.com/upload-files-directly-to-s3-w-backbone-on-heroku/http://codeartists.com/post/36892733572/how-to-directly-upload-files-to-amazon-s3-from-your的工作。

  createCORSRequest: (method, url) ->
    xhr = new XMLHttpRequest()

    if xhr.withCredentials?
      xhr.open method, url, true
    else if typeof XDomainRequest != "undefined"
      xhr = new XDomainRequest()
      xhr.open method, url
    else
      xhr = null

    xhr

  uploadToS3: (file, signature) ->
    this_s3upload = this
    this_s3upload.signature = signature
    url = signature.signed_request

    xhr = @createCORSRequest 'PUT', decodeURIComponent(signature.signed_request)

    if !xhr
      @onError 'CORS not supported'
    else
      xhr.onload = () ->
        if xhr.status == 200
          this_s3upload.onProgress 100, 'Upload completed.'
          this_s3upload.onFinishS3Put file, this_s3upload.signature
        else
          this_s3upload.onError file, 'Upload error: ' + xhr.status

      xhr.onerror = () ->
        this_s3upload.onError file, 'XHR error.', this_s3upload.signature

      xhr.upload.onprogress = (e) ->
        if e.lengthComputable
          percentLoaded = Math.round (e.loaded / e.total) * 100

          if percentLoaded == 100
            message = "Finalizing"
          else
            message = "Uploading"

          this_s3upload.onProgress xhr, file, percentLoaded, message, this_s3upload.signature

      xhr.onabort = ->
        this_s3upload.onAbort file, "XHR cancelled by user.", this_s3upload.signature

    xhr.setRequestHeader 'Content-Type', file.type
    xhr.setRequestHeader 'x-amz-acl', 'public-read'
    xhr.send file

更新

在这个问题上,我一直得到亚马逊的非常细心的支持。根据他们的建议,我创建了一个EC2 Windows实例,在其上加载了Chrome浏览器,并尝试使用我的代码上传5个文件10次。我没有看到错误一次。我偶尔会看到一些SignatureDoesNotMatch错误,但没有一个ERR_CONNECTION_RESET错误。我仍然在我使用的每个非EC2客户端/网络位置看到ERR_CONNECTION_RESET错误。

更新此处仍无解决方案。我已经从使用自动滚动签名算法转移到boto提供的算法。尽管如此,对ERR_CONNECTION_RESET问题没有影响。

3 个答案:

答案 0 :(得分:1)

我终于放弃了让它发挥作用。相反,我现在使用Fine Uploader来提供此功能。

答案 1 :(得分:1)

在使用预签名网址上传更大的文件(长请求)时遇到此问题,在Heroku's example之后(node aws sdk):

app.get('/sign-s3', (req, res) => {
  const s3 = new aws.S3();
  const fileName = req.query['file-name'];
  const fileType = req.query['file-type'];
  const s3Params = {
    Bucket: S3_BUCKET,
    Key: fileName,
    Expires: 60,
    ContentType: fileType,
    ACL: 'public-read'
  };

  s3.getSignedUrl('putObject', s3Params, (err, data) => {
    if(err){
      console.log(err);
      return res.end();
    }
    const returnData = {
      signedRequest: data,
      url: `https://${S3_BUCKET}.s3.amazonaws.com/${fileName}`
    };
    res.write(JSON.stringify(returnData));
    res.end();
  });
});

"到期"参数使签名的URL有效60秒。

我认为当签名的URL在上传过程中到期时,请求会崩溃(即使它在上传开始时有效)。

但它在60秒后并没有完全崩溃,而是随机地在60到120秒之间。大多数情况下,客户端会记录ERR_CONNECTION_RESET,有时会记录403 FORBIDDEN。

将它加速到3600后我就不再有问题了。

我怀疑这个问题在EC2上没有发生,因为它们的上传速度非常快。

答案 2 :(得分:0)

我认为这个问题没有决定。

尝试POST请求:https://aws.amazon.com/articles/1434

<form action="https://s3-bucket.s3.amazonaws.com/" method="post" enctype="multipart/form-data">
  <input type="hidden" name="key" value="uploads/${filename}">
  <input type="hidden" name="AWSAccessKeyId" value="YOUR_AWS_ACCESS_KEY"> 
  <input type="hidden" name="acl" value="private"> 
  <input type="hidden" name="success_action_redirect" value="http://localhost/">
  <input type="hidden" name="policy" value="YOUR_POLICY_DOCUMENT_BASE64_ENCODED">
  <input type="hidden" name="signature" value="YOUR_CALCULATED_SIGNATURE">
  <input type="hidden" name="Content-Type" value="image/jpeg">
  <!-- Include any additional input fields here -->

  File to upload to S3: 
  <input name="file" type="file"> 
  <br> 
  <input type="submit" value="Upload File to S3"> 
</form> 

如果您更喜欢发送AJAX,请使用FormData附加字段

在nodejs中签署请求:Amazon S3 POST api, and signing a policy with NodeJS