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, andotto monitorcommands. - 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
-
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
-
Run an instruction:
otto -l my_lab run deploy --debug
-
Run a test suite:
otto -l my_lab test TestDevice --firmware 2.1
-
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 stepsdocs/guide/— detailed guides for each CLI commanddocs/cookbook/— recipes for common asyncio patternsdocs/api/— full API reference for all otto packages
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
231e7cb600f1dfc3b73d2687fcbcdeb679b99dde4240765a55a107fd293defcc
|
|
| MD5 |
3e15ec921f21df8dbe7270d09f1f445e
|
|
| BLAKE2b-256 |
d742bf770b00cd183d5074a3c38c513306618eb5a787cd53d5cad84cb097b6eb
|
Provenance
The following attestation bundles were made for otto_sh-0.2.1.tar.gz:
Publisher:
release.yml on ludachrish3/otto-sh
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
otto_sh-0.2.1.tar.gz -
Subject digest:
231e7cb600f1dfc3b73d2687fcbcdeb679b99dde4240765a55a107fd293defcc - Sigstore transparency entry: 1441382259
- Sigstore integration time:
-
Permalink:
ludachrish3/otto-sh@96628ee13827b38c1d2e170435e4196777b090c4 -
Branch / Tag:
refs/tags/v0.2.1 - Owner: https://github.com/ludachrish3
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@96628ee13827b38c1d2e170435e4196777b090c4 -
Trigger Event:
push
-
Statement type:
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
af7dfca3c8fe6f1c8d53a89d777ea0ed1a0367712e6780f970aa89512763a2e4
|
|
| MD5 |
0d348d9d94a0a3d0ea273014a4eb01a2
|
|
| BLAKE2b-256 |
e496eeb60e5a1ecaa4d5551bc35899f4f11aa8fad3a1f99b3b5dfc8f09ef70a8
|
Provenance
The following attestation bundles were made for otto_sh-0.2.1-py3-none-any.whl:
Publisher:
release.yml on ludachrish3/otto-sh
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
otto_sh-0.2.1-py3-none-any.whl -
Subject digest:
af7dfca3c8fe6f1c8d53a89d777ea0ed1a0367712e6780f970aa89512763a2e4 - Sigstore transparency entry: 1441382347
- Sigstore integration time:
-
Permalink:
ludachrish3/otto-sh@96628ee13827b38c1d2e170435e4196777b090c4 -
Branch / Tag:
refs/tags/v0.2.1 - Owner: https://github.com/ludachrish3
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@96628ee13827b38c1d2e170435e4196777b090c4 -
Trigger Event:
push
-
Statement type: