在 python 中有一个很好的库'base64',使用这个模块可以根据文件名和内容对文件进行编码/解码。我正在开发一个具有上传文件功能的 API。我设法编写了一些代码来编码和解码这些文件,但是通过 HTTP 请求发送这个编码的 base64 消息的正确方法是什么?
- 通过 POST 并将文件名和编码消息放入 post 参数?
- 在标题里面?
欢迎任何想法,如果可能的话显示一个小的http请求你应该怎么做。
我最近不得不这样做,因为django-tastypie不支持multipart/form-data。klipstein创建了Base64FileField以通过使用文件的 base64 编码来支持文件上传。
下面是一个基于 Base64FileField 的示例:
#models.py
from django.db import models
class Person(models.Model):
name = models.CharField("name", max_length=40)
avatar = models.ImageField("avatar", upload_to="uploads/person_avatars")
def __unicode__(self):
return self.name
#views.py
import json
import base64
import sys
from django.core.files.uploadedfile import SimpleUploadedFile
from django.http import HttpResponse, HttpResponseBadRequest
from django.views.decorators.csrf import csrf_exempt
from .models import Person
MAX_SIZE = 10*1024*1024 #10MB
@csrf_exempt
def save_person(request):
if (request.method == "POST" and request.META.get("CONTENT_TYPE")
== "application/json"):
data = json.loads(request.raw_post_data)
avatar = data.get("avatar")
file = base64.b64decode(avatar["file"])
if sys.getsizeof(file) > MAX_SIZE:
response = {"error": {
"code": "invalid_request",
"message": "Max file size exceeded."
}}
return HttpResponseBadRequest(json.dumps(response),
mimetype="application/json")
person = Person(name=data.get("name"), avatar=SimpleUploadedFile(
avatar["name"],
file,
getattr(avatar, "content_type", "application/octet-stream")))
person.save()
response = {"id": person.id, "name": person.name,
"avatar": person.avatar.url}
return HttpResponse(json.dumps(response), mimetype="application/json")
response = {"error": {
"code": "invalid_request",
"message": "Method Not Allowed."
}}
return HttpResponseBadRequest(json.dumps(response),
mimetype="application/json")
这是要为上述视图发布的示例 JSON 正文
{
"name": "Test User",
"avatar": {
"name": "myfile.png",
"file": "a long base64 encoded string ",
"content_type": "image/png"
}
}
示例响应是:
{
"id": 1,
"avatar": "/media/uploads/person_avatars/myfile.png",
"name": "Test User"
}
什么是最好的方法?
不确定这是否是最好的方法,但是,它简单明了。