diff --git a/silverback/_cli.py b/silverback/_cli.py index b925cbd2..f3b5b8f2 100644 --- a/silverback/_cli.py +++ b/silverback/_cli.py @@ -37,7 +37,7 @@ from ape.contracts import ContractInstance from fief_client.integrations.cli import FiefAuth - from silverback.cluster.client import ClusterClient, PlatformClient + from silverback.cluster.client import Bot, ClusterClient, PlatformClient from silverback.cluster.types import VariableGroupInfo LOCAL_DATETIME = "%Y-%m-%d %H:%M:%S %Z" @@ -1061,8 +1061,29 @@ def new_bot( def list_bots(cluster: "ClusterClient"): """List all bots in a CLUSTER (Regardless of status)""" - if bot_names := list(cluster.bots): - click.echo(yaml.safe_dump(bot_names)) + if bot_names := cluster.bots_list(): + grouped_bots: dict[str, dict[str, list[Bot]]] = {} + for bot_list in bot_names.values(): + for bot in bot_list: + ecosystem, network, provider = bot.network.split("-") + network_key = f"{network}-{provider}" + grouped_bots.setdefault(ecosystem, {}).setdefault(network_key, []).append(bot) + + for ecosystem in sorted(grouped_bots.keys()): + grouped_bots[ecosystem] = { + network: sorted(bots, key=lambda b: b.name) + for network, bots in sorted(grouped_bots[ecosystem].items()) + } + + output = "" + for ecosystem in grouped_bots: + output += f"{ecosystem}:\n" + for network in grouped_bots[ecosystem]: + output += f" {network}:\n" + for bot in grouped_bots[ecosystem][network]: + output += f" - {bot.name}\n" + + click.echo(output) else: click.secho("No bots in this cluster", bold=True, fg="red") diff --git a/silverback/cluster/client.py b/silverback/cluster/client.py index 1167b181..ad85910e 100644 --- a/silverback/cluster/client.py +++ b/silverback/cluster/client.py @@ -1,3 +1,4 @@ +from collections import defaultdict from datetime import datetime from functools import cache from typing import ClassVar, Literal @@ -301,6 +302,14 @@ def bots_by_network(self, network: str) -> dict[str, Bot]: handle_error_with_response(response) return {bot.name: bot for bot in map(Bot.model_validate, response.json())} + def bots_list(self) -> dict[str, list[Bot]]: + response = self.get("/bots") + handle_error_with_response(response) + bots_dict = defaultdict(list) + for bot in map(Bot.model_validate, response.json()): + bots_dict[bot.name].append(bot) + return dict(bots_dict) + def new_bot( self, name: str,