Skip to main content

A strictly-typed, generics-first plugin framework for Python 3.13: hooks with derived return types

Project description

pluginkit

PyPI Python CI Docs License: MIT

A small, strictly-typed, generics-first plugin framework for Python 3.13+. Declare extension points, let plugins implement them, discover plugins via entry points - and, unlike untyped hook systems, get the right return type for every call, derived from the spec and checked by your type checker.

pm.caller(spec) hands back a caller whose result type matches the dispatch mode - list[R] for collecting, R | None for firstresult, R for pipeline - with no hand-annotations and no drift. Zero runtime dependencies, a py.typed marker, and a few readable files.

uv add pluginkit   # or: pip install pluginkit
from pluginkit import Extension, ExtensionPoint, PluginManager

extension_point = ExtensionPoint("greeter")
extension = Extension("greeter")


class Specs:
    @staticmethod
    @extension_point
    def greeting(name: str) -> str:
        """Return a greeting for the given name."""


class Casual:
    @extension
    def greeting(self, name: str) -> str:
        return f"hey {name}!"


pm = PluginManager("greeter")
pm.add_extension_points(Specs)
pm.register(Casual(), name="casual")

greetings = pm.caller(Specs.greeting)(name="Ada")   # typed list[str] - derived, not asserted
print(greetings)                                     # ['hey Ada!']

What it supports

  • collecting, firstresult, and pipeline (fold/middleware) hooks;
  • call ordering with tryfirst / trylast, plus optional and target;
  • generator wrappers that decorate results and observe exceptions safely;
  • historic hooks replayed to plugins registered later;
  • async dispatch via AsyncPluginManager (awaits coroutine impls);
  • plugin lifecycle: register, unregister (by name or object), set_blocked, lookup, call_extra;
  • registration-time validation and call-time argument checking (failures are loud);
  • external plugin discovery via the stdlib importlib.metadata (no setuptools);
  • thread-safe registry mutation.

Layout

src/pluginkit/             the library (pure - no demo code)
examples/                  everything that uses the library (not shipped):
  cookbook/                  worked examples: bite-size scripts + full apps
  tour/                      pluginkit-tour: a guided CLI walkthrough
  external-plugin/           a separate distribution discovered via entry points
docs/                      mkdocs + Material documentation
tests/                     library, tour, and cookbook tests

Everything that demonstrates the library lives under examples/. The cookbook holds standalone examples (from one-mechanism snippets to complete FastAPI/Click/pytest apps); the tour is a guided walkthrough on one host; the external-plugin shows cross-package discovery via entry points.

Use it

make install              # uv sync (library + tour + external plugin)
make test                 # pytest (framework, tour, examples)
make lint                 # ruff + mypy + pyright
make docs-serve           # serve the docs at http://127.0.0.1:8000
make docs-build           # build the docs (strict)

Two ways to learn it

The tour (examples/tour/) walks through one mechanism at a time on a single host:

make run                  # run every step
make run DEMO=wrapper      # run one
uv run pluginkit-tour list

The cookbook applies the library to realistic domains and frameworks - see examples/cookbook/:

uv run python examples/cookbook/report_builder.py
uv run python examples/cookbook/fastapi_app.py
uv run python examples/cookbook/cli_app.py --help
uv run python examples/cookbook/app_lifecycle.py

Documentation

Full docs (concepts, one page per mechanism, production/hardening notes, and a generated API reference) live under docs/. Serve them with make docs-serve.

Is it production ready?

It is solid - exception-safe wrappers, fail-fast validation, lifecycle management, resilient discovery, thread-safe mutation, strict typing, and a test suite. But for anything you ship, prefer pluggy itself: it is maintained and battle tested by pytest, tox, and datasette. See docs/production/vs-pluggy.md for the honest inventory of what differs.

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

pluginkit-0.4.5.tar.gz (15.8 kB view details)

Uploaded Source

Built Distribution

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

pluginkit-0.4.5-py3-none-any.whl (18.6 kB view details)

Uploaded Python 3

File details

Details for the file pluginkit-0.4.5.tar.gz.

File metadata

  • Download URL: pluginkit-0.4.5.tar.gz
  • Upload date:
  • Size: 15.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for pluginkit-0.4.5.tar.gz
Algorithm Hash digest
SHA256 551535c4a20fe8dcf7ae8e2a2da4ec5b58d644bde0a8697cc31bfea4dc12df6e
MD5 1724c301f9272b6b3c89c9c3ca8d8522
BLAKE2b-256 dd666fd3900109f815939d545a98767f8e394853c14c8f3163f47917bad81d90

See more details on using hashes here.

Provenance

The following attestation bundles were made for pluginkit-0.4.5.tar.gz:

Publisher: release.yml on winterop-com/pluginkit

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pluginkit-0.4.5-py3-none-any.whl.

File metadata

  • Download URL: pluginkit-0.4.5-py3-none-any.whl
  • Upload date:
  • Size: 18.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for pluginkit-0.4.5-py3-none-any.whl
Algorithm Hash digest
SHA256 91f144e9ef1e3d35147dab6380cdac0f5bd48265e1a8a1046abb502335dfd116
MD5 43ada2a2313c29eb45d5670368e8f356
BLAKE2b-256 81cf54312e211f44beac38e166245897d78e8201310ea1338004b9cce261c4d1

See more details on using hashes here.

Provenance

The following attestation bundles were made for pluginkit-0.4.5-py3-none-any.whl:

Publisher: release.yml on winterop-com/pluginkit

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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