.. _quickstart: Quickstart ********** Ready to write your first API client with Uplink? This guide will walk you through what you'll need to know to get started. **First**, make sure you've :ref:`installed (or updated) Uplink `: :: $ pip install -U uplink Defining an API Client ====================== Writing a **structured** API client with Uplink is very simple. To start, create a subclass of :class:`~uplink.Consumer`. For example, here's the beginning of our GitHub client (we'll add some methods to this class soon): .. code-block:: python from uplink import Consumer class GitHub(Consumer): ... When creating an instance of this consumer, we can use the :obj:`base_url` constructor argument to identify the target service. In our case, it's GitHub's public API: .. code-block:: python github = GitHub(base_url="https://api.github.com/") .. note:: :obj:`base_url` is especially useful for creating clients that target separate services with similar APIs; for example, we could use this GitHub consumer to also create clients for any GitHub Enterprise instance for projects hosted outside of the public `GitHub.com `_ service. Another example is creating separate clients for a company's production and staging environments, which are typically hosted on separate domains but expose the same API. So far, this class looks like any other Python class. The real magic happens when you define methods to interact with the webservice using Uplink's HTTP method decorators, which we cover next. .. _making-a-request: Making a Request ================ With Uplink, making a request to a webservice is as simple as invoking a method. Any method of a :class:`Consumer` subclass can be decorated with one of Uplink's HTTP method decorators: :class:`@get `, :class:`@post `, :class:`@put `, :class:`@patch `, :class:`@head `, and :class:`@delete `: .. code-block:: python class GitHub(Consumer): @get("repositories") def get_repos(self): """List all public repositories.""" As shown above, the method's body can be left empty. The decorator's first argument is the resource endpoint: i.e., the relative path from :class:`base_url`, which we covered above: .. code-block:: python @get("repositories") .. note:: To build a request's absolute URL, Uplink resolves the relative path against the :obj:`Consumer`'s base url using :func:`urljoin `, which implements the `RFC 3986`_ standards. For a simplified overview of these standards, see `these recommendations and examples`_ from Retrofit's documentation. .. _RFC 3986: https://tools.ietf.org/html/rfc3986#section-5 .. _these recommendations and examples: https://square.github.io/retrofit/2.x/retrofit/retrofit2/Retrofit.Builder.html#baseUrl-okhttp3.HttpUrl- You can also specify query parameters: .. code-block:: python @get("repositories?since=364") Finally, invoke the method to send a request: .. code-block:: python >>> github = GitHub(base_url="https://api.github.com/") >>> github.get_repos() >>> _.url https://api.github.com/repositories By default, uplink uses `Requests `_, so the response we get back from GitHub is wrapped inside a :class:`requests.Response` instance. (If you want, you can :ref:`swap out ` Requests for a different backing HTTP client, such as :ref:`aiohttp `.) .. |aiohttp| replace:: ``aiohttp`` Path Parameters =============== Resource endpoints can include `URI template parameters `__ that depend on method arguments. A simple URI parameter is an alphanumeric string surrounded by ``{`` and ``}``. To match the parameter with a method argument, either match the argument's name with the alphanumeric string, like so: .. code-block:: python @get("users/{username}") def get_user(self, username): pass or use the :py:class:`~uplink.Path` annotation. .. code-block:: python @get("users/{username}") def get_user(self, name: Path("username")): pass Query Parameters ================ Query parameters can be added dynamically using the :py:class:`~uplink.Query` argument annotation. .. code-block:: python @get("users/{username}/repos") def get_repos(self, username, sort: Query): pass Setting a default value for the query parameter works like you'd expect it to: .. code-block:: python @get("users/{username}/repos") def get_repos(self, username, sort: Query = "created"): pass To make the query parameter optional, set the argument's default value to :obj:`None`. Then, if the argument is not specified at runtime, the parameter will not appear in the request. .. code-block:: python @get("users/{username}/repos") def get_repos(self, username, sort: Query = None): pass Useful for "catch-all" or complex query parameter combinations, the :py:class:`~uplink.QueryMap` annotation accepts a mapping of query parameters: .. code-block:: python @get("users/{username}/repos") def get_repos(self, username, **options: QueryMap): pass You can set static query parameters for a method using the :py:class:`@params ` decorator. .. code-block:: python @params({"client_id": "my-client", "client_secret": "****"}) @get("users/{username}") def get_user(self, username): pass :py:class:`@params ` can be used as a class decorator for query parameters that need to be included with every request: .. code-block:: python @params({"client_id": "my-client", "client_secret": "****"}) class GitHub(Consumer): ... Request Headers =============== You can set static headers for a method using the :py:class:`@headers ` decorator. .. code-block:: python @headers({ "Accept": "application/vnd.github.v3.full+json", "User-Agent": "Uplink-Sample-App" }) @get("users/{username}") def get_user(self, username): pass :py:class:`@headers ` can be used as a class decorator for headers that need to be added to every request: .. code-block:: python @headers({ "Accept": "application/vnd.github.v3.full+json", "User-Agent": "Uplink-Sample-App" }) class GitHub(Consumer): ... A request header can depend on the value of a method argument by using the :py:class:`~uplink.Header` function parameter annotation: .. code-block:: python @get("user") def get_user(self, authorization: Header("Authorization"): """Get an authenticated user.""" Request Body ============ The :py:class:`~uplink.Body` annotation identifies a method argument as the HTTP request body: .. code-block:: python @post("user/repos") def create_repo(self, repo: Body): pass This annotation works well with the **keyword arguments** parameter (denoted by the ``**`` prefix): .. code-block:: python @post("user/repos") def create_repo(self, **repo_info: Body): pass Moreover, this annotation is useful when using supported serialization formats, such as :ref:`JSON ` and `Protocol Buffers `_. Take a look at :ref:`this guide ` for more about serialization with Uplink. .. _json: Form Encoded, Multipart, and JSON Requests ========================================== Methods can also be declared to send form-encoded, multipart, and JSON data. Form-encoded data is sent when :py:class:`@form_url_encoded ` decorates the method. Each key-value pair is annotated with a :py:class:`~uplink.Field` annotation: .. code-block:: python @form_url_encoded @patch("user") def update_user(self, name: Field, email: Field): pass Multipart requests are used when :py:class:`@multipart ` decorates the method. Parts are declared using the :py:class:`~uplink.Part` annotation: .. code-block:: python @multipart @put("user/photo") def upload_photo(self, photo: Part, description: Part): pass JSON data is sent when :py:class:`@json ` decorates the method. The :py:class:`~uplink.Body` annotation declares the JSON payload: .. code-block:: python @json @patch("user") def update_user(self, **user_info: uplink.Body): """Update an authenticated user.""" Alternatively, the :py:class:`~uplink.Field` annotation declares a JSON field: .. code-block:: python @json @patch("user") def update_user_bio(self, bio: Field): """Update the authenticated user's profile bio.""" Handling JSON Responses ======================= Many modern public APIs serve JSON responses to their clients. If your :class:`~uplink.Consumer` subclass accesses a JSON API, you can decorate any method with :class:`@returns.json ` to directly return the JSON response, instead of a response object, when invoked: .. code-block:: python class GitHub(Consumer): @returns.json @get("users/{username}") def get_user(self, username): """Get a single user.""" .. code-block:: python >>> github = GitHub("https://api.github.com") >>> github.get_user("prkumar") {'login': 'prkumar', 'id': 10181244, ... You can also target a specific field of the JSON response by using the decorator's ``key`` argument to select the target JSON field name: .. code-block:: python class GitHub(Consumer): @returns.json(key="blog") @get("users/{username}") def get_blog_url(self, username): """Get the user's blog URL.""" .. code-block:: python >>> github.get_blog_url("prkumar") "https://prkumar.io" .. note:: JSON responses may represent existing Python classes in your application (for example, a ``GitHubUser``). Uplink supports this kind of conversion (i.e., deserialization), and we detail this support in :ref:`the next guide `. .. _`session`: Persistence Across Requests from a :obj:`Consumer` ================================================== The :py:obj:`session` property of a :class:`~uplink.Consumer` instance exposes the instance's configuration and allows for the persistence of certain properties across requests sent from that instance. You can provide default headers and query parameters for requests sent from a consumer instance through its :py:obj:`session` property, like so: .. code-block:: python class GitHub(Consumer): def __init__(self, base_url, username, password): super(GitHub, self).__init__(base_url=base_url) # Creates the API token for this user api_key = create_api_key(username, password) # Send the API token as a query parameter with each request. self.session.params["access_token"] = api_key @get("user/repos") def get_user_repos(self, sort_by: Query("sort")): """Lists public repositories for the authenticated user.""" Headers and query parameters added through the :obj:`session` are applied to all requests sent from the consumer instance. .. code-block:: python github = GitHub("prkumar", "****") # Both `access_token` and `sort` are sent with the request. github.get_user_repos(sort_by="created") Notably, in case of conflicts, the method-level headers and parameters override the session-level, but the method-level properties are not persisted across requests. .. _`custom response handler`: Response and Error Handling =========================== Sometimes, you need to validate a response before it is returned or even calculate a new return value from the response. Or, you may need to handle errors from the underlying client before they reach your users. With Uplink, you can address these concerns by registering a callback with one of these decorators: :class:`@response_handler ` and :class:`@error_handler `. :class:`@response_handler ` registers a callback to intercept responses before they are returned (or deserialized): .. code-block:: python def raise_for_status(response): """Checks whether or not the response was successful.""" if 200 <= response.status_code < 300: # Pass through the response. return response raise UnsuccessfulRequest(response.url) class GitHub(Consumer): @response_handler(raise_for_status) @post("user/repo") def create_repo(self, name: Field): """Create a new repository.""" :class:`@error_handler ` registers a callback to handle an exception thrown by the underlying HTTP client (e.g., :class:`requests.Timeout`): .. code-block:: python def raise_api_error(exc_type, exc_val, exc_tb): """Wraps client error with custom API error""" raise MyApiError(exc_val) class GitHub(Consumer): @error_handler(raise_api_error) @post("user/repo") def create_repo(self, name: Field): """Create a new repository.""" To apply a handler onto all methods of a :py:class:`~uplink.Consumer` subclass, you can simply decorate the class itself: .. code-block:: python @error_handler(raise_api_error) class GitHub(Consumer): ... Notably, the decorators can be stacked on top of one another to chain their behaviors: .. code-block:: python @response_handler(check_expected_headers) # Second, check headers @response_handler(raise_for_status) # First, check success class GitHub(Consumer): ... Lastly, both decorators support the optional argument :obj:`requires_consumer`. When this option is set to :obj:`True`, the registered callback should accept a reference to the :class:`~Consumer` instance as its leading argument: .. code-block:: python :emphasize-lines: 1-2, 11 @error_handler(requires_consumer=True) def raise_api_error(consumer, exc_type, exc_val, exc_tb): """Wraps client error with custom API error""" ... class GitHub(Consumer): @raise_api_error @post("user/repo") def create_repo(self, name: Field): """Create a new repository.""" Retrying ======== `Networks are unreliable `_. Requests can fail for various reasons. In some cases, such as after a connection timeout, simply retrying a failed request is appropriate. The :class:`@retry ` decorator can handle this for you: .. code-block:: python :emphasize-lines: 1,4 from uplink import retry, Consumer, get class GitHub(Consumer): @retry @get("user/{username}") def get_user(self, username): """Get user by username.""" Without any further configuration, the decorator will retry requests that fail *for any reasons*. To constrain which exceptions should prompt a retry attempt, use the ``on_exception`` argument: .. code-block:: python :emphasize-lines: 4,5 from uplink import retry, Consumer, get class GitHub(Consumer): # Retry only on failure to connect to the remote server. @retry(on_exception=retry.CONNECTION_TIMEOUT) @get("user/{username}") def get_user(self, username): """Get user by username.""" Further, as long as the expected exception is thrown, the decorator will repeatedly retry until a response is rendered. If you'd like to cease retrying after a specific number of attempts, use the ``max_attempts`` argument: .. code-block:: python :emphasize-lines: 4,5 from uplink import retry, Consumer, get class GitHub(Consumer): # Try four times, then fail hard if no response. @retry(max_attempts=4) @get("user/{username}") def get_user(self, username): """Get user by username.""" The :class:`@retry ` decorators offers a bunch of other features! Below is a contrived example... checkout the :ref:`API documentation ` for more: .. code-block:: python from uplink import retry, Consumer, get class GitHub(Consumer): @retry( # Retry on 503 response status code or any exception. when=retry.when.status(503) | retry.when.raises(Exception) # Stop after 5 attempts or when backoff exceeds 10 seconds. stop=retry.stop.after_attempt(5) | retry.stop.after_delay(10) # Use exponential backoff with added randomness. backoff=retry.backoff.jittered(multiplier=0.5) ) @get("user/{username}") def get_user(self, username): """Get user by username.""" Finally, like other Uplink decorators, you can decorate a :class:`Consumer` subclass with :class:`@retry ` to :ref:`add retry support to all methods of that class `. .. note:: Response and error handlers (see :ref:`here `) are invoked after the retry condition breaks or after all retry attempts are exhausted, whatever comes first. These callbacks will receive the first response/exception that triggers the retry's ``stop`` condition or doesn't match its ``when`` filter. Client-Side Rate Limiting ========================= Often, an organization may enforce a strict limit on the number of requests a client can make to their public API within a fixed time period (e.g., 15 calls every 15 minutes) to help prevent denial-of-service (DoS) attacks and other issues caused by misbehaving clients. On the client-side, we can avoid exceeding these server-side limits by imposing our own rate limit. The :class:`@ratelimit` decorator enforces a constraint of *X calls every Y seconds*: .. code-block:: python :emphasize-lines: 1, 4 from uplink import ratelimit, Consumer, get class GitHub(Consumer): @ratelimit(calls=15, period=900) # 15 calls every 15 minutes. @get("user/{username}") def get_user(self, username): """Get user by username.""" When the consumer reaches the limit, it will wait until the next period before executing any subsequent requests. For blocking HTTP clients, such as Requests, this means the main thread is blocked until then. On the other hand, :ref:`using a non-blocking client `, such as :mod:`aiohttp`, enables you to continue making progress elsewhere while the consumer waits for the current period to lapse. Alternatively, you can fail fast when the limit is exceeded by setting the ``raise_on_limit`` argument: .. code-block:: python :emphasize-lines: 2,3 class GitHub(Consumer): # Raise Exception when the client exceeds the rate limit. @ratelimit(calls=15, period=900, raise_on_limit=Exception) @get("user/{username}") def get_user(self, username): """Get user by username.""" Like other Uplink decorators, you can decorate a :class:`Consumer` subclass with :class:`@ratelimit ` to :ref:`add rate limiting to all methods of that class `.