我通过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问题没有影响。
答案 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