sanic/sanic/request.py

171 lines
5.2 KiB
Python
Raw Normal View History

2016-10-15 20:59:00 +01:00
from cgi import parse_header
from collections import namedtuple
from http.cookies import SimpleCookie
2016-10-15 20:59:00 +01:00
from httptools import parse_url
from urllib.parse import parse_qs
from ujson import loads as json_loads
from .log import log
class RequestParameters(dict):
"""
Hosts a dict with lists as values where get returns the first
value of the list and getlist returns the whole shebang
"""
def __init__(self, *args, **kwargs):
self.super = super()
self.super.__init__(*args, **kwargs)
def get(self, name, default=None):
values = self.super.get(name)
return values[0] if values else default
def getlist(self, name, default=None):
return self.super.get(name, default)
class Request:
"""
Properties of an HTTP request such as URL, headers, etc.
"""
2016-10-15 20:59:00 +01:00
__slots__ = (
'url', 'headers', 'version', 'method', '_cookies',
2016-10-15 20:59:00 +01:00
'query_string', 'body',
'parsed_json', 'parsed_args', 'parsed_form', 'parsed_files',
)
def __init__(self, url_bytes, headers, version, method):
# TODO: Content-Encoding detection
url_parsed = parse_url(url_bytes)
self.url = url_parsed.path.decode('utf-8')
self.headers = headers
self.version = version
self.method = method
2016-10-16 14:01:59 +01:00
self.query_string = None
if url_parsed.query:
self.query_string = url_parsed.query.decode('utf-8')
2016-10-15 20:59:00 +01:00
# Init but do not inhale
self.body = None
self.parsed_json = None
self.parsed_form = None
self.parsed_files = None
self.parsed_args = None
self._cookies = None
2016-10-15 20:59:00 +01:00
@property
def json(self):
if not self.parsed_json:
try:
self.parsed_json = json_loads(self.body)
2016-10-16 14:01:59 +01:00
except Exception:
2016-10-15 20:59:00 +01:00
pass
return self.parsed_json
@property
def form(self):
if self.parsed_form is None:
self.parsed_form = {}
self.parsed_files = {}
2016-10-16 14:01:59 +01:00
content_type, parameters = parse_header(
2016-11-03 05:34:55 +00:00
self.headers.get('Content-Type', ''))
2016-10-15 20:59:00 +01:00
try:
2016-10-16 14:01:59 +01:00
is_url_encoded = (
content_type == 'application/x-www-form-urlencoded')
if content_type is None or is_url_encoded:
self.parsed_form = RequestParameters(
parse_qs(self.body.decode('utf-8')))
2016-10-15 20:59:00 +01:00
elif content_type == 'multipart/form-data':
# TODO: Stream this instead of reading to/from memory
boundary = parameters['boundary'].encode('utf-8')
2016-10-16 14:01:59 +01:00
self.parsed_form, self.parsed_files = (
parse_multipart_form(self.body, boundary))
2016-10-15 20:59:00 +01:00
except Exception as e:
log.exception(e)
pass
return self.parsed_form
@property
def files(self):
if self.parsed_files is None:
2016-10-16 10:21:24 +01:00
self.form # compute form to get files
2016-10-15 20:59:00 +01:00
return self.parsed_files
@property
def args(self):
if self.parsed_args is None:
if self.query_string:
2016-10-16 14:01:59 +01:00
self.parsed_args = RequestParameters(
parse_qs(self.query_string))
2016-10-15 20:59:00 +01:00
else:
self.parsed_args = {}
return self.parsed_args
@property
def cookies(self):
if self._cookies is None:
if 'Cookie' in self.headers:
cookies = SimpleCookie()
cookies.load(self.headers['Cookie'])
self._cookies = {name: cookie.value
for name, cookie in cookies.items()}
else:
self._cookies = {}
return self._cookies
2016-10-15 20:59:00 +01:00
File = namedtuple('File', ['type', 'body', 'name'])
def parse_multipart_form(body, boundary):
"""
Parses a request body and returns fields and files
:param body: Bytes request body
:param boundary: Bytes multipart boundary
:return: fields (dict), files (dict)
"""
files = {}
fields = {}
form_parts = body.split(boundary)
for form_part in form_parts[1:-1]:
file_name = None
file_type = None
field_name = None
line_index = 2
line_end_index = 0
while not line_end_index == -1:
line_end_index = form_part.find(b'\r\n', line_index)
form_line = form_part[line_index:line_end_index].decode('utf-8')
line_index = line_end_index + 2
if not form_line:
break
colon_index = form_line.index(':')
form_header_field = form_line[0:colon_index]
2016-10-16 14:01:59 +01:00
form_header_value, form_parameters = parse_header(
form_line[colon_index + 2:])
2016-10-15 20:59:00 +01:00
if form_header_field == 'Content-Disposition':
if 'filename' in form_parameters:
file_name = form_parameters['filename']
field_name = form_parameters.get('name')
elif form_header_field == 'Content-Type':
file_type = form_header_value
post_data = form_part[line_index:-4]
if file_name or file_type:
2016-10-16 14:01:59 +01:00
files[field_name] = File(
type=file_type, name=file_name, body=post_data)
2016-10-15 20:59:00 +01:00
else:
fields[field_name] = post_data.decode('utf-8')
return fields, files