Skip to main content

The agent-native CLI framework for Python

Project description

Tooli

The agent-native CLI framework for Python.

Tooli turns typed Python functions into command-line tools that work for both humans and AI agents: rich output in a terminal, strict structured output in automation, and self-describing schemas for tool calling and orchestration.

The name comes from "tool" + "CLI" = "tooli".

Why Tooli?

AI agents invoke lots of local commands, but typical CLIs are optimized for humans:

  • Huge, unstructured stdout that burns context windows
  • Opaque errors that don't suggest a fix
  • Fragile pipelines that mix logs with machine output
  • Undocumented flags that agents hallucinate

Tooli is built to be machine-consumable by default while still feeling great for humans.

Install

pip install tooli

Optional extras:

pip install "tooli[mcp]"   # MCP server support
pip install "tooli[api]"   # HTTP API + OpenAPI export (experimental)

Quick Start

Create file_tools.py:

from __future__ import annotations

from pathlib import Path
from typing import Annotated

from tooli import Argument, Option, Tooli
from tooli.annotations import Idempotent, ReadOnly

app = Tooli(name="file-tools", description="File utilities", version="1.2.0")


@app.command(annotations=ReadOnly | Idempotent, paginated=True, list_processing=True)
def find_files(
    pattern: Annotated[str, Argument(help="Glob to match")],
    root: Annotated[Path, Option(help="Root directory")] = Path("."),
) -> list[dict[str, str]]:
    return [{"path": str(p)} for p in root.rglob(pattern)]


if __name__ == "__main__":
    app()

Run it:

python file_tools.py find-files "*.py" --root .
python file_tools.py find-files "*.py" --root . --output json
python file_tools.py find-files --schema

Structured Output (JSON / JSONL)

Tooli supports dual-mode output:

  • Human mode: pretty output when attached to a TTY
  • Agent mode: strict envelopes when using --output json or --output jsonl

JSON envelope shape:

{
  "ok": true,
  "result": {"...": "..."},
  "meta": {
    "tool": "file-tools.find-files",
    "version": "1.2.0",
    "duration_ms": 12,
    "annotations": {"readOnlyHint": true, "idempotentHint": true}
  }
}

Structured Errors With Recovery Hints

When a command fails, Tooli emits a structured error with an actionable suggestion:

{
  "ok": false,
  "error": {
    "code": "E1004",
    "category": "input",
    "message": "Exact search string was not found in source.",
    "suggestion": {
      "action": "adjust search text",
      "fix": "Double-check exact spacing/newlines. Did you mean: \"...\"?"
    },
    "is_retryable": true
  }
}

Schemas, Docs, and Orchestration

Tooli can generate tool schemas and agent-facing docs directly from type hints and metadata:

python file_tools.py find-files --schema
python file_tools.py generate-skill > SKILL.md
python file_tools.py docs llms
python file_tools.py docs man

Run as an MCP server (one tool per command):

python file_tools.py mcp serve --transport stdio
python file_tools.py mcp serve --transport http --host 127.0.0.1 --port 8080
python file_tools.py mcp serve --transport sse --host 127.0.0.1 --port 8080

Universal Input (files / URLs / stdin)

Use StdinOr[T] to accept a file path, a URL, or piped stdin with one parameter.

from pathlib import Path
from typing import Annotated

from tooli import Argument, StdinOr, Tooli

app = Tooli(name="log-tools")


@app.command()
def head(
    source: Annotated[StdinOr[str], Argument(help="Path, URL, or '-' for stdin")],
) -> dict[str, int]:
    return {"bytes": len(source)}  # `source` resolves to the content

Built-In Guardrails

Tooli provides primitives for safer automation:

  • ReadOnly, Idempotent, Destructive, OpenWorld annotations on commands
  • --dry-run planning support via @dry_run_support + record_dry_action(...)
  • SecretInput[T] with automatic redaction in outputs and errors
  • Cursor pagination (--limit, --cursor, --fields, --filter) for list-shaped results

Example Apps (agent pain points)

The GitHub repo includes sample apps under examples/ that target common agent failure modes:

  • code-lens: token-efficient symbol outlines from Python ASTs (avoid dumping whole files)
  • safe-patch: self-healing file edits with dry-run plans and recovery hints
  • log-sift: pipeline-friendly log extraction with strict JSON/JSONL output
  • sqlite-probe: read-only SQLite exploration with pagination and guardrails

Links

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

tooli-1.2.0.tar.gz (73.9 kB view details)

Uploaded Source

Built Distribution

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

tooli-1.2.0-py3-none-any.whl (59.6 kB view details)

Uploaded Python 3

File details

Details for the file tooli-1.2.0.tar.gz.

File metadata

  • Download URL: tooli-1.2.0.tar.gz
  • Upload date:
  • Size: 73.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for tooli-1.2.0.tar.gz
Algorithm Hash digest
SHA256 c7a2ad3c55b4f4c190a528932d14e668b9af3a2aab89ed6390ac711afb900170
MD5 a3d52c7b2f08d3d47404dac9f161e0dc
BLAKE2b-256 7b5a414f2841a289b0cdf3a220c0bf9ec24e07bd1a27576341018b1c75091ff1

See more details on using hashes here.

Provenance

The following attestation bundles were made for tooli-1.2.0.tar.gz:

Publisher: publish.yml on weisberg/tooli

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

File details

Details for the file tooli-1.2.0-py3-none-any.whl.

File metadata

  • Download URL: tooli-1.2.0-py3-none-any.whl
  • Upload date:
  • Size: 59.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for tooli-1.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 edf509ca80eab2e0766c6b901039228d34f5eeab5f2af5fb76095a059054a959
MD5 cb2a82981c125b308adb235e6952eea3
BLAKE2b-256 37af29b3e117b410275e3100ceb43e573eac3dce6f2dc2f8ab2c3e1eab9e36c3

See more details on using hashes here.

Provenance

The following attestation bundles were made for tooli-1.2.0-py3-none-any.whl:

Publisher: publish.yml on weisberg/tooli

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