我必须使用node.js从S3存储桶下载多个文件。为此,我必须写一个for loop
&调用s3.getObject(param)
方法进行下载。下载文件后,我必须合并其内容。
我写得像这样:
var fileContentList = new ArrayList();
for(i=0; i<fileNameList.length i++){
s3.getObject({ Bucket: "my-bucket", Key: fileNameList.get(i) }, function (error, data) {
if (error != null) {
alert("Failed to retrieve an object: " + error);
} else {
alert("Loaded " + data.ContentLength + " bytes");
fileContentList.add(data.Body.toString());
}
}
);
}
//Do merging with the fileContentList.
但是当s3.getObject
是一个异步调用时,当前线程会继续运行。在进行合并时,fileContentList
没有添加任何内容。
我该如何解决这个问题?有什么想法?
他们在aws-sdk中的任何同步方法是下载文件吗?
答案 0 :(得分:4)
承诺是更好的方式,
var getObject = function(keyFile) {
return new Promise(function(success, reject) {
s3.getObject(
{ Bucket: "my-bucket", Key: keyFile },
function (error, data) {
if(error) {
reject(error);
} else {
success(data);
}
}
);
});
}
var promises = [];
var fileContentList = new ArrayList();
for(i=0; i<fileNameList.length i++){
promises.push(getObject(fileNameList.get(i)));
}
Promise.all(promises)
.then(function(results) {
for(var index in results) {
var data = results[index];
fileContentList.add(data.Body.toString());
}
// continue your process here
})
.catch(function(err) {
alert(err);
});
答案 1 :(得分:0)
你可以在这里使用async,它将并行下载所有文件。 在此示例中,如果某些文件失败,则下载将继续,如果您想在发生错误时停止下载文件,请使用错误调用回调,这将立即调用最终回调。
var async = require('async');
var fileContentList = new ArrayList();
function downloadS3Multiple(done){
async.each([
function (callback) {
s3.getObject({Bucket: "my-bucket", Key: fileNameList.get(i)}, function (err, res) {
if (err) {
alert("Failed to retrieve an object: " + error);
callback();
}
else {
alert("Loaded " + data.ContentLength + " bytes");
fileContentList.add(data.Body.toString());
callback();
}
})
}
], function (err, results) {
done(err, fileContentList)
});
}
答案 2 :(得分:0)
跟踪您在单独列表中发起的下载,并在每次下载完成后检查它们是否全部完成。
var fileContentList = new ArrayList();
var completedList = new ArrayList();
// function to setDone and initiate merge if all download attempts completed.
function setDone(i) {
completedList[i]=true;
var allDone= true;
for(i=0; i<completedList.length && allDone=completedList[i] && allDone; i++);
if(allDone) {
mergeFiles();
}
}
// fill completedList with a false value for each fill to be downloaded
for(i=0; i<fileNameList.length;i++) completedList.add(false);
// initiate the downloads
for(i=0; i<fileNameList.length; i++){
s3.getObject({ Bucket: "my-bucket", Key: fileNameList.get(i) }, function (error, data) {
if (error != null) {
alert("Failed to retrieve an object: " + error);
} else {
alert("Loaded " + data.ContentLength + " bytes");
fileContentList.add(data.Body.toString());
}
setDone(i);
}
);
}
更优雅的解决方案,如果您只想在所有下载成功完成后合并文件:
var fileContentList = new ArrayList();
for(i=0; i<fileNameList.length i++){
s3.getObject({ Bucket: "my-bucket", Key: fileNameList.get(i) }, function (error, data) {
if (error != null) {
alert("Failed to retrieve an object: " + error);
} else {
alert("Loaded " + data.ContentLength + " bytes");
fileContentList.add(data.Body.toString());
}
if(fileContentList.length==fileNameList.length) combineFiles();
}
);
}
答案 3 :(得分:0)
我已经解决了这个问题。虽然我没有尝试亚历山大,莉娜和他的答案。塞巴斯蒂安我相信他们提到的每个答案也适用于这种情况。很多人都感谢他们的快速回复:
Async.eachSeries(casCustomersList, function (customerName, next){
if(casCustomersList.length>0 && customerName != customerId) {
var paramToAws = {
Bucket: bucketName,
Key: folderPath +'applicationContext-security-' + customerName + '.xml' //file name
};
AWSFileAccessManager.downloadFile(paramToAws, function (error, result) {
if (error) {
next(error);
} else {
customerApplicationContext.add(result.Body.toString());
next();
}
});
} else{
next();
}
}, function(err) {
//Write the rest of your logic here to process synchronously as it is the callback function
}