forked from phoenix-oss/llama-stack-mirror
		
	# What does this PR do? - as title, cleaning up `import *`'s - upgrade tests to make them more robust to bad model outputs - remove import *'s in llama_stack/apis/* (skip __init__ modules) <img width="465" alt="image" src="https://github.com/user-attachments/assets/d8339c13-3b40-4ba5-9c53-0d2329726ee2" /> - run `sh run_openapi_generator.sh`, no types gets affected ## Test Plan ### Providers Tests **agents** ``` pytest -v -s llama_stack/providers/tests/agents/test_agents.py -m "together" --safety-shield meta-llama/Llama-Guard-3-8B --inference-model meta-llama/Llama-3.1-405B-Instruct-FP8 ``` **inference** ```bash # meta-reference torchrun $CONDA_PREFIX/bin/pytest -v -s -k "meta_reference" --inference-model="meta-llama/Llama-3.1-8B-Instruct" ./llama_stack/providers/tests/inference/test_text_inference.py torchrun $CONDA_PREFIX/bin/pytest -v -s -k "meta_reference" --inference-model="meta-llama/Llama-3.2-11B-Vision-Instruct" ./llama_stack/providers/tests/inference/test_vision_inference.py # together pytest -v -s -k "together" --inference-model="meta-llama/Llama-3.1-8B-Instruct" ./llama_stack/providers/tests/inference/test_text_inference.py pytest -v -s -k "together" --inference-model="meta-llama/Llama-3.2-11B-Vision-Instruct" ./llama_stack/providers/tests/inference/test_vision_inference.py pytest ./llama_stack/providers/tests/inference/test_prompt_adapter.py ``` **safety** ``` pytest -v -s llama_stack/providers/tests/safety/test_safety.py -m together --safety-shield meta-llama/Llama-Guard-3-8B ``` **memory** ``` pytest -v -s llama_stack/providers/tests/memory/test_memory.py -m "sentence_transformers" --env EMBEDDING_DIMENSION=384 ``` **scoring** ``` pytest -v -s -m llm_as_judge_scoring_together_inference llama_stack/providers/tests/scoring/test_scoring.py --judge-model meta-llama/Llama-3.2-3B-Instruct pytest -v -s -m basic_scoring_together_inference llama_stack/providers/tests/scoring/test_scoring.py pytest -v -s -m braintrust_scoring_together_inference llama_stack/providers/tests/scoring/test_scoring.py ``` **datasetio** ``` pytest -v -s -m localfs llama_stack/providers/tests/datasetio/test_datasetio.py pytest -v -s -m huggingface llama_stack/providers/tests/datasetio/test_datasetio.py ``` **eval** ``` pytest -v -s -m meta_reference_eval_together_inference llama_stack/providers/tests/eval/test_eval.py pytest -v -s -m meta_reference_eval_together_inference_huggingface_datasetio llama_stack/providers/tests/eval/test_eval.py ``` ### Client-SDK Tests ``` LLAMA_STACK_BASE_URL=http://localhost:5000 pytest -v ./tests/client-sdk ``` ### llama-stack-apps ``` PORT=5000 LOCALHOST=localhost python -m examples.agents.hello $LOCALHOST $PORT python -m examples.agents.inflation $LOCALHOST $PORT python -m examples.agents.podcast_transcript $LOCALHOST $PORT python -m examples.agents.rag_as_attachments $LOCALHOST $PORT python -m examples.agents.rag_with_memory_bank $LOCALHOST $PORT python -m examples.safety.llama_guard_demo_mm $LOCALHOST $PORT python -m examples.agents.e2e_loop_with_custom_tools $LOCALHOST $PORT # Vision model python -m examples.interior_design_assistant.app python -m examples.agent_store.app $LOCALHOST $PORT ``` ### CLI ``` which llama llama model prompt-format -m Llama3.2-11B-Vision-Instruct llama model list llama stack list-apis llama stack list-providers inference llama stack build --template ollama --image-type conda ``` ### Distributions Tests **ollama** ``` llama stack build --template ollama --image-type conda ollama run llama3.2:1b-instruct-fp16 llama stack run ./llama_stack/templates/ollama/run.yaml --env INFERENCE_MODEL=meta-llama/Llama-3.2-1B-Instruct ``` **fireworks** ``` llama stack build --template fireworks --image-type conda llama stack run ./llama_stack/templates/fireworks/run.yaml ``` **together** ``` llama stack build --template together --image-type conda llama stack run ./llama_stack/templates/together/run.yaml ``` **tgi** ``` llama stack run ./llama_stack/templates/tgi/run.yaml --env TGI_URL=http://0.0.0.0:5009 --env INFERENCE_MODEL=meta-llama/Llama-3.1-8B-Instruct ``` ## Sources Please link relevant resources if necessary. ## Before submitting - [ ] This PR fixes a typo or improves the docs (you can dismiss the other checks if that's the case). - [ ] Ran pre-commit to handle lint / formatting issues. - [ ] Read the [contributor guideline](https://github.com/meta-llama/llama-stack/blob/main/CONTRIBUTING.md), Pull Request section? - [ ] Updated relevant documentation. - [ ] Wrote necessary unit or integration tests.
		
			
				
	
	
		
			263 lines
		
	
	
	
		
			9.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
	
		
			9.8 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 threading
 | |
