Clients ******* To use a common English metaphor: Uplink stands on the shoulders of giants. Uplink doesn't implement any code to handle HTTP protocol stuff directly; for that, the library delegates to an actual HTTP client, such as Requests or Aiohttp. Whatever backing client you choose, when a request method on a :class:`~uplink.Consumer` subclass is invoked, Uplink ultimately interacts with the backing library's interface, at minimum to submit requests and read responses. This section covers the interaction between Uplink and the backing HTTP client library of your choosing, including how to specify your selection. .. _swap_default_http_client: Swapping Out the Default HTTP Session ===================================== By default, Uplink sends requests using the Requests library. You can configure the backing HTTP client object using the :obj:`client` parameter of the :py:class:`~uplink.Consumer` constructor: .. code-block:: python github = GitHub(BASE_URL, client=...) For example, you can use the :obj:`client` parameter to pass in your own `Requests session `_ object: .. code-block:: python session = requests.Session() session.verify = False github = GitHub(BASE_URL, client=session) Further, this also applies for session objects from other HTTP client libraries that Uplink supports, such as :mod:`aiohttp` (i.e., a custom :class:`~aiohttp.ClientSession` works here, as well). Following the above example, the :obj:`client` parameter also accepts an instance of any :class:`requests.Session` subclass. This makes it easy to leverage functionality from third-party Requests extensions, such as `requests-oauthlib`_, with minimal integration overhead: .. code-block:: python from requests_oauthlib import OAuth2Session session = OAuth2Session(...) api = MyApi(BASE_URL, client=session) .. |requests-oauthlib| replace:: ``requests-oauthlib`` .. _`requests-oauthlib`: https://github.com/requests/requests-oauthlib .. _sync_vs_async: Synchronous vs. Asynchronous ============================ Notably, Requests blocks while waiting for a response from the server. For non-blocking requests, Uplink comes with built-in (but optional) support for :mod:`aiohttp` and :mod:`twisted`. For instance, you can provide the :class:`~uplink.AiohttpClient` when constructing a :class:`~uplink.Consumer` instance: .. code:: python from uplink import AiohttpClient github = GitHub(BASE_URL, client=AiohttpClient()) Checkout `this example on GitHub `_ for more. Handling Exceptions From the Underlying HTTP Client Library =========================================================== Each :class:`~uplink.Consumer` instance has an :attr:`exceptions ` property that exposes an enum of standard HTTP client exceptions that can be handled: .. code-block:: python try: repo = github.create_repo(name="myproject", auto_init=True) except github.exceptions.ConnectionError: # Handle client socket error: ... This approach to handling exceptions decouples your code from the backing HTTP client, improving code reuse and testability. Here are the HTTP client exceptions that are exposed through this property: - :class:`BaseClientException`: Base exception for client connection errors. - :class:`ConnectionError`: A client socket error occurred. - :class:`ConnectionTimeout`: The request timed out while trying to connect to the remote server. - :class:`ServerTimeout`: The server did not send any data in the allotted amount of time. - :class:`SSLError`: An SSL error occurred. - :class:`InvalidURL`: URL used for fetching is malformed. Of course, you can also explicitly catch a particular client error from the backing client (e.g., :class:`requests.FileModeWarning`). This may be useful for handling exceptions that are not exposed through the :attr:`Consumer.exceptions ` property, for example: .. code-block:: python try: repo = github.create_repo(name="myproject", auto_init=True) except aiohttp.ContentTypeError: ... Handling Client Exceptions within an :code:`@error_handler` ----------------------------------------------------------- The :class:`@error_handler ` decorator registers a callback to deal with exceptions thrown by the backing HTTP client. To provide the decorated callback a reference to the :class:`Consumer` instance at runtime, set the decorator's optional argument :attr:`requires_consumer` to :obj:`True`. This enables the error handler to leverage the consumer's :attr:`exceptions ` property: .. code-block:: python :emphasize-lines: 1-2, 9 @error_handler(requires_consumer=True) def raise_api_error(consumer, exc_type, exc_val, exc_tb): """Wraps client error with custom API error""" if isinstance(exc_val, consumer.exceptions.ServerTimeout): # Handle the server timeout specifically: ... class GitHub(Consumer): @raise_api_error @post("user/repo") def create_repo(self, name: Field): """Create a new repository."""