在Google App Engine中上传文件

81

我计划创建一个Web应用程序,允许用户降级他们的Visual Studio项目文件。但是,似乎Google App Engine通过db.TextPropertydb.BlobProperty接受文件上传和平面文件存储在Google服务器上。

如果有人能提供如何完成这项任务的代码示例(包括客户端和服务器端),我将非常感谢。


@user858915 链接已经失效了 :( - Marco
11个回答

50

实际上,这个问题在App Engine文档中有解答。请参考上传用户图片的示例。

<form></form>标签内的HTML代码:

<input type="file" name="img"/>

Python 代码:

class Guestbook(webapp.RequestHandler):
  def post(self):
    greeting = Greeting()
    if users.get_current_user():
      greeting.author = users.get_current_user()
    greeting.content = self.request.get("content")
    avatar = self.request.get("img")
    greeting.avatar = db.Blob(avatar)
    greeting.put()
    self.redirect('/')

不喜欢这种方法,因为你会失去 MIME 类型信息。 - santiagobasulto
@santiagobasulto:为什么不自己检查一下呢? - vietean
我不想检查它。当您需要显示图像时,必须提供MIME类型信息(图像是JPG、GIF等),因此,您必须提供content-type HTTP标头。如果您使用所提供的解决方案上传图像,则将来无法知道图片的内容类型,因此无法设置标头,因此旧的浏览器将无法显示图像(由于缺少内容类型)。 - santiagobasulto
1
如果您不检查MIME类型,那么您就是在信任客户端,这会让您面临黑帽攻击或客户端中配置错误的MIME类型。如果您打算这样做,那么您只能信任文件扩展名本身。 - Nacho Coloma

45

这里是一个完整的、可工作的文件。我从Google网站上提取了原始文件并进行了修改,使其更加符合实际情况。

需要注意以下几点:

  1. 此代码使用BlobStore API
  2. ServeHandler类中此行代码的目的是“修复”键,以消除浏览器中可能发生的任何名称混淆(我在Chrome中没有观察到)。

    blob_key = str(urllib.unquote(blob_key))
    

    这里最后的"save_as"子句很重要,它将确保文件名在发送到浏览器时不会被篡改。如果将其删除,则会观察到发生了什么。

  3. 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_handler中的self.__uploadsNone - Tim

10

有一个关于此的 Google Groups 线程:

上传文件

那个讨论中有很多有用的代码,对我上传文件非常有帮助。


6

我今天尝试了一下,它的工作原理如下:

我的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() 

6

谷歌推出了一个用于存储大文件的服务。请查看blobstore API文档。如果您的文件大小超过1MB,建议使用此服务。


如何保存大于1MB的文件。我正在使用github.com/ckopanos/django-google-cloud-storage。 - Geo Jacob

3
如果您仍然遇到问题,请检查表单标签中是否使用了enctype属性。
否:
<form encoding="multipart/form-data" action="/upload">

是的:

<form enctype="multipart/form-data" action="/upload">

在我实施了你的答案之前,我一直遇到编码错误。 - Jader Dias
1
我在做这个项目时遇到了一个非常烦人的问题,那就是没有为文件输入类型包括“size”。我当时使用的是Safari进行测试,它似乎默认具有非常短的文件长度(?),而我在GAE中得到的只是文件名。这里提醒大家注意一下,以免浪费时间。 - John Carter

1

就个人而言,我发现在使用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'属性的表单。删除它可以解决问题,但这意味着我不能上传任何文件。


1
可能是因为你正在使用get方法,请尝试将其设置为post。我不确定它是否有效,但值得一试。 - slashnick

1

由于没有传统的文件系统,您无法存储文件。您只能将它们存储在自己的DataStore中(在定义为BlobProperty的字段中)

在上一个链接中有一个示例:

class MyModel(db.Model):
  blob = db.BlobProperty()

obj = MyModel()
obj.blob = db.Blob( file_contents )

0

我在使用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变量在Google Chrome中是str(),但在Firefox中是unicode。而且,正如你所知道的,db.Blob()构造函数需要一个字符串,如果你传递一个unicode字符串,它会抛出一个错误。

有人知道如何解决这个问题吗?

另外,我发现非常奇怪的是,当我复制并粘贴带有头像的留言板应用程序时,它可以完美地工作。我在我的代码中完全按照相同的方式做了一切,但它就是不起作用。我快要抓狂了。


2
表单上写着:mutlipart/form-data 而不是 multipart/form-data。Chrome 聪明到可以自动修正拼写错误,但 Firefox 不能。 - Honza Pokorny

0

Google App Engine中没有扁平文件存储。所有内容都必须存储到Datastore中,它类似于关系数据库但不完全相同。

您可以将文件存储为TextPropertyBlobProperty属性。

DataStore条目有1MB的限制,这可能是个问题也可能不是。


如何保存大于1MB的文件。我正在使用https://github.com/ckopanos/django-google-cloud-storage - Geo Jacob

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接