Used Content-Disposition filename (#104).

This commit is contained in:
Jakub Roztocil 2013-04-12 10:19:49 -03:00
parent 40bd8f65af
commit 9d043eb745

View File

@ -34,7 +34,7 @@ class ContentRangeError(ValueError):
pass
def _parse_content_range(content_range, resumed_from):
def parse_content_range(content_range, resumed_from):
"""
Parse and validate Content-Range header.
@ -90,6 +90,48 @@ def _parse_content_range(content_range, resumed_from):
return last_byte_pos + 1
def filename_from_content_disposition(content_disposition):
"""
Extract and validate filename from a Content-Disposition header.
:param content_disposition: Content-Disposition value
:return: the filename if present and valid, otherwise `None`
"""
# attachment; filename=jkbr-httpie-0.4.1-20-g40bd8f6.tar.gz
match = re.search('filename=(\S+)', content_disposition)
if match and match.group(1):
fn = match.group(1).lstrip('.')
if re.match('^[a-zA-Z0-9._-]+$', fn):
return fn
def filename_from_url(url, content_type):
fn = urlsplit(url).path.rstrip('/')
fn = os.path.basename(fn) if fn else 'index'
if '.' not in fn and content_type:
content_type = content_type.split(';')[0]
if content_type == 'text/plain':
# mimetypes returns '.ksh'
ext = '.txt'
else:
ext = mimetypes.guess_extension(content_type)
if ext:
fn += ext
return fn
def get_unique_filename(fn, exists=os.path.exists):
attempt = 0
while True:
suffix = '-' + str(attempt) if attempt > 0 else ''
if not exists(fn + suffix):
return fn + suffix
attempt += 1
class Download(object):
def __init__(self, output_file=None,
@ -161,7 +203,7 @@ class Download(object):
if self._resume and response.status_code == PARTIAL_CONTENT:
content_range = response.headers.get('Content-Range')
if content_range:
total_size = _parse_content_range(
total_size = parse_content_range(
content_range, self._resumed_from)
else:
@ -171,14 +213,16 @@ class Download(object):
else:
# TODO: Should the filename be taken from response.history[0].url?
# Output file not specified. Pick a name that doesn't exist yet.
content_type = response.headers.get('Content-Type', '')
self._output_file = open(
self._get_unique_output_filename(
fn = None
if 'Content-Disposition' in response.headers:
fn = filename_from_content_disposition(
response.headers['Content-Disposition'])
if not fn:
fn = filename_from_url(
url=response.url,
content_type=content_type,
),
mode='a+b'
content_type=response.headers.get('Content-Type'),
)
self._output_file = open(get_unique_filename(fn), mode='a+b')
self._progress.started(
resumed_from=self._resumed_from,
@ -225,29 +269,6 @@ class Download(object):
"""
self._progress.chunk_downloaded(len(chunk))
def _get_unique_output_filename(self, url, content_type):
suffix = 0
while True:
fn = self._get_output_filename(url, content_type, suffix)
if not os.path.exists(fn):
return fn
suffix += 1
def _get_output_filename(self, url, content_type, suffix=None):
suffix = '' if not suffix else '-' + str(suffix)
fn = urlsplit(url).path.rstrip('/')
fn = os.path.basename(fn) if fn else 'index'
if '.' in fn:
base, ext = os.path.splitext(fn)
else:
base = fn
ext = mimetypes.guess_extension(content_type.split(';')[0]) or ''
return base + suffix + ext
class Progress(object):