Skip to main content

A fast and durable bidirectional JSON RPC channel over Websockets and FastApi.

Project description

⚡ FASTAPI Websocket RPC

Tests

A fast and durable bidirectional JSON RPC channel over Websockets. The easiest way to create a live async channel between two nodes via Python.

  • Both server and clients can easily expose Python methods that can be called by the other side. Method return values are sent back as RPC responses, which the other side can wait on.
  • Remote methods are easily called via the .other.method() wrapper
  • Connections are kept alive with a configurable retry mechanism (using Tenacity)

Supports and tested on Python >= 3.7

Installation 🛠️

pip install fastapi_websocket_rpc

RPC call example:

Say the server exposes a "add" method, e.g. :

class RpcCalculator(RpcMethodsBase):
    async def add(self, a, b):
        return a + b

Calling it is as easy as calling the method under the client's "other" property:

response = await client.other.add(a=1,b=2)
print(response.result) # 3

getting the response with the return value.

Usage example:

Server:

import uvicorn
from fastapi import FastAPI
from fastapi_websocket_rpc import RpcMethodsBase, WebsocketRPCEndpoint

# Methods to expose to the clients
class ConcatServer(RpcMethodsBase):
    async def concat(self, a="", b=""):
        return a + b

# Init the FAST-API app
app =  FastAPI()
# Create an endpoint and load it with the methods to expose
endpoint = WebsocketRPCEndpoint(ConcatServer())
# add the endpoint to the app
endpoint.register_route(app, "/ws")

# Start the server itself
uvicorn.run(app, host="0.0.0.0", port=9000)

Client

import asyncio
from fastapi_websocket_rpc import RpcMethodsBase, WebSocketRpcClient

async def run_client(uri):
    async with WebSocketRpcClient(uri, RpcMethodsBase()) as client:
        # call concat on the other side
        response = await client.other.concat(a="hello", b=" world")
        # print result
        print(response.result)  # will print "hello world"

# run the client until it completes interaction with server
asyncio.get_event_loop().run_until_complete(
    run_client("ws://localhost:9000/ws")
)

See the examples and tests folders for more server and client examples

Server calling client example:

  • Clients can call client.other.method()
    • which is a shortcut for channel.other.method()
  • Servers also get the channel object and can call remote methods via channel.other.method()
  • See the bidirectional call example for calling client from server and server events (e.g. on_connect).

What can I do with this?

Websockets are ideal to create bi-directional realtime connections over the web.

  • Push updates
  • Remote control mechanism
  • Pub / Sub (see fastapi_websocket_pubsub)
  • Trigger events (see "tests/trigger_flow_test.py")
  • Node negotiations (see "tests/advanced_rpc_test.py :: test_recursive_rpc_calls")

Concepts

  • RpcChannel - implements the RPC-protocol over the websocket

    • Sending RpcRequests per method call
    • Creating promises to track them (via unique call ids), and allow waiting for responses
    • Executing methods on the remote side and serializing return values as
    • Receiving RpcResponses and delivering them to waiting callers
  • RpcMethods - classes passed to both client and server-endpoint inits to expose callable methods to the other side.

    • Simply derive from RpcMethodsBase and add your own async methods
    • Note currently only key-word arguments are supported
    • Checkout RpcUtilityMethods for example methods, which are also useful debugging utilities
  • Foundations:

    • Based on asyncio for the power of Python coroutines

    • Server Endpoint:

      • Based on FAST-API: enjoy all the benefits of a full ASGI platform, including Async-io and dependency injections (for example to authenticate connections)

      • Based on Pydnatic: easily serialize structured data as part of RPC requests and responses (see 'tests/basic_rpc_test.py :: test_structured_response' for an example)

    • Client :

      • Based on Tenacity: allowing configurable retries to keep to connection alive
        • see WebSocketRpcClient.init's retry_config
      • Based on python websockets - a more comprehensive client than the one offered by Fast-api

Pull requests - welcome!

  • Please include tests for new features

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

fastapi_websocket_rpc-0.1.5.tar.gz (13.8 kB view details)

Uploaded Source

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

fastapi_websocket_rpc-0.1.5-py3.8.egg (25.7 kB view details)

Uploaded Egg

fastapi_websocket_rpc-0.1.5-py3-none-any.whl (14.0 kB view details)

Uploaded Python 3

File details

Details for the file fastapi_websocket_rpc-0.1.5.tar.gz.

File metadata

  • Download URL: fastapi_websocket_rpc-0.1.5.tar.gz
  • Upload date:
  • Size: 13.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.3.0 pkginfo/1.7.0 requests/2.25.0 setuptools/49.2.1 requests-toolbelt/0.9.1 tqdm/4.56.0 CPython/3.8.6

File hashes

Hashes for fastapi_websocket_rpc-0.1.5.tar.gz
Algorithm Hash digest
SHA256 f206d400478c309c580fecea1cec1375603d824efc3caa40243d7f3b2414b19b
MD5 28287f35c3223338639fc9e0202b6ad1
BLAKE2b-256 121309bc107b0565ad38c1574a67b69e340197e708e6ea533fdb06e452992953

See more details on using hashes here.

File details

Details for the file fastapi_websocket_rpc-0.1.5-py3.8.egg.

File metadata

  • Download URL: fastapi_websocket_rpc-0.1.5-py3.8.egg
  • Upload date:
  • Size: 25.7 kB
  • Tags: Egg
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.4.1 importlib_metadata/3.7.3 pkginfo/1.7.0 requests/2.25.1 requests-toolbelt/0.9.1 tqdm/4.59.0 CPython/3.8.3

File hashes

Hashes for fastapi_websocket_rpc-0.1.5-py3.8.egg
Algorithm Hash digest
SHA256 ba920aafab72d6e002dd58b8f5e5a489a1b825cadc3d2a2fff3c434119d2cfb7
MD5 ef1bb0c661e1bf592a988ac8cc809f06
BLAKE2b-256 3557274bc1fe2333094e8414890b0deb1d234aa828af03f1960ad3461aeb128e

See more details on using hashes here.

File details

Details for the file fastapi_websocket_rpc-0.1.5-py3-none-any.whl.

File metadata

  • Download URL: fastapi_websocket_rpc-0.1.5-py3-none-any.whl
  • Upload date:
  • Size: 14.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.3.0 pkginfo/1.7.0 requests/2.25.0 setuptools/49.2.1 requests-toolbelt/0.9.1 tqdm/4.56.0 CPython/3.8.6

File hashes

Hashes for fastapi_websocket_rpc-0.1.5-py3-none-any.whl
Algorithm Hash digest
SHA256 abf66fae5c2b996ab924ff723a400a8d297d312da5192ea23755d9f732c65094
MD5 6c006221ad2ba44bdd19866932120cd2
BLAKE2b-256 084118c5cefea0c52101ff8063cc40fb141418e7bb357b144e2122716406f9ff

See more details on using hashes here.

Supported by

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