Python SDK for nexo subprocess plugins
Project description
nexoai — nexo plugin SDK (Python)
Child-side SDK for nexo subprocess plugins written in Python 3.10+.
Mirrors the Rust counterpart in
crates/microapp-sdk/, the TypeScript
counterpart in typescript/ and
the PHP counterpart in php/ — same wire
format (nexo-plugin-contract.md),
different language.
pip install nexoai
Distribution name vs import name — the PyPI package is
nexoai(thenexo-plugin-sdkname was already taken); the importable module isnexo_plugin_sdk(from nexo_plugin_sdk import PluginAdapter).
The reference plugin template lives at
the Python plugin template (or run nexo plugin new --lang python);
copy that directory to start a new plugin.
Public API
from nexo_plugin_sdk import (
PluginAdapter, # async dispatch loop
BrokerSender, # write-only handle to publish events back
Event, # dataclass mirror of the host's broker event
EventHandler, # type alias: Callable[[str, Event, BrokerSender], Awaitable[None]]
ShutdownHandler, # type alias: Callable[[], Awaitable[None]]
PluginError, # base exception
ManifestError, # raised when nexo-plugin.toml is malformed
WireError, # raised on malformed / oversized JSON-RPC frames
read_manifest, # standalone manifest TOML parser + validator
install_stdout_guard, # defensive guard installable independently
uninstall_stdout_guard,
is_stdout_guard_installed,
STDOUT_GUARD_MARKER, # sentinel prefixed onto diverted stdout lines
MAX_FRAME_BYTES, # default inbound frame cap (1 MiB)
JSONRPC_VERSION,
serialize_frame, build_response, build_error_response, build_notification,
)
Minimal example
import asyncio
from nexo_plugin_sdk import PluginAdapter, Event
MANIFEST = open("nexo-plugin.toml").read()
async def on_event(topic: str, event: Event, broker) -> None:
out = Event.new(
"plugin.inbound.my_kind",
"my_plugin",
{"echoed": event.payload},
)
await broker.publish("plugin.inbound.my_kind", out)
async def main() -> None:
adapter = PluginAdapter(manifest_toml=MANIFEST, on_event=on_event)
await adapter.run()
if __name__ == "__main__":
asyncio.run(main())
Robustness defaults
The PluginAdapter constructor defaults are picked to make the most
common plugin-author mistakes recoverable rather than fatal — matching
the TypeScript and PHP SDKs:
| Default | What it gives you |
|---|---|
enable_stdout_guard=True |
A stray print("hi") from your handler (or a chatty transitive dep) is diverted to stderr tagged [stdout-guard] rather than corrupting the JSON-RPC frame stream the host parses. The SDK's own replies and broker.publish frames write through the captured original stdout, so they bypass the guard. |
max_frame_bytes=1<<20 |
Inbound JSON-RPC frames larger than 1 MiB are rejected with a WireError log; dispatch continues. An adversarial host cannot OOM the plugin via a single huge line. |
handle_process_signals=True |
SIGTERM / SIGINT trigger a graceful shutdown — in-flight handler tasks are awaited (no mid-publish cancellation), then the process exits 0 instead of the default -15. Uses loop.add_signal_handler, falling back to signal.signal where that is unavailable (Windows ProactorEventLoop / non-main-thread). |
In-flight drain on shutdown |
Handlers spawned for broker.event are awaited via asyncio.gather(...) before the SDK replies {ok: true} to a host shutdown request. Same idiom as the TypeScript SDK's Promise.allSettled([...inflight]) and the PHP SDK's Scheduler::drain(). |
The stdin reader is fully async (loop.connect_read_pipe + an
asyncio.StreamReader) — no threadpool worker — so signal-driven
cancellation is clean.
Stdout guard limitation
The guard replaces sys.stdout with a line-buffering proxy, so it only
intercepts the text-stream API (print, sys.stdout.write). A C
extension or subprocess that writes to file descriptor 1 directly
bypasses it. Plugin authors who need stdout output should use
print() / sys.stdout.write(); those are guarded.
What the daemon expects
| Method | Direction | Reply |
|---|---|---|
initialize |
host → child | { manifest, server_version } automatically — the SDK reads + validates your manifest TOML at construction time (incl. the ^[a-z][a-z0-9_]{0,31}$ plugin.id slug regex the host enforces). |
broker.event (notification) |
host → child | No JSON reply. Your on_event handler runs in a detached task so the dispatch loop continues reading stdin while the handler awaits broker round-trips. |
shutdown |
host → child | { ok: true } after draining in-flight handler tasks + invoking your on_shutdown (if set). |
Full spec: nexo-plugin-contract.md.
Tests
cd python
PYTHONPATH=. python3 -m unittest discover -v tests/
21 tests covering: the handshake (initialize reply, unknown method
-32601, unknown notification ignored), manifest validation (missing
id, invalid TOML, id-regex violation), dispatch (handler invocation,
non-blocking reader, in-flight drain, oversized frame rejected with
continued dispatch), the stdout guard (idempotent install, install /
uninstall round-trip, divert vs passthrough, chunked print-style
writes, partial-line flush, attr delegation, handler-print diverted
while the blessed frame stays clean), the broker.publish back
channel, and lifecycle (double run() rejected, SIGTERM exits 0,
SIGTERM drains an in-flight handler before exiting).
Phase tracking
- 31.4 (shipped) — child-side SDK + 6 tests.
- 31.4.c (shipped, this package) — robustness parity with the
TypeScript/PHP SDKs: default-on stdout guard, 1 MiB inbound frame
cap, SIGTERM/SIGINT graceful drain, async stdin reader,
plugin.idregex validation, 21 tests,pyproject.tomlpublish-ready. - 31.4.b (deferred) — per-target Python tarballs (
pyXY-<triple>targets) for plugins that need native extensions. - 31.8 (shipped) — extracted to the
nexo-plugin-sdksmono-repo and published to PyPI asnexoai(thenexo-plugin-sdkname was taken). Taggedpython-vX.Y.Z.
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 nexoai-0.2.0.tar.gz.
File metadata
- Download URL: nexoai-0.2.0.tar.gz
- Upload date:
- Size: 20.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 |
2ce0f6c89729fab6af984f34758213cda720c77b1532a3dca5b4c3d961ccdc1f
|
|
| MD5 |
e70d09f7f482f0280887896c79944c42
|
|
| BLAKE2b-256 |
4fa6f96342d5567e7d7e7b0c3c696ad023c5d313e499c909eaac1b10d2a61e1e
|
Provenance
The following attestation bundles were made for nexoai-0.2.0.tar.gz:
Publisher:
python.yml on lordmacu/nexo-plugin-sdks
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
nexoai-0.2.0.tar.gz -
Subject digest:
2ce0f6c89729fab6af984f34758213cda720c77b1532a3dca5b4c3d961ccdc1f - Sigstore transparency entry: 1507452129
- Sigstore integration time:
-
Permalink:
lordmacu/nexo-plugin-sdks@99ca41f3405bc21b91609b71075e2f0d7545aa1c -
Branch / Tag:
refs/tags/python-v0.2.0 - Owner: https://github.com/lordmacu
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python.yml@99ca41f3405bc21b91609b71075e2f0d7545aa1c -
Trigger Event:
push
-
Statement type:
File details
Details for the file nexoai-0.2.0-py3-none-any.whl.
File metadata
- Download URL: nexoai-0.2.0-py3-none-any.whl
- Upload date:
- Size: 15.7 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 |
8dd6d6af28f0a2e142dcd3be11f9bc6e0b28296c033c979358a4db1899fa1e83
|
|
| MD5 |
f67bdefea217ac91a66a0266bf42537a
|
|
| BLAKE2b-256 |
fe3e5c8adddc8adadd936243a799de31b31ce838ecd1c70a280199c894da5bc9
|
Provenance
The following attestation bundles were made for nexoai-0.2.0-py3-none-any.whl:
Publisher:
python.yml on lordmacu/nexo-plugin-sdks
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
nexoai-0.2.0-py3-none-any.whl -
Subject digest:
8dd6d6af28f0a2e142dcd3be11f9bc6e0b28296c033c979358a4db1899fa1e83 - Sigstore transparency entry: 1507452375
- Sigstore integration time:
-
Permalink:
lordmacu/nexo-plugin-sdks@99ca41f3405bc21b91609b71075e2f0d7545aa1c -
Branch / Tag:
refs/tags/python-v0.2.0 - Owner: https://github.com/lordmacu
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python.yml@99ca41f3405bc21b91609b71075e2f0d7545aa1c -
Trigger Event:
push
-
Statement type: