llama-stack 0.0.42__py3-none-any.whl → 0.3.4__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- llama_stack/__init__.py +5 -0
- llama_stack/apis/agents/__init__.py +1 -1
- llama_stack/apis/agents/agents.py +700 -281
- llama_stack/apis/agents/openai_responses.py +1311 -0
- llama_stack/{providers/adapters/memory/sample/config.py → apis/batches/__init__.py} +2 -5
- llama_stack/apis/batches/batches.py +100 -0
- llama_stack/apis/benchmarks/__init__.py +7 -0
- llama_stack/apis/benchmarks/benchmarks.py +108 -0
- llama_stack/apis/common/content_types.py +143 -0
- llama_stack/apis/common/errors.py +103 -0
- llama_stack/apis/common/job_types.py +38 -0
- llama_stack/apis/common/responses.py +36 -0
- llama_stack/apis/common/training_types.py +36 -5
- llama_stack/apis/common/type_system.py +158 -0
- llama_stack/apis/conversations/__init__.py +31 -0
- llama_stack/apis/conversations/conversations.py +286 -0
- llama_stack/apis/datasetio/__init__.py +7 -0
- llama_stack/apis/datasetio/datasetio.py +59 -0
- llama_stack/apis/datasets/__init__.py +7 -0
- llama_stack/apis/datasets/datasets.py +251 -0
- llama_stack/apis/datatypes.py +160 -0
- llama_stack/apis/eval/__init__.py +7 -0
- llama_stack/apis/eval/eval.py +169 -0
- llama_stack/apis/files/__init__.py +7 -0
- llama_stack/apis/files/files.py +199 -0
- llama_stack/apis/inference/__init__.py +1 -1
- llama_stack/apis/inference/inference.py +1169 -113
- llama_stack/apis/inspect/__init__.py +1 -1
- llama_stack/apis/inspect/inspect.py +69 -16
- llama_stack/apis/models/__init__.py +1 -1
- llama_stack/apis/models/models.py +148 -21
- llama_stack/apis/post_training/__init__.py +1 -1
- llama_stack/apis/post_training/post_training.py +265 -120
- llama_stack/{providers/adapters/agents/sample/config.py → apis/prompts/__init__.py} +2 -5
- llama_stack/apis/prompts/prompts.py +204 -0
- llama_stack/apis/providers/__init__.py +7 -0
- llama_stack/apis/providers/providers.py +69 -0
- llama_stack/apis/resource.py +37 -0
- llama_stack/apis/safety/__init__.py +1 -1
- llama_stack/apis/safety/safety.py +95 -12
- llama_stack/apis/scoring/__init__.py +7 -0
- llama_stack/apis/scoring/scoring.py +93 -0
- llama_stack/apis/scoring_functions/__init__.py +7 -0
- llama_stack/apis/scoring_functions/scoring_functions.py +208 -0
- llama_stack/apis/shields/__init__.py +1 -1
- llama_stack/apis/shields/shields.py +76 -33
- llama_stack/apis/synthetic_data_generation/__init__.py +1 -1
- llama_stack/apis/synthetic_data_generation/synthetic_data_generation.py +40 -17
- llama_stack/apis/telemetry/__init__.py +1 -1
- llama_stack/apis/telemetry/telemetry.py +322 -31
- llama_stack/apis/{dataset → tools}/__init__.py +2 -1
- llama_stack/apis/tools/rag_tool.py +218 -0
- llama_stack/apis/tools/tools.py +221 -0
- llama_stack/apis/vector_io/__init__.py +7 -0
- llama_stack/apis/vector_io/vector_io.py +960 -0
- llama_stack/apis/vector_stores/__init__.py +7 -0
- llama_stack/apis/vector_stores/vector_stores.py +51 -0
- llama_stack/apis/version.py +9 -0
- llama_stack/cli/llama.py +13 -5
- llama_stack/cli/stack/_list_deps.py +182 -0
- llama_stack/cli/stack/list_apis.py +1 -1
- llama_stack/cli/stack/list_deps.py +55 -0
- llama_stack/cli/stack/list_providers.py +24 -10
- llama_stack/cli/stack/list_stacks.py +56 -0
- llama_stack/cli/stack/remove.py +115 -0
- llama_stack/cli/stack/run.py +169 -56
- llama_stack/cli/stack/stack.py +18 -4
- llama_stack/cli/stack/utils.py +151 -0
- llama_stack/cli/table.py +23 -61
- llama_stack/cli/utils.py +29 -0
- llama_stack/core/access_control/access_control.py +131 -0
- llama_stack/core/access_control/conditions.py +129 -0
- llama_stack/core/access_control/datatypes.py +107 -0
- llama_stack/core/build.py +164 -0
- llama_stack/core/client.py +205 -0
- llama_stack/core/common.sh +37 -0
- llama_stack/{distribution → core}/configure.py +74 -55
- llama_stack/core/conversations/conversations.py +309 -0
- llama_stack/core/datatypes.py +625 -0
- llama_stack/core/distribution.py +276 -0
- llama_stack/core/external.py +54 -0
- llama_stack/core/id_generation.py +42 -0
- llama_stack/core/inspect.py +86 -0
- llama_stack/core/library_client.py +539 -0
- llama_stack/core/prompts/prompts.py +234 -0
- llama_stack/core/providers.py +137 -0
- llama_stack/core/request_headers.py +115 -0
- llama_stack/core/resolver.py +506 -0
- llama_stack/core/routers/__init__.py +101 -0
- llama_stack/core/routers/datasets.py +73 -0
- llama_stack/core/routers/eval_scoring.py +155 -0
- llama_stack/core/routers/inference.py +645 -0
- llama_stack/core/routers/safety.py +85 -0
- llama_stack/core/routers/tool_runtime.py +91 -0
- llama_stack/core/routers/vector_io.py +442 -0
- llama_stack/core/routing_tables/benchmarks.py +62 -0
- llama_stack/core/routing_tables/common.py +254 -0
- llama_stack/core/routing_tables/datasets.py +91 -0
- llama_stack/core/routing_tables/models.py +163 -0
- llama_stack/core/routing_tables/scoring_functions.py +66 -0
- llama_stack/core/routing_tables/shields.py +61 -0
- llama_stack/core/routing_tables/toolgroups.py +129 -0
- llama_stack/core/routing_tables/vector_stores.py +292 -0
- llama_stack/core/server/auth.py +187 -0
- llama_stack/core/server/auth_providers.py +494 -0
- llama_stack/core/server/quota.py +110 -0
- llama_stack/core/server/routes.py +141 -0
- llama_stack/core/server/server.py +542 -0
- llama_stack/core/server/tracing.py +80 -0
- llama_stack/core/stack.py +546 -0
- llama_stack/core/start_stack.sh +117 -0
- llama_stack/core/storage/datatypes.py +283 -0
- llama_stack/{cli/model → core/store}/__init__.py +1 -1
- llama_stack/core/store/registry.py +199 -0
- llama_stack/core/testing_context.py +49 -0
- llama_stack/core/ui/app.py +55 -0
- llama_stack/core/ui/modules/api.py +32 -0
- llama_stack/core/ui/modules/utils.py +42 -0
- llama_stack/core/ui/page/distribution/datasets.py +18 -0
- llama_stack/core/ui/page/distribution/eval_tasks.py +20 -0
- llama_stack/core/ui/page/distribution/models.py +18 -0
- llama_stack/core/ui/page/distribution/providers.py +27 -0
- llama_stack/core/ui/page/distribution/resources.py +48 -0
- llama_stack/core/ui/page/distribution/scoring_functions.py +18 -0
- llama_stack/core/ui/page/distribution/shields.py +19 -0
- llama_stack/core/ui/page/evaluations/app_eval.py +143 -0
- llama_stack/core/ui/page/evaluations/native_eval.py +253 -0
- llama_stack/core/ui/page/playground/chat.py +130 -0
- llama_stack/core/ui/page/playground/tools.py +352 -0
- llama_stack/core/utils/config.py +30 -0
- llama_stack/{distribution → core}/utils/config_dirs.py +3 -6
- llama_stack/core/utils/config_resolution.py +125 -0
- llama_stack/core/utils/context.py +84 -0
- llama_stack/core/utils/exec.py +96 -0
- llama_stack/{providers/impls/meta_reference/codeshield/config.py → core/utils/image_types.py} +4 -3
- llama_stack/{distribution → core}/utils/model_utils.py +2 -2
- llama_stack/{distribution → core}/utils/prompt_for_config.py +30 -63
- llama_stack/{apis/batch_inference → distributions/dell}/__init__.py +1 -1
- llama_stack/distributions/dell/build.yaml +33 -0
- llama_stack/distributions/dell/dell.py +158 -0
- llama_stack/distributions/dell/run-with-safety.yaml +141 -0
- llama_stack/distributions/dell/run.yaml +132 -0
- llama_stack/distributions/meta-reference-gpu/__init__.py +7 -0
- llama_stack/distributions/meta-reference-gpu/build.yaml +32 -0
- llama_stack/distributions/meta-reference-gpu/meta_reference.py +163 -0
- llama_stack/distributions/meta-reference-gpu/run-with-safety.yaml +154 -0
- llama_stack/distributions/meta-reference-gpu/run.yaml +139 -0
- llama_stack/{apis/evals → distributions/nvidia}/__init__.py +1 -1
- llama_stack/distributions/nvidia/build.yaml +29 -0
- llama_stack/distributions/nvidia/nvidia.py +154 -0
- llama_stack/distributions/nvidia/run-with-safety.yaml +137 -0
- llama_stack/distributions/nvidia/run.yaml +116 -0
- llama_stack/distributions/open-benchmark/__init__.py +7 -0
- llama_stack/distributions/open-benchmark/build.yaml +36 -0
- llama_stack/distributions/open-benchmark/open_benchmark.py +303 -0
- llama_stack/distributions/open-benchmark/run.yaml +252 -0
- llama_stack/distributions/postgres-demo/__init__.py +7 -0
- llama_stack/distributions/postgres-demo/build.yaml +23 -0
- llama_stack/distributions/postgres-demo/postgres_demo.py +125 -0
- llama_stack/distributions/postgres-demo/run.yaml +115 -0
- llama_stack/{apis/memory → distributions/starter}/__init__.py +1 -1
- llama_stack/distributions/starter/build.yaml +61 -0
- llama_stack/distributions/starter/run-with-postgres-store.yaml +285 -0
- llama_stack/distributions/starter/run.yaml +276 -0
- llama_stack/distributions/starter/starter.py +345 -0
- llama_stack/distributions/starter-gpu/__init__.py +7 -0
- llama_stack/distributions/starter-gpu/build.yaml +61 -0
- llama_stack/distributions/starter-gpu/run-with-postgres-store.yaml +288 -0
- llama_stack/distributions/starter-gpu/run.yaml +279 -0
- llama_stack/distributions/starter-gpu/starter_gpu.py +20 -0
- llama_stack/distributions/template.py +456 -0
- llama_stack/distributions/watsonx/__init__.py +7 -0
- llama_stack/distributions/watsonx/build.yaml +33 -0
- llama_stack/distributions/watsonx/run.yaml +133 -0
- llama_stack/distributions/watsonx/watsonx.py +95 -0
- llama_stack/env.py +24 -0
- llama_stack/log.py +314 -0
- llama_stack/models/llama/checkpoint.py +164 -0
- llama_stack/models/llama/datatypes.py +164 -0
- llama_stack/models/llama/hadamard_utils.py +86 -0
- llama_stack/models/llama/llama3/args.py +74 -0
- llama_stack/models/llama/llama3/chat_format.py +286 -0
- llama_stack/models/llama/llama3/generation.py +376 -0
- llama_stack/models/llama/llama3/interface.py +255 -0
- llama_stack/models/llama/llama3/model.py +304 -0
- llama_stack/models/llama/llama3/multimodal/__init__.py +12 -0
- llama_stack/models/llama/llama3/multimodal/encoder_utils.py +180 -0
- llama_stack/models/llama/llama3/multimodal/image_transform.py +409 -0
- llama_stack/models/llama/llama3/multimodal/model.py +1430 -0
- llama_stack/models/llama/llama3/multimodal/utils.py +26 -0
- llama_stack/models/llama/llama3/prompt_templates/__init__.py +22 -0
- llama_stack/models/llama/llama3/prompt_templates/base.py +39 -0
- llama_stack/models/llama/llama3/prompt_templates/system_prompts.py +319 -0
- llama_stack/models/llama/llama3/prompt_templates/tool_response.py +62 -0
- llama_stack/models/llama/llama3/quantization/loader.py +316 -0
- llama_stack/models/llama/llama3/template_data.py +116 -0
- llama_stack/models/llama/llama3/tokenizer.model +128000 -0
- llama_stack/models/llama/llama3/tokenizer.py +198 -0
- llama_stack/models/llama/llama3/tool_utils.py +266 -0
- llama_stack/models/llama/llama3_1/__init__.py +12 -0
- llama_stack/models/llama/llama3_1/prompt_format.md +358 -0
- llama_stack/models/llama/llama3_1/prompts.py +258 -0
- llama_stack/models/llama/llama3_2/prompts_text.py +229 -0
- llama_stack/models/llama/llama3_2/prompts_vision.py +126 -0
- llama_stack/models/llama/llama3_2/text_prompt_format.md +286 -0
- llama_stack/models/llama/llama3_2/vision_prompt_format.md +141 -0
- llama_stack/models/llama/llama3_3/prompts.py +259 -0
- llama_stack/models/llama/llama4/args.py +107 -0
- llama_stack/models/llama/llama4/chat_format.py +317 -0
- llama_stack/models/llama/llama4/datatypes.py +56 -0
- llama_stack/models/llama/llama4/ffn.py +58 -0
- llama_stack/models/llama/llama4/generation.py +313 -0
- llama_stack/models/llama/llama4/model.py +437 -0
- llama_stack/models/llama/llama4/moe.py +214 -0
- llama_stack/models/llama/llama4/preprocess.py +435 -0
- llama_stack/models/llama/llama4/prompt_format.md +304 -0
- llama_stack/models/llama/llama4/prompt_templates/system_prompts.py +136 -0
- llama_stack/models/llama/llama4/prompts.py +279 -0
- llama_stack/models/llama/llama4/quantization/__init__.py +5 -0
- llama_stack/models/llama/llama4/quantization/loader.py +226 -0
- llama_stack/models/llama/llama4/tokenizer.model +200000 -0
- llama_stack/models/llama/llama4/tokenizer.py +263 -0
- llama_stack/models/llama/llama4/vision/__init__.py +5 -0
- llama_stack/models/llama/llama4/vision/embedding.py +210 -0
- llama_stack/models/llama/llama4/vision/encoder.py +412 -0
- llama_stack/models/llama/prompt_format.py +191 -0
- llama_stack/models/llama/quantize_impls.py +316 -0
- llama_stack/models/llama/sku_list.py +1029 -0
- llama_stack/models/llama/sku_types.py +233 -0
- llama_stack/models/llama/tokenizer_utils.py +40 -0
- llama_stack/providers/datatypes.py +136 -107
- llama_stack/providers/inline/__init__.py +5 -0
- llama_stack/providers/inline/agents/__init__.py +5 -0
- llama_stack/providers/{impls/meta_reference/agents → inline/agents/meta_reference}/__init__.py +12 -5
- llama_stack/providers/inline/agents/meta_reference/agent_instance.py +1024 -0
- llama_stack/providers/inline/agents/meta_reference/agents.py +383 -0
- llama_stack/providers/inline/agents/meta_reference/config.py +37 -0
- llama_stack/providers/inline/agents/meta_reference/persistence.py +228 -0
- llama_stack/providers/inline/agents/meta_reference/responses/__init__.py +5 -0
- llama_stack/providers/inline/agents/meta_reference/responses/openai_responses.py +423 -0
- llama_stack/providers/inline/agents/meta_reference/responses/streaming.py +1226 -0
- llama_stack/providers/inline/agents/meta_reference/responses/tool_executor.py +449 -0
- llama_stack/providers/inline/agents/meta_reference/responses/types.py +194 -0
- llama_stack/providers/inline/agents/meta_reference/responses/utils.py +365 -0
- llama_stack/providers/inline/agents/meta_reference/safety.py +52 -0
- llama_stack/providers/inline/batches/__init__.py +5 -0
- llama_stack/providers/inline/batches/reference/__init__.py +36 -0
- llama_stack/providers/inline/batches/reference/batches.py +679 -0
- llama_stack/providers/inline/batches/reference/config.py +40 -0
- llama_stack/providers/inline/datasetio/__init__.py +5 -0
- llama_stack/providers/inline/datasetio/localfs/__init__.py +20 -0
- llama_stack/providers/inline/datasetio/localfs/config.py +23 -0
- llama_stack/providers/inline/datasetio/localfs/datasetio.py +113 -0
- llama_stack/providers/inline/eval/__init__.py +5 -0
- llama_stack/providers/inline/eval/meta_reference/__init__.py +28 -0
- llama_stack/providers/inline/eval/meta_reference/config.py +23 -0
- llama_stack/providers/inline/eval/meta_reference/eval.py +259 -0
- llama_stack/providers/inline/files/localfs/__init__.py +20 -0
- llama_stack/providers/inline/files/localfs/config.py +31 -0
- llama_stack/providers/inline/files/localfs/files.py +219 -0
- llama_stack/providers/inline/inference/__init__.py +5 -0
- llama_stack/providers/{impls/meta_reference/inference → inline/inference/meta_reference}/__init__.py +4 -4
- llama_stack/providers/inline/inference/meta_reference/common.py +24 -0
- llama_stack/providers/inline/inference/meta_reference/config.py +68 -0
- llama_stack/providers/inline/inference/meta_reference/generators.py +211 -0
- llama_stack/providers/inline/inference/meta_reference/inference.py +158 -0
- llama_stack/providers/inline/inference/meta_reference/model_parallel.py +96 -0
- llama_stack/providers/{impls/meta_reference/inference → inline/inference/meta_reference}/parallel_utils.py +56 -73
- llama_stack/providers/inline/inference/sentence_transformers/__init__.py +22 -0
- llama_stack/providers/{impls/meta_reference/agents → inline/inference/sentence_transformers}/config.py +6 -4
- llama_stack/providers/inline/inference/sentence_transformers/sentence_transformers.py +83 -0
- llama_stack/providers/inline/post_training/__init__.py +5 -0
- llama_stack/providers/inline/post_training/common/__init__.py +5 -0
- llama_stack/providers/inline/post_training/common/utils.py +35 -0
- llama_stack/providers/inline/post_training/common/validator.py +36 -0
- llama_stack/providers/inline/post_training/huggingface/__init__.py +27 -0
- llama_stack/providers/inline/post_training/huggingface/config.py +83 -0
- llama_stack/providers/inline/post_training/huggingface/post_training.py +208 -0
- llama_stack/providers/inline/post_training/huggingface/recipes/__init__.py +5 -0
- llama_stack/providers/inline/post_training/huggingface/recipes/finetune_single_device.py +519 -0
- llama_stack/providers/inline/post_training/huggingface/recipes/finetune_single_device_dpo.py +485 -0
- llama_stack/providers/inline/post_training/huggingface/utils.py +269 -0
- llama_stack/providers/inline/post_training/torchtune/__init__.py +27 -0
- llama_stack/providers/inline/post_training/torchtune/common/__init__.py +5 -0
- llama_stack/providers/inline/post_training/torchtune/common/checkpointer.py +240 -0
- llama_stack/providers/inline/post_training/torchtune/common/utils.py +99 -0
- llama_stack/providers/inline/post_training/torchtune/config.py +20 -0
- llama_stack/providers/inline/post_training/torchtune/datasets/__init__.py +5 -0
- llama_stack/providers/inline/post_training/torchtune/datasets/format_adapter.py +57 -0
- llama_stack/providers/inline/post_training/torchtune/datasets/sft.py +78 -0
- llama_stack/providers/inline/post_training/torchtune/post_training.py +178 -0
- llama_stack/providers/inline/post_training/torchtune/recipes/__init__.py +5 -0
- llama_stack/providers/inline/post_training/torchtune/recipes/lora_finetuning_single_device.py +588 -0
- llama_stack/providers/inline/safety/__init__.py +5 -0
- llama_stack/providers/{impls/meta_reference/codeshield → inline/safety/code_scanner}/__init__.py +4 -2
- llama_stack/providers/inline/safety/code_scanner/code_scanner.py +128 -0
- llama_stack/providers/{impls/meta_reference/memory → inline/safety/code_scanner}/config.py +5 -3
- llama_stack/providers/inline/safety/llama_guard/__init__.py +19 -0
- llama_stack/providers/inline/safety/llama_guard/config.py +19 -0
- llama_stack/providers/inline/safety/llama_guard/llama_guard.py +489 -0
- llama_stack/providers/{adapters/memory/sample → inline/safety/prompt_guard}/__init__.py +4 -4
- llama_stack/providers/inline/safety/prompt_guard/config.py +32 -0
- llama_stack/providers/inline/safety/prompt_guard/prompt_guard.py +131 -0
- llama_stack/providers/inline/scoring/__init__.py +5 -0
- llama_stack/providers/inline/scoring/basic/__init__.py +25 -0
- llama_stack/providers/{adapters/memory/weaviate → inline/scoring/basic}/config.py +5 -7
- llama_stack/providers/inline/scoring/basic/scoring.py +126 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/__init__.py +5 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/docvqa_scoring_fn.py +240 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/equality_scoring_fn.py +41 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/fn_defs/__init__.py +5 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/fn_defs/docvqa.py +21 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/fn_defs/equality.py +21 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/fn_defs/ifeval.py +23 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/fn_defs/regex_parser_math_response.py +27 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/fn_defs/regex_parser_multiple_choice_answer.py +71 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/fn_defs/subset_of.py +21 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/ifeval_scoring_fn.py +80 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/regex_parser_math_response_scoring_fn.py +66 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/regex_parser_scoring_fn.py +58 -0
- llama_stack/providers/inline/scoring/basic/scoring_fn/subset_of_scoring_fn.py +38 -0
- llama_stack/providers/inline/scoring/basic/utils/__init__.py +5 -0
- llama_stack/providers/inline/scoring/basic/utils/ifeval_utils.py +3319 -0
- llama_stack/providers/inline/scoring/basic/utils/math_utils.py +330 -0
- llama_stack/providers/inline/scoring/braintrust/__init__.py +27 -0
- llama_stack/providers/inline/scoring/braintrust/braintrust.py +230 -0
- llama_stack/providers/inline/scoring/braintrust/config.py +21 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/__init__.py +5 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/__init__.py +5 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/answer_correctness.py +24 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/answer_relevancy.py +24 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/answer_similarity.py +24 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/context_entity_recall.py +24 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/context_precision.py +24 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/context_recall.py +24 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/context_relevancy.py +23 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/factuality.py +24 -0
- llama_stack/providers/inline/scoring/braintrust/scoring_fn/fn_defs/faithfulness.py +24 -0
- llama_stack/providers/inline/scoring/llm_as_judge/__init__.py +21 -0
- llama_stack/providers/inline/scoring/llm_as_judge/config.py +14 -0
- llama_stack/providers/inline/scoring/llm_as_judge/scoring.py +113 -0
- llama_stack/providers/inline/scoring/llm_as_judge/scoring_fn/__init__.py +5 -0
- llama_stack/providers/inline/scoring/llm_as_judge/scoring_fn/fn_defs/__init__.py +5 -0
- llama_stack/providers/inline/scoring/llm_as_judge/scoring_fn/fn_defs/llm_as_judge_405b_simpleqa.py +96 -0
- llama_stack/providers/inline/scoring/llm_as_judge/scoring_fn/fn_defs/llm_as_judge_base.py +20 -0
- llama_stack/providers/inline/scoring/llm_as_judge/scoring_fn/llm_as_judge_scoring_fn.py +81 -0
- llama_stack/providers/inline/telemetry/__init__.py +5 -0
- llama_stack/providers/inline/telemetry/meta_reference/__init__.py +21 -0
- llama_stack/providers/inline/telemetry/meta_reference/config.py +47 -0
- llama_stack/providers/inline/telemetry/meta_reference/telemetry.py +252 -0
- llama_stack/providers/inline/tool_runtime/__init__.py +5 -0
- llama_stack/providers/inline/tool_runtime/rag/__init__.py +19 -0
- llama_stack/providers/{impls/meta_reference/telemetry → inline/tool_runtime/rag}/config.py +5 -3
- llama_stack/providers/inline/tool_runtime/rag/context_retriever.py +77 -0
- llama_stack/providers/inline/tool_runtime/rag/memory.py +332 -0
- llama_stack/providers/inline/vector_io/__init__.py +5 -0
- llama_stack/providers/inline/vector_io/chroma/__init__.py +19 -0
- llama_stack/providers/inline/vector_io/chroma/config.py +30 -0
- llama_stack/providers/inline/vector_io/faiss/__init__.py +21 -0
- llama_stack/providers/inline/vector_io/faiss/config.py +26 -0
- llama_stack/providers/inline/vector_io/faiss/faiss.py +293 -0
- llama_stack/providers/inline/vector_io/milvus/__init__.py +19 -0
- llama_stack/providers/inline/vector_io/milvus/config.py +29 -0
- llama_stack/providers/inline/vector_io/qdrant/__init__.py +20 -0
- llama_stack/providers/inline/vector_io/qdrant/config.py +29 -0
- llama_stack/providers/inline/vector_io/sqlite_vec/__init__.py +20 -0
- llama_stack/providers/inline/vector_io/sqlite_vec/config.py +26 -0
- llama_stack/providers/inline/vector_io/sqlite_vec/sqlite_vec.py +483 -0
- llama_stack/providers/registry/agents.py +16 -18
- llama_stack/providers/registry/batches.py +26 -0
- llama_stack/providers/registry/datasetio.py +49 -0
- llama_stack/providers/registry/eval.py +46 -0
- llama_stack/providers/registry/files.py +31 -0
- llama_stack/providers/registry/inference.py +273 -118
- llama_stack/providers/registry/post_training.py +69 -0
- llama_stack/providers/registry/safety.py +46 -41
- llama_stack/providers/registry/scoring.py +51 -0
- llama_stack/providers/registry/tool_runtime.py +87 -0
- llama_stack/providers/registry/vector_io.py +828 -0
- llama_stack/providers/remote/__init__.py +5 -0
- llama_stack/providers/remote/agents/__init__.py +5 -0
- llama_stack/providers/remote/datasetio/__init__.py +5 -0
- llama_stack/providers/{adapters/memory/chroma → remote/datasetio/huggingface}/__init__.py +7 -4
- llama_stack/providers/remote/datasetio/huggingface/config.py +23 -0
- llama_stack/providers/remote/datasetio/huggingface/huggingface.py +99 -0
- llama_stack/providers/remote/datasetio/nvidia/__init__.py +23 -0
- llama_stack/providers/remote/datasetio/nvidia/config.py +61 -0
- llama_stack/providers/remote/datasetio/nvidia/datasetio.py +116 -0
- llama_stack/providers/remote/eval/__init__.py +5 -0
- llama_stack/providers/remote/eval/nvidia/__init__.py +31 -0
- llama_stack/providers/remote/eval/nvidia/config.py +29 -0
- llama_stack/providers/remote/eval/nvidia/eval.py +162 -0
- llama_stack/providers/remote/files/s3/__init__.py +19 -0
- llama_stack/providers/remote/files/s3/config.py +42 -0
- llama_stack/providers/remote/files/s3/files.py +313 -0
- llama_stack/providers/remote/inference/__init__.py +5 -0
- llama_stack/providers/{adapters/safety/sample → remote/inference/anthropic}/__init__.py +4 -6
- llama_stack/providers/remote/inference/anthropic/anthropic.py +36 -0
- llama_stack/providers/remote/inference/anthropic/config.py +28 -0
- llama_stack/providers/{impls/meta_reference/telemetry → remote/inference/azure}/__init__.py +4 -4
- llama_stack/providers/remote/inference/azure/azure.py +25 -0
- llama_stack/providers/remote/inference/azure/config.py +61 -0
- llama_stack/providers/{adapters → remote}/inference/bedrock/__init__.py +18 -17
- llama_stack/providers/remote/inference/bedrock/bedrock.py +142 -0
- llama_stack/providers/{adapters/inference/sample → remote/inference/bedrock}/config.py +3 -4
- llama_stack/providers/remote/inference/bedrock/models.py +29 -0
- llama_stack/providers/remote/inference/cerebras/__init__.py +19 -0
- llama_stack/providers/remote/inference/cerebras/cerebras.py +28 -0
- llama_stack/providers/remote/inference/cerebras/config.py +30 -0
- llama_stack/providers/{adapters → remote}/inference/databricks/__init__.py +4 -5
- llama_stack/providers/remote/inference/databricks/config.py +37 -0
- llama_stack/providers/remote/inference/databricks/databricks.py +44 -0
- llama_stack/providers/{adapters → remote}/inference/fireworks/__init__.py +8 -4
- llama_stack/providers/remote/inference/fireworks/config.py +27 -0
- llama_stack/providers/remote/inference/fireworks/fireworks.py +27 -0
- llama_stack/providers/{adapters/memory/pgvector → remote/inference/gemini}/__init__.py +4 -4
- llama_stack/providers/remote/inference/gemini/config.py +28 -0
- llama_stack/providers/remote/inference/gemini/gemini.py +82 -0
- llama_stack/providers/remote/inference/groq/__init__.py +15 -0
- llama_stack/providers/remote/inference/groq/config.py +34 -0
- llama_stack/providers/remote/inference/groq/groq.py +18 -0
- llama_stack/providers/remote/inference/llama_openai_compat/__init__.py +15 -0
- llama_stack/providers/remote/inference/llama_openai_compat/config.py +34 -0
- llama_stack/providers/remote/inference/llama_openai_compat/llama.py +46 -0
- llama_stack/providers/remote/inference/nvidia/__init__.py +23 -0
- llama_stack/providers/remote/inference/nvidia/config.py +64 -0
- llama_stack/providers/remote/inference/nvidia/nvidia.py +61 -0
- llama_stack/providers/{adapters/safety/sample/config.py → remote/inference/nvidia/utils.py} +3 -4
- llama_stack/providers/{impls/vllm → remote/inference/ollama}/__init__.py +4 -6
- llama_stack/providers/remote/inference/ollama/config.py +25 -0
- llama_stack/providers/remote/inference/ollama/ollama.py +102 -0
- llama_stack/providers/{adapters/telemetry/opentelemetry → remote/inference/openai}/__init__.py +4 -4
- llama_stack/providers/remote/inference/openai/config.py +39 -0
- llama_stack/providers/remote/inference/openai/openai.py +38 -0
- llama_stack/providers/remote/inference/passthrough/__init__.py +23 -0
- llama_stack/providers/remote/inference/passthrough/config.py +34 -0
- llama_stack/providers/remote/inference/passthrough/passthrough.py +122 -0
- llama_stack/providers/remote/inference/runpod/__init__.py +16 -0
- llama_stack/providers/remote/inference/runpod/config.py +32 -0
- llama_stack/providers/remote/inference/runpod/runpod.py +42 -0
- llama_stack/providers/remote/inference/sambanova/__init__.py +16 -0
- llama_stack/providers/remote/inference/sambanova/config.py +34 -0
- llama_stack/providers/remote/inference/sambanova/sambanova.py +28 -0
- llama_stack/providers/{adapters → remote}/inference/tgi/__init__.py +3 -4
- llama_stack/providers/remote/inference/tgi/config.py +76 -0
- llama_stack/providers/remote/inference/tgi/tgi.py +85 -0
- llama_stack/providers/{adapters → remote}/inference/together/__init__.py +8 -4
- llama_stack/providers/remote/inference/together/config.py +27 -0
- llama_stack/providers/remote/inference/together/together.py +102 -0
- llama_stack/providers/remote/inference/vertexai/__init__.py +15 -0
- llama_stack/providers/remote/inference/vertexai/config.py +48 -0
- llama_stack/providers/remote/inference/vertexai/vertexai.py +54 -0
- llama_stack/providers/remote/inference/vllm/__init__.py +22 -0
- llama_stack/providers/remote/inference/vllm/config.py +59 -0
- llama_stack/providers/remote/inference/vllm/vllm.py +111 -0
- llama_stack/providers/remote/inference/watsonx/__init__.py +15 -0
- llama_stack/providers/remote/inference/watsonx/config.py +45 -0
- llama_stack/providers/remote/inference/watsonx/watsonx.py +336 -0
- llama_stack/providers/remote/post_training/__init__.py +5 -0
- llama_stack/providers/remote/post_training/nvidia/__init__.py +23 -0
- llama_stack/providers/remote/post_training/nvidia/config.py +113 -0
- llama_stack/providers/remote/post_training/nvidia/models.py +27 -0
- llama_stack/providers/remote/post_training/nvidia/post_training.py +430 -0
- llama_stack/providers/remote/post_training/nvidia/utils.py +63 -0
- llama_stack/providers/remote/safety/__init__.py +5 -0
- llama_stack/providers/remote/safety/bedrock/bedrock.py +111 -0
- llama_stack/providers/remote/safety/bedrock/config.py +14 -0
- llama_stack/providers/{adapters/inference/sample → remote/safety/nvidia}/__init__.py +5 -4
- llama_stack/providers/remote/safety/nvidia/config.py +40 -0
- llama_stack/providers/remote/safety/nvidia/nvidia.py +161 -0
- llama_stack/providers/{adapters/agents/sample → remote/safety/sambanova}/__init__.py +5 -4
- llama_stack/providers/remote/safety/sambanova/config.py +37 -0
- llama_stack/providers/remote/safety/sambanova/sambanova.py +98 -0
- llama_stack/providers/remote/tool_runtime/__init__.py +5 -0
- llama_stack/providers/remote/tool_runtime/bing_search/__init__.py +21 -0
- llama_stack/providers/remote/tool_runtime/bing_search/bing_search.py +112 -0
- llama_stack/providers/remote/tool_runtime/bing_search/config.py +22 -0
- llama_stack/providers/remote/tool_runtime/brave_search/__init__.py +20 -0
- llama_stack/providers/remote/tool_runtime/brave_search/brave_search.py +148 -0
- llama_stack/providers/remote/tool_runtime/brave_search/config.py +27 -0
- llama_stack/providers/remote/tool_runtime/model_context_protocol/__init__.py +15 -0
- llama_stack/providers/remote/tool_runtime/model_context_protocol/config.py +20 -0
- llama_stack/providers/remote/tool_runtime/model_context_protocol/model_context_protocol.py +73 -0
- llama_stack/providers/remote/tool_runtime/tavily_search/__init__.py +20 -0
- llama_stack/providers/remote/tool_runtime/tavily_search/config.py +27 -0
- llama_stack/providers/remote/tool_runtime/tavily_search/tavily_search.py +84 -0
- llama_stack/providers/remote/tool_runtime/wolfram_alpha/__init__.py +22 -0
- llama_stack/providers/remote/tool_runtime/wolfram_alpha/config.py +21 -0
- llama_stack/providers/remote/tool_runtime/wolfram_alpha/wolfram_alpha.py +140 -0
- llama_stack/providers/remote/vector_io/__init__.py +5 -0
- llama_stack/providers/remote/vector_io/chroma/__init__.py +17 -0
- llama_stack/providers/remote/vector_io/chroma/chroma.py +215 -0
- llama_stack/providers/remote/vector_io/chroma/config.py +28 -0
- llama_stack/providers/remote/vector_io/milvus/__init__.py +18 -0
- llama_stack/providers/remote/vector_io/milvus/config.py +35 -0
- llama_stack/providers/remote/vector_io/milvus/milvus.py +375 -0
- llama_stack/providers/remote/vector_io/pgvector/__init__.py +17 -0
- llama_stack/providers/remote/vector_io/pgvector/config.py +47 -0
- llama_stack/providers/remote/vector_io/pgvector/pgvector.py +460 -0
- llama_stack/providers/remote/vector_io/qdrant/__init__.py +17 -0
- llama_stack/providers/remote/vector_io/qdrant/config.py +37 -0
- llama_stack/providers/remote/vector_io/qdrant/qdrant.py +265 -0
- llama_stack/providers/remote/vector_io/weaviate/__init__.py +17 -0
- llama_stack/providers/remote/vector_io/weaviate/config.py +32 -0
- llama_stack/providers/remote/vector_io/weaviate/weaviate.py +393 -0
- llama_stack/providers/utils/bedrock/__init__.py +5 -0
- llama_stack/providers/utils/bedrock/client.py +74 -0
- llama_stack/providers/utils/bedrock/config.py +64 -0
- llama_stack/providers/utils/bedrock/refreshable_boto_session.py +112 -0
- llama_stack/providers/utils/common/__init__.py +5 -0
- llama_stack/providers/utils/common/data_schema_validator.py +103 -0
- llama_stack/providers/utils/datasetio/__init__.py +5 -0
- llama_stack/providers/utils/datasetio/url_utils.py +47 -0
- llama_stack/providers/utils/files/__init__.py +5 -0
- llama_stack/providers/utils/files/form_data.py +69 -0
- llama_stack/providers/utils/inference/__init__.py +8 -7
- llama_stack/providers/utils/inference/embedding_mixin.py +101 -0
- llama_stack/providers/utils/inference/inference_store.py +264 -0
- llama_stack/providers/utils/inference/litellm_openai_mixin.py +336 -0
- llama_stack/providers/utils/inference/model_registry.py +173 -23
- llama_stack/providers/utils/inference/openai_compat.py +1261 -49
- llama_stack/providers/utils/inference/openai_mixin.py +506 -0
- llama_stack/providers/utils/inference/prompt_adapter.py +365 -67
- llama_stack/providers/utils/kvstore/api.py +6 -6
- llama_stack/providers/utils/kvstore/config.py +28 -48
- llama_stack/providers/utils/kvstore/kvstore.py +61 -15
- llama_stack/providers/utils/kvstore/mongodb/__init__.py +9 -0
- llama_stack/providers/utils/kvstore/mongodb/mongodb.py +82 -0
- llama_stack/providers/utils/kvstore/postgres/__init__.py +7 -0
- llama_stack/providers/utils/kvstore/postgres/postgres.py +114 -0
- llama_stack/providers/utils/kvstore/redis/redis.py +33 -9
- llama_stack/providers/utils/kvstore/sqlite/config.py +2 -1
- llama_stack/providers/utils/kvstore/sqlite/sqlite.py +123 -22
- llama_stack/providers/utils/memory/file_utils.py +1 -1
- llama_stack/providers/utils/memory/openai_vector_store_mixin.py +1304 -0
- llama_stack/providers/utils/memory/vector_store.py +220 -82
- llama_stack/providers/utils/pagination.py +43 -0
- llama_stack/providers/utils/responses/__init__.py +5 -0
- llama_stack/providers/utils/responses/responses_store.py +292 -0
- llama_stack/providers/utils/scheduler.py +270 -0
- llama_stack/providers/utils/scoring/__init__.py +5 -0
- llama_stack/providers/utils/scoring/aggregation_utils.py +75 -0
- llama_stack/providers/utils/scoring/base_scoring_fn.py +114 -0
- llama_stack/providers/utils/scoring/basic_scoring_utils.py +26 -0
- llama_stack/providers/utils/sqlstore/__init__.py +5 -0
- llama_stack/providers/utils/sqlstore/api.py +128 -0
- llama_stack/providers/utils/sqlstore/authorized_sqlstore.py +319 -0
- llama_stack/providers/utils/sqlstore/sqlalchemy_sqlstore.py +343 -0
- llama_stack/providers/utils/sqlstore/sqlstore.py +70 -0
- llama_stack/providers/utils/telemetry/trace_protocol.py +142 -0
- llama_stack/providers/utils/telemetry/tracing.py +192 -53
- llama_stack/providers/utils/tools/__init__.py +5 -0
- llama_stack/providers/utils/tools/mcp.py +148 -0
- llama_stack/providers/utils/tools/ttl_dict.py +70 -0
- llama_stack/providers/utils/vector_io/__init__.py +5 -0
- llama_stack/providers/utils/vector_io/vector_utils.py +156 -0
- llama_stack/schema_utils.py +118 -0
- llama_stack/strong_typing/__init__.py +19 -0
- llama_stack/strong_typing/auxiliary.py +228 -0
- llama_stack/strong_typing/classdef.py +440 -0
- llama_stack/strong_typing/core.py +46 -0
- llama_stack/strong_typing/deserializer.py +877 -0
- llama_stack/strong_typing/docstring.py +409 -0
- llama_stack/strong_typing/exception.py +23 -0
- llama_stack/strong_typing/inspection.py +1085 -0
- llama_stack/strong_typing/mapping.py +40 -0
- llama_stack/strong_typing/name.py +182 -0
- llama_stack/strong_typing/py.typed +0 -0
- llama_stack/strong_typing/schema.py +792 -0
- llama_stack/strong_typing/serialization.py +97 -0
- llama_stack/strong_typing/serializer.py +500 -0
- llama_stack/strong_typing/slots.py +27 -0
- llama_stack/strong_typing/topological.py +89 -0
- llama_stack/testing/__init__.py +5 -0
- llama_stack/testing/api_recorder.py +956 -0
- llama_stack/ui/node_modules/flatted/python/flatted.py +149 -0
- llama_stack-0.3.4.dist-info/METADATA +261 -0
- llama_stack-0.3.4.dist-info/RECORD +625 -0
- {llama_stack-0.0.42.dist-info → llama_stack-0.3.4.dist-info}/WHEEL +1 -1
- llama_stack/apis/agents/client.py +0 -292
- llama_stack/apis/agents/event_logger.py +0 -184
- llama_stack/apis/batch_inference/batch_inference.py +0 -72
- llama_stack/apis/common/deployment_types.py +0 -31
- llama_stack/apis/dataset/dataset.py +0 -63
- llama_stack/apis/evals/evals.py +0 -122
- llama_stack/apis/inference/client.py +0 -197
- llama_stack/apis/inspect/client.py +0 -82
- llama_stack/apis/memory/client.py +0 -155
- llama_stack/apis/memory/memory.py +0 -65
- llama_stack/apis/memory_banks/__init__.py +0 -7
- llama_stack/apis/memory_banks/client.py +0 -101
- llama_stack/apis/memory_banks/memory_banks.py +0 -78
- llama_stack/apis/models/client.py +0 -83
- llama_stack/apis/reward_scoring/__init__.py +0 -7
- llama_stack/apis/reward_scoring/reward_scoring.py +0 -55
- llama_stack/apis/safety/client.py +0 -105
- llama_stack/apis/shields/client.py +0 -79
- llama_stack/cli/download.py +0 -340
- llama_stack/cli/model/describe.py +0 -82
- llama_stack/cli/model/download.py +0 -24
- llama_stack/cli/model/list.py +0 -62
- llama_stack/cli/model/model.py +0 -34
- llama_stack/cli/model/prompt_format.py +0 -112
- llama_stack/cli/model/safety_models.py +0 -52
- llama_stack/cli/stack/build.py +0 -299
- llama_stack/cli/stack/configure.py +0 -178
- llama_stack/distribution/build.py +0 -123
- llama_stack/distribution/build_conda_env.sh +0 -136
- llama_stack/distribution/build_container.sh +0 -142
- llama_stack/distribution/common.sh +0 -40
- llama_stack/distribution/configure_container.sh +0 -47
- llama_stack/distribution/datatypes.py +0 -139
- llama_stack/distribution/distribution.py +0 -58
- llama_stack/distribution/inspect.py +0 -67
- llama_stack/distribution/request_headers.py +0 -57
- llama_stack/distribution/resolver.py +0 -323
- llama_stack/distribution/routers/__init__.py +0 -48
- llama_stack/distribution/routers/routers.py +0 -158
- llama_stack/distribution/routers/routing_tables.py +0 -173
- llama_stack/distribution/server/endpoints.py +0 -48
- llama_stack/distribution/server/server.py +0 -343
- llama_stack/distribution/start_conda_env.sh +0 -42
- llama_stack/distribution/start_container.sh +0 -64
- llama_stack/distribution/templates/local-bedrock-conda-example-build.yaml +0 -10
- llama_stack/distribution/templates/local-build.yaml +0 -10
- llama_stack/distribution/templates/local-databricks-build.yaml +0 -10
- llama_stack/distribution/templates/local-fireworks-build.yaml +0 -10
- llama_stack/distribution/templates/local-hf-endpoint-build.yaml +0 -10
- llama_stack/distribution/templates/local-hf-serverless-build.yaml +0 -10
- llama_stack/distribution/templates/local-ollama-build.yaml +0 -10
- llama_stack/distribution/templates/local-tgi-build.yaml +0 -10
- llama_stack/distribution/templates/local-together-build.yaml +0 -10
- llama_stack/distribution/templates/local-vllm-build.yaml +0 -10
- llama_stack/distribution/utils/exec.py +0 -105
- llama_stack/providers/adapters/agents/sample/sample.py +0 -18
- llama_stack/providers/adapters/inference/bedrock/bedrock.py +0 -451
- llama_stack/providers/adapters/inference/bedrock/config.py +0 -55
- llama_stack/providers/adapters/inference/databricks/config.py +0 -21
- llama_stack/providers/adapters/inference/databricks/databricks.py +0 -125
- llama_stack/providers/adapters/inference/fireworks/config.py +0 -20
- llama_stack/providers/adapters/inference/fireworks/fireworks.py +0 -130
- llama_stack/providers/adapters/inference/ollama/__init__.py +0 -19
- llama_stack/providers/adapters/inference/ollama/ollama.py +0 -175
- llama_stack/providers/adapters/inference/sample/sample.py +0 -23
- llama_stack/providers/adapters/inference/tgi/config.py +0 -43
- llama_stack/providers/adapters/inference/tgi/tgi.py +0 -200
- llama_stack/providers/adapters/inference/together/config.py +0 -22
- llama_stack/providers/adapters/inference/together/together.py +0 -143
- llama_stack/providers/adapters/memory/chroma/chroma.py +0 -157
- llama_stack/providers/adapters/memory/pgvector/config.py +0 -17
- llama_stack/providers/adapters/memory/pgvector/pgvector.py +0 -211
- llama_stack/providers/adapters/memory/sample/sample.py +0 -23
- llama_stack/providers/adapters/memory/weaviate/__init__.py +0 -15
- llama_stack/providers/adapters/memory/weaviate/weaviate.py +0 -190
- llama_stack/providers/adapters/safety/bedrock/bedrock.py +0 -113
- llama_stack/providers/adapters/safety/bedrock/config.py +0 -16
- llama_stack/providers/adapters/safety/sample/sample.py +0 -23
- llama_stack/providers/adapters/safety/together/__init__.py +0 -18
- llama_stack/providers/adapters/safety/together/config.py +0 -26
- llama_stack/providers/adapters/safety/together/together.py +0 -101
- llama_stack/providers/adapters/telemetry/opentelemetry/config.py +0 -12
- llama_stack/providers/adapters/telemetry/opentelemetry/opentelemetry.py +0 -201
- llama_stack/providers/adapters/telemetry/sample/__init__.py +0 -17
- llama_stack/providers/adapters/telemetry/sample/config.py +0 -12
- llama_stack/providers/adapters/telemetry/sample/sample.py +0 -18
- llama_stack/providers/impls/meta_reference/agents/agent_instance.py +0 -844
- llama_stack/providers/impls/meta_reference/agents/agents.py +0 -161
- llama_stack/providers/impls/meta_reference/agents/persistence.py +0 -84
- llama_stack/providers/impls/meta_reference/agents/rag/context_retriever.py +0 -74
- llama_stack/providers/impls/meta_reference/agents/safety.py +0 -57
- llama_stack/providers/impls/meta_reference/agents/tests/code_execution.py +0 -93
- llama_stack/providers/impls/meta_reference/agents/tests/test_chat_agent.py +0 -305
- llama_stack/providers/impls/meta_reference/agents/tools/base.py +0 -20
- llama_stack/providers/impls/meta_reference/agents/tools/builtin.py +0 -375
- llama_stack/providers/impls/meta_reference/agents/tools/ipython_tool/code_env_prefix.py +0 -133
- llama_stack/providers/impls/meta_reference/agents/tools/ipython_tool/code_execution.py +0 -256
- llama_stack/providers/impls/meta_reference/agents/tools/ipython_tool/matplotlib_custom_backend.py +0 -87
- llama_stack/providers/impls/meta_reference/agents/tools/ipython_tool/utils.py +0 -21
- llama_stack/providers/impls/meta_reference/agents/tools/safety.py +0 -43
- llama_stack/providers/impls/meta_reference/codeshield/code_scanner.py +0 -58
- llama_stack/providers/impls/meta_reference/inference/config.py +0 -45
- llama_stack/providers/impls/meta_reference/inference/generation.py +0 -376
- llama_stack/providers/impls/meta_reference/inference/inference.py +0 -280
- llama_stack/providers/impls/meta_reference/inference/model_parallel.py +0 -99
- llama_stack/providers/impls/meta_reference/inference/quantization/fp8_impls.py +0 -184
- llama_stack/providers/impls/meta_reference/inference/quantization/fp8_txest_disabled.py +0 -76
- llama_stack/providers/impls/meta_reference/inference/quantization/loader.py +0 -97
- llama_stack/providers/impls/meta_reference/inference/quantization/scripts/quantize_checkpoint.py +0 -161
- llama_stack/providers/impls/meta_reference/memory/__init__.py +0 -19
- llama_stack/providers/impls/meta_reference/memory/faiss.py +0 -113
- llama_stack/providers/impls/meta_reference/safety/__init__.py +0 -17
- llama_stack/providers/impls/meta_reference/safety/base.py +0 -57
- llama_stack/providers/impls/meta_reference/safety/config.py +0 -48
- llama_stack/providers/impls/meta_reference/safety/llama_guard.py +0 -268
- llama_stack/providers/impls/meta_reference/safety/prompt_guard.py +0 -145
- llama_stack/providers/impls/meta_reference/safety/safety.py +0 -112
- llama_stack/providers/impls/meta_reference/telemetry/console.py +0 -89
- llama_stack/providers/impls/vllm/config.py +0 -35
- llama_stack/providers/impls/vllm/vllm.py +0 -241
- llama_stack/providers/registry/memory.py +0 -78
- llama_stack/providers/registry/telemetry.py +0 -44
- llama_stack/providers/tests/agents/test_agents.py +0 -210
- llama_stack/providers/tests/inference/test_inference.py +0 -257
- llama_stack/providers/tests/inference/test_prompt_adapter.py +0 -126
- llama_stack/providers/tests/memory/test_memory.py +0 -136
- llama_stack/providers/tests/resolver.py +0 -100
- llama_stack/providers/tests/safety/test_safety.py +0 -77
- llama_stack-0.0.42.dist-info/METADATA +0 -137
- llama_stack-0.0.42.dist-info/RECORD +0 -256
- /llama_stack/{distribution → core}/__init__.py +0 -0
- /llama_stack/{distribution/server → core/access_control}/__init__.py +0 -0
- /llama_stack/{distribution/utils → core/conversations}/__init__.py +0 -0
- /llama_stack/{providers/adapters → core/prompts}/__init__.py +0 -0
- /llama_stack/{providers/adapters/agents → core/routing_tables}/__init__.py +0 -0
- /llama_stack/{providers/adapters/inference → core/server}/__init__.py +0 -0
- /llama_stack/{providers/adapters/memory → core/storage}/__init__.py +0 -0
- /llama_stack/{providers/adapters/safety → core/ui}/__init__.py +0 -0
- /llama_stack/{providers/adapters/telemetry → core/ui/modules}/__init__.py +0 -0
- /llama_stack/{providers/impls → core/ui/page}/__init__.py +0 -0
- /llama_stack/{providers/impls/meta_reference → core/ui/page/distribution}/__init__.py +0 -0
- /llama_stack/{providers/impls/meta_reference/agents/rag → core/ui/page/evaluations}/__init__.py +0 -0
- /llama_stack/{providers/impls/meta_reference/agents/tests → core/ui/page/playground}/__init__.py +0 -0
- /llama_stack/{providers/impls/meta_reference/agents/tools → core/utils}/__init__.py +0 -0
- /llama_stack/{distribution → core}/utils/dynamic.py +0 -0
- /llama_stack/{distribution → core}/utils/serialize.py +0 -0
- /llama_stack/{providers/impls/meta_reference/agents/tools/ipython_tool → distributions}/__init__.py +0 -0
- /llama_stack/{providers/impls/meta_reference/inference/quantization → models}/__init__.py +0 -0
- /llama_stack/{providers/impls/meta_reference/inference/quantization/scripts → models/llama}/__init__.py +0 -0
- /llama_stack/{providers/tests → models/llama/llama3}/__init__.py +0 -0
- /llama_stack/{providers/tests/agents → models/llama/llama3/quantization}/__init__.py +0 -0
- /llama_stack/{providers/tests/inference → models/llama/llama3_2}/__init__.py +0 -0
- /llama_stack/{providers/tests/memory → models/llama/llama3_3}/__init__.py +0 -0
- /llama_stack/{providers/tests/safety → models/llama/llama4}/__init__.py +0 -0
- /llama_stack/{scripts → models/llama/llama4/prompt_templates}/__init__.py +0 -0
- /llama_stack/providers/{adapters → remote}/safety/bedrock/__init__.py +0 -0
- {llama_stack-0.0.42.dist-info → llama_stack-0.3.4.dist-info}/entry_points.txt +0 -0
- {llama_stack-0.0.42.dist-info → llama_stack-0.3.4.dist-info/licenses}/LICENSE +0 -0
- {llama_stack-0.0.42.dist-info → llama_stack-0.3.4.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,494 @@
|
|
|
1
|
+
# Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
2
|
+
# All rights reserved.
|
|
3
|
+
#
|
|
4
|
+
# This source code is licensed under the terms described in the LICENSE file in
|
|
5
|
+
# the root directory of this source tree.
|
|
6
|
+
|
|
7
|
+
import ssl
|
|
8
|
+
from abc import ABC, abstractmethod
|
|
9
|
+
from urllib.parse import parse_qs, urljoin, urlparse
|
|
10
|
+
|
|
11
|
+
import httpx
|
|
12
|
+
import jwt
|
|
13
|
+
from pydantic import BaseModel, Field
|
|
14
|
+
|
|
15
|
+
from llama_stack.apis.common.errors import TokenValidationError
|
|
16
|
+
from llama_stack.core.datatypes import (
|
|
17
|
+
AuthenticationConfig,
|
|
18
|
+
CustomAuthConfig,
|
|
19
|
+
GitHubTokenAuthConfig,
|
|
20
|
+
KubernetesAuthProviderConfig,
|
|
21
|
+
OAuth2TokenAuthConfig,
|
|
22
|
+
User,
|
|
23
|
+
)
|
|
24
|
+
from llama_stack.log import get_logger
|
|
25
|
+
|
|
26
|
+
logger = get_logger(name=__name__, category="core::auth")
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
class AuthResponse(BaseModel):
|
|
30
|
+
"""The format of the authentication response from the auth endpoint."""
|
|
31
|
+
|
|
32
|
+
principal: str
|
|
33
|
+
# further attributes that may be used for access control decisions
|
|
34
|
+
attributes: dict[str, list[str]] | None = None
|
|
35
|
+
message: str | None = Field(
|
|
36
|
+
default=None, description="Optional message providing additional context about the authentication result."
|
|
37
|
+
)
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
class AuthRequestContext(BaseModel):
|
|
41
|
+
path: str = Field(description="The path of the request being authenticated")
|
|
42
|
+
|
|
43
|
+
headers: dict[str, str] = Field(description="HTTP headers from the original request (excluding Authorization)")
|
|
44
|
+
|
|
45
|
+
params: dict[str, list[str]] = Field(default_factory=dict, description="Query parameters from the original request")
|
|
46
|
+
|
|
47
|
+
|
|
48
|
+
class AuthRequest(BaseModel):
|
|
49
|
+
api_key: str = Field(description="The API key extracted from the Authorization header")
|
|
50
|
+
|
|
51
|
+
request: AuthRequestContext = Field(description="Context information about the request being authenticated")
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
class AuthProvider(ABC):
|
|
55
|
+
"""Abstract base class for authentication providers."""
|
|
56
|
+
|
|
57
|
+
@abstractmethod
|
|
58
|
+
async def validate_token(self, token: str, scope: dict | None = None) -> User:
|
|
59
|
+
"""Validate a token and return access attributes."""
|
|
60
|
+
pass
|
|
61
|
+
|
|
62
|
+
@abstractmethod
|
|
63
|
+
async def close(self):
|
|
64
|
+
"""Clean up any resources."""
|
|
65
|
+
pass
|
|
66
|
+
|
|
67
|
+
def get_auth_error_message(self, scope: dict | None = None) -> str:
|
|
68
|
+
"""Return provider-specific authentication error message."""
|
|
69
|
+
return "Authentication required"
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
def get_attributes_from_claims(claims: dict[str, str], mapping: dict[str, str]) -> dict[str, list[str]]:
|
|
73
|
+
attributes: dict[str, list[str]] = {}
|
|
74
|
+
for claim_key, attribute_key in mapping.items():
|
|
75
|
+
# First try dot notation for nested traversal (e.g., "resource_access.llamastack.roles")
|
|
76
|
+
# Then fall back to literal key with dots (e.g., "my.dotted.key")
|
|
77
|
+
claim: object = claims
|
|
78
|
+
keys = claim_key.split(".")
|
|
79
|
+
for key in keys:
|
|
80
|
+
if isinstance(claim, dict) and key in claim:
|
|
81
|
+
claim = claim[key]
|
|
82
|
+
else:
|
|
83
|
+
claim = None
|
|
84
|
+
break
|
|
85
|
+
|
|
86
|
+
if claim is None and claim_key in claims:
|
|
87
|
+
# Fall back to checking if claim_key exists as a literal key
|
|
88
|
+
claim = claims[claim_key]
|
|
89
|
+
|
|
90
|
+
if claim is None:
|
|
91
|
+
continue
|
|
92
|
+
|
|
93
|
+
if isinstance(claim, list):
|
|
94
|
+
values = claim
|
|
95
|
+
elif isinstance(claim, str):
|
|
96
|
+
values = claim.split()
|
|
97
|
+
else:
|
|
98
|
+
continue
|
|
99
|
+
|
|
100
|
+
if attribute_key in attributes:
|
|
101
|
+
attributes[attribute_key].extend(values)
|
|
102
|
+
else:
|
|
103
|
+
attributes[attribute_key] = values
|
|
104
|
+
return attributes
|
|
105
|
+
|
|
106
|
+
|
|
107
|
+
class OAuth2TokenAuthProvider(AuthProvider):
|
|
108
|
+
"""
|
|
109
|
+
JWT token authentication provider that validates a JWT token and extracts access attributes.
|
|
110
|
+
|
|
111
|
+
This should be the standard authentication provider for most use cases.
|
|
112
|
+
"""
|
|
113
|
+
|
|
114
|
+
def __init__(self, config: OAuth2TokenAuthConfig):
|
|
115
|
+
self.config = config
|
|
116
|
+
self._jwks_client: jwt.PyJWKClient | None = None
|
|
117
|
+
|
|
118
|
+
async def validate_token(self, token: str, scope: dict | None = None) -> User:
|
|
119
|
+
if self.config.jwks:
|
|
120
|
+
return await self.validate_jwt_token(token, scope)
|
|
121
|
+
if self.config.introspection:
|
|
122
|
+
return await self.introspect_token(token, scope)
|
|
123
|
+
raise ValueError("One of jwks or introspection must be configured")
|
|
124
|
+
|
|
125
|
+
def _get_jwks_client(self) -> jwt.PyJWKClient:
|
|
126
|
+
if self._jwks_client is None:
|
|
127
|
+
ssl_context = None
|
|
128
|
+
if not self.config.verify_tls:
|
|
129
|
+
# Disable SSL verification if verify_tls is False
|
|
130
|
+
ssl_context = ssl.create_default_context()
|
|
131
|
+
ssl_context.check_hostname = False
|
|
132
|
+
ssl_context.verify_mode = ssl.CERT_NONE
|
|
133
|
+
elif self.config.tls_cafile:
|
|
134
|
+
# Use custom CA file if provided
|
|
135
|
+
ssl_context = ssl.create_default_context(
|
|
136
|
+
cafile=self.config.tls_cafile.as_posix(),
|
|
137
|
+
)
|
|
138
|
+
# If verify_tls is True and no tls_cafile, ssl_context remains None (use system defaults)
|
|
139
|
+
|
|
140
|
+
# Prepare headers for JWKS request - this is needed for Kubernetes to authenticate
|
|
141
|
+
# to the JWK endpoint, we must use the token in the config to authenticate
|
|
142
|
+
headers = {}
|
|
143
|
+
if self.config.jwks and self.config.jwks.token:
|
|
144
|
+
headers["Authorization"] = f"Bearer {self.config.jwks.token}"
|
|
145
|
+
|
|
146
|
+
self._jwks_client = jwt.PyJWKClient(
|
|
147
|
+
self.config.jwks.uri if self.config.jwks else None,
|
|
148
|
+
cache_keys=True,
|
|
149
|
+
max_cached_keys=10,
|
|
150
|
+
lifespan=self.config.jwks.key_recheck_period if self.config.jwks else None,
|
|
151
|
+
headers=headers,
|
|
152
|
+
ssl_context=ssl_context,
|
|
153
|
+
)
|
|
154
|
+
return self._jwks_client
|
|
155
|
+
|
|
156
|
+
async def validate_jwt_token(self, token: str, scope: dict | None = None) -> User:
|
|
157
|
+
"""Validate a token using the JWT token."""
|
|
158
|
+
try:
|
|
159
|
+
jwks_client: jwt.PyJWKClient = self._get_jwks_client()
|
|
160
|
+
signing_key = jwks_client.get_signing_key_from_jwt(token)
|
|
161
|
+
algorithm = jwt.get_unverified_header(token)["alg"]
|
|
162
|
+
claims = jwt.decode(
|
|
163
|
+
token,
|
|
164
|
+
signing_key.key,
|
|
165
|
+
algorithms=[algorithm],
|
|
166
|
+
audience=self.config.audience,
|
|
167
|
+
issuer=self.config.issuer,
|
|
168
|
+
options={"verify_exp": True, "verify_aud": True, "verify_iss": True},
|
|
169
|
+
)
|
|
170
|
+
|
|
171
|
+
# Decode and verify the JWT
|
|
172
|
+
claims = jwt.decode(
|
|
173
|
+
token,
|
|
174
|
+
signing_key.key,
|
|
175
|
+
algorithms=[algorithm],
|
|
176
|
+
audience=self.config.audience,
|
|
177
|
+
issuer=self.config.issuer,
|
|
178
|
+
options={"verify_exp": True, "verify_aud": True, "verify_iss": True},
|
|
179
|
+
)
|
|
180
|
+
except Exception as exc:
|
|
181
|
+
raise ValueError("Invalid JWT token") from exc
|
|
182
|
+
|
|
183
|
+
# There are other standard claims, the most relevant of which is `scope`.
|
|
184
|
+
# We should incorporate these into the access attributes.
|
|
185
|
+
principal = claims["sub"]
|
|
186
|
+
access_attributes = get_attributes_from_claims(claims, self.config.claims_mapping)
|
|
187
|
+
return User(
|
|
188
|
+
principal=principal,
|
|
189
|
+
attributes=access_attributes,
|
|
190
|
+
)
|
|
191
|
+
|
|
192
|
+
async def introspect_token(self, token: str, scope: dict | None = None) -> User:
|
|
193
|
+
"""Validate a token using token introspection as defined by RFC 7662."""
|
|
194
|
+
form = {
|
|
195
|
+
"token": token,
|
|
196
|
+
}
|
|
197
|
+
if self.config.introspection is None:
|
|
198
|
+
raise ValueError("Introspection is not configured")
|
|
199
|
+
|
|
200
|
+
if self.config.introspection.send_secret_in_body:
|
|
201
|
+
form["client_id"] = self.config.introspection.client_id
|
|
202
|
+
form["client_secret"] = self.config.introspection.client_secret
|
|
203
|
+
auth = None
|
|
204
|
+
else:
|
|
205
|
+
auth = (self.config.introspection.client_id, self.config.introspection.client_secret)
|
|
206
|
+
ssl_ctxt = None
|
|
207
|
+
if self.config.tls_cafile:
|
|
208
|
+
ssl_ctxt = ssl.create_default_context(cafile=self.config.tls_cafile.as_posix())
|
|
209
|
+
try:
|
|
210
|
+
async with httpx.AsyncClient(verify=ssl_ctxt) as client:
|
|
211
|
+
response = await client.post(
|
|
212
|
+
self.config.introspection.url,
|
|
213
|
+
data=form,
|
|
214
|
+
auth=auth,
|
|
215
|
+
timeout=10.0, # Add a reasonable timeout
|
|
216
|
+
)
|
|
217
|
+
if response.status_code != httpx.codes.OK:
|
|
218
|
+
logger.warning(f"Token introspection failed with status code: {response.status_code}")
|
|
219
|
+
raise ValueError(f"Token introspection failed: {response.status_code}")
|
|
220
|
+
|
|
221
|
+
fields = response.json()
|
|
222
|
+
if not fields["active"]:
|
|
223
|
+
raise ValueError("Token not active")
|
|
224
|
+
principal = fields["sub"] or fields["username"]
|
|
225
|
+
access_attributes = get_attributes_from_claims(fields, self.config.claims_mapping)
|
|
226
|
+
return User(
|
|
227
|
+
principal=principal,
|
|
228
|
+
attributes=access_attributes,
|
|
229
|
+
)
|
|
230
|
+
except httpx.TimeoutException:
|
|
231
|
+
logger.exception("Token introspection request timed out")
|
|
232
|
+
raise
|
|
233
|
+
except ValueError:
|
|
234
|
+
# Re-raise ValueError exceptions to preserve their message
|
|
235
|
+
raise
|
|
236
|
+
except Exception as e:
|
|
237
|
+
logger.exception("Error during token introspection")
|
|
238
|
+
raise ValueError("Token introspection error") from e
|
|
239
|
+
|
|
240
|
+
async def close(self):
|
|
241
|
+
pass
|
|
242
|
+
|
|
243
|
+
def get_auth_error_message(self, scope: dict | None = None) -> str:
|
|
244
|
+
"""Return OAuth2-specific authentication error message."""
|
|
245
|
+
if self.config.issuer:
|
|
246
|
+
return f"Authentication required. Please provide a valid OAuth2 Bearer token from {self.config.issuer}"
|
|
247
|
+
elif self.config.introspection:
|
|
248
|
+
# Extract domain from introspection URL for a cleaner message
|
|
249
|
+
domain = urlparse(self.config.introspection.url).netloc
|
|
250
|
+
return f"Authentication required. Please provide a valid OAuth2 Bearer token validated by {domain}"
|
|
251
|
+
else:
|
|
252
|
+
return "Authentication required. Please provide a valid OAuth2 Bearer token in the Authorization header"
|
|
253
|
+
|
|
254
|
+
|
|
255
|
+
class CustomAuthProvider(AuthProvider):
|
|
256
|
+
"""Custom authentication provider that uses an external endpoint."""
|
|
257
|
+
|
|
258
|
+
def __init__(self, config: CustomAuthConfig):
|
|
259
|
+
self.config = config
|
|
260
|
+
self._client = None
|
|
261
|
+
|
|
262
|
+
async def validate_token(self, token: str, scope: dict | None = None) -> User:
|
|
263
|
+
"""Validate a token using the custom authentication endpoint."""
|
|
264
|
+
if scope is None:
|
|
265
|
+
scope = {}
|
|
266
|
+
|
|
267
|
+
headers = dict(scope.get("headers", []))
|
|
268
|
+
path = scope.get("path", "")
|
|
269
|
+
request_headers = {k.decode(): v.decode() for k, v in headers.items()}
|
|
270
|
+
|
|
271
|
+
# Remove sensitive headers
|
|
272
|
+
if "authorization" in request_headers:
|
|
273
|
+
del request_headers["authorization"]
|
|
274
|
+
|
|
275
|
+
query_string = scope.get("query_string", b"").decode()
|
|
276
|
+
params = parse_qs(query_string)
|
|
277
|
+
|
|
278
|
+
# Build the auth request model
|
|
279
|
+
auth_request = AuthRequest(
|
|
280
|
+
api_key=token,
|
|
281
|
+
request=AuthRequestContext(
|
|
282
|
+
path=path,
|
|
283
|
+
headers=request_headers,
|
|
284
|
+
params=params,
|
|
285
|
+
),
|
|
286
|
+
)
|
|
287
|
+
|
|
288
|
+
# Validate with authentication endpoint
|
|
289
|
+
try:
|
|
290
|
+
async with httpx.AsyncClient() as client:
|
|
291
|
+
response = await client.post(
|
|
292
|
+
self.config.endpoint,
|
|
293
|
+
json=auth_request.model_dump(),
|
|
294
|
+
timeout=10.0, # Add a reasonable timeout
|
|
295
|
+
)
|
|
296
|
+
if response.status_code != httpx.codes.OK:
|
|
297
|
+
logger.warning(f"Authentication failed with status code: {response.status_code}")
|
|
298
|
+
raise ValueError(f"Authentication failed: {response.status_code}")
|
|
299
|
+
|
|
300
|
+
# Parse and validate the auth response
|
|
301
|
+
try:
|
|
302
|
+
response_data = response.json()
|
|
303
|
+
auth_response = AuthResponse(**response_data)
|
|
304
|
+
return User(principal=auth_response.principal, attributes=auth_response.attributes)
|
|
305
|
+
except Exception as e:
|
|
306
|
+
logger.exception("Error parsing authentication response")
|
|
307
|
+
raise ValueError("Invalid authentication response format") from e
|
|
308
|
+
|
|
309
|
+
except httpx.TimeoutException:
|
|
310
|
+
logger.exception("Authentication request timed out")
|
|
311
|
+
raise
|
|
312
|
+
except ValueError:
|
|
313
|
+
# Re-raise ValueError exceptions to preserve their message
|
|
314
|
+
raise
|
|
315
|
+
except Exception as e:
|
|
316
|
+
logger.exception("Error during authentication")
|
|
317
|
+
raise ValueError("Authentication service error") from e
|
|
318
|
+
|
|
319
|
+
async def close(self):
|
|
320
|
+
"""Close the HTTP client."""
|
|
321
|
+
if self._client:
|
|
322
|
+
await self._client.aclose()
|
|
323
|
+
self._client = None
|
|
324
|
+
|
|
325
|
+
def get_auth_error_message(self, scope: dict | None = None) -> str:
|
|
326
|
+
"""Return custom auth provider-specific authentication error message."""
|
|
327
|
+
domain = urlparse(self.config.endpoint).netloc
|
|
328
|
+
if domain:
|
|
329
|
+
return f"Authentication required. Please provide your API key as a Bearer token (validated by {domain})"
|
|
330
|
+
else:
|
|
331
|
+
return "Authentication required. Please provide your API key as a Bearer token in the Authorization header"
|
|
332
|
+
|
|
333
|
+
|
|
334
|
+
class GitHubTokenAuthProvider(AuthProvider):
|
|
335
|
+
"""
|
|
336
|
+
GitHub token authentication provider that validates GitHub access tokens directly.
|
|
337
|
+
|
|
338
|
+
This provider accepts GitHub personal access tokens or OAuth tokens and verifies
|
|
339
|
+
them against the GitHub API to get user information.
|
|
340
|
+
"""
|
|
341
|
+
|
|
342
|
+
def __init__(self, config: GitHubTokenAuthConfig):
|
|
343
|
+
self.config = config
|
|
344
|
+
|
|
345
|
+
async def validate_token(self, token: str, scope: dict | None = None) -> User:
|
|
346
|
+
"""Validate a GitHub token by calling the GitHub API.
|
|
347
|
+
|
|
348
|
+
This validates tokens issued by GitHub (personal access tokens or OAuth tokens).
|
|
349
|
+
"""
|
|
350
|
+
try:
|
|
351
|
+
user_info = await _get_github_user_info(token, self.config.github_api_base_url)
|
|
352
|
+
except httpx.HTTPStatusError as e:
|
|
353
|
+
logger.warning(f"GitHub token validation failed: {e}")
|
|
354
|
+
raise ValueError("GitHub token validation failed. Please check your token and try again.") from e
|
|
355
|
+
|
|
356
|
+
principal = user_info["user"]["login"]
|
|
357
|
+
|
|
358
|
+
github_data = {
|
|
359
|
+
"login": user_info["user"]["login"],
|
|
360
|
+
"id": str(user_info["user"]["id"]),
|
|
361
|
+
"organizations": user_info.get("organizations", []),
|
|
362
|
+
}
|
|
363
|
+
|
|
364
|
+
access_attributes = get_attributes_from_claims(github_data, self.config.claims_mapping)
|
|
365
|
+
|
|
366
|
+
return User(
|
|
367
|
+
principal=principal,
|
|
368
|
+
attributes=access_attributes,
|
|
369
|
+
)
|
|
370
|
+
|
|
371
|
+
async def close(self):
|
|
372
|
+
"""Clean up any resources."""
|
|
373
|
+
pass
|
|
374
|
+
|
|
375
|
+
def get_auth_error_message(self, scope: dict | None = None) -> str:
|
|
376
|
+
"""Return GitHub-specific authentication error message."""
|
|
377
|
+
return "Authentication required. Please provide a valid GitHub access token (https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/managing-your-personal-access-tokens) in the Authorization header (Bearer <token>)"
|
|
378
|
+
|
|
379
|
+
|
|
380
|
+
async def _get_github_user_info(access_token: str, github_api_base_url: str) -> dict:
|
|
381
|
+
"""Fetch user info and organizations from GitHub API."""
|
|
382
|
+
headers = {
|
|
383
|
+
"Authorization": f"Bearer {access_token}",
|
|
384
|
+
"Accept": "application/vnd.github.v3+json",
|
|
385
|
+
"User-Agent": "llama-stack",
|
|
386
|
+
}
|
|
387
|
+
|
|
388
|
+
async with httpx.AsyncClient() as client:
|
|
389
|
+
user_response = await client.get(f"{github_api_base_url}/user", headers=headers, timeout=10.0)
|
|
390
|
+
user_response.raise_for_status()
|
|
391
|
+
user_data = user_response.json()
|
|
392
|
+
|
|
393
|
+
return {
|
|
394
|
+
"user": user_data,
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
|
|
398
|
+
class KubernetesAuthProvider(AuthProvider):
|
|
399
|
+
"""
|
|
400
|
+
Kubernetes authentication provider that validates tokens using the Kubernetes SelfSubjectReview API.
|
|
401
|
+
This provider integrates with Kubernetes API server by using the
|
|
402
|
+
/apis/authentication.k8s.io/v1/selfsubjectreviews endpoint to validate tokens and extract user information.
|
|
403
|
+
"""
|
|
404
|
+
|
|
405
|
+
def __init__(self, config: KubernetesAuthProviderConfig):
|
|
406
|
+
self.config = config
|
|
407
|
+
|
|
408
|
+
def _httpx_verify_value(self) -> bool | str:
|
|
409
|
+
"""
|
|
410
|
+
Build the value for httpx's `verify` parameter.
|
|
411
|
+
- False disables verification.
|
|
412
|
+
- Path string points to a CA bundle.
|
|
413
|
+
- True uses system defaults.
|
|
414
|
+
"""
|
|
415
|
+
if not self.config.verify_tls:
|
|
416
|
+
return False
|
|
417
|
+
if self.config.tls_cafile:
|
|
418
|
+
return self.config.tls_cafile.as_posix()
|
|
419
|
+
return True
|
|
420
|
+
|
|
421
|
+
async def validate_token(self, token: str, scope: dict | None = None) -> User:
|
|
422
|
+
"""Validate a token using Kubernetes SelfSubjectReview API endpoint."""
|
|
423
|
+
# Build the Kubernetes SelfSubjectReview API endpoint URL
|
|
424
|
+
review_api_url = urljoin(self.config.api_server_url, "/apis/authentication.k8s.io/v1/selfsubjectreviews")
|
|
425
|
+
|
|
426
|
+
# Create SelfSubjectReview request body
|
|
427
|
+
review_request = {"apiVersion": "authentication.k8s.io/v1", "kind": "SelfSubjectReview"}
|
|
428
|
+
verify = self._httpx_verify_value()
|
|
429
|
+
|
|
430
|
+
try:
|
|
431
|
+
async with httpx.AsyncClient(verify=verify, timeout=10.0) as client:
|
|
432
|
+
response = await client.post(
|
|
433
|
+
review_api_url,
|
|
434
|
+
json=review_request,
|
|
435
|
+
headers={
|
|
436
|
+
"Authorization": f"Bearer {token}",
|
|
437
|
+
"Content-Type": "application/json",
|
|
438
|
+
},
|
|
439
|
+
)
|
|
440
|
+
|
|
441
|
+
if response.status_code == httpx.codes.UNAUTHORIZED:
|
|
442
|
+
raise TokenValidationError("Invalid token")
|
|
443
|
+
if response.status_code != httpx.codes.CREATED:
|
|
444
|
+
logger.warning(f"Kubernetes SelfSubjectReview API failed with status code: {response.status_code}")
|
|
445
|
+
raise TokenValidationError(f"Token validation failed: {response.status_code}")
|
|
446
|
+
|
|
447
|
+
review_response = response.json()
|
|
448
|
+
# Extract user information from SelfSubjectReview response
|
|
449
|
+
status = review_response.get("status", {})
|
|
450
|
+
if not status:
|
|
451
|
+
raise ValueError("No status found in SelfSubjectReview response")
|
|
452
|
+
|
|
453
|
+
user_info = status.get("userInfo", {})
|
|
454
|
+
if not user_info:
|
|
455
|
+
raise ValueError("No userInfo found in SelfSubjectReview response")
|
|
456
|
+
|
|
457
|
+
username = user_info.get("username")
|
|
458
|
+
if not username:
|
|
459
|
+
raise ValueError("No username found in SelfSubjectReview response")
|
|
460
|
+
|
|
461
|
+
# Build user attributes from Kubernetes user info
|
|
462
|
+
user_attributes = get_attributes_from_claims(user_info, self.config.claims_mapping)
|
|
463
|
+
|
|
464
|
+
return User(
|
|
465
|
+
principal=username,
|
|
466
|
+
attributes=user_attributes,
|
|
467
|
+
)
|
|
468
|
+
|
|
469
|
+
except httpx.TimeoutException:
|
|
470
|
+
logger.warning("Kubernetes SelfSubjectReview API request timed out")
|
|
471
|
+
raise ValueError("Token validation timeout") from None
|
|
472
|
+
except Exception as e:
|
|
473
|
+
logger.warning(f"Error during token validation: {str(e)}")
|
|
474
|
+
raise ValueError(f"Token validation error: {str(e)}") from e
|
|
475
|
+
|
|
476
|
+
async def close(self):
|
|
477
|
+
"""Close any resources."""
|
|
478
|
+
pass
|
|
479
|
+
|
|
480
|
+
|
|
481
|
+
def create_auth_provider(config: AuthenticationConfig) -> AuthProvider:
|
|
482
|
+
"""Factory function to create the appropriate auth provider."""
|
|
483
|
+
provider_config = config.provider_config
|
|
484
|
+
|
|
485
|
+
if isinstance(provider_config, CustomAuthConfig):
|
|
486
|
+
return CustomAuthProvider(provider_config)
|
|
487
|
+
elif isinstance(provider_config, OAuth2TokenAuthConfig):
|
|
488
|
+
return OAuth2TokenAuthProvider(provider_config)
|
|
489
|
+
elif isinstance(provider_config, GitHubTokenAuthConfig):
|
|
490
|
+
return GitHubTokenAuthProvider(provider_config)
|
|
491
|
+
elif isinstance(provider_config, KubernetesAuthProviderConfig):
|
|
492
|
+
return KubernetesAuthProvider(provider_config)
|
|
493
|
+
else:
|
|
494
|
+
raise ValueError(f"Unknown authentication provider config type: {type(provider_config)}")
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
# Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
2
|
+
# All rights reserved.
|
|
3
|
+
#
|
|
4
|
+
# This source code is licensed under the terms described in the LICENSE file in
|
|
5
|
+
# the root directory of this source tree.
|
|
6
|
+
|
|
7
|
+
import json
|
|
8
|
+
import time
|
|
9
|
+
from datetime import UTC, datetime, timedelta
|
|
10
|
+
|
|
11
|
+
from starlette.types import ASGIApp, Receive, Scope, Send
|
|
12
|
+
|
|
13
|
+
from llama_stack.core.storage.datatypes import KVStoreReference, StorageBackendType
|
|
14
|
+
from llama_stack.log import get_logger
|
|
15
|
+
from llama_stack.providers.utils.kvstore.api import KVStore
|
|
16
|
+
from llama_stack.providers.utils.kvstore.kvstore import _KVSTORE_BACKENDS, kvstore_impl
|
|
17
|
+
|
|
18
|
+
logger = get_logger(name=__name__, category="core::server")
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
class QuotaMiddleware:
|
|
22
|
+
"""
|
|
23
|
+
ASGI middleware that enforces separate quotas for authenticated and anonymous clients
|
|
24
|
+
within a configurable time window.
|
|
25
|
+
|
|
26
|
+
- For authenticated requests, it reads the client ID from the
|
|
27
|
+
`Authorization: Bearer <client_id>` header.
|
|
28
|
+
- For anonymous requests, it falls back to the IP address of the client.
|
|
29
|
+
Requests are counted in a KV store (e.g., SQLite), and HTTP 429 is returned
|
|
30
|
+
once a client exceeds its quota.
|
|
31
|
+
"""
|
|
32
|
+
|
|
33
|
+
def __init__(
|
|
34
|
+
self,
|
|
35
|
+
app: ASGIApp,
|
|
36
|
+
kv_config: KVStoreReference,
|
|
37
|
+
anonymous_max_requests: int,
|
|
38
|
+
authenticated_max_requests: int,
|
|
39
|
+
window_seconds: int = 86400,
|
|
40
|
+
):
|
|
41
|
+
self.app = app
|
|
42
|
+
self.kv_config = kv_config
|
|
43
|
+
self.kv: KVStore | None = None
|
|
44
|
+
self.anonymous_max_requests = anonymous_max_requests
|
|
45
|
+
self.authenticated_max_requests = authenticated_max_requests
|
|
46
|
+
self.window_seconds = window_seconds
|
|
47
|
+
|
|
48
|
+
async def _get_kv(self) -> KVStore:
|
|
49
|
+
if self.kv is None:
|
|
50
|
+
self.kv = await kvstore_impl(self.kv_config)
|
|
51
|
+
backend_config = _KVSTORE_BACKENDS.get(self.kv_config.backend)
|
|
52
|
+
if backend_config and backend_config.type == StorageBackendType.KV_SQLITE:
|
|
53
|
+
logger.warning(
|
|
54
|
+
"QuotaMiddleware: Using SQLite backend. Expiry/TTL is not enforced; cleanup is manual. "
|
|
55
|
+
f"window_seconds={self.window_seconds}"
|
|
56
|
+
)
|
|
57
|
+
return self.kv
|
|
58
|
+
|
|
59
|
+
async def __call__(self, scope: Scope, receive: Receive, send: Send):
|
|
60
|
+
if scope["type"] == "http":
|
|
61
|
+
# pick key & limit based on auth
|
|
62
|
+
auth_id = scope.get("authenticated_client_id")
|
|
63
|
+
if auth_id:
|
|
64
|
+
key_id = auth_id
|
|
65
|
+
limit = self.authenticated_max_requests
|
|
66
|
+
else:
|
|
67
|
+
# fallback to IP
|
|
68
|
+
client = scope.get("client")
|
|
69
|
+
key_id = client[0] if client else "anonymous"
|
|
70
|
+
limit = self.anonymous_max_requests
|
|
71
|
+
|
|
72
|
+
current_window = int(time.time() // self.window_seconds)
|
|
73
|
+
key = f"quota:{key_id}:{current_window}"
|
|
74
|
+
|
|
75
|
+
try:
|
|
76
|
+
kv = await self._get_kv()
|
|
77
|
+
prev = await kv.get(key) or "0"
|
|
78
|
+
count = int(prev) + 1
|
|
79
|
+
|
|
80
|
+
if int(prev) == 0:
|
|
81
|
+
# Set with expiration datetime when it is the first request in the window.
|
|
82
|
+
expiration = datetime.now(UTC) + timedelta(seconds=self.window_seconds)
|
|
83
|
+
await kv.set(key, str(count), expiration=expiration)
|
|
84
|
+
else:
|
|
85
|
+
await kv.set(key, str(count))
|
|
86
|
+
except Exception:
|
|
87
|
+
logger.exception("Failed to access KV store for quota")
|
|
88
|
+
return await self._send_error(send, 500, "Quota service error")
|
|
89
|
+
|
|
90
|
+
if count > limit:
|
|
91
|
+
logger.warning(
|
|
92
|
+
"Quota exceeded for client %s: %d/%d",
|
|
93
|
+
key_id,
|
|
94
|
+
count,
|
|
95
|
+
limit,
|
|
96
|
+
)
|
|
97
|
+
return await self._send_error(send, 429, "Quota exceeded")
|
|
98
|
+
|
|
99
|
+
return await self.app(scope, receive, send)
|
|
100
|
+
|
|
101
|
+
async def _send_error(self, send: Send, status: int, message: str):
|
|
102
|
+
await send(
|
|
103
|
+
{
|
|
104
|
+
"type": "http.response.start",
|
|
105
|
+
"status": status,
|
|
106
|
+
"headers": [[b"content-type", b"application/json"]],
|
|
107
|
+
}
|
|
108
|
+
)
|
|
109
|
+
body = json.dumps({"error": {"message": message}}).encode()
|
|
110
|
+
await send({"type": "http.response.body", "body": body})
|