在搜索 SO、示例和文档两天后遇到了无法解决的问题。我真的希望这里有一个解决方案。
我正在开发一项用于在 Python 和 Google App Engine 中自动创建 VK 广告的服务。广告图像首先上传到我的服务器(第 1 部分),然后在适当的时间从那里上传到 VK 服务器(第 2.1 和 2.2 部分)。看起来与加载时相同的字节存储在图像服务器上,但以防万一,我附上了代码。
要将图片上传到广告,您首先需要获取一个链接(这很简单,所以我们跳过它),然后使用此链接发送一个 POST 请求,其中包含图片文件的文件字段(doc 链接)。我以两种方式(2.1 和 2.2)实现了发送图像,但都返回errcode: 2,这意味着повреждённый файл. 我认为问题在于请求本身的结构,但是,我不会错过在我的服务器上加载/存储图像或 Vk.Ads API 的特定工作中出现问题的可能性。我将非常感谢您的回答和建议。
1. 将照片上传到您的服务器
import webapp2
from google.appengine.ext import ndb
# для хранения изображения на сервере
class Photo(ndb.Model):
name = ndb.StringProperty()
img = ndb.BlobProperty()
@staticmethod
def get(name):
retval = Photo.query(Photo.name == name).get()
return retval
@staticmethod
def create(name, blob):
retval = Photo()
retval.name = name
retval.img = blob
return retval
class PhotosPage(webapp2.RequestHandler):
def get(self):
# основное содержимое страницы:
html = '''<form action="/photos" method="post" enctype="multipart/form-data">
<input type="file" name="flimg"/>
<input value="new_pic" name="flname"/>
<input type="submit" value="Upload"/> </form>'''
def post(self):
n = str(self.request.get('flname'))
f = self.request.get('flimg')
p = Photo.get(n)
if p:
p.img = f
else:
p = Photo.create(n, f)
p.put()
2.1。发送照片、方法 #1、urlfetch和海报库:
from poster.encode import multipart_encode, MultipartParam
from google.appengine.api import urlfetch
name = 'file'
content = ... # бинарник файла
where = ... # полученная ссылка
options = {
'file': MultipartParam(
name=name,
value=content,
filename=name,
filetype='image/png',
filesize=len(content))
}
data, headers = multipart_encode(options)
pocket = "".join(data)
result = urlfetch.fetch(
url=where,
payload=pocket,
method=urlfetch.POST,
headers=headers)
2.2. 发送照片,方法 #2,请求库:
import requests
name = 'file'
content = ... # бинарник файла
where = ... # полученная ссылка
# пробовал и без этого словаря; всегда ли он нужен?
data = {
'fileName': name,
'fileSize': len(content),
'description': 'undefined',
}
result = requests.post(where, files={name: StringIO(content)}, data=data)
此外,对于第二种方法,我能够获取正在发送的消息的内容:
POST
https://pu.vk.com/c.../upload.php?act=ads_add&mid=...&size=m&rdsn=1&hash_time=...&hash=...&rhash=...&api=1
Content-Length: 15946
Content-Type: multipart/form-data; boundary=b4b260eace4e4a7082a99753b74cf51f
--b4b260eace4e4a7082a99753b74cf51f
Content-Disposition: form-data; name="description"
undefined
--b4b260eace4e4a7082a99753b74cf51f
Content-Disposition: form-data; name="fileSize"
15518
--b4b260eace4e4a7082a99753b74cf51f
Content-Disposition: form-data; name="fileName"
file
--b4b260eace4e4a7082a99753b74cf51f
Content-Disposition: form-data; name="file"; filename="file"
< File binary content >
--b4b260eace4e4a7082a99753b74cf51f--
解决方案被证明是微不足道的,但并不明显。由于对 HTTP 请求的内容进行了大量的实验和研究,我发现了工作请求和非工作请求之间的唯一区别。区别只有 4 个字节:文件名必须包含扩展名。如果您在文件名中指定扩展名(或其他),VK API 会忽略
Content-Type: image/png并成功处理没有它的请求。.pngfilename通常,此代码不起作用:
而这个处理成功:
就像这个在 GAE 中不可用的简单选项一样:
两者
StringIO都StringIO同样适合这项任务。就像我说的,Content-Type没关系,它可能只是multipart/form-data。