Skip to main content

Swagger + Flask + Grequests + Bravado = Client/Server auto-spawning

Project description

Klue Client-Server

Dynamically generate client libraries and Flask servers based on Swagger specifications of REST apis.

Purpose

A typical micro-service will expose a REST API on its server side, while making client calls to a number of other services nearby.

Given a set of Swagger specifications describing the APIs of these services, Klue Client-Server will populate a Flask app with auto-generated server endpoints for all specified routes, as well as generate client libraries for every client API. Both client and server stubs handle marshaling/unmarshaling of json documents to and from objects modeled upon the Swagger definitions for each API, as well as provide format verifications of these objects.

Klue Client-Server relies on bravado-core for marshaling/unmarshaling and format validation.

The generated client library support both synchronous and asynchronous/parallel calls using grequests.

Disclaimer

Klue Client-Server is under active development. Its API is subject to change. It has been tested only on python 2.7.

Usage

First, load the Swagger specifications of all the services your server will use:

from klue.swagger import ApiPool

ApiPool.add('public', yaml_path='public.yaml')
ApiPool.add('login', yaml_path='login.yaml')
ApiPool.add('user', yaml_path='user.yaml', timeout=20)

Usage - Generating Server

In the Swagger spec describing the server side, each endpoint that you want to have auto-generated into the Flask app should have the ‘x-bind-server’ attribute set to the path of a python method that will take as argument an object modelled on the endpoint’s argument, and return an object matching that of the endpoint’s reponses (See bravado-core for details):

/login:
  post:
    summary: Login a user.
    produces:
      - application/json
    x-bind-server: myserver.handlers.do_login
    parameters:
      - in: body
        name: body
        description: User login credentials.
        required: true
        schema:
          $ref: "#/definitions/Credentials"
    responses:
      200:
        description: API version
        schema:
          $ref: '#/definitions/Welcome'
      default:
        description: Error
        schema:
          $ref: '#/definitions/Error'

Populate a Flask app with server endpoints for the ‘login’ api:

app = Flask(__name__)
ApiPool.login.spawn_api(app)

# Optionaly: wrap all server endpoints with a decorator
def analytics_wrapper(f):
    ...
ApiPool.login.spawn_api(app, decorator=analytics_wrapper)

Implement the ‘do_login’ endpoint:

from flask import jsonify
from klue.swagger import ApiPool
from klue.exceptions import KlueExeption

def do_login(credentials):
    if authenticate_user(credentials):
        # Get the class representing bravado-core Welcome objects
        Welcome = ApiPool.login.model.Welcome
        # Instantiate Welcome and return it
        return Welcome(message="Welcome!")
    else:
        # Rise an error in the API's error format, directly as
        # a Flask response object
        r = jsonify({'error': 'INVALID_CREDENTIALS'})
        r.status_code = 401
        return r

Usage - Generating Client

In the Swagger spec describing the server you want to call, each endpoint that you want to have auto-generated into the client library should have the ‘x-bind-client’ attribute set to the path of a python method that will take as argument an object modelled on the endpoint’s argument, and return an object matching that of the endpoint’s reponses (See bravado-core for details):

/version:
  get:
    summary: Return the API''s version.
    produces:
      - application/json
    x-bind-client: version
    responses:
      200:
        description: API version
        schema:
          $ref: '#/definitions/Version'

Calling that server now looks like (assuming the server api is called ‘public’):

from klue.swagger import ApiPool

# Call the /version endpoint on the host:port specified in the Swagger
# spec, and return a Version object:
version = ApiPool.public.client.version().call()

To call multiple server endpoints in parallel:

from klue.swagger import ApiPool
from klue.swagger.client import async_call

# Call two endpoints in parallel:
[result_version, result_login]
    = async_call(
         ApiPool.public.client.version(),
         ApiPool.login.client.login(credentials),
    )

Usage - Authentication

TODO: describe the ‘x-decorate-request’ and ‘x-decorate-server’ attributes of the swagger spec + give example of using them to add-on authentication support.

Usage - Handling Errors

Klue-client-server may raise exceptions, for example if the server stub gets an invalid request according to the swagger specification.

However klue-client-server does not know how to format internal errors into an object model fitting that of the loaded swagger specification. Instead, you should provide the apipool with a callback to format exceptions into whatever object you wish to return instead. Something like:

from klue.swagger import ApiPool

def my_error_formatter(e):
    """Take an exception and return a proper swagger Error object"""
    return ApiPool.public.model.Error(
        type=type(e).__name__,
        raw=str(e),
    )

ApiPool.add('public', yaml_path='public.yaml', error_callback=my_error_formatter)

Internal errors raised by klue-client-server are instances of klue.exceptions.KlueException

Install

pip install klue-client-server

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

klue-client-server-0.0.42.tar.gz (14.6 kB view hashes)

Uploaded Source

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page