# -*- coding: utf-8 -*-
requests.models ~~~~~~~~~~~~~~~
This module contains the primary objects that power Requests. """
# Import encoding now, to avoid implicit import later. # Implicit import within threads may cause LookupError when standard library is in a ZIP, # such as in Embedded Python. See https://github.com/psf/requests/issues/3578.
DecodeError, ReadTimeoutError, ProtocolError, LocationParseError)
HTTPError, MissingSchema, InvalidURL, ChunkedEncodingError, ContentDecodingError, ConnectionError, StreamConsumedError) guess_filename, get_auth_from_url, requote_uri, stream_decode_response_unicode, to_key_val_list, parse_header_links, iter_slices, guess_json_utf, super_len, check_header_validity) Callable, Mapping, cookielib, urlunparse, urlsplit, urlencode, str, bytes, is_py2, chardet, builtin_str, basestring)
#: The set of HTTP status codes that indicate an automatically #: processable redirect. codes.moved, # 301 codes.found, # 302 codes.other, # 303 codes.temporary_redirect, # 307 codes.permanent_redirect, # 308 )
def path_url(self): """Build the path URL to use."""
path = '/'
url.append('?') url.append(query)
def _encode_params(data): """Encode parameters in a piece of data.
Will successfully encode parameters when passed as a dict or a list of 2-tuples. Order is retained if data is a list of 2-tuples but arbitrary if parameters are supplied as a dict. """
return data return data if isinstance(vs, basestring) or not hasattr(vs, '__iter__'): vs = [vs] for v in vs: if v is not None: result.append( (k.encode('utf-8') if isinstance(k, str) else k, v.encode('utf-8') if isinstance(v, str) else v)) else: return data
def _encode_files(files, data): """Build the body for a multipart/form-data request.
Will successfully encode files when passed as a dict or a list of tuples. Order is retained if data is a list of tuples but arbitrary if parameters are supplied as a dict. The tuples may be 2-tuples (filename, fileobj), 3-tuples (filename, fileobj, contentype) or 4-tuples (filename, fileobj, contentype, custom_headers). """ if (not files): raise ValueError("Files must be provided.") elif isinstance(data, basestring): raise ValueError("Data must not be a string.")
new_fields = [] fields = to_key_val_list(data or {}) files = to_key_val_list(files or {})
for field, val in fields: if isinstance(val, basestring) or not hasattr(val, '__iter__'): val = [val] for v in val: if v is not None: # Don't call str() on bytestrings: in Py3 it all goes wrong. if not isinstance(v, bytes): v = str(v)
new_fields.append( (field.decode('utf-8') if isinstance(field, bytes) else field, v.encode('utf-8') if isinstance(v, str) else v))
for (k, v) in files: # support for explicit filename ft = None fh = None if isinstance(v, (tuple, list)): if len(v) == 2: fn, fp = v elif len(v) == 3: fn, fp, ft = v else: fn, fp, ft, fh = v else: fn = guess_filename(v) or k fp = v
if isinstance(fp, (str, bytes, bytearray)): fdata = fp elif hasattr(fp, 'read'): fdata = fp.read() elif fp is None: continue else: fdata = fp
rf = RequestField(name=k, data=fdata, filename=fn, headers=fh) rf.make_multipart(content_type=ft) new_fields.append(rf)
body, content_type = encode_multipart_formdata(new_fields)
return body, content_type
"""Properly register a hook."""
raise ValueError('Unsupported event specified, with event name "%s"' % (event))
self.hooks[event].append(hook)
"""Deregister a previously registered hook. Returns True if the hook existed, False if not. """
try: self.hooks[event].remove(hook) return True except ValueError: return False
"""A user-created :class:`Request <Request>` object.
Used to prepare a :class:`PreparedRequest <PreparedRequest>`, which is sent to the server.
:param method: HTTP method to use. :param url: URL to send. :param headers: dictionary of headers to send. :param files: dictionary of {filename: fileobject} files to multipart upload. :param data: the body to attach to the request. If a dictionary or list of tuples ``[(key, value)]`` is provided, form-encoding will take place. :param json: json for the body to attach to the request (if files or data is not specified). :param params: URL parameters to append to the URL. If a dictionary or list of tuples ``[(key, value)]`` is provided, form-encoding will take place. :param auth: Auth handler or (user, pass) tuple. :param cookies: dictionary or CookieJar of cookies to attach to this request. :param hooks: dictionary of callback hooks, for internal usage.
Usage::
>>> import requests >>> req = requests.Request('GET', 'https://httpbin.org/get') >>> req.prepare() <PreparedRequest [GET]> """
method=None, url=None, headers=None, files=None, data=None, params=None, auth=None, cookies=None, hooks=None, json=None):
# Default empty dicts for dict params.
self.register_hook(event=k, hook=v)
def __repr__(self): return '<Request [%s]>' % (self.method)
"""Constructs a :class:`PreparedRequest <PreparedRequest>` for transmission and returns it.""" p = PreparedRequest() p.prepare( method=self.method, url=self.url, headers=self.headers, files=self.files, data=self.data, json=self.json, params=self.params, auth=self.auth, cookies=self.cookies, hooks=self.hooks, ) return p
"""The fully mutable :class:`PreparedRequest <PreparedRequest>` object, containing the exact bytes that will be sent to the server.
Generated from either a :class:`Request <Request>` object or manually.
Usage::
>>> import requests >>> req = requests.Request('GET', 'https://httpbin.org/get') >>> r = req.prepare() >>> r <PreparedRequest [GET]>
>>> s = requests.Session() >>> s.send(r) <Response [200]> """
#: HTTP verb to send to the server. #: HTTP URL to send the request to. #: dictionary of HTTP headers. # The `CookieJar` used to create the Cookie header will be stored here # after prepare_cookies is called #: request body to send to the server. #: dictionary of callback hooks, for internal usage. #: integer denoting starting position of a readable file-like body.
method=None, url=None, headers=None, files=None, data=None, params=None, auth=None, cookies=None, hooks=None, json=None): """Prepares the entire request with the given parameters."""
# Note that prepare_auth must be last to enable authentication schemes # such as OAuth to work on a fully prepared request.
# This MUST go after prepare_auth. Authenticators could add a hook
def __repr__(self): return '<PreparedRequest [%s]>' % (self.method)
"""Prepares the given HTTP method."""
def _get_idna_encoded_host(host): import idna
try: host = idna.encode(host, uts46=True).decode('utf-8') except idna.IDNAError: raise UnicodeError return host
"""Prepares the given HTTP URL.""" #: Accept objects that have string representations. #: We're unable to blindly call unicode/str functions #: as this will include the bytestring indicator (b'') #: on python 3.x. #: https://github.com/psf/requests/pull/2238 url = url.decode('utf8') else:
# Remove leading whitespaces from url
# Don't do any URL preparation for non-HTTP schemes like `mailto`, # `data` etc to work around exceptions from `url_parse`, which # handles RFC 3986 only. self.url = url return
# Support for unicode domain names and paths. except LocationParseError as e: raise InvalidURL(*e.args)
error = ("Invalid URL {0!r}: No schema supplied. Perhaps you meant http://{0}?") error = error.format(to_native_string(url, 'utf8'))
raise MissingSchema(error)
raise InvalidURL("Invalid URL %r: No host supplied" % url)
# In general, we want to try IDNA encoding the hostname if the string contains # non-ASCII characters. This allows users to automatically get the correct IDNA # behaviour. For strings containing only ASCII characters, we need to also verify # it doesn't start with a wildcard (*), before allowing the unencoded hostname. try: host = self._get_idna_encoded_host(host) except UnicodeError: raise InvalidURL('URL has an invalid label.') raise InvalidURL('URL has an invalid label.')
# Carefully reconstruct the network location netloc += '@' netloc += ':' + str(port)
# Bare domains aren't valid URLs. path = '/'
if isinstance(scheme, str): scheme = scheme.encode('utf-8') if isinstance(netloc, str): netloc = netloc.encode('utf-8') if isinstance(path, str): path = path.encode('utf-8') if isinstance(query, str): query = query.encode('utf-8') if isinstance(fragment, str): fragment = fragment.encode('utf-8')
params = to_native_string(params)
if query: query = '%s&%s' % (query, enc_params) else: query = enc_params
"""Prepares the given HTTP headers."""
# Raise exception on invalid header value.
"""Prepares the given HTTP body data."""
# Check if file, fo, generator, iterator. # If not, run through normal process.
# Nottin' on you.
# urllib3 requires a bytes-like body. Python 2's json.dumps # provides this natively, but Python 3 gives a Unicode string. content_type = 'application/json' body = complexjson.dumps(json) if not isinstance(body, bytes): body = body.encode('utf-8')
hasattr(data, '__iter__'), not isinstance(data, (basestring, list, tuple, Mapping)) ])
try: length = super_len(data) except (TypeError, AttributeError, UnsupportedOperation): length = None
body = data
if getattr(body, 'tell', None) is not None: # Record the current file position before reading. # This will allow us to rewind a file in the event # of a redirect. try: self._body_position = body.tell() except (IOError, OSError): # This differentiates from None, allowing us to catch # a failed `tell()` later when trying to rewind the body self._body_position = object()
if files: raise NotImplementedError('Streamed bodies and files are mutually exclusive.')
if length: self.headers['Content-Length'] = builtin_str(length) else: self.headers['Transfer-Encoding'] = 'chunked' else: # Multi-part file uploads. (body, content_type) = self._encode_files(files, data) else: body = self._encode_params(data) if isinstance(data, basestring) or hasattr(data, 'read'): content_type = None else: content_type = 'application/x-www-form-urlencoded'
# Add content-type if it wasn't explicitly provided. self.headers['Content-Type'] = content_type
"""Prepare Content-Length header based on request method and body""" length = super_len(body) if length: # If length exists, set it. Otherwise, we fallback # to Transfer-Encoding: chunked. self.headers['Content-Length'] = builtin_str(length) # Set Content-Length to 0 for methods that can have a body # but don't provide one. (i.e. not GET or HEAD) self.headers['Content-Length'] = '0'
"""Prepares the given HTTP auth data."""
# If no Auth is explicitly provided, extract it from the URL first.
if isinstance(auth, tuple) and len(auth) == 2: # special-case basic HTTP auth auth = HTTPBasicAuth(*auth)
# Allow auth to make its changes. r = auth(self)
# Update self to reflect the auth changes. self.__dict__.update(r.__dict__)
# Recompute Content-Length self.prepare_content_length(self.body)
"""Prepares the given HTTP cookie data.
This function eventually generates a ``Cookie`` header from the given cookies using cookielib. Due to cookielib's design, the header will not be regenerated if it already exists, meaning this function can only be called once for the life of the :class:`PreparedRequest <PreparedRequest>` object. Any subsequent calls to ``prepare_cookies`` will have no actual effect, unless the "Cookie" header is removed beforehand. """ else: self._cookies = cookiejar_from_dict(cookies)
self.headers['Cookie'] = cookie_header
"""Prepares the given hooks.""" # hooks can be passed as None to the prepare method and to this # method. To prevent iterating over None, simply use an empty list # if hooks is False-y
"""The :class:`Response <Response>` object, which contains a server's response to an HTTP request. """
'_content', 'status_code', 'headers', 'url', 'history', 'encoding', 'reason', 'cookies', 'elapsed', 'request' ]
#: Integer Code of responded HTTP Status, e.g. 404 or 200.
#: Case-insensitive Dictionary of Response Headers. #: For example, ``headers['content-encoding']`` will return the #: value of a ``'Content-Encoding'`` response header.
#: File-like object representation of response (for advanced usage). #: Use of ``raw`` requires that ``stream=True`` be set on the request. #: This requirement does not apply for use internally to Requests.
#: Final URL location of Response.
#: Encoding to decode with when accessing r.text.
#: A list of :class:`Response <Response>` objects from #: the history of the Request. Any redirect responses will end #: up here. The list is sorted from the oldest to the most recent request.
#: Textual reason of responded HTTP Status, e.g. "Not Found" or "OK".
#: A CookieJar of Cookies the server sent back.
#: The amount of time elapsed between sending the request #: and the arrival of the response (as a timedelta). #: This property specifically measures the time taken between sending #: the first byte of the request and finishing parsing the headers. It #: is therefore unaffected by consuming the response content or the #: value of the ``stream`` keyword argument.
#: The :class:`PreparedRequest <PreparedRequest>` object to which this #: is a response.
return self
self.close()
# Consume everything; accessing the content attribute makes # sure the content has been fully read. if not self._content_consumed: self.content
return {attr: getattr(self, attr, None) for attr in self.__attrs__}
for name, value in state.items(): setattr(self, name, value)
# pickled objects do not have .raw setattr(self, '_content_consumed', True) setattr(self, 'raw', None)
def __repr__(self): return '<Response [%s]>' % (self.status_code)
"""Returns True if :attr:`status_code` is less than 400.
This attribute checks if the status code of the response is between 400 and 600 to see if there was a client error or a server error. If the status code, is between 200 and 400, this will return True. This is **not** a check to see if the response code is ``200 OK``. """ return self.ok
"""Returns True if :attr:`status_code` is less than 400.
This attribute checks if the status code of the response is between 400 and 600 to see if there was a client error or a server error. If the status code, is between 200 and 400, this will return True. This is **not** a check to see if the response code is ``200 OK``. """ return self.ok
"""Allows you to use a response as an iterator.""" return self.iter_content(128)
def ok(self): """Returns True if :attr:`status_code` is less than 400, False if not.
This attribute checks if the status code of the response is between 400 and 600 to see if there was a client error or a server error. If the status code is between 200 and 400, this will return True. This is **not** a check to see if the response code is ``200 OK``. """ try: self.raise_for_status() except HTTPError: return False return True
def is_redirect(self): """True if this Response is a well-formed HTTP redirect that could have been processed automatically (by :meth:`Session.resolve_redirects`). """
def is_permanent_redirect(self): """True if this Response one of the permanent versions of redirect.""" return ('location' in self.headers and self.status_code in (codes.moved_permanently, codes.permanent_redirect))
def next(self): """Returns a PreparedRequest for the next request in a redirect chain, if there is one.""" return self._next
def apparent_encoding(self): """The apparent encoding, provided by the chardet library.""" return chardet.detect(self.content)['encoding']
"""Iterates over the response data. When stream=True is set on the request, this avoids reading the content at once into memory for large responses. The chunk size is the number of bytes it should read into memory. This is not necessarily the length of each item returned as decoding can take place.
chunk_size must be of type int or None. A value of None will function differently depending on the value of `stream`. stream=True will read data as it arrives in whatever size the chunks are received. If stream=False, data is returned as a single chunk.
If decode_unicode is True, content will be decoded using the best available encoding based on the response. """
# Special case for urllib3. except ProtocolError as e: raise ChunkedEncodingError(e) except DecodeError as e: raise ContentDecodingError(e) except ReadTimeoutError as e: raise ConnectionError(e) else: # Standard file-like object. while True: chunk = self.raw.read(chunk_size) if not chunk: break yield chunk
raise StreamConsumedError() raise TypeError("chunk_size must be an int, it is instead a %s." % type(chunk_size)) # simulate reading small chunks of the content
chunks = stream_decode_response_unicode(chunks, self)
"""Iterates over the response data, one line at a time. When stream=True is set on the request, this avoids reading the content at once into memory for large responses.
.. note:: This method is not reentrant safe. """
pending = None
for chunk in self.iter_content(chunk_size=chunk_size, decode_unicode=decode_unicode):
if pending is not None: chunk = pending + chunk
if delimiter: lines = chunk.split(delimiter) else: lines = chunk.splitlines()
if lines and lines[-1] and chunk and lines[-1][-1] == chunk[-1]: pending = lines.pop() else: pending = None
for line in lines: yield line
if pending is not None: yield pending
def content(self): """Content of the response, in bytes."""
# Read the contents. raise RuntimeError( 'The content for this response was already consumed')
self._content = None else:
# don't need to release the connection; that's been handled by urllib3 # since we exhausted the data.
def text(self): """Content of the response, in unicode.
If Response.encoding is None, encoding will be guessed using ``chardet``.
The encoding of the response content is determined based solely on HTTP headers, following RFC 2616 to the letter. If you can take advantage of non-HTTP knowledge to make a better guess at the encoding, you should set ``r.encoding`` appropriately before accessing this property. """
# Try charset from content-type
return str('')
# Fallback to auto-detected encoding. encoding = self.apparent_encoding
# Decode unicode from given encoding. except (LookupError, TypeError): # A LookupError is raised if the encoding was not found which could # indicate a misspelling or similar mistake. # # A TypeError can be raised if encoding is None # # So we try blindly encoding. content = str(self.content, errors='replace')
r"""Returns the json-encoded content of a response, if any.
:param \*\*kwargs: Optional arguments that ``json.loads`` takes. :raises ValueError: If the response body does not contain valid json. """
if not self.encoding and self.content and len(self.content) > 3: # No encoding set. JSON RFC 4627 section 3 states we should expect # UTF-8, -16 or -32. Detect which one to use; If the detection or # decoding fails, fall back to `self.text` (using chardet to make # a best guess). encoding = guess_json_utf(self.content) if encoding is not None: try: return complexjson.loads( self.content.decode(encoding), **kwargs ) except UnicodeDecodeError: # Wrong UTF codec detected; usually because it's not UTF-8 # but some other 8-bit codec. This is an RFC violation, # and the server didn't bother to tell us what codec *was* # used. pass return complexjson.loads(self.text, **kwargs)
def links(self): """Returns the parsed header links of the response, if any."""
header = self.headers.get('link')
# l = MultiDict() l = {}
if header: links = parse_header_links(header)
for link in links: key = link.get('rel') or link.get('url') l[key] = link
return l
def raise_for_status(self): """Raises :class:`HTTPError`, if one occurred."""
http_error_msg = '' if isinstance(self.reason, bytes): # We attempt to decode utf-8 first because some servers # choose to localize their reason strings. If the string # isn't utf-8, we fall back to iso-8859-1 for all other # encodings. (See PR #3538) try: reason = self.reason.decode('utf-8') except UnicodeDecodeError: reason = self.reason.decode('iso-8859-1') else: reason = self.reason
if 400 <= self.status_code < 500: http_error_msg = u'%s Client Error: %s for url: %s' % (self.status_code, reason, self.url)
elif 500 <= self.status_code < 600: http_error_msg = u'%s Server Error: %s for url: %s' % (self.status_code, reason, self.url)
if http_error_msg: raise HTTPError(http_error_msg, response=self)
"""Releases the connection back to the pool. Once this method has been called the underlying ``raw`` object must not be accessed again.
*Note: Should not normally need to be called explicitly.* """ self.raw.close()
|