Загрузить на s3 с помощью curl, используя предварительно подписанный URL (получая 403) - PullRequest
16 голосов
/ 30 января 2012

Я использую curl для вызова API Java ReST для получения URL. Затем Java генерирует предварительно подписанный URL-адрес для загрузки S3, используя мои учетные данные S3, и возвращает его в ответе ReST. Curl берет URL-адрес и использует его для загрузки на S3, но S3 возвращает 403 "Считаемая нами подпись запроса не соответствует предоставленной вами подписи. Проверьте свой ключ и метод подписи."

Вот код, который я использую для создания предварительно подписанного URL:

public class S3Util {
    static final AmazonS3 s3 = new AmazonS3Client( new AWSCredentials() {
        @Override
        public String getAWSAccessKeyId() {
            return "XXXXXXX";
        }
        @Override
        public String getAWSSecretKey() {
            return "XXXXXXXXXXXXXX";
        }
    });
    static final String BUCKET = "XXXXXXXXXXXXXXXXXXXXXXXXXXX";

    static public URL getMediaChunkURL( MediaChunk mc, HttpMethod method ) {
        String key = ...
        //way in the future (for testing)...
        Date expiration = new Date( System.currentTimeMillis() + CalendarUtil.ONE_MINUTE_IN_MILLISECONDS*60*1000 );

        GeneratePresignedUrlRequest req = new GeneratePresignedUrlRequest(BUCKET, key, method);
        req.setExpiration(expiration);
        req.addRequestParameter("Content-Type", "application/octet-stream");

        //this gets passed to the end user:
        return s3.generatePresignedUrl(req);
    }
}

и в curl, запустить из bash, я выполняю это:

echo Will try to upload chunk to ${location}
curl -i -X POST \
        -F 'Content-Type=application/octet-stream' \
        -F "file=@${fileName}" \
        ${location} || (echo upload chunk failed. ; exit 1 )

Помимо прочего, я попробовал PUT и «Content-type» (строчная буква T). Я понимаю, что упускаю что-то (или что-то) очевидное, но после прочтения соответствующих документов, поиска в Google и просмотра множества похожих вопросов я не уверен, что это такое. Я вижу много подсказок о необходимых заголовках, но я думал, что отказавшийся URL-адрес должен был устранить эти потребности. Может нет?

ТИА!

Обновление:

Просто чтобы прояснить ситуацию, я проверил загрузки, и это прекрасно работает.

Java выглядит так:

GeneratePresignedUrlRequest req = new GeneratePresignedUrlRequest(BUCKET, key, HttpMethod.GET);
req.setExpiration(expiration);

и curl это просто:

curl -i ${location}

Ответы [ 4 ]

24 голосов
/ 31 января 2012

Мне удалось создать предварительно подписанный URL-адрес с помощью C # и затем загрузить его с помощью curl , как и ожидалось.Учитывая мои тесты, я подозреваю, что вы действительно не используете curl правильно - я смог загрузить файл примерно так:

curl -v --upload-file ${fileName} ${location}

Параметр -v выводит как запрос, так и ответЗаголовки (а также рукопожатие SSL) для целей отладки и иллюстрации:

> PUT [...] HTTP/1.1
> User-Agent: curl/7.21.0 [...]
> Host: [...]
> Accept: */*
> Content-Length: 12
> Expect: 100-continue

Обратите внимание, что --upload-file (или -T) облегчает PUT, как и ожидалось, но добавляет дополнительные заголовки по мере необходимости, давая правильный ответ в ответ:

< HTTP/1.1 100 Continue
< HTTP/1.1 200 OK
< x-amz-id-2: [...]
< x-amz-request-id:  [...]
< Date: Tue, 31 Jan 2012 18:34:56 GMT
< ETag: "253801c0d260f076b0d5db5b62c54824"
< Content-Length: 0
< Server: AmazonS3
6 голосов
/ 23 сентября 2012

когда вы делаете это с помощью curl, вам нужно поместить URL в одинарные кавычки, иначе половина строки запроса будет обрезана (часть с ключом / подписью).

1 голос
/ 28 марта 2015

Способ генерации URL:

private static URL generateRUL(String objectKey, String ACCESS_KEY, String SECRET_KEY, String BUCKET_NAME) {
    AmazonS3 s3Client = new AmazonS3Client(new BasicAWSCredentials(ACCESS_KEY, SECRET_KEY));
    URL url = null;

    try {
        GeneratePresignedUrlRequest request  = new GeneratePresignedUrlRequest(BUCKET_NAME, objectKey);
        request.setMethod(com.amazonaws.HttpMethod.PUT);
        request.setExpiration(new Date( System.currentTimeMillis() + (60 * 60 * 1000)));

        // Very important ! It won't work without adding this! 
        // And request.addRequestParameter("Content-Type", "application/octet-stream") won't work neither
        request.setContentType("application/octet-stream");

        url = s3Client.generatePresignedUrl(request ); 
    } catch (AmazonServiceException exception) { 
    } catch (AmazonClientException ace) { }

    return url;
}

Способ загрузки файла:

public int upload(byte[] fileBytes, URL url) {
    HttpURLConnection connection = (HttpURLConnection) url.openConnection();
    connection.setDoOutput(true);
    connection.setRequestMethod("PUT");
    connection.setRequestProperty("Content-Type", "application/octet-stream"); // Very important ! It won't work without adding this!
    OutputStream output = connection.getOutputStream();

    InputStream input = new ByteArrayInputStream(fileBytes);
    byte[] buffer = new byte[4096];
    int length;
    while ((length = input.read(buffer)) > 0) {
        output.write(buffer, 0, length);
    }
    output.flush();

    return connection.getResponseCode();
}
0 голосов
/ 31 января 2012

Несмотря на то, что GeneratePresignedUrlRequest принимает аргумент метода http (и имеет функцию setMethod), он кажется непригодным для всего, кроме GET.

http://wiki.nercomp.org/wiki/images/0/05/AmazonWebServices.pdf гласит: «Практика подписания запроса и передачи его третьему лицу для исполнения подходит только для простых GET-запросов объекта». Возможно, что-то может быть использовано для установки другого метода, но, видимо, не для этого.

Итак, вместо этого я должен был следовать инструкциям здесь:

http://aws.amazon.com/articles/1434?_encoding=UTF8&jiveRedirect=1

Это более сложно, потому что клиент должен публиковать полную форму, а не просто использовать URL-адрес, а также означает, что вся эта информация о сообщении должна передаваться клиенту отдельно, но, похоже, это работает.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...