Skip to main content

Xliner's CLI Framework

Project description

Xclif

uv Code style: black Ruff Imports: isort Checked with mypy codecov

CI PyPI - Python Version PyPI PyPI - License Read the Docs

Xliner's CLI Framework

Read the Manifesto to understand why Xclif exists and how it compares to Click, Typer, and argparse.

Installation

pip install xclif

Or with uv:

uv add xclif

Quick Start

Your directory structure is your command tree:

myapp/
├── __init__.py
├── __main__.py
└── routes/
    ├── __init__.py       →  myapp
    ├── greet.py          →  myapp greet
    └── server/
        ├── __init__.py   →  myapp server
        ├── start.py      →  myapp server start
        └── stop.py       →  myapp server stop
# routes/greet.py
from xclif import command

@command()
def _(name: str, template: str = "Hello, {}!") -> None:
    """Greet someone by name."""
    print(template.format(name))
# __main__.py
from xclif import Cli
from . import routes

cli = Cli.from_routes(routes)
if __name__ == "__main__":
    cli()

No default → positional argument. Has default → --template option. Docstring → help text. Drop a file in the right folder and the command exists.

Features

  • File-based routing — directory structure is the command tree
  • Decorator + type-hint API — function signatures define the CLI contract
  • Rich integration — beautiful help pages, formatted errors, progress indicators
  • Built-in logging and verbosity--verbose / -v wired up automatically
  • Config managementWithConfig[T] reads from config files or environment variables (CLI flag > env var > config file > default)
  • Autogenerated shell completions — bash, zsh, fish
  • Minimal overhead — custom parser built from scratch for fast startup; xclif compile pre-builds a static manifest to eliminate route-walking cost
  • Plugin discovery (planned) — third-party subcommands via entry points (like Git or cargo)
  • Easy testingcommand.execute(["greet", "Alice"]) with explicit arg lists, no mocking needed

Performance

Performance is not a primary focus of Xclif. If startup latency is a hard constraint, Python is probably the wrong tool for the job. These numbers are here for fun. That being said, we do recommend switching to a manifest-based setup for large codebases.

Startup time

Benchmarked on macOS (Apple Silicon, Python 3.12, 30 iterations + 3 warmup, wall-clock subprocess time):

Scenario Click Typer Xclif (from_routes) Xclif (flat) Xclif (manifest)
greet World 28.0 39.8 41.0 27.2 26.9
greet + options 27.9 37.8 40.9 26.3 26.7
config set 27.6 38.0 40.5 26.1 26.9
config get 27.9 38.1 40.3 26.2 26.8
--help 29.2 82.2 59.6 46.6 47.2
greet --help 29.7 83.8 59.3 46.8 47.6

Xclif (flat) uses the decorator API (Command.command() / Command.group()) instead of from_routes, and is the fastest framework for command execution — edging out Click by ~1–2 ms. The --help gap (~20 ms vs Click) is Rich's lazy-import cost. If better scaling is desirable for large codebases, the manifest compiler pre-builds a static manifest that matches flat API performance without requiring manual command registration.

Typer is the slowest overall: it wraps Click with extra overhead, and its Rich-based help rendering adds ~52–53 ms on --help scenarios.

from_routes adds ~13–15 ms for the package walker on top of Xclif (flat) — the trade-off for zero-registration file-based routing.

from_manifest eliminates that cost. Running xclif compile myapp.routes once generates a _xclif_manifest.py next to your routes package; loading it with Cli.from_manifest() skips the filesystem walk entirely, matching flat API performance.

# __main__.py — manifest variant
from xclif import Cli
from myapp import _xclif_manifest

cli = Cli.from_manifest(_xclif_manifest)
if __name__ == "__main__":
    cli()
# regenerate after adding or removing routes
xclif compile myapp.routes

To reproduce (requires hyperfinebrew install hyperfine on macOS):

bash benchmarks/bench_frameworks.sh

Parse and dispatch latency

Measured in-process (no subprocess, no import cost) on the same machine, 5 000 iterations:

Scenario Click Typer Xclif
greet World 72 µs 496 µs 2.7 µs
greet + options 77 µs 490 µs 4.0 µs
config set 82 µs 578 µs 3.4 µs
config get 79 µs 553 µs 3.2 µs
--help 131 µs 1 879 µs 483 µs
greet --help 142 µs 2 281 µs 553 µs

Xclif's custom parser is ~25× faster than Click and ~170× faster than Typer for command dispatch. The --help cases are slower than Click because Rich is doing real formatting work; Typer is dramatically slower there due to its reflective help generation.

uv run python benchmarks/bench_parsing.py

Contributing

See CONTRIBUTING.md.

License

This project is licensed under the MIT license.

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

xclif-0.3.0.tar.gz (120.9 kB view details)

Uploaded Source

Built Distribution

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

xclif-0.3.0-py3-none-any.whl (30.0 kB view details)

Uploaded Python 3

File details

Details for the file xclif-0.3.0.tar.gz.

File metadata

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

File hashes

Hashes for xclif-0.3.0.tar.gz
Algorithm Hash digest
SHA256 ba3086e806f504bc18ecbdecd5b369e735d78e0e472cfc808b4c1dcbb6b0925c
MD5 9da6f75429dc70762cb039db8f65200e
BLAKE2b-256 dba478c46f999f2e9cae861eb4f3996adeaa8e32e8ec26b1f33f4e3557e3c147

See more details on using hashes here.

Provenance

The following attestation bundles were made for xclif-0.3.0.tar.gz:

Publisher: release.yml on ThatXliner/xclif

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

File details

Details for the file xclif-0.3.0-py3-none-any.whl.

File metadata

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

File hashes

Hashes for xclif-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 1298b2d85b7970c1c1883da3576b311e65ee16f9484debe8b0dbab9332dda294
MD5 1c8e4c152032f3c142b5803d8b900912
BLAKE2b-256 05f05a9992f5e2e33270d721df7f107c4a49d9479e25a44e05e2e266655c2817

See more details on using hashes here.

Provenance

The following attestation bundles were made for xclif-0.3.0-py3-none-any.whl:

Publisher: release.yml on ThatXliner/xclif

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