Skip to main content

noot library

Project description

noot

PyPI version License CI

Test interactive CLIs. Think Stagehand, but for the terminal.

  • f.step("Select 'pyright' from dropdown"): Define CLI interactions in plain English.
  • f.expect("Linting options must contain 'pyright'"): Define expected CLI states in plain English too.
  • assert "pyright" in f.screen().lower(): Or make assertions on CLI state in good ol' Python.
  • Record with LLM once, replay locally and in CI/CD.

Installation

pip install noot

Requires tmux and an ANTHROPIC_API_KEY environment variable.

Quick Start

Scaffold a new project:

noot init

Or add noot to an existing project:

from noot import Flow

def test_create_web_project():
    with Flow.spawn('python setup_wizard.py') as f:
        f.expect('Welcome to Project Setup Wizard')

        f.step("Enter project name 'mywebapp' and press enter")

        # `expect` parses assertions from natural language
        f.expect('Web Application project option is available')

        f.step('Press enter to select Web Application')

        # or specify assertions on screen state directly
        assert "author name" in f.screen()

        f.step("Enter author name 'Alice' and press enter")

Run your tests:

pytest tests/test_cli.py

The first run records LLM responses to the cassette file. Subsequent runs replay from the cassette, so no API calls are made.

API

Method Description
Flow.spawn(cmd) Context manager. Start a CLI process in a managed terminal session
f.step(instruction) Execute a natural language instruction (e.g., "Press enter", "Type 'hello'")
f.expect(condition) Assert the screen matches a natural language condition
f.screen() Return the current terminal output as a string

Recording Modes

Control recording behavior with the RECORD_MODE environment variable:

RECORD_MODE Behavior
once (Default) Record if cassette is missing, replay if it exists.
none Replay only. Fails if a request isn't cached. Use this in CI.
all Always re-record, overwriting existing cassettes.

By default you don't have to think about recording and replay:

pytest tests/test_cli.py
# Subsequent runs will use cache

Example - force re-recording:

RECORD_MODE=all pytest tests/test_cli.py

Example - CI mode (fail if cassette is missing):

RECORD_MODE=none pytest tests/test_cli.py

Cassettes are stored in <project_root>/.cassettes/:

  • CLI cassettes (LLM responses): .cassettes/cli/
  • HTTP cassettes (API recordings): .cassettes/http/

CI/CD

Run tests in CI with RECORD_MODE=none to replay from cached cassettes (no API key needed):

# .github/workflows/test.yml
name: Tests
on: [push, pull_request]

jobs:
  test:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4
      - uses: actions/setup-python@v5
        with:
          python-version: '3.12'
      - name: Install tmux
        run: sudo apt-get install -y tmux
      - name: Install dependencies
        run: pip install -e ".[dev]"
      - name: Run tests
        run: RECORD_MODE=none pytest

Commit your .cassettes/ directory to version control so CI can replay recordings.

Contributing

Issues and PRs welcome.

License

Apache 2.0

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

noot-0.1.0.tar.gz (20.2 kB view details)

Uploaded Source

Built Distribution

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

noot-0.1.0-py3-none-any.whl (27.0 kB view details)

Uploaded Python 3

File details

Details for the file noot-0.1.0.tar.gz.

File metadata

  • Download URL: noot-0.1.0.tar.gz
  • Upload date:
  • Size: 20.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.22 {"installer":{"name":"uv","version":"0.9.22","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for noot-0.1.0.tar.gz
Algorithm Hash digest
SHA256 4f8ef66784734fcf7a961ec1f740ba34a7c14dfce2abfd6b9d4bea1edfad6eff
MD5 ac6db3883647533f4b239129a3a77c21
BLAKE2b-256 2b98895f85374ac81b8575c7d40ce7a7dc8d76716184e14af4070ad81451d4d4

See more details on using hashes here.

File details

Details for the file noot-0.1.0-py3-none-any.whl.

File metadata

  • Download URL: noot-0.1.0-py3-none-any.whl
  • Upload date:
  • Size: 27.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.22 {"installer":{"name":"uv","version":"0.9.22","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for noot-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 abb9b8705464f265b596f803192b09a6647398ad59a2a0caac6887e2322fed88
MD5 2413ad5bec37a0b4274824a6261491c9
BLAKE2b-256 254752fdffef530a41d469dada0cee97dbf8b8bf5b86e788dd7eeb67facbe56d

See more details on using hashes here.

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