mirror of
				https://github.com/meta-llama/llama-stack.git
				synced 2025-10-25 09:05:37 +00:00 
			
		
		
		
	# What does this PR do? when --image-name is not provided the build script default to the image_name in the config, this makes sure the same is done for the run script ## Test Plan llama stack build w/o --image-name
		
			
				
	
	
		
			484 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			484 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright (c) Meta Platforms, Inc. and affiliates.
 | |
| # All rights reserved.
 | |
| #
 | |
| # This source code is licensed under the terms described in the LICENSE file in
 | |
| # the root directory of this source tree.
 | |
| 
 | |
| import argparse
 | |
| import importlib.resources
 | |
| import json
 | |
| import os
 | |
| import shutil
 | |
| import sys
 | |
| import textwrap
 | |
| from functools import lru_cache
 | |
| from importlib.abc import Traversable
 | |
| from pathlib import Path
 | |
| 
 | |
| import yaml
 | |
| from prompt_toolkit import prompt
 | |
| from prompt_toolkit.completion import WordCompleter
 | |
| from prompt_toolkit.validation import Validator
 | |
| from termcolor import colored, cprint
 | |
| 
 | |
| from llama_stack.cli.stack.utils import ImageType
 | |
| from llama_stack.cli.table import print_table
 | |
| from llama_stack.distribution.build import (
 | |
|     SERVER_DEPENDENCIES,
 | |
|     build_image,
 | |
|     get_provider_dependencies,
 | |
| )
 | |
| from llama_stack.distribution.configure import parse_and_maybe_upgrade_config
 | |
| from llama_stack.distribution.datatypes import (
 | |
|     BuildConfig,
 | |
|     BuildProvider,
 | |
|     DistributionSpec,
 | |
|     Provider,
 | |
|     StackRunConfig,
 | |
| )
 | |
| from llama_stack.distribution.distribution import get_provider_registry
 | |
| from llama_stack.distribution.external import load_external_apis
 | |
| from llama_stack.distribution.resolver import InvalidProviderError
 | |
| from llama_stack.distribution.stack import replace_env_vars
 | |
| from llama_stack.distribution.utils.config_dirs import DISTRIBS_BASE_DIR, EXTERNAL_PROVIDERS_DIR
 | |
| from llama_stack.distribution.utils.dynamic import instantiate_class_type
 | |
| from llama_stack.distribution.utils.exec import formulate_run_args, run_command
 | |
| from llama_stack.distribution.utils.image_types import LlamaStackImageType
 | |
| from llama_stack.providers.datatypes import Api
 | |
| 
 | |
| TEMPLATES_PATH = Path(__file__).parent.parent.parent / "templates"
 | |
| 
 | |
| 
 | |
| @lru_cache
 | |
| def available_templates_specs() -> dict[str, BuildConfig]:
 | |
|     import yaml
 | |
| 
 | |
|     template_specs = {}
 | |
|     for p in TEMPLATES_PATH.rglob("*build.yaml"):
 | |
|         template_name = p.parent.name
 | |
|         with open(p) as f:
 | |
|             build_config = BuildConfig(**yaml.safe_load(f))
 | |
|             template_specs[template_name] = build_config
 | |
|     return template_specs
 | |
| 
 | |
| 
 | |
| def run_stack_build_command(args: argparse.Namespace) -> None:
 | |
|     if args.list_templates:
 | |
|         return _run_template_list_cmd()
 | |
| 
 | |
|     if args.image_type == ImageType.VENV.value:
 | |
|         current_venv = os.environ.get("VIRTUAL_ENV")
 | |
|         image_name = args.image_name or current_venv
 | |
|     elif args.image_type == ImageType.CONDA.value:
 | |
|         current_conda_env = os.environ.get("CONDA_DEFAULT_ENV")
 | |
|         image_name = args.image_name or current_conda_env
 | |
|     else:
 | |
|         image_name = args.image_name
 | |
| 
 | |
|     if args.template:
 | |
|         available_templates = available_templates_specs()
 | |
|         if args.template not in available_templates:
 | |
|             cprint(
 | |
|                 f"Could not find template {args.template}. Please run `llama stack build --list-templates` to check out the available templates",
 | |
|                 color="red",
 | |
|                 file=sys.stderr,
 | |
|             )
 | |
|             sys.exit(1)
 | |
|         build_config = available_templates[args.template]
 | |
|         if args.image_type:
 | |
|             build_config.image_type = args.image_type
 | |
|         else:
 | |
|             cprint(
 | |
|                 f"Please specify a image-type ({' | '.join(e.value for e in ImageType)}) for {args.template}",
 | |
|                 color="red",
 | |
|                 file=sys.stderr,
 | |
|             )
 | |
|             sys.exit(1)
 | |
|     elif args.providers:
 | |
|         provider_list: dict[str, list[BuildProvider]] = dict()
 | |
|         for api_provider in args.providers.split(","):
 | |
|             if "=" not in api_provider:
 | |
|                 cprint(
 | |
|                     "Could not parse `--providers`. Please ensure the list is in the format api1=provider1,api2=provider2",
 | |
|                     color="red",
 | |
|                     file=sys.stderr,
 | |
|                 )
 | |
|                 sys.exit(1)
 | |
|             api, provider_type = api_provider.split("=")
 | |
|             providers_for_api = get_provider_registry().get(Api(api), None)
 | |
|             if providers_for_api is None:
 | |
|                 cprint(
 | |
|                     f"{api} is not a valid API.",
 | |
|                     color="red",
 | |
|                     file=sys.stderr,
 | |
|                 )
 | |
|                 sys.exit(1)
 | |
|             if provider_type in providers_for_api:
 | |
|                 provider = BuildProvider(
 | |
|                     provider_type=provider_type,
 | |
|                     module=None,
 | |
|                 )
 | |
|                 provider_list.setdefault(api, []).append(provider)
 | |
|             else:
 | |
|                 cprint(
 | |
|                     f"{provider} is not a valid provider for the {api} API.",
 | |
|                     color="red",
 | |
|                     file=sys.stderr,
 | |
|                 )
 | |
|                 sys.exit(1)
 | |
|         distribution_spec = DistributionSpec(
 | |
|             providers=provider_list,
 | |
|             description=",".join(args.providers),
 | |
|         )
 | |
|         if not args.image_type:
 | |
|             cprint(
 | |
|                 f"Please specify a image-type (container | conda | venv) for {args.template}",
 | |
|                 color="red",
 | |
|                 file=sys.stderr,
 | |
|             )
 | |
|             sys.exit(1)
 | |
| 
 | |
|         build_config = BuildConfig(image_type=args.image_type, distribution_spec=distribution_spec)
 | |
|     elif not args.config and not args.template:
 | |
|         name = prompt(
 | |
|             "> Enter a name for your Llama Stack (e.g. my-local-stack): ",
 | |
|             validator=Validator.from_callable(
 | |
|                 lambda x: len(x) > 0,
 | |
|                 error_message="Name cannot be empty, please enter a name",
 | |
|             ),
 | |
|         )
 | |
| 
 | |
|         image_type = prompt(
 | |
|             "> Enter the image type you want your Llama Stack to be built as (use <TAB> to see options): ",
 | |
|             completer=WordCompleter([e.value for e in ImageType]),
 | |
|             complete_while_typing=True,
 | |
|             validator=Validator.from_callable(
 | |
|                 lambda x: x in [e.value for e in ImageType],
 | |
|                 error_message="Invalid image type. Use <TAB> to see options",
 | |
|             ),
 | |
|         )
 | |
| 
 | |
|         if image_type == ImageType.CONDA.value:
 | |
|             if not image_name:
 | |
|                 cprint(
 | |
|                     f"No current conda environment detected or specified, will create a new conda environment with the name `llamastack-{name}`",
 | |
|                     color="yellow",
 | |
|                     file=sys.stderr,
 | |
|                 )
 | |
|                 image_name = f"llamastack-{name}"
 | |
|             else:
 | |
|                 cprint(
 | |
|                     f"Using conda environment {image_name}",
 | |
|                     color="green",
 | |
|                     file=sys.stderr,
 | |
|                 )
 | |
|         else:
 | |
|             image_name = f"llamastack-{name}"
 | |
| 
 | |
|         cprint(
 | |
|             textwrap.dedent(
 | |
|                 """
 | |
|             Llama Stack is composed of several APIs working together. Let's select
 | |
|             the provider types (implementations) you want to use for these APIs.
 | |
|             """,
 | |
|             ),
 | |
|             color="green",
 | |
|             file=sys.stderr,
 | |
|         )
 | |
| 
 | |
|         cprint("Tip: use <TAB> to see options for the providers.\n", color="green", file=sys.stderr)
 | |
| 
 | |
|         providers: dict[str, list[BuildProvider]] = dict()
 | |
|         for api, providers_for_api in get_provider_registry().items():
 | |
|             available_providers = [x for x in providers_for_api.keys() if x not in ("remote", "remote::sample")]
 | |
|             if not available_providers:
 | |
|                 continue
 | |
|             api_provider = prompt(
 | |
|                 f"> Enter provider for API {api.value}: ",
 | |
|                 completer=WordCompleter(available_providers),
 | |
|                 complete_while_typing=True,
 | |
|                 validator=Validator.from_callable(
 | |
|                     lambda x: x in available_providers,  # noqa: B023 - see https://github.com/astral-sh/ruff/issues/7847
 | |
|                     error_message="Invalid provider, use <TAB> to see options",
 | |
|                 ),
 | |
|             )
 | |
| 
 | |
|             string_providers = api_provider.split(" ")
 | |
| 
 | |
|             for provider in string_providers:
 | |
|                 providers.setdefault(api.value, []).append(BuildProvider(provider_type=provider))
 | |
| 
 | |
|         description = prompt(
 | |
|             "\n > (Optional) Enter a short description for your Llama Stack: ",
 | |
|             default="",
 | |
|         )
 | |
| 
 | |
|         distribution_spec = DistributionSpec(
 | |
|             providers=providers,
 | |
|             description=description,
 | |
|         )
 | |
| 
 | |
|         build_config = BuildConfig(image_type=image_type, distribution_spec=distribution_spec)
 | |
|     else:
 | |
|         with open(args.config) as f:
 | |
|             try:
 | |
|                 contents = yaml.safe_load(f)
 | |
|                 contents = replace_env_vars(contents)
 | |
|                 build_config = BuildConfig(**contents)
 | |
|                 if args.image_type:
 | |
|                     build_config.image_type = args.image_type
 | |
|             except Exception as e:
 | |
|                 cprint(
 | |
|                     f"Could not parse config file {args.config}: {e}",
 | |
|                     color="red",
 | |
|                     file=sys.stderr,
 | |
|                 )
 | |
|                 sys.exit(1)
 | |
| 
 | |
|     if args.print_deps_only:
 | |
|         print(f"# Dependencies for {args.template or args.config or image_name}")
 | |
|         normal_deps, special_deps, external_provider_dependencies = get_provider_dependencies(build_config)
 | |
|         normal_deps += SERVER_DEPENDENCIES
 | |
|         print(f"uv pip install {' '.join(normal_deps)}")
 | |
|         for special_dep in special_deps:
 | |
|             print(f"uv pip install {special_dep}")
 | |
|         for external_dep in external_provider_dependencies:
 | |
|             print(f"uv pip install {external_dep}")
 | |
|         return
 | |
| 
 | |
|     try:
 | |
|         run_config = _run_stack_build_command_from_build_config(
 | |
|             build_config,
 | |
|             image_name=image_name,
 | |
|             config_path=args.config,
 | |
|             template_name=args.template,
 | |
|         )
 | |
| 
 | |
|     except (Exception, RuntimeError) as exc:
 | |
|         import traceback
 | |
| 
 | |
|         cprint(
 | |
|             f"Error building stack: {exc}",
 | |
|             color="red",
 | |
|             file=sys.stderr,
 | |
|         )
 | |
|         cprint("Stack trace:", color="red", file=sys.stderr)
 | |
|         traceback.print_exc()
 | |
|         sys.exit(1)
 | |
| 
 | |
|     if run_config is None:
 | |
|         cprint(
 | |
|             "Run config path is empty",
 | |
|             color="red",
 | |
|             file=sys.stderr,
 | |
|         )
 | |
|         sys.exit(1)
 | |
| 
 | |
|     if args.run:
 | |
|         config_dict = yaml.safe_load(run_config.read_text())
 | |
|         config = parse_and_maybe_upgrade_config(config_dict)
 | |
|         if config.external_providers_dir and not config.external_providers_dir.exists():
 | |
|             config.external_providers_dir.mkdir(exist_ok=True)
 | |
|         run_args = formulate_run_args(args.image_type, image_name or config.image_name)
 | |
|         run_args.extend([str(os.getenv("LLAMA_STACK_PORT", 8321)), "--config", str(run_config)])
 | |
|         run_command(run_args)
 | |
| 
 | |
| 
 | |
| def _generate_run_config(
 | |
|     build_config: BuildConfig,
 | |
|     build_dir: Path,
 | |
|     image_name: str,
 | |
| ) -> Path:
 | |
|     """
 | |
|     Generate a run.yaml template file for user to edit from a build.yaml file
 | |
|     """
 | |
|     apis = list(build_config.distribution_spec.providers.keys())
 | |
|     run_config = StackRunConfig(
 | |
|         container_image=(image_name if build_config.image_type == LlamaStackImageType.CONTAINER.value else None),
 | |
|         image_name=image_name,
 | |
|         apis=apis,
 | |
|         providers={},
 | |
|         external_providers_dir=build_config.external_providers_dir
 | |
|         if build_config.external_providers_dir
 | |
|         else EXTERNAL_PROVIDERS_DIR,
 | |
|     )
 | |
|     # build providers dict
 | |
|     provider_registry = get_provider_registry(build_config)
 | |
|     for api in apis:
 | |
|         run_config.providers[api] = []
 | |
|         providers = build_config.distribution_spec.providers[api]
 | |
| 
 | |
|         for provider in providers:
 | |
|             pid = provider.provider_type.split("::")[-1]
 | |
| 
 | |
|             p = provider_registry[Api(api)][provider.provider_type]
 | |
|             if p.deprecation_error:
 | |
|                 raise InvalidProviderError(p.deprecation_error)
 | |
| 
 | |
|             try:
 | |
|                 config_type = instantiate_class_type(provider_registry[Api(api)][provider.provider_type].config_class)
 | |
|             except (ModuleNotFoundError, ValueError) as exc:
 | |
|                 # HACK ALERT:
 | |
|                 # This code executes after building is done, the import cannot work since the
 | |
|                 # package is either available in the venv or container - not available on the host.
 | |
|                 # TODO: use a "is_external" flag in ProviderSpec to check if the provider is
 | |
|                 # external
 | |
|                 cprint(
 | |
|                     f"Failed to import provider {provider.provider_type} for API {api} - assuming it's external, skipping: {exc}",
 | |
|                     color="yellow",
 | |
|                     file=sys.stderr,
 | |
|                 )
 | |
|                 # Set config_type to None to avoid UnboundLocalError
 | |
|                 config_type = None
 | |
| 
 | |
|             if config_type is not None and hasattr(config_type, "sample_run_config"):
 | |
|                 config = config_type.sample_run_config(__distro_dir__=f"~/.llama/distributions/{image_name}")
 | |
|             else:
 | |
|                 config = {}
 | |
| 
 | |
|             p_spec = Provider(
 | |
|                 provider_id=pid,
 | |
|                 provider_type=provider.provider_type,
 | |
|                 config=config,
 | |
|                 module=provider.module,
 | |
|             )
 | |
|             run_config.providers[api].append(p_spec)
 | |
| 
 | |
|     run_config_file = build_dir / f"{image_name}-run.yaml"
 | |
| 
 | |
|     with open(run_config_file, "w") as f:
 | |
|         to_write = json.loads(run_config.model_dump_json())
 | |
|         f.write(yaml.dump(to_write, sort_keys=False))
 | |
| 
 | |
|     # Only print this message for non-container builds since it will be displayed before the
 | |
|     # container is built
 | |
|     # For non-container builds, the run.yaml is generated at the very end of the build process so it
 | |
|     # makes sense to display this message
 | |
|     if build_config.image_type != LlamaStackImageType.CONTAINER.value:
 | |
|         cprint(f"You can now run your stack with `llama stack run {run_config_file}`", color="green", file=sys.stderr)
 | |
|     return run_config_file
 | |
| 
 | |
| 
 | |
| def _run_stack_build_command_from_build_config(
 | |
|     build_config: BuildConfig,
 | |
|     image_name: str | None = None,
 | |
|     template_name: str | None = None,
 | |
|     config_path: str | None = None,
 | |
| ) -> Path | Traversable:
 | |
|     image_name = image_name or build_config.image_name
 | |
|     if build_config.image_type == LlamaStackImageType.CONTAINER.value:
 | |
|         if template_name:
 | |
|             image_name = f"distribution-{template_name}"
 | |
|         else:
 | |
|             if not image_name:
 | |
|                 raise ValueError("Please specify an image name when building a container image without a template")
 | |
|     elif build_config.image_type == LlamaStackImageType.CONDA.value:
 | |
|         if not image_name:
 | |
|             raise ValueError("Please specify an image name when building a conda image")
 | |
|     elif build_config.image_type == LlamaStackImageType.VENV.value:
 | |
|         if not image_name and os.environ.get("UV_SYSTEM_PYTHON"):
 | |
|             image_name = "__system__"
 | |
|         if not image_name:
 | |
|             raise ValueError("Please specify an image name when building a venv image")
 | |
| 
 | |
|     # At this point, image_name should be guaranteed to be a string
 | |
|     if image_name is None:
 | |
|         raise ValueError("image_name should not be None after validation")
 | |
| 
 | |
|     if template_name:
 | |
|         build_dir = DISTRIBS_BASE_DIR / template_name
 | |
