mirror of
https://github.com/meta-llama/llama-stack.git
synced 2025-10-24 16:57:21 +00:00
Some checks failed
SqlStore Integration Tests / test-postgres (3.13) (push) Failing after 1s
Installer CI / lint (push) Failing after 3s
Test External Providers Installed via Module / test-external-providers-from-module (venv) (push) Has been skipped
Integration Tests (Replay) / Integration Tests (, , , client=, ) (push) Failing after 3s
Python Package Build Test / build (3.13) (push) Failing after 4s
SqlStore Integration Tests / test-postgres (3.12) (push) Failing after 6s
Python Package Build Test / build (3.12) (push) Failing after 5s
Unit Tests / unit-tests (3.12) (push) Failing after 5s
Installer CI / smoke-test-on-dev (push) Failing after 11s
Unit Tests / unit-tests (3.13) (push) Failing after 8s
API Conformance Tests / check-schema-compatibility (push) Successful in 15s
Vector IO Integration Tests / test-matrix (push) Failing after 18s
Test External API and Providers / test-external (venv) (push) Failing after 17s
Integration Auth Tests / test-matrix (oauth2_token) (push) Failing after 44s
UI Tests / ui-tests (22) (push) Successful in 1m28s
Pre-commit / pre-commit (push) Successful in 2m27s
# What does this PR do? ## Test Plan .venv ❯ sh ./scripts/install.sh ⚠️ Found existing container(s) for 'ollama-server', removing... ⚠️ Found existing container(s) for 'llama-stack', removing... ⚠️ Found existing container(s) for 'jaeger', removing... ⚠️ Found existing container(s) for 'otel-collector', removing... ⚠️ Found existing container(s) for 'prometheus', removing... ⚠️ Found existing container(s) for 'grafana', removing... 📡 Starting telemetry stack... 🦙 Starting Ollama... ⏳ Waiting for Ollama daemon... 📦 Ensuring model is pulled: llama3.2:3b... 🦙 Starting Llama Stack... ⏳ Waiting for Llama Stack API... .. 🎉 Llama Stack is ready! 👉 API endpoint: http://localhost:8321 📖 Documentation: https://llamastack.github.io/latest/references/api_reference/index.html 💻 To access the llama stack CLI, exec into the container: docker exec -ti llama-stack bash 📡 Telemetry dashboards: Jaeger UI: http://localhost:16686 Prometheus UI: http://localhost:9090 Grafana UI: http://localhost:3000 (admin/admin) OTEL Collector: http://localhost:4318 🐛 Report an issue @ https://github.com/llamastack/llama-stack/issues if you think it's a bug
451 lines
13 KiB
Bash
Executable file
451 lines
13 KiB
Bash
Executable file
#!/usr/bin/env bash
|
||
# 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.
|
||
|
||
[ -z "${BASH_VERSION:-}" ] && exec /usr/bin/env bash "$0" "$@"
|
||
if set -o | grep -Eq 'posix[[:space:]]+on'; then
|
||
exec /usr/bin/env bash "$0" "$@"
|
||
fi
|
||
|
||
set -Eeuo pipefail
|
||
|
||
PORT=8321
|
||
OLLAMA_PORT=11434
|
||
MODEL_ALIAS="llama3.2:3b"
|
||
SERVER_IMAGE="docker.io/llamastack/distribution-starter:latest"
|
||
WAIT_TIMEOUT=30
|
||
TEMP_LOG=""
|
||
WITH_TELEMETRY=true
|
||
TELEMETRY_SERVICE_NAME="llama-stack"
|
||
TELEMETRY_SINKS="otel_trace,otel_metric"
|
||
OTEL_EXPORTER_OTLP_ENDPOINT="http://otel-collector:4318"
|
||
TEMP_TELEMETRY_DIR=""
|
||
|
||
materialize_telemetry_configs() {
|
||
local dest="$1"
|
||
mkdir -p "$dest"
|
||
local otel_cfg="${dest}/otel-collector-config.yaml"
|
||
local prom_cfg="${dest}/prometheus.yml"
|
||
local graf_cfg="${dest}/grafana-datasources.yaml"
|
||
|
||
for asset in "$otel_cfg" "$prom_cfg" "$graf_cfg"; do
|
||
if [ -e "$asset" ]; then
|
||
die "Telemetry asset ${asset} already exists; refusing to overwrite"
|
||
fi
|
||
done
|
||
|
||
cat <<'EOF' > "$otel_cfg"
|
||
receivers:
|
||
otlp:
|
||
protocols:
|
||
grpc:
|
||
endpoint: 0.0.0.0:4317
|
||
http:
|
||
endpoint: 0.0.0.0:4318
|
||
|
||
processors:
|
||
batch:
|
||
timeout: 1s
|
||
send_batch_size: 1024
|
||
|
||
exporters:
|
||
# Export traces to Jaeger
|
||
otlp/jaeger:
|
||
endpoint: jaeger:4317
|
||
tls:
|
||
insecure: true
|
||
|
||
# Export metrics to Prometheus
|
||
prometheus:
|
||
endpoint: 0.0.0.0:9464
|
||
namespace: llama_stack
|
||
|
||
# Debug exporter for troubleshooting
|
||
debug:
|
||
verbosity: detailed
|
||
|
||
service:
|
||
pipelines:
|
||
traces:
|
||
receivers: [otlp]
|
||
processors: [batch]
|
||
exporters: [otlp/jaeger, debug]
|
||
|
||
metrics:
|
||
receivers: [otlp]
|
||
processors: [batch]
|
||
exporters: [prometheus, debug]
|
||
EOF
|
||
|
||
cat <<'EOF' > "$prom_cfg"
|
||
global:
|
||
scrape_interval: 15s
|
||
evaluation_interval: 15s
|
||
|
||
scrape_configs:
|
||
- job_name: 'prometheus'
|
||
static_configs:
|
||
- targets: ['localhost:9090']
|
||
|
||
- job_name: 'otel-collector'
|
||
static_configs:
|
||
- targets: ['otel-collector:9464']
|
||
EOF
|
||
|
||
cat <<'EOF' > "$graf_cfg"
|
||
apiVersion: 1
|
||
|
||
datasources:
|
||
- name: Prometheus
|
||
type: prometheus
|
||
access: proxy
|
||
url: http://prometheus:9090
|
||
isDefault: true
|
||
editable: true
|
||
|
||
- name: Jaeger
|
||
type: jaeger
|
||
access: proxy
|
||
url: http://jaeger:16686
|
||
editable: true
|
||
EOF
|
||
}
|
||
|
||
# Cleanup function to remove temporary files
|
||
cleanup() {
|
||
if [ -n "$TEMP_LOG" ] && [ -f "$TEMP_LOG" ]; then
|
||
rm -f "$TEMP_LOG"
|
||
fi
|
||
if [ -n "$TEMP_TELEMETRY_DIR" ] && [ -d "$TEMP_TELEMETRY_DIR" ]; then
|
||
rm -rf "$TEMP_TELEMETRY_DIR"
|
||
fi
|
||
}
|
||
|
||
# Set up trap to clean up on exit, error, or interrupt
|
||
trap cleanup EXIT ERR INT TERM
|
||
|
||
log(){ printf "\e[1;32m%s\e[0m\n" "$*"; }
|
||
die(){
|
||
printf "\e[1;31m❌ %s\e[0m\n" "$*" >&2
|
||
printf "\e[1;31m🐛 Report an issue @ https://github.com/llamastack/llama-stack/issues if you think it's a bug\e[0m\n" >&2
|
||
exit 1
|
||
}
|
||
|
||
# Helper function to execute command with logging
|
||
execute_with_log() {
|
||
local cmd=("$@")
|
||
TEMP_LOG=$(mktemp)
|
||
if ! "${cmd[@]}" > "$TEMP_LOG" 2>&1; then
|
||
log "❌ Command failed; dumping output:"
|
||
log "Command that failed: ${cmd[*]}"
|
||
log "Command output:"
|
||
cat "$TEMP_LOG"
|
||
return 1
|
||
fi
|
||
return 0
|
||
}
|
||
|
||
wait_for_service() {
|
||
local url="$1"
|
||
local pattern="$2"
|
||
local timeout="$3"
|
||
local name="$4"
|
||
local start ts
|
||
log "⏳ Waiting for ${name}..."
|
||
start=$(date +%s)
|
||
while true; do
|
||
if curl --retry 5 --retry-delay 1 --retry-max-time "$timeout" --retry-all-errors --silent --fail "$url" 2>/dev/null | grep -q "$pattern"; then
|
||
break
|
||
fi
|
||
ts=$(date +%s)
|
||
if (( ts - start >= timeout )); then
|
||
return 1
|
||
fi
|
||
printf '.'
|
||
done
|
||
printf '\n'
|
||
return 0
|
||
}
|
||
|
||
usage() {
|
||
cat << EOF
|
||
📚 Llama Stack Deployment Script
|
||
|
||
Description:
|
||
This script sets up and deploys Llama Stack with Ollama integration in containers.
|
||
It handles both Docker and Podman runtimes and includes automatic platform detection.
|
||
|
||
Usage:
|
||
$(basename "$0") [OPTIONS]
|
||
|
||
Options:
|
||
-p, --port PORT Server port for Llama Stack (default: ${PORT})
|
||
-o, --ollama-port PORT Ollama service port (default: ${OLLAMA_PORT})
|
||
-m, --model MODEL Model alias to use (default: ${MODEL_ALIAS})
|
||
-i, --image IMAGE Server image (default: ${SERVER_IMAGE})
|
||
-t, --timeout SECONDS Service wait timeout in seconds (default: ${WAIT_TIMEOUT})
|
||
--with-telemetry Provision Jaeger, OTEL Collector, Prometheus, and Grafana (default: enabled)
|
||
--no-telemetry, --without-telemetry
|
||
Skip provisioning the telemetry stack
|
||
--telemetry-service NAME Service name reported to telemetry (default: ${TELEMETRY_SERVICE_NAME})
|
||
--telemetry-sinks SINKS Comma-separated telemetry sinks (default: ${TELEMETRY_SINKS})
|
||
--otel-endpoint URL OTLP endpoint provided to Llama Stack (default: ${OTEL_EXPORTER_OTLP_ENDPOINT})
|
||
-h, --help Show this help message
|
||
|
||
For more information:
|
||
Documentation: https://llamastack.github.io/latest/
|
||
GitHub: https://github.com/llamastack/llama-stack
|
||
|
||
Report issues:
|
||
https://github.com/llamastack/llama-stack/issues
|
||
EOF
|
||
}
|
||
|
||
# Parse command line arguments
|
||
while [[ $# -gt 0 ]]; do
|
||
case $1 in
|
||
-h|--help)
|
||
usage
|
||
exit 0
|
||
;;
|
||
-p|--port)
|
||
PORT="$2"
|
||
shift 2
|
||
;;
|
||
-o|--ollama-port)
|
||
OLLAMA_PORT="$2"
|
||
shift 2
|
||
;;
|
||
-m|--model)
|
||
MODEL_ALIAS="$2"
|
||
shift 2
|
||
;;
|
||
-i|--image)
|
||
SERVER_IMAGE="$2"
|
||
shift 2
|
||
;;
|
||
-t|--timeout)
|
||
WAIT_TIMEOUT="$2"
|
||
shift 2
|
||
;;
|
||
--with-telemetry)
|
||
WITH_TELEMETRY=true
|
||
shift
|
||
;;
|
||
--no-telemetry|--without-telemetry)
|
||
WITH_TELEMETRY=false
|
||
shift
|
||
;;
|
||
--telemetry-service)
|
||
TELEMETRY_SERVICE_NAME="$2"
|
||
shift 2
|
||
;;
|
||
--telemetry-sinks)
|
||
TELEMETRY_SINKS="$2"
|
||
shift 2
|
||
;;
|
||
--otel-endpoint)
|
||
OTEL_EXPORTER_OTLP_ENDPOINT="$2"
|
||
shift 2
|
||
;;
|
||
*)
|
||
die "Unknown option: $1"
|
||
;;
|
||
esac
|
||
done
|
||
|
||
if command -v docker &> /dev/null; then
|
||
ENGINE="docker"
|
||
elif command -v podman &> /dev/null; then
|
||
ENGINE="podman"
|
||
else
|
||
die "Docker or Podman is required. Install Docker: https://docs.docker.com/get-docker/ or Podman: https://podman.io/getting-started/installation"
|
||
fi
|
||
|
||
# Explicitly set the platform for the host architecture
|
||
HOST_ARCH="$(uname -m)"
|
||
if [ "$HOST_ARCH" = "arm64" ]; then
|
||
if [ "$ENGINE" = "docker" ]; then
|
||
PLATFORM_OPTS=( --platform linux/amd64 )
|
||
else
|
||
PLATFORM_OPTS=( --os linux --arch amd64 )
|
||
fi
|
||
else
|
||
PLATFORM_OPTS=()
|
||
fi
|
||
|
||
# macOS + Podman: ensure VM is running before we try to launch containers
|
||
# If you need GPU passthrough under Podman on macOS, init the VM with libkrun:
|
||
# CONTAINERS_MACHINE_PROVIDER=libkrun podman machine init
|
||
if [ "$ENGINE" = "podman" ] && [ "$(uname -s)" = "Darwin" ]; then
|
||
if ! podman info &>/dev/null; then
|
||
log "⌛️ Initializing Podman VM..."
|
||
podman machine init &>/dev/null || true
|
||
podman machine start &>/dev/null || true
|
||
|
||
log "⌛️ Waiting for Podman API..."
|
||
until podman info &>/dev/null; do
|
||
sleep 1
|
||
done
|
||
log "✅ Podman VM is up."
|
||
fi
|
||
fi
|
||
|
||
# Clean up any leftovers from earlier runs
|
||
containers=(ollama-server llama-stack)
|
||
if [ "$WITH_TELEMETRY" = true ]; then
|
||
containers+=(jaeger otel-collector prometheus grafana)
|
||
fi
|
||
for name in "${containers[@]}"; do
|
||
ids=$($ENGINE ps -aq --filter "name=^${name}$")
|
||
if [ -n "$ids" ]; then
|
||
log "⚠️ Found existing container(s) for '${name}', removing..."
|
||
if ! execute_with_log $ENGINE rm -f "$ids"; then
|
||
die "Container cleanup failed"
|
||
fi
|
||
fi
|
||
done
|
||
|
||
###############################################################################
|
||
# 0. Create a shared network
|
||
###############################################################################
|
||
if ! $ENGINE network inspect llama-net >/dev/null 2>&1; then
|
||
log "🌐 Creating network..."
|
||
if ! execute_with_log $ENGINE network create llama-net; then
|
||
die "Network creation failed"
|
||
fi
|
||
fi
|
||
|
||
###############################################################################
|
||
# Telemetry Stack
|
||
###############################################################################
|
||
if [ "$WITH_TELEMETRY" = true ]; then
|
||
TEMP_TELEMETRY_DIR="$(mktemp -d)"
|
||
TELEMETRY_ASSETS_DIR="$TEMP_TELEMETRY_DIR"
|
||
log "🧰 Materializing telemetry configs..."
|
||
materialize_telemetry_configs "$TELEMETRY_ASSETS_DIR"
|
||
|
||
log "📡 Starting telemetry stack..."
|
||
|
||
if ! execute_with_log $ENGINE run -d "${PLATFORM_OPTS[@]}" --name jaeger \
|
||
--network llama-net \
|
||
-e COLLECTOR_ZIPKIN_HOST_PORT=:9411 \
|
||
-p 16686:16686 \
|
||
-p 14250:14250 \
|
||
-p 9411:9411 \
|
||
docker.io/jaegertracing/all-in-one:latest > /dev/null 2>&1; then
|
||
die "Jaeger startup failed"
|
||
fi
|
||
|
||
if ! execute_with_log $ENGINE run -d "${PLATFORM_OPTS[@]}" --name otel-collector \
|
||
--network llama-net \
|
||
-p 4318:4318 \
|
||
-p 4317:4317 \
|
||
-p 9464:9464 \
|
||
-p 13133:13133 \
|
||
-v "${TELEMETRY_ASSETS_DIR}/otel-collector-config.yaml:/etc/otel-collector-config.yaml:Z" \
|
||
docker.io/otel/opentelemetry-collector-contrib:latest \
|
||
--config /etc/otel-collector-config.yaml > /dev/null 2>&1; then
|
||
die "OpenTelemetry Collector startup failed"
|
||
fi
|
||
|
||
if ! execute_with_log $ENGINE run -d "${PLATFORM_OPTS[@]}" --name prometheus \
|
||
--network llama-net \
|
||
-p 9090:9090 \
|
||
-v "${TELEMETRY_ASSETS_DIR}/prometheus.yml:/etc/prometheus/prometheus.yml:Z" \
|
||
docker.io/prom/prometheus:latest \
|
||
--config.file=/etc/prometheus/prometheus.yml \
|
||
--storage.tsdb.path=/prometheus \
|
||
--web.console.libraries=/etc/prometheus/console_libraries \
|
||
--web.console.templates=/etc/prometheus/consoles \
|
||
--storage.tsdb.retention.time=200h \
|
||
--web.enable-lifecycle > /dev/null 2>&1; then
|
||
die "Prometheus startup failed"
|
||
fi
|
||
|
||
if ! execute_with_log $ENGINE run -d "${PLATFORM_OPTS[@]}" --name grafana \
|
||
--network llama-net \
|
||
-p 3000:3000 \
|
||
-e GF_SECURITY_ADMIN_PASSWORD=admin \
|
||
-e GF_USERS_ALLOW_SIGN_UP=false \
|
||
-v "${TELEMETRY_ASSETS_DIR}/grafana-datasources.yaml:/etc/grafana/provisioning/datasources/datasources.yaml:Z" \
|
||
docker.io/grafana/grafana:11.0.0 > /dev/null 2>&1; then
|
||
die "Grafana startup failed"
|
||
fi
|
||
fi
|
||
|
||
###############################################################################
|
||
# 1. Ollama
|
||
###############################################################################
|
||
log "🦙 Starting Ollama..."
|
||
if ! execute_with_log $ENGINE run -d "${PLATFORM_OPTS[@]}" --name ollama-server \
|
||
--network llama-net \
|
||
-p "${OLLAMA_PORT}:${OLLAMA_PORT}" \
|
||
docker.io/ollama/ollama > /dev/null 2>&1; then
|
||
die "Ollama startup failed"
|
||
fi
|
||
|
||
if ! wait_for_service "http://localhost:${OLLAMA_PORT}/" "Ollama" "$WAIT_TIMEOUT" "Ollama daemon"; then
|
||
log "❌ Ollama daemon did not become ready in ${WAIT_TIMEOUT}s; dumping container logs:"
|
||
$ENGINE logs --tail 200 ollama-server
|
||
die "Ollama startup failed"
|
||
fi
|
||
|
||
log "📦 Ensuring model is pulled: ${MODEL_ALIAS}..."
|
||
if ! execute_with_log $ENGINE exec ollama-server ollama pull "${MODEL_ALIAS}"; then
|
||
log "❌ Failed to pull model ${MODEL_ALIAS}; dumping container logs:"
|
||
$ENGINE logs --tail 200 ollama-server
|
||
die "Model pull failed"
|
||
fi
|
||
|
||
###############################################################################
|
||
# 2. Llama‑Stack
|
||
###############################################################################
|
||
server_env_opts=()
|
||
if [ "$WITH_TELEMETRY" = true ]; then
|
||
server_env_opts+=(
|
||
-e TELEMETRY_SINKS="${TELEMETRY_SINKS}"
|
||
-e OTEL_EXPORTER_OTLP_ENDPOINT="${OTEL_EXPORTER_OTLP_ENDPOINT}"
|
||
-e OTEL_SERVICE_NAME="${TELEMETRY_SERVICE_NAME}"
|
||
)
|
||
fi
|
||
|
||
cmd=( run -d "${PLATFORM_OPTS[@]}" --name llama-stack \
|
||
--network llama-net \
|
||
-p "${PORT}:${PORT}" \
|
||
"${server_env_opts[@]}" \
|
||
-e OLLAMA_URL="http://ollama-server:${OLLAMA_PORT}" \
|
||
"${SERVER_IMAGE}" --port "${PORT}")
|
||
|
||
log "🦙 Starting Llama Stack..."
|
||
if ! execute_with_log $ENGINE "${cmd[@]}"; then
|
||
die "Llama Stack startup failed"
|
||
fi
|
||
|
||
if ! wait_for_service "http://127.0.0.1:${PORT}/v1/health" "OK" "$WAIT_TIMEOUT" "Llama Stack API"; then
|
||
log "❌ Llama Stack did not become ready in ${WAIT_TIMEOUT}s; dumping container logs:"
|
||
$ENGINE logs --tail 200 llama-stack
|
||
die "Llama Stack startup failed"
|
||
fi
|
||
|
||
###############################################################################
|
||
# Done
|
||
###############################################################################
|
||
log ""
|
||
log "🎉 Llama Stack is ready!"
|
||
log "👉 API endpoint: http://localhost:${PORT}"
|
||
log "📖 Documentation: https://llamastack.github.io/latest/references/api_reference/index.html"
|
||
log "💻 To access the llama stack CLI, exec into the container:"
|
||
log " $ENGINE exec -ti llama-stack bash"
|
||
if [ "$WITH_TELEMETRY" = true ]; then
|
||
log "📡 Telemetry dashboards:"
|
||
log " Jaeger UI: http://localhost:16686"
|
||
log " Prometheus UI: http://localhost:9090"
|
||
log " Grafana UI: http://localhost:3000 (admin/admin)"
|
||
log " OTEL Collector: http://localhost:4318"
|
||
fi
|
||
log "🐛 Report an issue @ https://github.com/llamastack/llama-stack/issues if you think it's a bug"
|
||
log ""
|