2012-08-10 34 views
11

私はpycurlを使った例を見ましたが、これがどうやって行くのか分からなかったでしょうか?いくつかの例が役に立ちます。ありがとう。python-tornadoで画像をHTML形式でアップロードするには?

class FileHandler(tornado.web.RequestHandler): 
    # get post data 
    file_body = self.request.files['filefieldname'][0]['body'] 
    img = Image.open(StringIO.StringIO(file_body)) 
    img.save("../img/", img.format) 

が、すべてアップロードされたデータをRAMにロードされているので、それは、お勧めできません;:Pythonで

<form action="/file" methods="POST"><!--your code--></form> 

答えて

16

それは簡単です最も良い方法はnginx loadupモジュールを使用することですが、これは複雑です。

+0

ここでdir()は必要ですか? /../img/ディレクトリにファイルをアップロードしたい – gora

+0

@AniruddhaJanaは必要ではなく、コンソールに出力するだけです。 –

+0

@AniruddhaJana申し訳ありませんが、コードはエラーです。私は修正した。 –

15

ここには竜巻アップロードを実装するdemo applicationがあります。ここで

は、サーバーのコードです:

import tornado.httpserver, tornado.ioloop, tornado.options, tornado.web, os.path, random, string 
from tornado.options import define, options 

define("port", default=8888, help="run on the given port", type=int) 

class Application(tornado.web.Application): 
    def __init__(self): 
     handlers = [ 
      (r"/", IndexHandler), 
      (r"/upload", UploadHandler) 
     ] 
     tornado.web.Application.__init__(self, handlers) 

class IndexHandler(tornado.web.RequestHandler): 
    def get(self): 
     self.render("upload_form.html") 

class UploadHandler(tornado.web.RequestHandler): 
    def post(self): 
     file1 = self.request.files['file1'][0] 
     original_fname = file1['filename'] 
     extension = os.path.splitext(original_fname)[1] 
     fname = ''.join(random.choice(string.ascii_lowercase + string.digits) for x in range(6)) 
     final_filename= fname+extension 
     output_file = open("uploads/" + final_filename, 'w') 
     output_file.write(file1['body']) 
     self.finish("file" + final_filename + " is uploaded") 

def main(): 
    http_server = tornado.httpserver.HTTPServer(Application()) 
    http_server.listen(options.port) 
    tornado.ioloop.IOLoop.instance().start() 

if __name__ == "__main__": 
    main() 

唯一のこと、あなたは、このコードからself.request.files[<file_input_name>][0]に位置して、そのファイルの内容を理解する必要があります。必ず、そのフォームはenctype="multipart/form-data"を持っている -

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/> 
<title>Tornado Upload Application</title> 
</head> 
<body> 
<p><h1>Tornado Upload App</h1></p> 
<form enctype="multipart/form-data" action="/upload" method="post"> 
File: <input type="file" name="file1" /> 
<br /> 
<br /> 
<input type="submit" value="upload" /> 
</form> 

ファイルで作業:

はここでhtmlコードです。

+0

小さな一言:python3.xでは、ファイルをバイナリ書き込みモードで開く必要があります。 'output_file = open(" uploads/"+ final_filename、 'wb')'これはpython3の文字列(Unicode)とバイト文字列の扱いが異なるためです。 [more here](https://stackoverflow.com/questions/6224052/what-is-the-difference-between-a-string-and-byte-string) – klaas

4

前のコードでは、ファイル名が正しくなく、エンコーディングが正しくありません。 [「」]構文、なぜわからないと、ファイルのプロパティにアクセスするときに問題に

import tornado.httpserver, tornado.ioloop, tornado.options, tornado.web, os.path, random, string 




class Application(tornado.web.Application): 
    def __init__(self): 
     handlers = [ 
      (r"/", IndexHandler), 
      (r"/upload", UploadHandler) 
     ] 
     tornado.web.Application.__init__(self, handlers) 

class IndexHandler(tornado.web.RequestHandler): 
    def get(self): 
     self.render("tornadoUpload.html") 

class UploadHandler(tornado.web.RequestHandler): 
    def post(self): 
     file1 = self.request.files['file1'][0] 
     original_fname = file1['filename'] 

     output_file = open("uploads/" + original_fname, 'wb') 
     output_file.write(file1['body']) 

     self.finish("file " + original_fname + " is uploaded") 

settings = { 
'template_path': 'templates', 
'static_path': 'static', 
"xsrf_cookies": False 

} 
application = tornado.web.Application([ 
    (r"/", IndexHandler), 
      (r"/upload", UploadHandler) 


], debug=True,**settings) 



print "Server started." 
if __name__ == "__main__": 
    application.listen(8888) 
    tornado.ioloop.IOLoop.instance().start() 
3

は私が走っていたが、私は、構文をドットに切り替えてデータを読み取ることができました: 次のコードは動作します。私はWindowsマシン上にあるので、open( "static/public /" + file_name、 "w")をopen( "static/public /" + file_name、 "wb")に変更しなければなりませんでした。 'wb'がなければ、ファイルは壊れていました。

def uploadFile(self,input_name,file_type): 
      a_file = self.request.files[input_name][0] 
      extension = os.path.splitext(a_file.filename)[1] 

      if file_type is 'photo': 
       type_list = ['.png','.jpg','.jpeg','.gif'] 
      elif file_type is 'attachment': 
       type_list = ['.pdf','.doc','.docx','.xls'] 

      if extension in type_list: 
       file_name = ''.join(random.choice(string.ascii_lowercase + string.digits) for x in range(16)) 
       output_file = open("static/public/" + file_name + extension, 'wb') 
       output_file.write(a_file.body) 
       return (a_file.filename + " has been uploaded.") 
0

tornado.web.RequestHandlerself.request.files方法があります。 結果は

{u'file': [{'body':FILEBODY, 'content_type':CONTENT_TYPE, 'filename': FILENAME}],...} 
関連する問題