diff options
| author | Tom Christie | 2014-09-06 07:13:12 +0100 | 
|---|---|---|
| committer | Tom Christie | 2014-09-06 07:13:12 +0100 | 
| commit | 5bbfef36f46979591ad599c56126a8698a47513a (patch) | |
| tree | ace412c0b1eebcac8c7db85cd87a5ddb59257135 | |
| parent | c419fb0f9e395e805a04b9a8c28709a0b6deba31 (diff) | |
| parent | 3f7fad2e5a1dd2cccaff97d662745ac24cc18efa (diff) | |
| download | django-rest-framework-5bbfef36f46979591ad599c56126a8698a47513a.tar.bz2 | |
Merge pull request #1838 from jbittel/fix-encoded-filename-rfc6266
Support RFC6266 encoded filenames
| -rw-r--r-- | rest_framework/parsers.py | 20 | ||||
| -rw-r--r-- | tests/test_parsers.py | 24 | 
2 files changed, 42 insertions, 2 deletions
| diff --git a/rest_framework/parsers.py b/rest_framework/parsers.py index aa4fd3f1..c287908d 100644 --- a/rest_framework/parsers.py +++ b/rest_framework/parsers.py @@ -11,7 +11,7 @@ from django.http import QueryDict  from django.http.multipartparser import MultiPartParser as DjangoMultiPartParser  from django.http.multipartparser import MultiPartParserError, parse_header, ChunkIter  from django.utils import six -from rest_framework.compat import etree, yaml, force_text +from rest_framework.compat import etree, yaml, force_text, urlparse  from rest_framework.exceptions import ParseError  from rest_framework import renderers  import json @@ -290,6 +290,22 @@ class FileUploadParser(BaseParser):          try:              meta = parser_context['request'].META              disposition = parse_header(meta['HTTP_CONTENT_DISPOSITION'].encode('utf-8')) -            return force_text(disposition[1]['filename']) +            filename_parm = disposition[1] +            if 'filename*' in filename_parm: +                return self.get_encoded_filename(filename_parm) +            return force_text(filename_parm['filename'])          except (AttributeError, KeyError):              pass + +    def get_encoded_filename(self, filename_parm): +        """ +        Handle encoded filenames per RFC6266. See also: +        http://tools.ietf.org/html/rfc2231#section-4 +        """ +        encoded_filename = force_text(filename_parm['filename*']) +        try: +            charset, lang, filename = encoded_filename.split('\'', 2) +            filename = urlparse.unquote(filename) +        except (ValueError, LookupError): +            filename = force_text(filename_parm['filename']) +        return filename diff --git a/tests/test_parsers.py b/tests/test_parsers.py index 8af90677..3f2672df 100644 --- a/tests/test_parsers.py +++ b/tests/test_parsers.py @@ -1,3 +1,5 @@ +# -*- coding: utf-8 -*- +  from __future__ import unicode_literals  from rest_framework.compat import StringIO  from django import forms @@ -113,3 +115,25 @@ class TestFileUploadParser(TestCase):          parser = FileUploadParser()          filename = parser.get_filename(self.stream, None, self.parser_context)          self.assertEqual(filename, 'file.txt') + +    def test_get_encoded_filename(self): +        parser = FileUploadParser() + +        self.__replace_content_disposition('inline; filename*=utf-8\'\'ÀĥƦ.txt') +        filename = parser.get_filename(self.stream, None, self.parser_context) +        self.assertEqual(filename, 'ÀĥƦ.txt') + +        self.__replace_content_disposition('inline; filename=fallback.txt; filename*=utf-8\'\'ÀĥƦ.txt') +        filename = parser.get_filename(self.stream, None, self.parser_context) +        self.assertEqual(filename, 'ÀĥƦ.txt') + +        self.__replace_content_disposition('inline; filename=fallback.txt; filename*=utf-8\'en-us\'ÀĥƦ.txt') +        filename = parser.get_filename(self.stream, None, self.parser_context) +        self.assertEqual(filename, 'ÀĥƦ.txt') + +        self.__replace_content_disposition('inline; filename=fallback.txt; filename*=utf-8--ÀĥƦ.txt') +        filename = parser.get_filename(self.stream, None, self.parser_context) +        self.assertEqual(filename, 'fallback.txt') + +    def __replace_content_disposition(self, disposition): +        self.parser_context['request'].META['HTTP_CONTENT_DISPOSITION'] = disposition | 
