django - 从 next.js 上传到 django 以存储在 aws s3 中的图像是空白的
问题描述
我为 s3 设置了功能,我可以轻松地从 Django 管理界面上传图像。我有“createProduct”页面,我想通过从 next.js 发送表单数据来创建产品。Django 将 URL s3 存储桶存储在 DB 中,并在需要时从 s3 检索图像。
我设置了视图图像并裁剪它的功能,它也可以工作。为此,我有一个单独的组件:
<div className="form-group">
<label htmlFor="image">Image</label>
<FileLoader
onFileUpload={image => setValue('image', image._id)}
/>
</div>
我需要创建一个端点。admin,首先上传图片,图片保存到s3,然后admin获取图片的url,传给form data,这样form data就会发送到createProduct端点。
@api_view(['POST'])
@permission_classes([IsAdminUser])
def createProduct(request):
user=request.user
print("user in post",user)
print("request.data",request.data)
name=request.data['name']
price=request.data['price']
brand=request.data['brand']
countInStock=request.data['countInStock']
category=request.data['category']
description=request.data['description']
// Here i need to send the imag url I guess
image=request.FILES.get('image')
print("image",image)
product=Product.objects.create(user=user, name=name,
price=price,brand=brand,
countInStock=countInStock,category=category,description=description, image=image)
product.save()
serializer=ProductSerializer(product, many=False)
return Response(serializer.data)
这是 s3 上传的端点:
@api_view(['POST'])
def upload(request):
data=request.data
print("image data sent",data)
client=boto3.client("s3",
aws_access_key_id=os.environ.get('AWS_ACCESS_KEY_ID'),
aws_secret_access_key=os.environ.get('AWS_SECRET_ACCESS_KEY'),
# s3 does not need region
# region_name=os.environ.get(AWS_REGION_NAME),
)
response = client.upload_fileobj(data,"bingology-bucket","testing",{'ContentType': "image/jpeg"}
我收到成功上传的消息,图像在存储桶中,但它只显示黑色背景,“类型”是“-”。
我也尝试获取预签名的 url,我得到了 url,但它没有保存任何东西,所以 s3。由于权限问题,我在访问时无法查看 url,即使存储桶是公开的
解决方案
我认为我们需要另一个端点,具有使用此架构处理上传的逻辑:
- uploadImage(另一个端点的名称)从下一个 js 接收图像元数据。
- uploadImage 生成一个预签名的 Url(因为您使用的是 S3),将此 URL 返回给 nextjs,nextjs 使用此预签名的 url 执行上传,并调用 createProduct 端点传递 URL(uploadImage 可以过滤以返回对象的确切 URL)和要存储在数据库中的所有产品数据。
这种方法可以更好地使用 SAGA 模式来编排事物,因为由于您需要处理 2 个不同的流程(s3 上传和 db 事务),您可以编排这些事物,或者在单一架构中使用事务(所有业务逻辑在一个部署)。
几个小贴士:
在这种情况下使用 SAGA 模式,如果 createProduct 中有错误,您只需要检测作业以删除图像。您可以通过在前端使用逻辑将消息放在带有 lambda 表达式的 SNS 上来实现这一点,以从中读取并删除 S3 中的图像,因为产品未正确保存。
文档:
https://microservices.io/patterns/data/saga.html https://docs.aws.amazon.com/AmazonS3/latest/userguide/PresignedUrlUploadObject.html
推荐阅读
- python - Python:网格上的曲面积分
- python-3.x - 当 AWS RDS 导致连接冻结时更新 Postgres 上的表
- sql - 如何在 SQL 的 Openquery 中运行 Pivot?
- node.js - 错误:找不到模块 'webpack/lib/rules/DescriptionDataMatcherRulePlugin' 需要堆栈:
- c# - .NET Core 类中的结构对齐
- javascript - 未找到模块:包路径 ./react 未从包 E:\NextApp\portfolio_website-main\portfolio_website-main\node_modules\next-auth 导出
- kotlin - 聊天 Recyclerview 中的 MediaPlayer
- arrays - 转换 mongo 数组
- chart.js - ChartJS 条形图未正确启动
- firebase - 我的 Firebase 云功能链接不起作用