| from typing import Any, Dict, List, Optional
 | |
| 
 | |
| from opentelemetry import metrics, trace
 | |
| from opentelemetry.exporter.otlp.proto.http.metric_exporter import OTLPMetricExporter
 | |
| from opentelemetry.exporter.otlp.proto.http.trace_exporter import OTLPSpanExporter
 | |
| from opentelemetry.sdk.metrics import MeterProvider
 | |
| from opentelemetry.sdk.metrics.export import PeriodicExportingMetricReader
 | |
| from opentelemetry.sdk.resources import Resource
 | |
| from opentelemetry.sdk.trace import TracerProvider
 | |
| from opentelemetry.sdk.trace.export import BatchSpanProcessor
 | |
| from opentelemetry.semconv.resource import ResourceAttributes
 | |
| 
 | |
| from llama_stack.apis.telemetry import (
 | |
|     Event,
 | |
|     MetricEvent,
 | |
|     QueryCondition,
 | |
|     SpanEndPayload,
 | |
|     SpanStartPayload,
 | |
|     SpanStatus,
 | |
|     SpanWithStatus,
 | |
|     StructuredLogEvent,
 | |
|     Telemetry,
 | |
|     Trace,
 | |
|     UnstructuredLogEvent,
 | |
| )
 | |
| 
 | |
| from llama_stack.distribution.datatypes import Api
 | |
| 
 | |
| from llama_stack.providers.inline.telemetry.meta_reference.console_span_processor import (
 | |
|     ConsoleSpanProcessor,
 | |
| )
 | |
| 
 | |
| from llama_stack.providers.inline.telemetry.meta_reference.sqlite_span_processor import (
 | |
|     SQLiteSpanProcessor,
 | |
| )
 | |
| from llama_stack.providers.utils.telemetry.dataset_mixin import TelemetryDatasetMixin
 | |
| from llama_stack.providers.utils.telemetry.sqlite_trace_store import SQLiteTraceStore
 | |
| 
 | |
| from .config import TelemetryConfig, TelemetrySink
 | |
| 
 | |
| _GLOBAL_STORAGE = {
 | |
|     "active_spans": {},
 | |
|     "counters": {},
 | |
|     "gauges": {},
 | |
|     "up_down_counters": {},
 | |
| }
 | |
| _global_lock = threading.Lock()
 | |
| 
 | |
| 
 | |
| def string_to_trace_id(s: str) -> int:
 | |
|     # Convert the string to bytes and then to an integer
 | |
|     return int.from_bytes(s.encode(), byteorder="big", signed=False)
 | |
| 
 | |
| 
 | |
| def string_to_span_id(s: str) -> int:
 | |
|     # Use only the first 8 bytes (64 bits) for span ID
 | |
|     return int.from_bytes(s.encode()[:8], byteorder="big", signed=False)
 | |
| 
 | |
| 
 | |
| def is_tracing_enabled(tracer):
 | |
|     with tracer.start_as_current_span("check_tracing") as span:
 | |
|         return span.is_recording()
 | |
| 
 | |
| 
 | |
| class TelemetryAdapter(TelemetryDatasetMixin, Telemetry):
 | |
|     def __init__(self, config: TelemetryConfig, deps: Dict[str, Any]) -> None:
 | |
|         self.config = config
 | |
|         self.datasetio_api = deps[Api.datasetio]
 | |
| 
 | |
|         resource = Resource.create(
 | |
|             {
 | |
|                 ResourceAttributes.SERVICE_NAME: self.config.service_name,
 | |
|             }
 | |
|         )
 | |
| 
 | |
|         provider = TracerProvider(resource=resource)
 | |
|         trace.set_tracer_provider(provider)
 | |
