grpchook (grpc + hook) is an asynchronous Python gRPC bidirectional-streaming framework. Subclass BaseServer/BaseClient, override hooks — the base handles all gRPC plumbing.
Project description
grpchook
grpchook (gRPC + hook) is a Python framework for building asynchronous gRPC bidirectional-streaming services. Subclass BaseServer and BaseClient, override the hooks you need — the framework handles all gRPC plumbing.
Status: Work in Progress. The project is open source and will remain open source. Treat with caution. If you depend on it, pin your version.
Table of Contents
- Disclaimer
- When to Use and When Not to Use grpchook
- Requirements
- Installation
- Quick Start
- Minimal Examples
- Examples
- Testing
- Regenerating the gRPC Interface
- ToDos & Roadmap
- Known Issues & Troubleshooting
- Contributing
- License
- Release History
Disclaimer
Core concept by a human developer. AI was used to assist with unit and integration tests, examples, documentation, and selected code sections. Core logic has been reviewed by a human; the test suite has not been fully audited — there may be AI-introduced oversights. Please report any issues you find.
This software is provided "as is", without warranty of any kind. The developer is not responsible for any damage, data loss, security vulnerabilities, or other issues that may arise from using this software. You use it at your own risk. See LICENSE.txt for the full BSD 3-Clause terms.
When to Use and When Not to Use grpchook
When to Use grpchook
-
You need a simple, Python-based gRPC bidirectional streaming server and client.
-
You want a data exchange blueprint for developers or AI agents to build on top of.
-
You want a framework that can be extended with custom hooks for specific events.
-
You want to distribute clients to many different machines (e.g. voice recorder, voice to text, text to LLM, and vice versa until the final response is replayed)
Example — four clients on four machines, all routed through one grpchook server:
💡 Diagram requires the Markdown Preview Mermaid Support extension to render in VS Code.
flowchart LR subgraph M1["📦 Machine 1"] VR["🎤 Voice Recorder"] end subgraph M2["📦 Machine 2"] STT["📝 Speech-to-Text"] end subgraph M3["📦 Machine 3"] LLM["🤖 LLM Processor"] end subgraph M4["📦 Machine 4"] RP["🔊 Voice Replay"] end SRV(["⚙️ grpchook Server"]) VR -->|"① audio"| SRV SRV -->|"① audio"| STT STT -->|"② transcript"| SRV SRV -->|"② transcript"| LLM LLM -->|"③ llm_response"| SRV SRV -->|"③ llm_response"| RP
When Not to Use grpchook
- When you need a very large number of clients; the threading model may introduce overhead.
- When you need direct peer-to-peer communication without a server intermediary; grpchook routes all messages through a central server.
- You want a framework that supports multiple programming languages out of the box; grpchook is (currently) Python-only.
Requirements
- Python 3.10 or later
- A dedicated virtual environment is strongly recommended — gRPC version conflicts with other packages are common when using grpchook.
Installation
From PyPI
pip install grpchook
From Source
git clone https://github.com/fwkrumm/grpchook.git
cd grpchook
pip install -e .
Quick Start
Refer to HOW_TO.md for the full API reference and code examples. Alternatively run
python -m grpchook --generate-skeletons
to generate a very basic server and client skeleton in the current directory. Use
python -m grpchook --generate-interface-with-skeletons
to generate the skeletons along with a copy of the message.proto interface file in the current directory to modify which is then used by the skeletons.
Minimal Examples
Ultra-minimal — no subclassing required
The simplest possible working setup: start a server, connect two clients, exchange a message. Everything runs in a single script — no subclassing or hook overrides needed.
# example_minimal.py
import threading
from grpchook.baseserver import BaseServer
from grpchook.baseclient import BaseClient
from grpchook.tools import generate_message
# start the server in a background thread
server = BaseServer(port=50051, name="server")
threading.Thread(target=server.serve_forever, daemon=True).start()
# both clients declare the same channel name
# fan-out skips the sender, so client_b receives what client_a sends
client_a = BaseClient(port=50051, name="A", provides=["ping"], requires=["ping"])
client_b = BaseClient(port=50051, name="B", provides=["ping"], requires=["ping"])
client_a.send_data(generate_message("ping", byte_payload=b"hello"))
msg = client_b.get_data(timeout=5.0)
client_a.logger.info(msg.payload.bytePayload) # b"hello"
client_b.logger.info(msg.payload.bytePayload) # b"hello"
client_a.disconnect()
client_b.disconnect()
server.shutdown()
Request / response — subclass with hooks
For real workloads, subclass BaseServer to control routing and BaseClient to react to messages
via the on_receive hook.
server.py
from grpchook.baseserver import BaseServer, Peer
from grpchook.tools import generate_message
import grpchook.message_pb2 as pb2
class EchoServer(BaseServer):
def __init__(self):
super().__init__(port=50051, name="echo-server")
def on_receive(self, peer: Peer, request: pb2.Message) -> bool:
if request.metaInfo.messageName == "request":
reply = generate_message("response", byte_payload=request.payload.bytePayload)
self._data_register.add_data_for_message_name(
peer.client_id, "response", reply,
target_client_id=peer.client_id, # unicast back to sender
)
return False # skip default fan-out; routing handled above
return True
EchoServer().serve_forever()
client.py
from grpchook.baseclient import BaseClient
from grpchook.tools import generate_message
import grpchook.message_pb2 as pb2
class EchoClient(BaseClient):
def __init__(self):
super().__init__(port=50051, name="echo-client",
provides=["request"], requires=["response"])
def on_receive(self, data: pb2.Message):
print(f"Server replied: {data.payload.bytePayload.decode()}")
client = EchoClient()
client.send_data(generate_message("request", byte_payload=b"hello, grpchook!"))
client.spin(timeout=5.0) # calls on_receive() per message; returns on timeout/disconnect
client.disconnect()
Run the server first, then the client:
# terminal 1
python server.py
# terminal 2
python client.py
Examples
Runnable examples are available in two locations:
examples/— self-contained, scenario-focused examplestests/integration/— integration test scenarios covering a broad range of use cases
Run them on a machine with adequate resources; some scenarios are resource-intensive.
Testing
Install dev dependencies and run the unit tests:
pip install -r requirements_dev.txt
python -m unittest discover -s tests
Integration tests are in tests/integration/ and can be run via:
python tests/integration/run_integration_tests.py
Regenerating the gRPC Interface
If you modify grpchook/message.proto after cloning the repository, regenerate the Python bindings with:
python -m grpc_tools.protoc -I. --python_out=. --grpc_python_out=. --pyi_out=. grpchook/message.proto
Note that all clients which connect to a server have to use the same proto schema version i.e. the same proto file. The different signals for the clients must be used in substructures:
message Payload {
// For client A
SomeTypeA payloadClientA = 1;
// For client B
SomeTypeB payloadClientB = 2;
...
}
ToDos & Roadmap
Performance & Stability
- Evaluate replacing the threading model with
asyncioif the performance gain justifies the API tradeoff. - Verify behavior when connections are interrupted mid-stream; ensure no ghost threads or queue deadlocks occur.
Planned Features
- Multi-language client example (e.g., C++ or Rust).
- SSL/TLS usage example.
Known Issues & Troubleshooting
TBD
Contributing
Contributions are welcome. Please open an issue first for major changes so the approach can be discussed. For bug fixes and small improvements, a pull request is sufficient.
License
BSD 3-Clause — see LICENSE.txt.
Release History
| Version / Git Tag on Master | Description |
|---|---|
| 0.0.1 | Unpublished. |
| 0.0.2 | Initial public release. |
| 0.0.3 | Add ms timestamp resolution to log output and minor adjustments to readme. |
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 grpchook-0.0.3.tar.gz.
File metadata
- Download URL: grpchook-0.0.3.tar.gz
- Upload date:
- Size: 111.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5bc0f803acc070db207037257db4ecf56c98a361c06c28ce517ed29d5dae1d6e
|
|
| MD5 |
75193725255720c7307ed1110fa7ef37
|
|
| BLAKE2b-256 |
2a189aa813654decad05ad19a4e0a6d62ee29bdf7e3f74a92511155ba12fd179
|
Provenance
The following attestation bundles were made for grpchook-0.0.3.tar.gz:
Publisher:
pypi_publish.yml on fwkrumm/grpchook
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
grpchook-0.0.3.tar.gz -
Subject digest:
5bc0f803acc070db207037257db4ecf56c98a361c06c28ce517ed29d5dae1d6e - Sigstore transparency entry: 1919261183
- Sigstore integration time:
-
Permalink:
fwkrumm/grpchook@3b43ca605f8a9303a790c046a3478148f11b4eb8 -
Branch / Tag:
refs/heads/master - Owner: https://github.com/fwkrumm
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
pypi_publish.yml@3b43ca605f8a9303a790c046a3478148f11b4eb8 -
Trigger Event:
push
-
Statement type:
File details
Details for the file grpchook-0.0.3-py3-none-any.whl.
File metadata
- Download URL: grpchook-0.0.3-py3-none-any.whl
- Upload date:
- Size: 88.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8fd953a811b934bfc175c9b7765fceffa211afb2490c75ba1a6455a993d7329c
|
|
| MD5 |
a2a5eb63e39f3fffb9ca25713ce786ee
|
|
| BLAKE2b-256 |
e03ce73f6bd7502712d83ba354fab63e100cdcfc7001e781eb857224bb47cdf7
|
Provenance
The following attestation bundles were made for grpchook-0.0.3-py3-none-any.whl:
Publisher:
pypi_publish.yml on fwkrumm/grpchook
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
grpchook-0.0.3-py3-none-any.whl -
Subject digest:
8fd953a811b934bfc175c9b7765fceffa211afb2490c75ba1a6455a993d7329c - Sigstore transparency entry: 1919261295
- Sigstore integration time:
-
Permalink:
fwkrumm/grpchook@3b43ca605f8a9303a790c046a3478148f11b4eb8 -
Branch / Tag:
refs/heads/master - Owner: https://github.com/fwkrumm
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
pypi_publish.yml@3b43ca605f8a9303a790c046a3478148f11b4eb8 -
Trigger Event:
push
-
Statement type: