Base64 кодирует двоичные загруженные данные на AppEngine - PullRequest
2 голосов
/ 12 июля 2010

Я пытался закодировать 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 здесь .

1 Ответ

4 голосов
/ 12 июля 2010

код вашего магазина может быть таким ...

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

выполнение base64encode двоичных данных может увеличить размер самих данных и увеличить время кодирования и декодирования процессора.

и Blobstore использует ту же самую структуру хранения, что и хранилище данных, так что это просто упрощает использовать загрузку файла загрузки магазина.

...