|         if TelemetrySink.OTEL in self.config.sinks:
 | |
|             otlp_exporter = OTLPSpanExporter(
 | |
|                 endpoint=self.config.otel_endpoint,
 | |
|             )
 | |
|             span_processor = BatchSpanProcessor(otlp_exporter)
 | |
|             trace.get_tracer_provider().add_span_processor(span_processor)
 | |
|             metric_reader = PeriodicExportingMetricReader(
 | |
|                 OTLPMetricExporter(
 | |
|                     endpoint=self.config.otel_endpoint,
 | |
|                 )
 | |
|             )
 | |
|             metric_provider = MeterProvider(
 | |
|                 resource=resource, metric_readers=[metric_reader]
 | |
|             )
 | |
|             metrics.set_meter_provider(metric_provider)
 | |
|             self.meter = metrics.get_meter(__name__)
 | |
|         if TelemetrySink.SQLITE in self.config.sinks:
 | |
|             trace.get_tracer_provider().add_span_processor(
 | |
|                 SQLiteSpanProcessor(self.config.sqlite_db_path)
 | |
|             )
 | |
|             self.trace_store = SQLiteTraceStore(self.config.sqlite_db_path)
 | |
|         if TelemetrySink.CONSOLE in self.config.sinks:
 | |
|             trace.get_tracer_provider().add_span_processor(ConsoleSpanProcessor())
 | |
|         self._lock = _global_lock
 | |
| 
 | |
|     async def initialize(self) -> None:
 | |
|         pass
 | |
| 
 | |
|     async def shutdown(self) -> None:
 | |
|         trace.get_tracer_provider().force_flush()
 | |
|         trace.get_tracer_provider().shutdown()
 | |
|         metrics.get_meter_provider().shutdown()
 | |
| 
 | |
|     async def log_event(self, event: Event, ttl_seconds: int = 604800) -> None:
 | |
|         if isinstance(event, UnstructuredLogEvent):
 | |
|             self._log_unstructured(event, ttl_seconds)
 | |
|         elif isinstance(event, MetricEvent):
 | |
|             self._log_metric(event)
 | |
|         elif isinstance(event, StructuredLogEvent):
 | |
|             self._log_structured(event, ttl_seconds)
 | |
|         else:
 | |
|             raise ValueError(f"Unknown event type: {event}")
 | |
| 
 | |
|     def _log_unstructured(self, event: UnstructuredLogEvent, ttl_seconds: int) -> None:
 | |
|         with self._lock:
 | |
|             # Use global storage instead of instance storage
 | |
|             span_id = string_to_span_id(event.span_id)
 | |
|             span = _GLOBAL_STORAGE["active_spans"].get(span_id)
 | |
| 
 | |
|             if span:
 | |
|                 timestamp_ns = int(event.timestamp.timestamp() * 1e9)
 | |
|                 span.add_event(
 | |
|                     name=event.type,
 | |
|                     attributes={
 | |
|                         "message": event.message,
 | |
|                         "severity": event.severity.value,
 | |
|                         "__ttl__": ttl_seconds,
 | |
|                         **event.attributes,
 | |
|                     },
 | |
|                     timestamp=timestamp_ns,
 | |
|                 )
 | |
|             else:
 | |
|                 print(
 | |
|                     f"Warning: No active span found for span_id {span_id}. Dropping event: {event}"
 | |
|                 )
 | |
| 
 | |
|     def _get_or_create_counter(self, name: str, unit: str) -> metrics.Counter:
 | |
|         if name not in _GLOBAL_STORAGE["counters"]:
 | |
|             _GLOBAL_STORAGE["counters"][name] = self.meter.create_counter(
 | |
|                 name=name,
 | |
|                 unit=unit,
 | |
|                 description=f"Counter for {name}",
 | |
|             )
 | |
|         return _GLOBAL_STORAGE["counters"][name]
 | |
| 
 | |
|     def _get_or_create_gauge(self, name: str, unit: str) -> metrics.ObservableGauge:
 | |
|         if name not in _GLOBAL_STORAGE["gauges"]:
 | |
|             _GLOBAL_STORAGE["gauges"][name] = self.meter.create_gauge(
 | |
|                 name=name,
 | |
|                 unit=unit,
 | |
|                 description=f"Gauge for {name}",
 | |
|             )
 | |
|         return _GLOBAL_STORAGE["gauges"][name]
 | |
| 
 | |
|     def _log_metric(self, event: MetricEvent) -> None:
 | |
|         if isinstance(event.value, int):
 | |
