首页 > 解决方案 > 使用 API 生成多个预签名的 url

问题描述

我创建了一个 API,它为我的私有 s3 存储桶中的文件生成预签名的 url。我想将这些 url 存储到一个数组中,以便我可以从另一个应用程序调用它们。["FILE1 pre-signed url","FILE2 pre-signed url", etc..] 但是,我对我的 API 如何读取命令感到困惑,所以我来这里寻求帮助。

  1. 为什么我进入时托管 API 的页面不显示填充的数组?直到我刷新页面,它才最终填充了数组。

我希望每次都重置数组并替换链接而不是添加到数组的末尾。我还需要在页面加载后立即显示数组,而不是刷新它以查看结果。

这是当我进入提供 API 的页面时显示的内容: 在此处输入图像描述

当我点击刷新时,数组中填充了预签名的 url:

["https://bucketname.s3.amazonaws.com/file1.pdf?AWSAccessKeyId=ACCESSKEY&Expires=1572669513&Signature=SIGNATURE","https://bucketname.s3.amazonaws.com/file2.pdf?AWSAccessKeyId=ACCESSKEY&Expires=1572669513&Signature=SIGNATURE"]

但是,当我再次刷新页面时,它会再次推送这两个文件,使数组成为四个元素而不是两个。如果我不断刷新页面,它会继续添加另外两个 url,依此类推。(如果循环在 2 次迭代时停止,为什么要这样做)

const express = require('express');
const bodyParser = require('body-parser');
const morgan = require('morgan');
const uuid = require('uuid');
const AWS = require('aws-sdk');
let config = require('./config');

var s3 = new AWS.S3();
s3.config.update({
    accessKeyId: config.accessKeyId,
    secretAccessKey: config.secretAccessKey,
    region: 'us-east-1'
});

var app = express();
var router = express.Router();

// Create an array that houses pre-signed Urls


// Loop through x amt of times (depending on number of documents)
// Push each url into an array called 'preSignedUrls'
// Exit loop and reset array to re-generate new urls
let preSignedUrls = [];
    function getSignedURL(req, res, next) {
    Key = ['file1.pdf', 'file2.pdf']
        for ( i = 0; i < Key.length; i++ ){
            var params = {
                Bucket: 'bucketName',
                Key: Key[i],
                Expires: 36000
            };
                console.log(Key.length);
                console.log(i);
                s3.getSignedUrl('getObject', params, function(err, signedURL) {
                    if (err) {
                        console.log(err);
                        return next(err);
                    } else {
                        console.log(signedURL);
                        console.log(params);
                        preSignedUrls.push(signedURL);
                    }
                });
        }
        res.json(preSignedUrls);

    }
router.route('/get_signed_url')
    .get(getSignedURL);

app.use(morgan('combined'));
app.use(bodyParser.json());
app.use('/v1', router);

var PORT = process.env.PORT || 3001;
var HOST = process.env.HOST || '10.123.456.789';

console.log("Listening on", HOST, PORT);
app.listen(PORT, HOST);

这是 node.js 终端的输出(也就是我放入代码中以测试输出的日志命令)

2
0
2
1
https://bucketname.s3.amazonaws.com/file1.pdf?AWSAccessKeyId=ACCESSKEY&Expires=1572669520&Signature=SIGNATURE
{ Bucket: 'bucketname',
  Key: 'file2.pdf',
  Expires: 36000 }
https://bucketname.s3.amazonaws.com/file2.pdf?AWSAccessKeyId=ACCESSKEY&Expires=1572669520&Signature=SIGNATURE
{ Bucket: 'bucketname',
  Key: 'file2.pdf',
  Expires: 36000 }

我对此感到奇怪的是,它将对象参数记录为 file2.pdf,但它输出的预签名 URL 按应有的顺序显示 file1.pdf 和 file2.pdf。此外,长度和变量 i 显示在对象之前的事实让我感到困惑,因为我认为 for 循环按顺序执行命令。

标签: javascriptnode.jsapiamazon-s3pre-signed-url

解决方案


因为 signedURL 是来自 s3.getSignedUrl() 的响应。它是异步运行的,并且有自己的值和上下文。但另一方面,变量 params 将主要返回 for 循环的最后一个值。这是因为函数 s3.getSignedUrl 尚未执行,而 for 循环更早执行。for 循环不会等待 s3.getSignedUrl() 作为其异步完成。如果要保留 i 值,您可能希望使用箭头功能。

这是其他线程上执行此操作的方法。 async / await 循环遍历数组以获取 s3 签名的 URL


推荐阅读