Subcommands¶
In these examples, we show how tyro.cli()
can be used to create CLI
interfaces with subcommands.
Subcommands are Unions¶
All of tyro
’s subcommand features are built using unions over struct
types (typically dataclasses). Subcommands are used to choose between types in
the union; arguments are then populated from the chosen type.
Note
For configuring subcommands beyond what can be expressed with type annotations, see
tyro.conf.subcommand()
.
1# 01_subcommands.py
2from __future__ import annotations
3
4import dataclasses
5
6import tyro
7
8@dataclasses.dataclass
9class Checkout:
10 """Checkout a branch."""
11
12 branch: str
13
14@dataclasses.dataclass
15class Commit:
16 """Commit changes."""
17
18 message: str
19
20if __name__ == "__main__":
21 cmd = tyro.cli(Checkout | Commit)
22 print(cmd)
Print the helptext. This will show the available subcommands:
$ python ./01_subcommands.py --help usage: 01_subcommands.py [-h] {checkout,commit} ╭─ options ───────────────────────────────────────────────╮ │ -h, --help show this help message and exit │ ╰─────────────────────────────────────────────────────────╯ ╭─ subcommands ───────────────────────────────────────────╮ │ {checkout,commit} │ │ checkout Checkout a branch. │ │ commit Commit changes. │ ╰─────────────────────────────────────────────────────────╯
The commit subcommand:
$ python ./01_subcommands.py commit --help usage: 01_subcommands.py commit [-h] --message STR Commit changes. ╭─ options ────────────────────────────────────────────╮ │ -h, --help show this help message and exit │ │ --message STR (required) │ ╰──────────────────────────────────────────────────────╯
$ python ./01_subcommands.py commit --message hello
Commit(message='hello')
The checkout subcommand:
$ python ./01_subcommands.py checkout --help usage: 01_subcommands.py checkout [-h] --branch STR Checkout a branch. ╭─ options ───────────────────────────────────────────╮ │ -h, --help show this help message and exit │ │ --branch STR (required) │ ╰─────────────────────────────────────────────────────╯
$ python ./01_subcommands.py checkout --branch main
Checkout(branch='main')
Subcommands as Function Arguments¶
A subcommand will be created for each input annotated with a union over struct types.
Note
To prevent tyro.cli()
from converting a Union type into a subcommand,
use tyro.conf.AvoidSubcommands
.
Note
Argument ordering for subcommands can be tricky. In the example below,
--shared-arg
must always come before the subcommand. As an option for
alleviating this, see tyro.conf.ConsolidateSubcommandArgs
.
1# 02_subcommands_in_func.py
2from __future__ import annotations
3
4import dataclasses
5
6import tyro
7
8@dataclasses.dataclass
9class Checkout:
10 """Checkout a branch."""
11
12 branch: str
13
14@dataclasses.dataclass
15class Commit:
16 """Commit changes."""
17
18 message: str
19
20def main(
21 shared_arg: int,
22 cmd: Checkout | Commit = Checkout(branch="default"),
23):
24 print(f"{shared_arg=}")
25 print(cmd)
26
27if __name__ == "__main__":
28 tyro.cli(main)
Print the helptext. This will show the available subcommands:
$ python ./02_subcommands_in_func.py --help usage: 02_subcommands_in_func.py [-h] --shared-arg INT [{cmd:checkout,cmd:commit}] ╭─ options ───────────────────────────────────────────────╮ │ -h, --help show this help message and exit │ │ --shared-arg INT (required) │ ╰─────────────────────────────────────────────────────────╯ ╭─ optional subcommands ──────────────────────────────────╮ │ (default: cmd:checkout) │ │ ────────────────────────────────────────── │ │ [{cmd:checkout,cmd:commit}] │ │ cmd:checkout Checkout a branch. │ │ cmd:commit Commit changes. │ ╰─────────────────────────────────────────────────────────╯
Using the default subcommand:
$ python ./02_subcommands_in_func.py --shared-arg 100
shared_arg=100
Checkout(branch='default')
Choosing a different subcommand:
$ python ./02_subcommands_in_func.py --shared-arg 100 cmd:commit --cmd.message 'Hello!'
shared_arg=100
Commit(message='Hello!')
Sequenced Subcommands¶
Multiple unions over struct types are populated using a series of subcommands.
1# 03_multiple_subcommands.py
2from __future__ import annotations
3
4import dataclasses
5from typing import Literal
6
7import tyro
8
9# Possible dataset configurations.
10
11@dataclasses.dataclass
12class Mnist:
13 binary: bool = False
14 """Set to load binary version of MNIST dataset."""
15
16@dataclasses.dataclass
17class ImageNet:
18 subset: Literal[50, 100, 1000]
19 """Choose between ImageNet-50, ImageNet-100, ImageNet-1000, etc."""
20
21# Possible optimizer configurations.
22
23@dataclasses.dataclass
24class Adam:
25 learning_rate: float = 1e-3
26 betas: tuple[float, float] = (0.9, 0.999)
27
28@dataclasses.dataclass
29class Sgd:
30 learning_rate: float = 3e-4
31
32# Train script.
33
34def train(
35 dataset: Mnist | ImageNet = Mnist(),
36 optimizer: Adam | Sgd = Adam(),
37) -> None:
38 """Example training script.
39
40 Args:
41 dataset: Dataset to train on.
42 optimizer: Optimizer to train with.
43
44 Returns:
45 None:
46 """
47 print(dataset)
48 print(optimizer)
49
50if __name__ == "__main__":
51 tyro.cli(train, config=(tyro.conf.ConsolidateSubcommandArgs,))
We apply the tyro.conf.ConsolidateSubcommandArgs
flag. This
pushes all arguments to the end of the command:
$ python ./03_multiple_subcommands.py --help usage: 03_multiple_subcommands.py [-h] [{dataset:mnist,dataset:image-net}] Example training script. ╭─ options ─────────────────────────────────────────╮ │ -h, --help show this help message and exit │ ╰───────────────────────────────────────────────────╯ ╭─ optional subcommands ────────────────────────────╮ │ Dataset to train on. (default: dataset:mnist) │ │ ───────────────────────────────────────────── │ │ [{dataset:mnist,dataset:image-net}] │ │ dataset:mnist │ │ dataset:image-net │ ╰───────────────────────────────────────────────────╯
$ python ./03_multiple_subcommands.py dataset:mnist --help usage: 03_multiple_subcommands.py dataset:mnist [-h] [{optimizer:adam,optimizer:sgd }] ╭─ options ──────────────────────────────────────────╮ │ -h, --help show this help message and exit │ ╰────────────────────────────────────────────────────╯ ╭─ optional subcommands ─────────────────────────────╮ │ Optimizer to train with. (default: optimizer:adam) │ │ ────────────────────────────────────────────────── │ │ [{optimizer:adam,optimizer:sgd}] │ │ optimizer:adam │ │ optimizer:sgd │ ╰────────────────────────────────────────────────────╯
$ python ./03_multiple_subcommands.py dataset:mnist optimizer:adam --help usage: 03_multiple_subcommands.py dataset:mnist optimizer:adam [-h] [--optimizer.learning-rate FLOAT] [--optimizer.betas FLOAT FLOAT] [--dataset.binary | --dataset.no-binary] ╭─ options ─────────────────────────────────────────────────────────╮ │ -h, --help │ │ show this help message and exit │ ╰───────────────────────────────────────────────────────────────────╯ ╭─ optimizer options ───────────────────────────────────────────────╮ │ --optimizer.learning-rate FLOAT │ │ (default: 0.001) │ │ --optimizer.betas FLOAT FLOAT │ │ (default: 0.9 0.999) │ ╰───────────────────────────────────────────────────────────────────╯ ╭─ dataset options ─────────────────────────────────────────────────╮ │ --dataset.binary, --dataset.no-binary │ │ Set to load binary version of MNIST dataset. (default: False) │ ╰───────────────────────────────────────────────────────────────────╯
$ python ./03_multiple_subcommands.py dataset:mnist optimizer:adam --optimizer.learning-rate 3e-4 --dataset.binary
Mnist(binary=True)
Adam(learning_rate=0.0003, betas=(0.9, 0.999))
Decorator-based Subcommands¶
tyro.extras.SubcommandApp()
provides a decorator-based API for
subcommands, which is inspired by click.
1# 04_decorator_subcommands.py
2from tyro.extras import SubcommandApp
3
4app = SubcommandApp()
5
6@app.command
7def greet(name: str, loud: bool = False) -> None:
8 """Greet someone."""
9 greeting = f"Hello, {name}!"
10 if loud:
11 greeting = greeting.upper()
12 print(greeting)
13
14@app.command(name="addition")
15def add(a: int, b: int) -> None:
16 """Add two numbers."""
17 print(f"{a} + {b} = {a + b}")
18
19if __name__ == "__main__":
20 app.cli()
$ python 04_decorator_subcommands.py --help usage: 04_decorator_subcommands.py [-h] {greet,addition} ╭─ options ───────────────────────────────────────────────╮ │ -h, --help show this help message and exit │ ╰─────────────────────────────────────────────────────────╯ ╭─ subcommands ───────────────────────────────────────────╮ │ {greet,addition} │ │ greet Greet someone. │ │ addition Add two numbers. │ ╰─────────────────────────────────────────────────────────╯
$ python 04_decorator_subcommands.py greet --help usage: 04_decorator_subcommands.py greet [-h] --name STR [--loud | --no-loud] Greet someone. ╭─ options ───────────────────────────────────────────────╮ │ -h, --help show this help message and exit │ │ --name STR (required) │ │ --loud, --no-loud (default: False) │ ╰─────────────────────────────────────────────────────────╯
$ python 04_decorator_subcommands.py greet --name Alice
Hello, Alice!
$ python 04_decorator_subcommands.py greet --name Bob --loud
HELLO, BOB!
$ python 04_decorator_subcommands.py addition --help usage: 04_decorator_subcommands.py addition [-h] --a INT --b INT Add two numbers. ╭─ options ─────────────────────────────────────────╮ │ -h, --help show this help message and exit │ │ --a INT (required) │ │ --b INT (required) │ ╰───────────────────────────────────────────────────╯
$ python 04_decorator_subcommands.py addition --a 5 --b 3
5 + 3 = 8
Subcommands from Functions¶
We provide a shorthand for generating a subcommand CLI from a dictionary. This
is a thin wrapper around tyro.cli()
’s more verbose, type-based API. If
more generality is needed, the internal working are explained in the docs for
tyro.extras.subcommand_cli_from_dict()
.
1# 05_subcommands_func.py
2import tyro
3
4def checkout(branch: str) -> None:
5 """Check out a branch."""
6 print(f"{branch=}")
7
8def commit(message: str, all: bool = False) -> None:
9 """Make a commit."""
10 print(f"{message=} {all=}")
11
12if __name__ == "__main__":
13 tyro.extras.subcommand_cli_from_dict(
14 {
15 "checkout": checkout,
16 "commit": commit,
17 }
18 )
$ python ./05_subcommands_func.py --help usage: 05_subcommands_func.py [-h] {checkout,commit} ╭─ options ───────────────────────────────────────────────╮ │ -h, --help show this help message and exit │ ╰─────────────────────────────────────────────────────────╯ ╭─ subcommands ───────────────────────────────────────────╮ │ {checkout,commit} │ │ checkout Check out a branch. │ │ commit Make a commit. │ ╰─────────────────────────────────────────────────────────╯
$ python ./05_subcommands_func.py commit --help usage: 05_subcommands_func.py commit [-h] --message STR [--all | --no-all] Make a commit. ╭─ options ──────────────────────────────────────────────╮ │ -h, --help show this help message and exit │ │ --message STR (required) │ │ --all, --no-all (default: False) │ ╰────────────────────────────────────────────────────────╯
$ python ./05_subcommands_func.py commit --message hello --all
message='hello' all=True
$ python ./05_subcommands_func.py checkout --help usage: 05_subcommands_func.py checkout [-h] --branch STR Check out a branch. ╭─ options ───────────────────────────────────────────╮ │ -h, --help show this help message and exit │ │ --branch STR (required) │ ╰─────────────────────────────────────────────────────╯
$ python ./05_subcommands_func.py checkout --branch main
branch='main'