Nested SubCommandsΒΆ
We'll now see how these same ideas can be extended for deeply nested commands.
Let's imagine that the same CLI program from the previous examples now needs to handle lands
.
But a land could be a reign
or town
.
And each of those could have their own commands, like create
and delete
.
A CLI app for reignsΒΆ
Let's start with a file reigns.py
:
import typer
app = typer.Typer()
@app.command()
def conquer(name: str):
print(f"Conquering reign: {name}")
@app.command()
def destroy(name: str):
print(f"Destroying reign: {name}")
if __name__ == "__main__":
app()
This is already a simple CLI program to manage reigns:
A CLI app for townsΒΆ
And now the equivalent for managing towns in towns.py
:
import typer
app = typer.Typer()
@app.command()
def found(name: str):
print(f"Founding town: {name}")
@app.command()
def burn(name: str):
print(f"Burning town: {name}")
if __name__ == "__main__":
app()
With it, you can manage towns:
Manage the land in a CLI appΒΆ
Now let's put the reigns
and towns
together in the same CLI program in lands.py
:
import typer
import reigns
import towns
app = typer.Typer()
app.add_typer(reigns.app, name="reigns")
app.add_typer(towns.app, name="towns")
if __name__ == "__main__":
app()
And now we have a single CLI program with a command (or command group) reigns
that has its own commands. And another command towns
with its own subcommands.
Check it:
Now try it, manage the lands through the CLI:
Deeply nested subcommandsΒΆ
Now let's say that all these commands in the lands.py
CLI program should be part of the previous CLI program we built in the first example.
We want our CLI program to have these commands/command groups:
users
:create
delete
items
:create
delete
sell
lands
:reigns
:conquer
destroy
towns
:found
burn
This already is a quite deeply nested "tree" of commands/command groups.
But to achieve that, we just have to add the lands
Typer app to the same main.py
file we already had:
import typer
import items
import lands
import users
app = typer.Typer()
app.add_typer(users.app, name="users")
app.add_typer(items.app, name="items")
app.add_typer(lands.app, name="lands")
if __name__ == "__main__":
app()
And now we have everything in a single CLI program:
Review the filesΒΆ
Here are all the files if you want to review/copy them:
reigns.py
:
import typer
app = typer.Typer()
@app.command()
def conquer(name: str):
print(f"Conquering reign: {name}")
@app.command()
def destroy(name: str):
print(f"Destroying reign: {name}")
if __name__ == "__main__":
app()
towns.py
:
import typer
app = typer.Typer()
@app.command()
def found(name: str):
print(f"Founding town: {name}")
@app.command()
def burn(name: str):
print(f"Burning town: {name}")
if __name__ == "__main__":
app()
lands.py
:
import typer
import reigns
import towns
app = typer.Typer()
app.add_typer(reigns.app, name="reigns")
app.add_typer(towns.app, name="towns")
if __name__ == "__main__":
app()
users.py
:
import typer
app = typer.Typer()
@app.command()
def create(user_name: str):
print(f"Creating user: {user_name}")
@app.command()
def delete(user_name: str):
print(f"Deleting user: {user_name}")
if __name__ == "__main__":
app()
items.py
:
import typer
app = typer.Typer()
@app.command()
def create(item: str):
print(f"Creating item: {item}")
@app.command()
def delete(item: str):
print(f"Deleting item: {item}")
@app.command()
def sell(item: str):
print(f"Selling item: {item}")
if __name__ == "__main__":
app()
main.py
:
import typer
import items
import lands
import users
app = typer.Typer()
app.add_typer(users.app, name="users")
app.add_typer(items.app, name="items")
app.add_typer(lands.app, name="lands")
if __name__ == "__main__":
app()
Tip
All these files have an if __name__ == "__main__"
block just to demonstrate how each of them can also be an independent CLI app.
But for your final application, only main.py
would need it.
RecapΒΆ
That's it, you can just add Typer applications one inside another as much as you want and create complex CLI programs while writing simple code.
You can probably achieve a simpler CLI program design that's easier to use than the example here. But if your requirements are complex, Typer helps you build your CLI app easily.
Tip
Auto completion helps a lot, specially with complex programs.
Check the docs about adding auto completion to your CLI apps.