Typed wrapper for python-socketio with Pydantic validation and dependency injection.
Project description
ZnDraw SocketIO
This package provides an opinionated typed interface to the python-socketio library using pydantic models.
from zndraw_socketio import wrap
from pydantic import BaseModel
import socketio
sio = wrap(socketio.AsyncClient()) # or AsyncServer, Client, Server, etc.
Emit Pattern
class Ping(BaseModel):
message: str
# kwargs are passed to socketio's emit method
# emits {"message": "Hello, World!"} to "ping"
await sio.emit(Ping(message="Hello, World!"), **kwargs)
# emits {"message": "Hello, World!"} to "my-ping"
await sio.emit("my-ping", Ping(message="Hello, World!"), **kwargs)
# standard sio behaviour
await sio.emit("event", {"payload": ...})
Call / RPC Pattern
class Pong(BaseModel):
reply: str
# emits {"message": "Hello, World!"} to "ping" and receives Pong(reply=...) in return
response = await sio.call(Ping(message="Hello, World!"), response_model=Pong)
assert isinstance(response, Pong)
# emits {"message": "Hello, World!"} to "my-ping" and receives Pong(reply=...) in return
response = await sio.call("my-ping", Ping(message="Hello, World!"), response_model=Pong)
assert isinstance(response, Pong)
# standard sio behaviour
response = await sio.call("event", {"payload": ...})
# standard response obj, typically dict
Handler Registration
Handlers are registered with @sio.on() or @sio.event and get automatic Pydantic validation from type hints.
tsio = wrap(socketio.AsyncServer(async_mode="asgi"))
@tsio.on(Ping)
async def handle_ping(sid: str, data: Ping) -> Pong:
return Pong(reply=data.message)
# or use the function name as the event name
@tsio.event
async def ping(sid: str, data: Ping) -> Pong:
return Pong(reply=data.message)
Outbound Event Documentation
Handlers that emit events to other channels can declare them with emits:
class SessionLeft(BaseModel):
room_id: str
user_id: str
@tsio.on("disconnect", emits=[SessionLeft])
async def handle_disconnect(sid: str) -> None:
await tsio.emit(SessionLeft(room_id="room1", user_id="user1"), room="room1")
These appear as action: "send" operations in the generated AsyncAPI schema.
REST Endpoint Emits
REST endpoints can also declare socket event emissions using the Emits annotation. These are auto-discovered when tsio.app is set and appear as x-rest-triggers in the AsyncAPI schema.
from typing import Annotated
from fastapi import Depends, FastAPI
from zndraw_socketio import Emits, AsyncServerWrapper, wrap
app = FastAPI()
tsio = wrap(socketio.AsyncServer(async_mode="asgi"))
tsio.app = app
@app.put("/{key}/selection")
async def update_selection(
sio: Annotated[AsyncServerWrapper, Depends(tsio), Emits(SessionLeft)],
key: str,
) -> dict:
await sio.emit(SessionLeft(room_id=key, user_id="..."), room=key)
return {"status": "ok"}
AsyncAPI Schema Generation
Generate an AsyncAPI 3.0 specification from registered handlers:
schema = tsio.asyncapi_schema(title="My API", version="1.0.0")
Event Names
By default, the event name is the class name in snake_case. You can customize it by setting the event_name attribute.
class CustomEvent(BaseModel):
...
get_event_name(CustomEvent) == "custom_event"
You can override it like this:
from typing import ClassVar
class CustomEvent(BaseModel):
event_name: ClassVar[str] = "my_custom_event"
Dependency Injection (Depends)
Handlers support FastAPI-style Depends() for dependency injection.
from typing import Annotated
from zndraw_socketio import wrap, Depends
async def get_redis() -> Redis:
return Redis()
RedisDep = Annotated[Redis, Depends(get_redis)]
@tsio.on(Ping)
async def handle(sid: str, data: Ping, redis: RedisDep) -> Pong:
await redis.set("last_ping", data.message)
return Pong(reply=data.message)
FastAPI App Integration
To use dependencies that need request.app (e.g. accessing app.state), set the .app property on the wrapper. The app can be set after handler registration — it is resolved at event time, not at registration time.
from fastapi import FastAPI, Request
app = FastAPI()
tsio = wrap(socketio.AsyncServer(async_mode="asgi"))
def get_db(request: Request) -> Database:
return request.app.state.db
@tsio.on(Ping)
async def handle(sid: str, data: Ping, db: Annotated[Database, Depends(get_db)]) -> Pong:
...
# Set app later — e.g. in a lifespan, after handler registration
tsio.app = app
Exception Handlers
Server wrappers support exception handlers similar to FastAPI.
from zndraw_socketio import EventContext
@tsio.exception_handler(ValueError)
async def handle_error(ctx: EventContext, exc: ValueError):
return {"error": str(exc)}
Union Return Types
You might want to return Response | ErrorResponse from an event handler.
[!NOTE] If your responses share fields, it is recommended to add a discriminator field to avoid ambiguity.
class ProblemDetail(BaseModel):
"""RFC 9457 Problem Details."""
kind: Literal["error"] = "error"
type: str = "about:blank"
title: str
status: int
detail: str | None = None
instance: str | None = None
class Response(BaseModel):
kind: Literal["response"] = "response"
data: str
class ServerRequest(BaseModel):
query: str
response = await sio.call(
ServerRequest(query="..."),
response_model=Response | ProblemDetail,
)
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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file zndraw_socketio-0.1.5.tar.gz.
File metadata
- Download URL: zndraw_socketio-0.1.5.tar.gz
- Upload date:
- Size: 200.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.10.1 {"installer":{"name":"uv","version":"0.10.1","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
dc7fa8c0b3cff30eb46e2f6cc0714ec98e311498dba56cef7bd4deba91533904
|
|
| MD5 |
6054e62b3e9cb749833265438daf3144
|
|
| BLAKE2b-256 |
e5bc81b6c0a1dd06c84a49294b0a10c78be6ee8e7dda15f964e861de64b56db5
|
File details
Details for the file zndraw_socketio-0.1.5-py3-none-any.whl.
File metadata
- Download URL: zndraw_socketio-0.1.5-py3-none-any.whl
- Upload date:
- Size: 20.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.10.1 {"installer":{"name":"uv","version":"0.10.1","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ce56528a3e3767767476422f057e879078113eeaa65e88e2d1f788f210df3ce6
|
|
| MD5 |
386ceb82c59c165db3ee7a514831e0c5
|
|
| BLAKE2b-256 |
11fb8bd73053f02d9055d361ba855397084f634f30fec6bdee56a098a984d238
|