Python应用引擎:如何保存图像?

2024-10-01 07:48:52 发布

您现在位置:Python中文网/ 问答频道 /正文

这是我从flex 4文件引用上传得到的:

在自我请求=

    Request: POST /UPLOAD
    Accept: text/*
    Cache-Control: no-cache
    Connection: Keep-Alive
    Content-Length: 51386
    Content-Type: multipart/form-data; boundary=----------ei4cH2gL6ae0ei4ae0gL6GI3KM7ei4
    Host: localhost:8080
    User-Agent: Shockwave Flash

    ------------ei4cH2gL6ae0ei4ae0gL6GI3KM7ei4
    Content-Disposition: form-data; name="Filename"

    36823_117825034935819_100001249682611_118718_676534_n.jpg
    ------------ei4cH2gL6ae0ei4ae0gL6GI3KM7ei4
    Content-Disposition: form-data; name="Filedata"; filename="36823_117825034935819_100001249682611_118718_676534_n.jpg"
    Content-Type: application/octet-stream

    ���� [AND OTHER STRANGE CHARACTERS]

我的班级:

^{pr2}$

为了把文件保存到磁盘上,我在Upload类中缺少了什么? 我在contentvar中有一些奇怪的字符(在调试中查看)。在


Tags: 文件textnameformdatarequesttypecontent
1条回答
网友
1楼 · 发布于 2024-10-01 07:48:52

An App Engine application cannot:

  • write to the filesystem. Applications must use the App Engine datastore for storing persistent data.

您需要做的是向用户呈现带有文件上载字段的表单。
当表单提交时,文件被上传,Blobstore从文件内容创建一个blob,并返回一个blob键,用于以后检索和服务blob。
允许的最大对象大小为2GB。在

下面是一个可以按原样尝试的工作片段:

#!/usr/bin/env python
#

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>""")

class UploadHandler(blobstore_handlers.BlobstoreUploadHandler):
    def post(self):
        upload_files = self.get_uploads('file') 
        blob_info = upload_files[0]
        self.redirect('/serve/%s' % blob_info.key())

class ServeHandler(blobstore_handlers.BlobstoreDownloadHandler):
    def get(self, resource):
        resource = str(urllib.unquote(resource))
        blob_info = blobstore.BlobInfo.get(resource)
        self.send_blob(blob_info)

def main():
    application = webapp.WSGIApplication(
          [('/', MainHandler),
           ('/upload', UploadHandler),
           ('/serve/([^/]+)?', ServeHandler),
          ], debug=True)
    run_wsgi_app(application)

if __name__ == '__main__':
  main()

编辑1:
在您的特定情况下,您可以使用BlobProperty(限制为1MB)来存储您的请求:

^{pr2}$

然后调整您的webapp.RequestHandler以保存您的请求:

class Upload(webapp.RequestHandler):
    def post(self):
        image = self.request.get("Filedata")
        photo = Photo()
        photo.imageblob = db.Blob(image) 
        photo.put()

编辑2:
你不需要改变应用程序yaml,只需添加一个新的处理程序并将其映射到您的WSGI中。 要检索存储的照片,您应该添加另一个处理程序来处理您的照片:

class DownloadImage(webapp.RequestHandler):
    def get(self):
        photo= db.get(self.request.get("photo_id"))
        if photo:
            self.response.headers['Content-Type'] = "image/jpeg"
            self.response.out.write(photo.imageblob)
        else:
            self.response.out.write("Image not available")

然后映射新的DownloadImage类:

application = webapp.WSGIApplication([
    ...
    ('/i', DownloadImage),
    ...
], debug=True)

您可以使用url获取图像:

yourapp/i?photo_id = photo_key

根据要求,如果出于任何奇怪的原因,您确实希望使用这种url www.mysite.com/i/photo_key.jpg来提供图像,您可以尝试使用以下方法:

class Download(webapp.RequestHandler):
        def get(self, photo_id):
            photo= db.get(db.Key(photo_id))
            if photo:
                self.response.headers['Content-Type'] = "image/jpeg"
                self.response.out.write(photo.imageblob)
            else:
                self.response.out.write("Image not available")

使用稍微不同的映射:

application = webapp.WSGIApplication([
        ...
        ('/i/(\d+)\.jpg', DownloadImage),
        ...
    ], debug=True)

相关问题 更多 >