|             counter = self._get_or_create_counter(event.metric, event.unit)
 | |
|             counter.add(event.value, attributes=event.attributes)
 | |
|         elif isinstance(event.value, float):
 | |
|             up_down_counter = self._get_or_create_up_down_counter(
 | |
|                 event.metric, event.unit
 | |
|             )
 | |
|             up_down_counter.add(event.value, attributes=event.attributes)
 | |
| 
 | |
|     def _get_or_create_up_down_counter(
 | |
|         self, name: str, unit: str
 | |
|     ) -> metrics.UpDownCounter:
 | |
|         if name not in _GLOBAL_STORAGE["up_down_counters"]:
 | |
|             _GLOBAL_STORAGE["up_down_counters"][name] = (
 | |
|                 self.meter.create_up_down_counter(
 | |
|                     name=name,
 | |
|                     unit=unit,
 | |
|                     description=f"UpDownCounter for {name}",
 | |
|                 )
 | |
|             )
 | |
|         return _GLOBAL_STORAGE["up_down_counters"][name]
 | |
| 
 | |
|     def _log_structured(self, event: StructuredLogEvent, ttl_seconds: int) -> None:
 | |
|         with self._lock:
 | |
|             span_id = string_to_span_id(event.span_id)
 | |
|             trace_id = string_to_trace_id(event.trace_id)
 | |
|             tracer = trace.get_tracer(__name__)
 | |
|             if event.attributes is None:
 | |
|                 event.attributes = {}
 | |
|             event.attributes["__ttl__"] = ttl_seconds
 | |
| 
 | |
|             if isinstance(event.payload, SpanStartPayload):
 | |
|                 # Check if span already exists to prevent duplicates
 | |
|                 if span_id in _GLOBAL_STORAGE["active_spans"]:
 | |
|                     return
 | |
| 
 | |
|                 parent_span = None
 | |
|                 if event.payload.parent_span_id:
 | |
|                     parent_span_id = string_to_span_id(event.payload.parent_span_id)
 | |
|                     parent_span = _GLOBAL_STORAGE["active_spans"].get(parent_span_id)
 | |
| 
 | |
|                 context = trace.Context(trace_id=trace_id)
 | |
|                 if parent_span:
 | |
|                     context = trace.set_span_in_context(parent_span, context)
 | |
| 
 | |
|                 span = tracer.start_span(
 | |
|                     name=event.payload.name,
 | |
|                     context=context,
 | |
|                     attributes=event.attributes or {},
 | |
|                 )
 | |
|                 _GLOBAL_STORAGE["active_spans"][span_id] = span
 | |
| 
 | |
|             elif isinstance(event.payload, SpanEndPayload):
 | |
|                 span = _GLOBAL_STORAGE["active_spans"].get(span_id)
 | |
|                 if span:
 | |
|                     if event.attributes:
 | |
|                         span.set_attributes(event.attributes)
 | |
| 
 | |
|                     status = (
 | |
|                         trace.Status(status_code=trace.StatusCode.OK)
 | |
|                         if event.payload.status == SpanStatus.OK
 | |
|                         else trace.Status(status_code=trace.StatusCode.ERROR)
 | |
|                     )
 | |
|                     span.set_status(status)
 | |
|                     span.end()
 | |
|                     _GLOBAL_STORAGE["active_spans"].pop(span_id, None)
 | |
|             else:
 | |
|                 raise ValueError(f"Unknown structured log event: {event}")
 | |
| 
 | |
|     async def query_traces(
 | |
|         self,
 | |
|         attribute_filters: Optional[List[QueryCondition]] = None,
 | |
|         limit: Optional[int] = 100,
 | |
|         offset: Optional[int] = 0,
 | |
|         order_by: Optional[List[str]] = None,
 | |
|     ) -> List[Trace]:
 | |
|         return await self.trace_store.query_traces(
 | |
|             attribute_filters=attribute_filters,
 | |
|             limit=limit,
 | |
|             offset=offset,
 | |
|             order_by=order_by,
 | |
|         )
 | |
| 
 | |
|     async def get_span_tree(
 | |
|         self,
 | |
|         span_id: str,
 | |
|         attributes_to_return: Optional[List[str]] = None,
 | |
|         max_depth: Optional[int] = None,
 | |
|     ) -> Dict[str, SpanWithStatus]:
 | |
|         return await self.trace_store.get_span_tree(
 | |
|             span_id=span_id,
 | |
|             attributes_to_return=attributes_to_return,
 | |
|             max_depth=max_depth,
 | |
|         )
 |