Skip to main content
Guides Last updated: 24 February 2026

Plugin system

Untether uses Python entrypoints to extend engines, transports, and commands.

Untether uses Python entrypoints to extend engines, transports, and commands.

Why entrypoints

Entrypoints let Untether discover plugins without hard dependencies on plugin packages. Installed distributions declare what they provide, and Untether can list and load them at runtime.

This makes it possible to:

  • Add new engines/transports/commands without changing Untether itself.
  • Ship plugins independently.
  • Keep the core CLI small.

Why discovery is lazy

Untether lists plugin IDs without importing plugin code, then imports a plugin only when:

  • it is selected by routing (engine/transport), or
  • it is invoked as a command, or
  • you explicitly request loading via untether plugins --load.

This keeps untether --help fast and prevents a broken third-party plugin from bricking the CLI.

Entrypoint rules (what Untether expects)

Untether uses three entrypoint groups:

[project.entry-points."untether.engine_backends"]
myengine = "myengine.backend:BACKEND"

[project.entry-points."untether.transport_backends"]
mytransport = "mytransport.backend:BACKEND"

[project.entry-points."untether.command_backends"]
mycommand = "mycommand.backend:BACKEND"

Rules:

  • The entrypoint name is the plugin id.
  • The entrypoint value must resolve to a backend object:
    • engine backend: EngineBackend
    • transport backend: TransportBackend
    • command backend: CommandBackend
  • The backend object must have id == entrypoint name.

Why there is an enabled list

Plugin visibility can be restricted via:

=== “untether config”

```sh
untether config set plugins.enabled '["untether-engine-acme", "untether-transport-slack"]'
```

=== “toml”

```toml
[plugins]
enabled = ["untether-engine-acme", "untether-transport-slack"]
```

When set, Untether filters by distribution name (package metadata), not by entrypoint name. This lets you:

  • ship multiple entrypoints from one distribution, and
  • enable/disable whole plugin packages predictably.

IDs and collisions

Entrypoint names become plugin IDs and appear in user-facing surfaces (CLI subcommands, Telegram commands, /<engine-id> directives). Untether validates IDs and rejects collisions with reserved names.

Plugin IDs must match:

^[a-z0-9_]{1,32}$

Reserved IDs include core chat and CLI command names such as cancel, init, and plugins.

How to debug discovery and loading

untether plugins
untether plugins --load
Was this helpful?

Related Articles