我计划创建一个 Web 应用程序,允许用户降级他们的 Visual Studio 项目文件。但是,Google App Engine 似乎通过 和 接受文件上传和平面文件存储在 Google 服务器db.TextProperty
上db.BlobProperty
。
我很高兴任何人都可以提供代码示例(客户端和服务器端)来说明如何做到这一点。
我计划创建一个 Web 应用程序,允许用户降级他们的 Visual Studio 项目文件。但是,Google App Engine 似乎通过 和 接受文件上传和平面文件存储在 Google 服务器db.TextProperty
上db.BlobProperty
。
我很高兴任何人都可以提供代码示例(客户端和服务器端)来说明如何做到这一点。
事实上,这个问题在 App Egnine 文档中得到了解答。请参阅上传用户图像的示例。
<form></form> 内的 HTML 代码:
<input type="file" name="img"/>
Python代码:
类留言簿(webapp.RequestHandler): def post(自我): 问候=问候() 如果 users.get_current_user(): greeting.author = users.get_current_user() greeting.content = self.request.get("content") 头像 = self.request.get("img") 问候.头像 = db.Blob(头像) 问候语.put() self.redirect('/')
这是一个完整的工作文件。我从谷歌网站上提取了原件并对其进行了修改,使其更加真实。
需要注意的几点:
ServeHandler 类中这一行的目的是“修复”密钥,以便它摆脱浏览器中可能发生的任何名称修改(我在 Chrome 中没有观察到任何名称)
blob_key = str(urllib.unquote(blob_key))
最后的“save_as”子句很重要。它将确保文件名在发送到您的浏览器时不会被破坏。摆脱它以观察会发生什么。
self.send_blob(blobstore.BlobInfo.get(blob_key), save_as=True)
祝你好运!
import os
import urllib
from google.appengine.ext import blobstore
from google.appengine.ext import webapp
from google.appengine.ext.webapp import blobstore_handlers
from google.appengine.ext.webapp import template
from google.appengine.ext.webapp.util import run_wsgi_app
class MainHandler(webapp.RequestHandler):
def get(self):
upload_url = blobstore.create_upload_url('/upload')
self.response.out.write('<html><body>')
self.response.out.write('<form action="%s" method="POST" enctype="multipart/form-data">' % upload_url)
self.response.out.write("""Upload File: <input type="file" name="file"><br> <input type="submit" name="submit" value="Submit"> </form></body></html>""")
for b in blobstore.BlobInfo.all():
self.response.out.write('<li><a href="/serve/%s' % str(b.key()) + '">' + str(b.filename) + '</a>')
class UploadHandler(blobstore_handlers.BlobstoreUploadHandler):
def post(self):
upload_files = self.get_uploads('file')
blob_info = upload_files[0]
self.redirect('/')
class ServeHandler(blobstore_handlers.BlobstoreDownloadHandler):
def get(self, blob_key):
blob_key = str(urllib.unquote(blob_key))
if not blobstore.get(blob_key):
self.error(404)
else:
self.send_blob(blobstore.BlobInfo.get(blob_key), save_as=True)
def main():
application = webapp.WSGIApplication(
[('/', MainHandler),
('/upload', UploadHandler),
('/serve/([^/]+)?', ServeHandler),
], debug=True)
run_wsgi_app(application)
if __name__ == '__main__':
main()
谷歌发布了一项存储大文件的服务。查看blobstore API 文档。如果您的文件 > 1MB,则应该使用它。
我今天试了一下,它的工作原理如下:
我的 sdk 版本是 1.3.x
html页面:
<form enctype="multipart/form-data" action="/upload" method="post" >
<input type="file" name="myfile" />
<input type="submit" />
</form>
服务器代码:
file_contents = self.request.POST.get('myfile').file.read()
如果您仍然有问题,请检查您是否在表单标签中使用 enctype
不:
<form encoding="multipart/form-data" action="/upload">
是的:
<form enctype="multipart/form-data" action="/upload">
您不能存储文件,因为没有传统的文件系统。您只能将它们存储在它们自己的 DataStore 中(在定义为BlobProperty的字段中)
上一个链接中有一个例子:
class MyModel(db.Model):
blob = db.BlobProperty()
obj = MyModel()
obj.blob = db.Blob( file_contents )
就我个人而言,我发现此处描述的教程在将 Java 运行时与 GAE 结合使用时很有用。出于某种原因,当我尝试使用上传文件时
<form action="/testservelet" method="get" enctype="multipart/form-data">
<div>
Myfile:<input type="file" name="file" size="50"/>
</div>
<div>
<input type="submit" value="Upload file">
</div>
</form>
我发现我的 HttpServlet 类由于某种原因不接受带有“enctype”属性的表单。删除它可以,但是,这意味着我无法上传任何文件。
Google App Engine 中没有存储平面文件。一切都必须进入数据存储区,这有点像关系数据库,但又不完全一样。
您可以将文件存储为TextProperty或BlobProperty属性。
DataStore 条目有 1MB 的限制,这可能是也可能不是问题。
在 App Engine 上上传文件时,我观察到一些奇怪的行为。当您提交以下表格时:
<form method="post" action="/upload" enctype="multipart/form-data">
<input type="file" name="img" />
...
</form>
然后你img
像这样从请求中提取:
img_contents = self.request.get('img')
该img_contents
变量str()
在 Google Chrome 中是 a,但在 Firefox 中是 unicode。就像你现在一样,db.Blob()
构造函数接受一个字符串,如果你传入一个 unicode 字符串,它将抛出一个错误。
有谁知道如何解决这个问题?
此外,我发现绝对奇怪的是,当我复制和粘贴留言簿应用程序(带有头像)时,它可以完美运行。我在我的代码中以完全相同的方式执行所有操作,但它不起作用。我非常接近把头发拉出来。
有一种使用平面文件系统的方法(从使用角度来看至少)
有这个Google App Engine Virtual FileSystem 项目。这是在数据存储和内存缓存 API 的帮助下实现的,以模拟普通文件系统。使用这个库,您可以在项目中使用类似的文件系统访问(读取和写入)。