Как загрузить изображение с помощью python-торнадо из HTML-формы?

Я видел примеры, которые использовали pycurl, но не мог быть уверен, что с этим связано. Некоторые примеры помогут. Благодарю.

Ответ 1

Это просто:

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

в Python:

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)

но это не рекомендуется, поскольку все загруженные данные загружаются в ОЗУ; лучший способ - использовать модуль загрузки nginx, но это сложно.

Ответ 2

Вот демонстрационное приложение которое реализует загрузку торнадо.

Вот код сервера:

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].

Вот html-код:

<!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>

При работе с файлами - убедитесь, что эта форма имеет enctype="multipart/form-data".

Ответ 3

Предыдущий код возвратил неправильное имя файла и неправильную кодировку. Работает следующий код:

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

Ответ 4

У меня возникали проблемы при доступе к свойствам файлов с синтаксисом [''], не знаю почему, но я переключился на точечный синтаксис и смог прочитать данные. Я нахожусь на машине с Windows, поэтому мне также пришлось изменить "open" ( "static/public/" + file_name, 'w') 'to' 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.")

Ответ 5

tornado.web.RequestHandler имеет метод self.request.files. это результат, как

{u'file': [{'body':FILEBODY, 'content_type':CONTENT_TYPE, 'filename': FILENAME}],...}