Skip to main content

Our Trusty Testing Orchestrator - framework and CLI for deploying, testing, and monitoring software across remote hosts.

Project description

otto

otto — Our Trusty Testing Orchestrator — is a framework for deploying products to remote hosts for testing and validation. It provides a CLI and a Python API for running commands on remote systems, transferring files, executing test suites, and monitoring host metrics in real time.

Who is otto for?

Otto is a general-purpose tool for developers and testers who need to interact with one or more remote machines as part of their workflow — deploying builds, validating firmware, running integration tests, or collecting performance data.

Two ways to use otto

  • CLI users — interact with otto through the otto run, otto test, and otto monitor commands.
  • API builders — import otto's Python packages to build higher-level automation on top of hosts, suites, and the monitor.

Key concepts

Hosts

A Host represents a machine otto can talk to. RemoteHost connects over SSH or Telnet; LocalHost runs commands on the local machine without any network connection.

Both expose the same core interface — run, oneshot, send / expect, and file-transfer methods (put, get).

run executes a command on a host's persistent shell session (state like the working directory and environment variables are preserved between calls). oneshot runs each call independently of the persistent shell and of other concurrent oneshot calls, making it safe to fan out via asyncio.gather().

Labs

Hosts can be reached through intermediate hops — SSH jump hosts that otto tunnels through automatically. Hops can be chained for multi-hop paths (otto -> hop1 -> hop2 -> target). All file transfer protocols (SCP, SFTP, FTP, netcat) work through hops. Set the hop field in a host's JSON definition or use --hop on the CLI.

A Lab is a JSON file that describes a set of hosts and their topology. Otto loads labs at startup (via --lab or the OTTO_LAB environment variable) and makes every host available to instructions, test suites, and the monitor. Multiple labs can be merged by passing several names.

Repos and settings

Otto discovers your project through a .otto/settings.toml file at the repository root. This file tells otto where to find your Python libraries, test suites, run instructions, and lab data:

name = "my_project"
version = "1.0.0"

labs  = ["${sutDir}/../lab_data"]
libs  = ["${sutDir}/pylib"]
tests = ["${sutDir}/tests"]
init  = ["my_instructions"]

${sutDir} is replaced with the repository root at load time. The init list names Python modules that otto imports at startup — this is where you register your instructions and shared options.

Instructions (otto run)

An instruction is an async function decorated with @instruction() that becomes a subcommand of otto run. Instructions have full access to the lab's hosts and can accept their own CLI options via Typer annotations:

from otto.cli.run import instruction
from otto.configmodule.configmodule import all_hosts
from otto.logger import getOttoLogger

logger = getOttoLogger()

@instruction()
async def deploy(
    debug: Annotated[bool, typer.Option("--field/--debug")] = False,
):
    for host in all_hosts():
        await host.run(["echo deploying", "make install"])
    logger.info("Done")
otto -l my_lab run deploy --debug

Test suites (otto test)

A suite is a class that extends OttoSuite and is registered with the @register_suite() decorator. Each suite becomes a subcommand of otto test. Suites can define their own Options dataclass whose fields appear as CLI flags:

from dataclasses import dataclass
from typing import Annotated

import typer
from otto.suite import OttoSuite, register_suite

@dataclass
class _Options:
    firmware: Annotated[str, typer.Option(help="Firmware version.")] = "latest"

@register_suite()
class TestDevice(OttoSuite[_Options]):
    Options = _Options

    async def test_device_reachable(self, suite_options: _Options) -> None:
        self.logger.info(f"firmware={suite_options.firmware}")
        assert True
otto -l my_lab test TestDevice --firmware 2.1
otto test --iterations 10 --threshold 95 TestDevice

Suites support pytest markers (timeout, retry, parametrize, integration), non-fatal assertions via self.expect(), per-test artifact directories, and built-in monitoring.

Both suites and instructions accept an options dataclass. For flags that are repo-wide (device type, lab environment, etc.), define a single RepoOptions dataclass in your pylib and inherit it from both sides.

Monitor (otto monitor)

The monitor collects live performance metrics (CPU, memory, disk, network) from one or more hosts and serves an interactive web dashboard:

otto -l my_lab monitor                     # all hosts, default 5 s interval
otto monitor host1,host2 --interval 2.0    # specific hosts, faster polling
otto monitor --db metrics.db               # persist data for later viewing
otto monitor --file metrics.db             # replay saved data

Monitoring can also be started from within a test suite using await self.startMonitor(hosts=...) and await self.stopMonitor().

Quick-start example

  1. Set the environment — point otto at your repo and lab:

    export OTTO_SUT_DIRS=/path/to/my_project
    otto --lab my_lab --list-hosts          # verify hosts are visible
    
  2. Run an instruction:

    otto -l my_lab run deploy --debug
    
  3. Run a test suite:

    otto -l my_lab test TestDevice --firmware 2.1
    
  4. Monitor hosts:

    otto -l my_lab monitor
    

Documentation

Hosted documentation: otto-sh.readthedocs.io.

The same content lives under docs/ and can be built locally with make docs — the generated HTML is written to docs/_build/html/. Key entry points:

  • docs/getting-started.md — installation and first steps
  • docs/guide/ — detailed guides for each CLI command
  • docs/cookbook/ — recipes for common asyncio patterns
  • docs/api/ — full API reference for all otto packages

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

otto_sh-0.2.1.tar.gz (1.5 MB view details)

Uploaded Source

Built Distribution

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

otto_sh-0.2.1-py3-none-any.whl (1.6 MB view details)

Uploaded Python 3

File details

Details for the file otto_sh-0.2.1.tar.gz.

File metadata

  • Download URL: otto_sh-0.2.1.tar.gz
  • Upload date:
  • Size: 1.5 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for otto_sh-0.2.1.tar.gz
Algorithm Hash digest
SHA256 231e7cb600f1dfc3b73d2687fcbcdeb679b99dde4240765a55a107fd293defcc
MD5 3e15ec921f21df8dbe7270d09f1f445e
BLAKE2b-256 d742bf770b00cd183d5074a3c38c513306618eb5a787cd53d5cad84cb097b6eb

See more details on using hashes here.

Provenance

The following attestation bundles were made for otto_sh-0.2.1.tar.gz:

Publisher: release.yml on ludachrish3/otto-sh

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

File details

Details for the file otto_sh-0.2.1-py3-none-any.whl.

File metadata

  • Download URL: otto_sh-0.2.1-py3-none-any.whl
  • Upload date:
  • Size: 1.6 MB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for otto_sh-0.2.1-py3-none-any.whl
Algorithm Hash digest
SHA256 af7dfca3c8fe6f1c8d53a89d777ea0ed1a0367712e6780f970aa89512763a2e4
MD5 0d348d9d94a0a3d0ea273014a4eb01a2
BLAKE2b-256 e496eeb60e5a1ecaa4d5551bc35899f4f11aa8fad3a1f99b3b5dfc8f09ef70a8

See more details on using hashes here.

Provenance

The following attestation bundles were made for otto_sh-0.2.1-py3-none-any.whl:

Publisher: release.yml on ludachrish3/otto-sh

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