|         build_file_path = build_dir / f"{template_name}-build.yaml"
 | |
|     else:
 | |
|         if image_name is None:
 | |
|             raise ValueError("image_name cannot be None")
 | |
|         build_dir = DISTRIBS_BASE_DIR / image_name
 | |
|         build_file_path = build_dir / f"{image_name}-build.yaml"
 | |
| 
 | |
|     os.makedirs(build_dir, exist_ok=True)
 | |
|     run_config_file = None
 | |
|     # Generate the run.yaml so it can be included in the container image with the proper entrypoint
 | |
|     # Only do this if we're building a container image and we're not using a template
 | |
|     if build_config.image_type == LlamaStackImageType.CONTAINER.value and not template_name and config_path:
 | |
|         cprint("Generating run.yaml file", color="yellow", file=sys.stderr)
 | |
|         run_config_file = _generate_run_config(build_config, build_dir, image_name)
 | |
| 
 | |
|     with open(build_file_path, "w") as f:
 | |
|         to_write = json.loads(build_config.model_dump_json(exclude_none=True))
 | |
|         f.write(yaml.dump(to_write, sort_keys=False))
 | |
| 
 | |
|     # We first install the external APIs so that the build process can use them and discover the
 | |
|     # providers dependencies
 | |
|     if build_config.external_apis_dir:
 | |
|         cprint("Installing external APIs", color="yellow", file=sys.stderr)
 | |
|         external_apis = load_external_apis(build_config)
 | |
|         if external_apis:
 | |
|             # install the external APIs
 | |
|             packages = []
 | |
|             for _, api_spec in external_apis.items():
 | |
|                 if api_spec.pip_packages:
 | |
|                     packages.extend(api_spec.pip_packages)
 | |
|                     cprint(
 | |
|                         f"Installing {api_spec.name} with pip packages {api_spec.pip_packages}",
 | |
|                         color="yellow",
 | |
|                         file=sys.stderr,
 | |
|                     )
 | |
|             return_code = run_command(["uv", "pip", "install", *packages])
 | |
|             if return_code != 0:
 | |
|                 packages_str = ", ".join(packages)
 | |
|                 raise RuntimeError(
 | |
|                     f"Failed to install external APIs packages: {packages_str} (return code: {return_code})"
 | |
|                 )
 | |
| 
 | |
|     return_code = build_image(
 | |
|         build_config,
 | |
|         build_file_path,
 | |
|         image_name,
 | |
|         template_or_config=template_name or config_path or str(build_file_path),
 | |
|         run_config=run_config_file.as_posix() if run_config_file else None,
 | |
|     )
 | |
|     if return_code != 0:
 | |
|         raise RuntimeError(f"Failed to build image {image_name}")
 | |
| 
 | |
|     if template_name:
 | |
|         # copy run.yaml from template to build_dir instead of generating it again
 | |
|         template_path = importlib.resources.files("llama_stack") / f"templates/{template_name}/run.yaml"
 | |
|         run_config_file = build_dir / f"{template_name}-run.yaml"
 | |
| 
 | |
|         with importlib.resources.as_file(template_path) as path:
 | |
|             shutil.copy(path, run_config_file)
 | |
| 
 | |
|         cprint("Build Successful!", color="green", file=sys.stderr)
 | |
|         cprint(f"You can find the newly-built template here: {run_config_file}", color="blue", file=sys.stderr)
 | |
|         cprint(
 | |
|             "You can run the new Llama Stack distro via: "
 | |
|             + colored(f"llama stack run {run_config_file} --image-type {build_config.image_type}", "blue"),
 | |
|             color="green",
 | |
|             file=sys.stderr,
 | |
|         )
 | |
|         return template_path
 | |
|     else:
 | |
|         return _generate_run_config(build_config, build_dir, image_name)
 | |
| 
 | |
| 
 | |
| def _run_template_list_cmd() -> None:
 | |
|     # eventually, this should query a registry at llama.meta.com/llamastack/distributions
 | |
|     headers = [
 | |
|         "Template Name",
 | |
|         # "Providers",
 | |
|         "Description",
 | |
|     ]
 | |
| 
 | |
|     rows = []
 | |
|     for template_name, spec in available_templates_specs().items():
 | |
|         rows.append(
 | |
|             [
 | |
|                 template_name,
 | |
|                 # json.dumps(spec.distribution_spec.providers, indent=2),
 | |
|                 spec.distribution_spec.description,
 | |
|             ]
 | |
|         )
 | |
|     print_table(
 | |
|         rows,
 | |
|         headers,
 | |
|         separate_rows=True,
 | |
|     )
 |