Flask / Python. 从上传文件中获取mimetype

21 投票
2 回答
25996 浏览
提问于 2025-04-16 02:34

我正在使用Flask这个微框架,版本是0.6,编程语言是Python 2.6。

我需要从上传的文件中获取文件类型(mimetype),这样我才能把它存储起来。

以下是相关的Python/Flask代码:

@app.route('/upload_file', methods=['GET', 'POST'])
def upload_file():
    if request.method == 'POST':
        file = request.files['file']
        mimetype = #FIXME
        if file:
            file.save(os.path.join(UPLOAD_FOLDER, 'File-Name')
            return redirect(url_for('uploaded_file'))
        else:
            return redirect(url_for('upload'))


这段代码是网页的部分:

<form action="upload_file" method=post enctype=multipart/form-data> 
Select file to upload: <input type=file name=file> 
<input type=submit value=Upload> 
</form> 


这段代码可以正常工作,但我需要在文件上传时获取文件类型。我查看了Flask的文档,地址是:http://flask.pocoo.org/docs/api/#incoming-request-data
我知道它可以获取文件类型,但我不知道怎么把它提取出来,想要的格式是文本字符串,比如'txt/plain'。

有什么建议吗?

谢谢。

2 个回答

10

理论上,你可以使用 request.files['YOUR_FILE_KEY'].content_type 来获取文件的类型,但这个实现(下面有具体代码,来源于 werkzeug.datastructures)要么是完全信任客户端提供的信息,要么就是用 mimetypes.guess_type 这个方法,它只检查文件的扩展名(你可以在Python文档中查看 这里)。

class FileMultiDict(MultiDict):

    """A special :class:`MultiDict` that has convenience methods to add
    files to it.  This is used for :class:`EnvironBuilder` and generally
    useful for unittesting.
    .. versionadded:: 0.5
    """

    def add_file(self, name, file, filename=None, content_type=None):
        """Adds a new file to the dict.  `file` can be a file name or
        a :class:`file`-like or a :class:`FileStorage` object.
        :param name: the name of the field.
        :param file: a filename or :class:`file`-like object
        :param filename: an optional filename
        :param content_type: an optional content type
        """
        if isinstance(file, FileStorage):
            value = file
        else:
            if isinstance(file, string_types):
                if filename is None:
                    filename = file
                file = open(file, 'rb')
            if filename and content_type is None:
                content_type = mimetypes.guess_type(filename)[0] or \
                    'application/octet-stream'
            value = FileStorage(file, filename, name, content_type)

        self.add(name, value)

根据你的具体需求,你可能想使用 python-magic,这个工具会根据实际的文件内容来判断文件类型。用法大概是这样的:

import magic


def get_mimetype(data: bytes) -> str:
    """Get the mimetype from file data."""
    f = magic.Magic(mime=True)
    return f.from_buffer(data)


get_mimetype(request.files['YOUR_FILE_KEY'].stream.read(MAX_LENGTH))
45

根据文档file.content_type包含了完整的类型信息和编码,而mimetype只包含了基本的mime类型。

@app.route('/upload_file', methods=['GET', 'POST'])
def upload_file():
    if request.method == 'POST':
        file = request.files.get('file')
        if file:
            mimetype = file.content_type
            filename = werkzeug.secure_filename(file.filename)
            file.save(os.path.join(UPLOAD_FOLDER, filename)
            return redirect(url_for('uploaded_file'))
        else:
            return redirect(url_for('upload'))

撰写回答