ユーザーがVisual StudioプロジェクトファイルをダウングレードできるWebアプリを作成する予定です。ただし、Google App Engineは、db.TextProperty
およびdb.BlobProperty
。
これを実現する方法については、誰でもコードサンプル(クライアント側とサーバー側の両方)を提供できてうれしいです。
これが完全な作業ファイルです。 Googleサイトからオリジナルを引き出して、少し現実に近づけるように修正しました。
注目すべきいくつかの点:
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 __== '__main__':
main()
実際、この質問はApp Egnineのドキュメントで回答されています。 ユーザー画像のアップロード の例を参照してください。
<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( '/')
私は今日それを試して、それは次のように機能します:
私の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()
Googleは大きなファイルを保存するサービスをリリースしました。 blobstore APIドキュメント をご覧ください。ファイルが1 MBを超える場合は、使用する必要があります。
まだ問題がある場合は、formタグでenctypeを使用していることを確認してください
いいえ:
<form encoding="multipart/form-data" action="/upload">
はい:
<form enctype="multipart/form-data" action="/upload">
従来のファイルシステムがないため、ファイルを保存できません。独自のデータストア( BlobProperty として定義されたフィールド)にのみ保存できます
前のリンクに例があります:
class MyModel(db.Model):
blob = db.BlobProperty()
obj = MyModel()
obj.blob = db.Blob( file_contents )
個人的には、JavaランタイムをGAEで使用するときに役立つ here )で説明されているチュートリアルが見つかりました。何らかの理由で、を使用してファイルをアップロードしようとしました
<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 Virtual FileSystem project データストアとmemcache APIを使用して実装され、通常のファイルシステムをエミュレートします。このライブラリを使用すると、同様のファイルシステムアクセス(読み取りおよび書き込み)をプロジェクトで使用できます。
Google App Engineに保存するフラットファイルはありません。すべてを Datastore に入力する必要があります。これはリレーショナルデータベースに少し似ていますが、完全ではありません。
ファイルを 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
_変数はGoogle Chromeではstr()
ですが、FirefoxではUnicodeです。そして今、あなたはdb.Blob()
コンストラクターが文字列を受け取り、ユニコード文字列を渡すとエラーを投げます。
誰もこれを修正する方法を知っていますか?
また、私がまったくおかしいのは、Guestbookアプリケーション(アバター付き)をコピーして貼り付けると、完全に機能することです。私は自分のコードですべてをまったく同じように行いますが、うまくいきません。私は髪を抜くのにとても近いです。