import click
from dask import config
from rich.console import Console
from rich.table import Table
from coiled._beta.core import setup_logging
from coiled.exceptions import BuildError
from coiled.types import ArchitectureTypesEnum
from ..core import Cloud, create_software_environment, delete_software_environment
from ..utils import COILED_SERVER
from .utils import CONTEXT_SETTINGS
console = Console()
@click.group(context_settings=CONTEXT_SETTINGS)
def env():
"""Commands for managing Coiled software environments"""
setup_logging()
@env.command(context_settings=CONTEXT_SETTINGS)
@click.option("-n", "--name", help="Name of software environment, it must be lowercase.")
@click.option("--container", default=None, help="Base docker image to use.")
@click.option(
"--conda",
default=None,
help="Conda environment file.",
type=click.Path(exists=True),
)
@click.option("--pip", default=None, help="Pip requirements file.", type=click.Path(exists=True))
@click.option(
"--force-rebuild",
default=False,
help="Skip checks for an existing software environment build.",
is_flag=True,
)
@click.option(
"--account",
default=None,
type=str,
help="Account to use for creating this software environment.",
)
@click.option(
"--gpu-enabled",
is_flag=True,
show_default=True,
default=False,
help="Set CUDA virtual package for Conda",
)
@click.option(
"--architecture",
type=click.Choice([e.value for e in ArchitectureTypesEnum]),
default=ArchitectureTypesEnum.X86_64.value,
show_default=True,
help="CPU architecture to use for the software environment",
)
def create(
name,
container,
conda,
pip,
force_rebuild,
account,
gpu_enabled,
architecture,
):
"""Create a Coiled software environment"""
try:
create_software_environment(
name=name,
container=container,
conda=conda,
pip=pip,
force_rebuild=force_rebuild,
account=account,
gpu_enabled=gpu_enabled,
architecture=architecture,
)
except BuildError as e:
raise click.ClickException(f"{e}")
@env.command(context_settings=CONTEXT_SETTINGS)
@click.argument("name")
def delete(name: str):
"""Delete a Coiled software environment"""
delete_software_environment(
name,
)
@env.command(context_settings=CONTEXT_SETTINGS)
@click.argument("account", default=None, required=False)
def list(account: str):
"""List the Coiled software environments in an account"""
with Cloud(account=account) as cloud:
environments = cloud.list_software_environments(
account,
)
account = cloud.default_account
table = Table(title="Software Environments")
table.add_column("Name", style="cyan")
table.add_column("Updated", style="magenta")
table.add_column("Link", style="magenta")
table.add_column("Status", style="green")
server = config.get("coiled.server", COILED_SERVER).replace("8000", "5173")
account = account or config.get("coiled.account")
for env_name, env_details in environments.items():
latest_build = env_details["latest_spec"].get("latest_build")
if latest_build:
build_status = latest_build["state"] if latest_build["state"] != "error" else "[red]error[/red]"
else:
build_status = "n/a"
table.add_row(
env_name,
env_details["updated"],
f"{server}/software/{env_details['id']}/latest?account={account}",
build_status,
)
console.print(table)
[docs]@env.command(
context_settings=CONTEXT_SETTINGS,
help="View the details of a Coiled software environment",
)
@click.argument("name")
def inspect(name: str):
"""View the details of a Coiled software environment
Parameters
----------
name
Identifier of the software environment to use, in the format (<account>/)<name>. If the software environment
is owned by the same account as that passed into "account", the (<account>/) prefix is optional.
For example, suppose your account is "wondercorp", but your friends at "friendlycorp" have an environment
named "xgboost" that you want to use; you can specify this with "friendlycorp/xgboost". If you simply
entered "xgboost", this is shorthand for "wondercorp/xgboost".
The "name" portion of (<account>/)<name> can only contain ASCII letters, hyphens and underscores.
Examples
--------
>>> import coiled
>>> coiled.inspect("coiled/default")
"""
with Cloud() as cloud:
results = cloud.get_software_info(name)
console.print(results)