A pure Python framework for building HAProxy SPOE agents
Project description
SPOE Forge
A pure Python framework for building SPOE (Stream Processing Offload Engine) agents that communicate with HAProxy using the SPOA protocol.
Overview
SPOE Forge provides a clean, decorator-based API for creating agents that process HAProxy messages and return actions. Built with async/await throughout, it's designed for high-performance production environments. Or at least as performant as python will allow.
Why SPOE Forge?
Originally created to power a Google OAuth2 authentication backend for HAProxy, it became clear the project could be converted to an abstracted framework. I noticed during the development of this project that there was a lack of well-maintained, easily understood implementations of the SPOA protocol in python.
Key Features
- Simple decorator-based API - Register message handlers with
@agent.message() - Full SPOP protocol support - Complete implementation of the SPOA protocol
- Health check support - Built-in HAProxy health check handling
Installation
Install from PyPI:
pip install spoe-forge
Quick Start
Basic Example
from spoe_forge import (
SpoeForge,
AgentContext,
SetVarAction,
ActionScope
)
# Create an agent
agent = SpoeForge(name="my-agent", debug=False)
# Register a message handler
@agent.message("check-request")
def handle_request(ctx: AgentContext) -> list[SetVarAction]:
"""Process incoming request and set HAProxy variables"""
# Get message arguments from HAProxy
client_ip = ctx.get_arg("client_ip")
request_path = ctx.get_arg("path")
# Your business logic here
is_allowed = check_access(client_ip, request_path)
# Return actions to set HAProxy variables
return [
SetVarAction(
scope=ActionScope.TRANSACTION,
name="access_allowed",
value=is_allowed
)
]
# Start the server
if __name__ == "__main__":
agent.run(host="0.0.0.0", port=12345)
HAProxy Configuration
SPOE Forge works with HAProxy's SPOE configuration. For details on configuring HAProxy to communicate with your agent, see the official HAProxy SPOE documentation.
Local Development
Running with Docker
A complete local development environment is provided using Docker Compose, including a sample SPOE agent, HAProxy, and a test backend service.
Quick start:
cd docker
docker compose up --build
This starts three services:
- SPOA Agent (
spoa) - Sample SPOE Forge agent running on port 8500 - Whoami (
whoami) - Simple backend service for testing - HAProxy (
haproxy) - Configured to communicate with the WhoAmI example BE Service, the SPOA agent, and is listening on port 8080
Test the setup:
# Open logs
docker compose logs
# Visit the dev url in your browser
http://localhost:8080
Check both the docker logs and the X-Test-Arg header displayed on the WhoAmI page.
Make any updates to the HAProxy configs or the sample_server.py files in ./docker/ to
support your testing.
Health Checks
SPOE Forge includes a built-in CLI healthcheck utility for monitoring agent health in containerized environments.
Basic usage:
# Basic usage
spoe-forge healthcheck --host 127.0.0.1 --port 8500
# Use default host and port
spoe-forge healthcheck
# Quiet mode, only exit codes
spoe-forge healthcheck --quiet
# Help info
spoe-forge --help
spoe-forge healthcheck --help
Docker Integration:
The provided docker-compose.yml includes a pre-configured healthcheck for the sample SPOE agent as a reference:
healthcheck:
test: ["CMD", "uv", "run", "spoe-forge", "healthcheck", "--host", "127.0.0.1", "--port", "8500", "--quiet"]
interval: 30s
timeout: 10s
retries: 3
start_period: 10s
View healthcheck status:
docker compose ps # Shows health status
docker inspect spoa # Detailed healthcheck info
Exit codes:
0: Health check passed1: Health check failed2: Invalid usage
Roadmap
Future enhancements under consideration with no timeline guaranteed:
- Middleware support
- Much more extended documentation and examples
License
This project is licensed under the MIT License - see the LICENSE file for details.
Contributing
Any and all contributions welcome.
Support
For issues and questions, please file an issue on GitHub.
Acknowledgments
Built to solve real-world production needs for HAProxy SPOA agents. Special thanks to the HAProxy team for excellent documentation of the SPOE protocol.
Extra shoutout to Christopher Faulet for responding to some questions about a few hiccups along the way.
Project details
Release history Release notifications | RSS feed
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 spoe_forge-0.0.3.tar.gz.
File metadata
- Download URL: spoe_forge-0.0.3.tar.gz
- Upload date:
- Size: 21.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
dbb9e2a8e18ee3401947771092a7061ecd52a6c90ea80374f3e259ae40da85cf
|
|
| MD5 |
fb497a6ad9f46da55cae87957b8ef1f8
|
|
| BLAKE2b-256 |
6b8ae6546d15898a8e37df412099cc049811841876719c6bce5fec801b11dfc4
|
Provenance
The following attestation bundles were made for spoe_forge-0.0.3.tar.gz:
Publisher:
release.yml on mwodonnell/spoe-forge
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
spoe_forge-0.0.3.tar.gz -
Subject digest:
dbb9e2a8e18ee3401947771092a7061ecd52a6c90ea80374f3e259ae40da85cf - Sigstore transparency entry: 782438249
- Sigstore integration time:
-
Permalink:
mwodonnell/spoe-forge@2708ca5b18ed247a524e7b9e218ced565556b91d -
Branch / Tag:
refs/tags/v0.0.3 - Owner: https://github.com/mwodonnell
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@2708ca5b18ed247a524e7b9e218ced565556b91d -
Trigger Event:
release
-
Statement type:
File details
Details for the file spoe_forge-0.0.3-py3-none-any.whl.
File metadata
- Download URL: spoe_forge-0.0.3-py3-none-any.whl
- Upload date:
- Size: 30.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e80d948574353aba32c06e4d9bd16c6e8fec16afa1f2f6a7c9b78aba0c693175
|
|
| MD5 |
c3eabfa50f3ed9470a4febe49d1f2574
|
|
| BLAKE2b-256 |
e55db701e6e7b7b7e4b9868d2851ed734cc239d7758bd96a6d19ec4fdeba6071
|
Provenance
The following attestation bundles were made for spoe_forge-0.0.3-py3-none-any.whl:
Publisher:
release.yml on mwodonnell/spoe-forge
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
spoe_forge-0.0.3-py3-none-any.whl -
Subject digest:
e80d948574353aba32c06e4d9bd16c6e8fec16afa1f2f6a7c9b78aba0c693175 - Sigstore transparency entry: 782438262
- Sigstore integration time:
-
Permalink:
mwodonnell/spoe-forge@2708ca5b18ed247a524e7b9e218ced565556b91d -
Branch / Tag:
refs/tags/v0.0.3 - Owner: https://github.com/mwodonnell
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@2708ca5b18ed247a524e7b9e218ced565556b91d -
Trigger Event:
release
-
Statement type: