如何将文件存储在Google App Engine上的URL上的Google存储中?
-
29-09-2019 - |
题
我想在Google App Engine(Python)上创建一个服务,该服务将接收图像的URL并将其存储在Google存储中。我设法使用本地文件上传 boto
或者 gsutil
命令行,但不能通过通过URL检索文件。我尝试使用 http请求(PUT
) 而且我会收到错误签名的错误响应。显然我在做错事,但不幸的是我不知道在哪里。
因此,我的问题是:如何从URL中检索文件并使用Python用于Google App Angine的Google存储中?
这是我所做的(使用另一个 回答):
class ImportPhoto(webapp.RequestHandler):
def get(self):
self.response.headers['Content-Type'] = 'text/plain'
srow = self.response.out.write
url = self.request.get('url')
srow('URL: %s\n' % (url))
image_response = urlfetch.fetch(url)
m = md5.md5()
m.update(image_response.content)
hash = m.hexdigest()
time = "%s" % datetime.utcnow().strftime("%a, %d %b %Y %H:%M:%S GMT")
str_to_sig = "PUT\n" + hash + "\n\n" +
time + "\nx-goog-acl:public-read\n/lipis/8418.png"
sig = base64.b64encode(hmac.new(
config_credentials.GS_SECRET_ACCESS_KEY,
str_to_sig, hashlib.sha1).digest())
total = len(image_response.content)
srow('Size: %d bytes\n' % (total))
header = {"Date": time,
"x-goog-acl": "public-read",
"Content-MD5": hash,
'Content-Length': total,
'Authorization': "GOOG1 %s:%s" %
(config_credentials.GS_ACCESS_KEY_ID, sig)}
conn = httplib.HTTPConnection("lipis.commondatastorage.googleapis.com")
conn.set_debuglevel(2)
conn.putrequest('PUT', "/8418.png")
for h in header:
conn.putheader(h, header[h])
conn.endheaders()
conn.send(image_response.content + '\r\n')
res = conn.getresponse()
srow('\n\n%d: %s\n' % (res.status, res.reason))
data = res.read()
srow(data)
conn.close()
我得到的回应:
URL: https://stackoverflow.com/users/flair/8418.png
Size: 9605 bytes
400: Bad Request
<?xml version='1.0' encoding='UTF-8'?><Error><Code>BadDigest</Code><Message>The Content-MD5 you specified did not match what we received.</Message><Details>lipis/hello.jpg</Details></Error>
解决方案
你阅读文档了吗 如何签署请求?要签名的字符串必须包括 Content-MD5
, Content-Type
和 Date
标头,除了自定义标题和资源路径。
其他提示
Content-MD5
标题是可选的 提出请求. 。尝试将其排除在测试中。
另外,所需的标题是 Authorization
, Date
和 Host
. 。看来您的要求缺失了 Host
标题。
不隶属于 StackOverflow