问题描述
我正在使用 React、API 网关和 Lambda (Python) 来获取预签名的 url 以将文件上传到 s3 存储桶。但是上传到 s3 存储桶的文件始终为空(0 字节)。出于某种原因,当我在 React 应用程序中上传 CSV 文件时。 s3 中上传的文件变成了 .XLS 文件,但仍命名为原始 csv。
这是我的代码:
<div>
<input type="file" name="fileOne" onChange={fileUploadOne} />
</div>
...
const fileUploadOne = (event) => {
const { files } = event.target;
getPresignedUrl(files[0]).then((response) => {
putToS3(files[0],response);
});
};
...
export async function getPresignedUrl(fileObject) {
const requestOptions = {
method: "GET",headers: {
"Content-Type": "application/json",},};
const response = await fetch(
"https://<api_gateway_url>.amazonaws.com/<api_name>?filename=" +
fileObject.name +
"&filetype=" +
fileObject.type,requestOptions
);
return await response.json();
}
export async function putToS3(fileObject,presignedUrl) {
const requestOptions = {
method: "PUT",headers: {
"Content-Type": fileObject.type,data: fileObject,};
const response = await fetch(presignedUrl,requestOptions);
return await response;
}
我的 Lambda 代码(通过 getPresignedUrl 函数访问):
import json
import boto3
from botocore.exceptions import ClientError
def lambda_handler(event,context):
params = event["querystringparameters"]
key = params.get('filename')
filetype = params.get('filetype')
s3_client = boto3.client('s3')
client_action = 'put_object'
bucket_name = '<bucket_name>'
resp = generate_presigned_url(
s3_client,client_action,{'Bucket': bucket_name,'Key': key,'ContentType':filetype},1000)
return {
'statusCode': 200,'headers': {
"Access-Control-Allow-Origin" : "*",# required for CORS support to work
},'body': json.dumps(resp)
}
def generate_presigned_url(s3_client,client_method,method_parameters,expires_in):
"""
Generate a presigned Amazon S3 URL that can be used to perform an action.
:param s3_client: A Boto3 Amazon S3 client.
:param client_method: The name of the client method that the URL performs.
:param method_parameters: The parameters of the specified client method.
:param expires_in: The number of seconds the presigned URL is valid for.
:return: The presigned URL.
"""
try:
url = s3_client.generate_presigned_url(
ClientMethod=client_method,Params=method_parameters,ExpiresIn=expires_in
)
print("Got presigned URL: %s",url)
except ClientError:
print(
"Couldn't get a presigned URL for client method '%s'.",client_method)
raise
return url
解决方法
我发现了我的错误。我的 HTTP PUT 请求应该在 body 中包含 fileObject 而不是数据。
export async function putToS3(fileObject,presignedUrl) {
const requestOptions = {
method: "PUT",headers: {
"Content-Type": fileObject.type,},body: fileObject,};
const response = await fetch(presignedUrl,requestOptions);
return await response;
}
这真的很简单,而且是一个愚蠢的错误。我花了几个小时才弄明白。