For most typical cases the usage should be fairly straightforward. Simply create a
subclass of decorest.RestClient and define methods, which will perform calls
to the actual REST service. You can declare how each function should perform
the request to the service solely using decorators attached to the
method definition. The method itself is not expected to have any implementation,
except for a docstring.
After your API client class definition is complete, simply create an instance
of it and you’re good to go. This library relies on the functionality provided
by either requests or httpx libraries, which means that any valid named argument,
which could be passed to a requests or httpx HTTP call can be also passed to the calls
of the client methods and will be forwarded as is.
For more information checkout sample clients in examples.
- decorest supports currently 2 backends:
-
To select a specific backend, simply pass it’s name to the constructor of the client:
client = DogClient('https://dog.ceo/api', backend='httpx')
Another option is to declare a specific default backend for the client using @backend()
decorator, for instance:
@decorest.backend('httpx')
class DogClient(decorest.RestClient):
@GET('breed/{breed_name}/list')
def list_subbreeds(self, breed_name):
"""List all sub-breeds"""
client = DogClient('https://dog.ceo/api')
If no backend is provided, requests is used by default. The client usage is largely
independent of the backend, however there some minor differences in handling streams
and multipart messages, please consult tests in httpbin test suite
and httpx compatibility guide.
Please note, that asyncio is only supported on the httpx backend.
Below is a list of all supported decorators along with short explanation and
examples. Some decorators can be attached to both client class as well as
methods, in which case the class-level decorator is applied to all HTTP methods
in that class. Furthermore, each decorator can be overridden directly during
the method call by providing a named argument with name equal to the decorator
name.
Marks the request with a specific HTTP method and the path relative to
endpoint provided as argument. The path can contain variables enclosed
in curly brackets, e.g.:
@GET('breed/{breed_name}/list')
def list_subbreeds(self, breed_name):
"""List all sub-breeds"""
which will be replaced by the arguments from the method definition.
These decorators apply only to methods.
Adds a query parameter to the request. URL encoding will be applied to
the value using urlencode, e.g.:
@GET('breed/{breed_name}/list')
@query('long_names', 'longNames')
@query('limit')
def list_subbreeds(self, breed_name, long_names, limit=100):
"""List all sub-breeds"""
This decorator can take a single string parameter, which determines the name
of the method argument whose value will be added as the query argument value
of the same name.
In case 2 arguments are provided, the second argument determines the actual
query key name, which will be used in the request query (if for some reason
it should be different than the method argument name).
Furthermore, if a default value is provided in a method declaration, it
will be used whenever a value for this argument is not provided during
invocation.
For example, the following invocation of the above method:
client.list_subbreeds('hound', 1)
will result in the following query:
https://dog.ceo/api/breed/hound?longNames=1&limit=100
This decorator can be added only to methods.
Adds a multipart parameter to the request. For example:
@POST('post')
@multipart('part1')
@multipart('part_2', 'part2')
@multipart('test')
def post_multipart(self, part1, part_2, test):
"""Return multipart POST data."""
The first parameter to the decorator is the name of the variable in the decorated
method and at the same time the name of the part in HTTP request (which will be
set in the Content-Disposition header. In case the method argument name
should be different than the part name in the request, a second parameter to the
decorator will determine the actual name for the part in the HTTP request.
The values for the arguments can be either strings, which will be added directly
as content in the appropriate part, or tuples. In case a tuple is passed, it will
be treated as a file, the same way as is treated by both backend libraries.
The above method can be thus called as follows:
f = '/tmp/test.dat'
res = client.post_multipart('TEST1', 'TEST2',
('filename', open(f, 'rb'), 'text/plain'))
- which will generate the following parts:
- part part1 with content TEST1
- part part2 with content TEST2
- part test with content read from file /tmp/test.dat
Body decorator enables to specify, which of the method parameters should provide
the body content to the request, e.g.:
@POST('pet')
@header('content-type', 'application/json')
@header('accept', 'application/json')
@body('pet')
def add_pet(self, pet):
"""Add a new pet to the store"""
@body decorator can take an optional argument which provides a serialization
handler, which will be invoked automatically before passing the argument as
body content, which can be a simple lambda or a more complex function with some
logic. For example:
@POST('pet')
@header('content-type', 'application/json')
@header('accept', 'application/json')
@body('pet', lambda p: json.dumps(p))
def add_pet(self, pet):
"""Add a new pet to the store"""
The above code will automatically stringify the dictionary provided as
value of pet argument using json.dumps() function.
By default the request method will not return requests or httpx response object,
but the response will depend on the content type of the response.
In case the HTTP request succeeds the following results are expected:
response.json() if the content type of response is JSON
response.content if the content type is binary
response.text otherwise
In case the request fails, response.raise_for_status() is called and
should be handled in the client code.
In case another behavior is required, custom handlers can be provided
for each method using lambdas or functions. The provided handler is
expected to take only a single argument, which is the requests or httpx
response object, e.g.:
@GET('breed/{breed_name}/list')
@header('accept', 'application/json')
@on(200, lambda r: r.json())
def list_subbreeds(self, breed_name):
"""List all sub-breeds"""
This decorator can be applied to both methods and classes, however when
applied to a class the handler will be called for the method which receives
the provided status code.
The first argument of this decorator must be an int. It is
also possible to pass … (i.e. Ellipsis) object, which is equivalent
to HttpStatus.ANY. Any other value passed for this argument will
raise TypeError.
This decorator is a shortcut for @header(‘content-type’, …), e.g:
@POST('pet')
@content('application/json')
@header('accept', 'application/json')
@body('pet', lambda p: json.dumps(p))
def add_pet(self, pet):
"""Add a new pet to the store"""
This decorator is a shortcut for @header(‘accept’, …), e.g:
@GET('breed/{breed_name}/list')
@content('application/json')
@accept('application/xml')
def list_subbreeds(self, breed_name):
"""List all sub-breeds"""
Multiple @accept() decorators can be added and will be joined into
a list, e.g.:
@GET('breed/{breed_name}/list')
@content('application/json')
@accept('application/xml')
@accept('application/json')
@accept('text/plain')
def list_subbreeds(self, breed_name):
"""List all sub-breeds"""
will submit the following header to the server:
Accept: text/plain, application/json, application/xml
This decorator enables to define a default endpoint for the service,
which then doesn’t have to be provided in the client constructor:
@endpoint('https://dog.ceo/api')
class DogClient(RestClient):
"""List all sub-breeds"""
...
The endpoint provided in the client constructor will take precedence
however.
Specifies a default timeout value (in seconds) for a method or entire API.
@endpoint('https://dog.ceo/api')
@timeout(5)
class DogClient(RestClient):
"""List all sub-breeds"""
...
This decorator allows to specify a method which returns binary stream of data.
Adding this decorator to a method will add a stream=True
argument to the requests or httpx call and will by default returns entire response
object, which then can be accessed for instance using iter_content() method.
...
class MyClient(RestClient):
...
@GET('stream/{n}/{m}')
@stream
@query('size')
@query('offset', 'off')
def stream(self, n, m, size, offset):
"""Get data range"""
...
with client.stream(2,4, 1024, 200) as r:
for b in r.iter_content(chunk_size=100):
content.append(b)
or for an async API:
...
@backend('httpx')
class MyClient(RestClient):
...
@GET('stream/{n}/{m}')
@stream
@query('size')
@query('offset', 'off')
async def stream(self, n, m, size, offset):
"""Get data range"""
...
async def main():
async with client.async_session_() as s:
r = await s.stream(5)
async for _ in r.aiter_raw(chunk_size=100):
content.append(b)
Specifies the default backend to use by the client, currently the only possible
values are ‘requests’ (default) and ‘httpx’, e.g.:
@endpoint('https://dog.ceo/api')
@backend('httpx')
class DogClient(RestClient):
"""List all sub-breeds"""
...
The backend provided in the constructor arguments when creating client instance has precedence
over the value provided in this decorator. This decorator can only be applied to classes.
Based on the functionality provided by the backend HTTP library in the form of
session objects, sessions can significantly improve the performance of the
client in case multiple responses are performed as well as maintain certain
information between requests such as session cookies.
Sessions in decorest can either be created and closed manually:
s = client.session_()
s.list_subbreeds('hound')
s.list_subbreeds('husky')
s.close_()
or can be used via the context manager with operator:
with client.session_() as s:
s.list_subbreeds('hound')
s.list_subbreeds('husky')
All session specific methods begin with a single underscore, in order not
to interfere with any possible API method names defined in the base client
class.
If some additional customization of the session is required, the underlying
requests session or httpx session`object can be retrieved from decorest_
session object using :py:`backend_session_ attribute:
with client.session_() as s:
s.backend_session_.verify = '/path/to/cert.pem'
s.list_subbreeds('hound')
s.list_subbreeds('husky')
Async sessions can be created in a similar manner, using async_session_() method,
for instance:
async def main():
async with client.async_session_() as s:
await s.list_subbreeds('hound')
await s.list_subbreeds('husky')
Since authentication is highly specific to actual invocation of the REST API,
and not to it’s specification, there is not decorator for authentication,
but instead an authentication object (compatible with requests_
or httpx_ authentication mechanism) can be set in the client object using
set_auth_() method, for example:
client.set_auth_(HTTPBasicAuth('user', 'password'))
with client.session_() as s:
s.backend_session_.verify = '/path/to/cert.pem'
s.list_subbreeds('hound')
s.list_subbreeds('husky')
The authentication object will be used in both regular API calls, as well
as when using sessions.
Furthermore, the auth object - specific for selected backend - can be also
passed to the client constructor, e.g.:
client = DogClient(backend='httpx', auth=httpx.BasicAuth('user', 'password'))
For larger API’s it can be useful to be able to split the API definition
into multiple files but still use it from a single instance in the code.
This can be achieved by creating separate client classes for each group
of operations and then create a common class, which inherits from all the
group clients and provides entire API from one instance.
class A(RestClient):
"""API One client"""
@GET('stuff/{sth}')
@on(200, lambda r: r.json())
def get(self, sth: str) -> typing.Any:
"""Get what"""
class B(RestClient):
"""API One client"""
@PUT('stuff/{sth}')
@body('body')
@on(204, lambda _: True)
def put_b(self, sth: str, body: bytes) -> typing.Any:
"""Put sth"""
@endpoint('https://put.example.com')
class BB(B):
"""API One client"""
@PUT('stuff/{sth}')
@body('body')
@on(204, lambda _: True)
def put_bb(self, sth: str, body: bytes) -> typing.Any:
"""Put sth"""
@endpoint('https://patches.example.com')
class C(RestClient):
"""API Three client"""
@PATCH('stuff/{sth}')
@body('body')
@on(204, lambda _: True)
@on(..., lambda _: False)
def patch(self, sth: str, body: bytes) -> typing.Any:
"""Patch sth"""
@accept('application/json')
@content('application/xml')
@header('X-Auth-Key', 'ABCD')
@endpoint('https://example.com')
@backend('httpx')
class InheritedClient(A, BB, C):
...
Please note that the @endpoint() decorator can be specified for each
sub API with a different value if necessary. It will be inherited by methods
backwards with respect to the inheritance chain, i.e. the more abstract class
will use the first endpoint specified in it’s subclass chain. In the above example
method B.put_b() will use ‘https://put.example.com’ endpoint, and
method C.patch() will use ‘https://patches.example.com’.
For real world example checkout the Petstore Swagger client example.