Base64在AppEngine上编码二进制上传数据

时间:2010-07-12 02:26:50

标签: python google-app-engine encoding base64

我一直在尝试Base64编码来自用户的图像数据(在这种情况下是一个受信任的管理员),以便尽可能多地跳过对BlobStore的调用。每当我尝试对其进行编码时,我都会收到一条错误消息:

Error uploading image: 'ascii' codec can't decode byte 0x89 in position 0: ordinal not in range(128)

我已经搜索了错误(不太重要的部分)并发现它可能与Unicode(?)有关。模板部分只是一个基本的上传表单,而处理程序包含以下代码:

  def post(self,id):
    logging.info("ImagestoreHandler#post %s", self.request.path)
    fileupload = self.request.POST.get("file",None)
    if fileupload is None : return self.error(400)

    content_type = fileupload.type or getContentType( fileupload.filename )
    if content_type is None: 
      self.error(400)
      self.response.headers['Content-Type'] = 'text/plain'
      self.response.out.write( "Unsupported image type: " + fileupload.filename )
      return
    logging.debug( "File upload: %s, mime type: %s", fileupload.filename, content_type )

    try:
      (img_name, img_url) = self._store_image(
        fileupload.filename, fileupload.file, content_type )
      self.response.headers['Location'] = img_url
      ex=None
    except Exception, err:
      logging.exception( "Error while storing image" )
      self.error(400)
      self.response.headers['Content-Type'] = 'text/plain'
      self.response.out.write("Error uploading image: " + str(err))
      return
    #self.redirect(urlBase % img.key() )  #dummy redirect is acceptable for non-AJAX clients,
    # location header should be acceptable for true REST clients, however AJAX requests 
    # might not be able to access the location header so we'll write a 200 response with 
    # the new URL in the response body:

    acceptType = self.request.accept.best_match( listRenderers.keys() )
    out = self.response.out
    if acceptType == 'application/json':
      self.response.headers['Content-Type'] = 'application/json'
      out.write( '{"name":"%s","href":"%s"}' % ( img_name, img_url ) )
    elif re.search( 'html|xml', acceptType ):
      self.response.headers['Content-Type'] = 'text/html'
      out.write( '<a href="%s">%s</a>' % ( img_url, img_name) )

  def _store_image(self, name, file, content_type):
    """POST handler delegates to this method for actual image storage; as
    a result, alternate implementation may easily override the storage
    mechanism without rewriting the same content-type handling. 

    This method returns a tuple of file name and image URL."""

    img_enc = base64.b64encode(file.read())
    img_enc_struct = "data:%s;base64,%s" % (content_type, img_enc)

    img = Image( name=name, data=img_enc_struct )
    img.put()
    logging.info("Saved image to key %s", img.key() ) 
    return ( str(img.name), img.key() )

我的图像模型:

from google.appengine.ext import db

class Image(db.Model):

  name = db.StringProperty(required=True)
  data = db.TextProperty(required=True)
  created = db.DateTimeProperty(auto_now_add=True)
  owner = db.UserProperty(auto_current_user_add=True)

非常感谢任何帮助。此代码减去_store_image中的图像编码,来自gvdent here的blooger分支。

1 个答案:

答案 0 :(得分:4)

您的商店图片代码可能是这样....

img = Image( name=name, data=file.read() )
img.put()
return ( str(img.name), img.key() )

执行二进制数据的base64encode可能会增加数据本身的大小并增加cpu编码和解码时间。

和Blobstore使用与数据存储相同的存储结构,因此它更容易实现 使用文件上传商店下载。