python - fontforge не может проанализировать SVG-файл на сервере Tornado

Я пытаюсь смонтировать сервер Tornado на виртуальной машине Ubuntu, задачей которой будет создание полного пакета шрифтов, начиная с файла.svg. Я запустил следующий скрипт без моего сервера, который работал вроде отлично. https://gist.github.com/jorgegarciadev/6127832

И теперь, когда я пытаюсь выполнить эту задачу на моем сервере, Fontforge, похоже, испытывает проблемы с файлом.svg. Сначала я подумал, что это была чанк-проблема, как будто мой файл не был прочитан или отправлен полностью, но когда я открываю недавно загруженный файл на сервере, он завершен и имеет тот же размер, что и исходный файл. И что еще более тревожно, так это то, что Fontforge сообщает мне, что этот файл не является файлом шрифта. Вот что заставило меня задуматься о чанке (потому что, похоже, он разбирает ошибки)

Файл SVG на наклейке прямо здесь: http://pastebin.com/ai5pr1DG

Код сервера Python:

import tornado, tornado.ioloop, tornado.web
import os, uuid
import fontforge, re, zipfile

__UPLOADS__ = "uploads/"
EXTS = [".woff", ".ttf", ".otf", ".svg", ".eot"]

class Userform(tornado.web.RequestHandler):
    def get(self):
        self.render("fileuploadform.html")

class Upload(tornado.web.RequestHandler):
    @staticmethod
    def cssGenerator(name, fullname):
        cssFile = name + ".css"
        template = "@font-face {\
            \n\tfont-family: '" + fullname + "';\
            \n\tsrc: url('" + name + ".eot');\
            \n\tsrc: url('" + name + ".eot?#iefix') format('embedded-opentype'),\
            \n\turl('" + name + ".woff') format('woff'),\
            \n\turl('" + name + ".ttf') format('truetype'),\
            \n\turl('" + name + ".svg#ywftsvg') format('svg');\
            \n\tfont-style: normal;\
            \n\tfont-weight: normal;\
            \n}\n\n"
        open(cssFile, 'w+').writelines(template)

    @staticmethod
    def fontGenerator(filename):
        #exts = ["woff", "ttf", "otf", "svg", "eot"]
        name = os.path.splitext(filename)[0]
        '''if not os.path.exists(name):
            os.makedirs(name)'''

        font = fontforge.open(filename)
        fullname = font.fullname

        for ext in EXTS:
            f = name + ext
            font.generate(f)

    def post(self):
        fileinfo = self.request.files['filearg'][0]
        fname = fileinfo['filename']
        extn = os.path.splitext(fname)[1]
        cname = str(uuid.uuid4()) + extn
        zname = os.path.splitext(fname)[0]
        fh = open(cname, 'w+')
        fh.write(fileinfo['body'])
        os.system("mv " + cname + " " + zname + extn)
        #zname is default, extn is .svg, fname is default.svg, cname is file name in the server
        self.cssGenerator(zname, 'testfont')
        self.fontGenerator(zname + extn)

application = tornado.web.Application([
    (r"/", Userform),
    (r"/upload", Upload),
    ], debug=True)

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

Вывод после запуска сервера и нажатия кнопки "загрузить" на клиенте:

strippedname:/home/bill/proto.svg
/home/bill/proto.svg:109: parser error : AttValue: ' expected
-18 403 -18 L 403 -18 C 352 -18 301 0 251 0 C 225 0 210 15 210 41 C 210 63 225 7
                                                                               ^
/home/bill/proto.svg:109: parser error : attributes construct error
-18 403 -18 L 403 -18 C 352 -18 301 0 251 0 C 225 0 210 15 210 41 C 210 63 225 7
                                                                               ^
/home/bill/proto.svg:109: parser error : Couldn't find end of Start Tag glyph line 109
-18 403 -18 L 403 -18 C 352 -18 301 0 251 0 C 225 0 210 15 210 41 C 210 63 225 7
                                                                               ^
/home/bill/proto.svg:109: parser error : Premature end of data in tag font line 6
-18 403 -18 L 403 -18 C 352 -18 301 0 251 0 C 225 0 210 15 210 41 C 210 63 225 7
                                                                               ^
/home/bill/proto.svg:109: parser error : Premature end of data in tag defs line 5
-18 403 -18 L 403 -18 C 352 -18 301 0 251 0 C 225 0 210 15 210 41 C 210 63 225 7
                                                                               ^
/home/bill/proto.svg:109: parser error : Premature end of data in tag svg line 3
-18 403 -18 L 403 -18 C 352 -18 301 0 251 0 C 225 0 210 15 210 41 C 210 63 225 7
                                                                               ^
Couldn't find a font file named /home/bill/proto.svg
proto.svg is not in a known format (or uses features of that format fontforge does not support, or is so badly corrupted as to be unreadable)
ERROR:tornado.application:Uncaught exception POST /upload (82.225.61.131)
HTTPServerRequest(protocol='http', host='py-fontconv.cloudapp.net', method='POST', uri='/upload', version='HTTP/1.1', remote_ip='82.225.61.131', headers={'Content-Length': '32378', 'Accept-Language': 'en-us,en;q=0.8,fr;q=0.5,fr-fr;q=0.3', 'Accept-Encoding': 'gzip, deflate', 'Host': 'py-fontconv.cloudapp.net', 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8', 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10.7; rv:28.0) Gecko/20100101 Firefox/28.0', 'Connection': 'keep-alive', 'Content-Type': 'multipart/form-data; boundary=---------------------------191335620930032341323215978'})
Traceback (most recent call last):
  File "/usr/local/lib/python2.7/dist-packages/tornado/web.py", line 1332, in _execute
    result = method(*self.path_args, **self.path_kwargs)
  File "tornadofileupload.py", line 53, in post
    self.fontGenerator(zname + extn)
  File "tornadofileupload.py", line 35, in fontGenerator
    font = fontforge.open(filename)
EnvironmentError: Open failed
ERROR:tornado.access:500 POST /upload (82.225.61.131) 283.75ms

Если кто-нибудь узнает, чего здесь не хватает, это было бы таким облегчением, потому что я действительно борюсь с этим.

1 ответ

Решение

Вам необходимо промыть или закрыть fh после записи на него. Лучший способ сделать это с with заявление:

with open(cname, 'w+') as f:
    f.write(fileinfo['body'])

Использование mv на следующей линии опасно небезопасно. Кто-то может загрузить файл с "именем" $(rm -rf *) и эта команда будет выполнена. Никогда не используйте os.system с ненадежным вводом. В этом случае shutil Модуль предоставляет альтернативы. Также проблематично использовать предоставленное пользователем имя файла как zname напрямую - всегда лучше генерировать ваши собственные имена файлов вместо того, чтобы использовать имя, предоставленное клиентом.

Другие вопросы по тегам