apps/opik-documentation/documentation/fern/docs/contributing/local-development.mdx
This guide provides detailed instructions for setting up your local Opik development environment. We offer multiple development modes optimized for different workflows.
Choose the approach that best fits your development needs:
| Development Mode | Use Case | Command | Speed |
|---|---|---|---|
| Docker Mode | Testing full stack, closest to production | ./opik.sh --build | Slow |
| Local Process Mode | Fast BE + FE development | scripts/dev-runner.sh | Fast |
| BE-Only Mode | Backend development only | scripts/dev-runner.sh --be-only-restart | Fast |
| Infrastructure | Manual with IDE development | ./opik.sh --infra --port-mapping | Medium |
# Check Docker
docker --version
docker compose version
# Check Java and Maven
java -version
mvn -version
# Check Node.js and npm
node --version
npm --version
# Check Python
python --version
Best for: Testing complete system, integration testing, or when you need an environment closest to production.
The opik.sh script manages Docker Compose profiles to start different combinations of services:
# Build and start all services (recommended for first time)
./opik.sh --build
# Start without rebuilding (if no code changes)
./opik.sh
# Enable port mapping (useful for debugging)
./opik.sh --build --port-mapping
# Enable debug logging
./opik.sh --build --debug
# Infrastructure only (MySQL, Redis, ClickHouse, ZooKeeper, MinIO)
./opik.sh --infra --port-mapping
# Infrastructure + Backend services
./opik.sh --backend --port-mapping
# All services EXCEPT backend (for local backend development)
./opik.sh --local-be --port-mapping
# Add guardrails services
./opik.sh --build --guardrails
# Check service health
./opik.sh --verify
# View system status
./opik.sh --info
# Stop all services and clean up
./opik.sh --stop
# Rebuild specific service
docker compose -f deployment/docker-compose/docker-compose.yaml build backend
Best for: Rapid backend and frontend development with instant code reloading.
The dev-runner.sh script:
# Full restart (stop, build, start) - DEFAULT
scripts/dev-runner.sh
# Or explicitly
scripts/dev-runner.sh --restart
# Start without rebuilding (faster if no dependency changes)
scripts/dev-runner.sh --start
# Stop all services
scripts/dev-runner.sh --stop
# Check status
scripts/dev-runner.sh --verify
# View logs
scripts/dev-runner.sh --logs
# Enable verbose logging
scripts/dev-runner.sh --restart --debug
# Or set environment variable
DEBUG_MODE=true scripts/dev-runner.sh --restart
Backend Process:
/tmp/opik-<worktree-id>-backend.log/tmp/opik-<worktree-id>-backend.pidapps/opik-backendFrontend Process:
/tmp/opik-<worktree-id>-frontend.log/tmp/opik-<worktree-id>-frontend.pidInfrastructure (Docker):
After starting, configure the SDK to use your local instance:
opik configure --use_local
IMPORTANT: You must manually edit ~/.opik.config to remove /api from the URL:
[opik]
# Change from:
url_override = http://localhost:8080/api/
# To:
url_override = http://localhost:8080
workspace = default
Or use environment variables:
export OPIK_URL_OVERRIDE='http://localhost:8080'
export OPIK_WORKSPACE='default'
Best for: Backend-focused development when you don't need to modify frontend code.
This mode:
The frontend in Docker proxies API calls to your local backend process.
# Full restart (stop, build backend, start)
scripts/dev-runner.sh --be-only-restart
# Start without rebuilding
scripts/dev-runner.sh --be-only-start
# Stop services
scripts/dev-runner.sh --be-only-stop
# Check status
scripts/dev-runner.sh --be-only-verify
Backend Process (Local):
/tmp/opik-backend.logFrontend (Docker):
Infrastructure (Docker):
Configure SDK without the manual edit requirement:
opik configure --use_local
# Use URL: http://localhost:5173
Or with environment variables:
export OPIK_URL_OVERRIDE='http://localhost:5173/api'
export OPIK_WORKSPACE='default'
Best for: SDK development, integration testing, or when you need just the databases.
# Start only infrastructure services
./opik.sh --infra --port-mapping
# Verify infrastructure is running
./opik.sh --infra --verify
# Stop infrastructure
./opik.sh --infra --stop
This gives you access to:
Opik supports running multiple development environments simultaneously from different git worktrees. This is useful when you need to work on multiple features or compare branches side-by-side.
Each worktree automatically gets:
The port offset (0-99) is deterministically calculated from an MD5 hash of your project path, ensuring consistent port assignments across restarts.
| Service | Base Port | With Offset (e.g., 42) |
|---|---|---|
| Backend | 8080 | 8122 |
| Frontend | 5174 | 5216 |
| MySQL | 3306 | 3348 |
| Redis | 6379 | 6421 |
| ClickHouse HTTP | 8123 | 8165 |
| ClickHouse Native | 9000 | 9042 |
| Python Backend | 8000 | 8042 |
| Zookeeper | 2181 | 2223 |
| MinIO API | 9001 | 9043 |
| MinIO Console | 9090 | 9132 |
# Terminal 1: Main branch
cd ~/opik
scripts/dev-runner.sh --restart
# Access at ports based on hash of ~/opik
# Terminal 2: Feature branch
cd ~/opik-worktrees/feature-xyz
scripts/dev-runner.sh --restart
# Access at different ports based on hash of ~/opik-worktrees/feature-xyz
If you need specific ports (e.g., to use standard ports or avoid conflicts):
# Use standard ports (offset 0)
OPIK_PORT_OFFSET=0 scripts/dev-runner.sh --restart
# Use a specific offset
OPIK_PORT_OFFSET=10 scripts/dev-runner.sh --restart
The script automatically checks for port conflicts before starting:
# If ports are in use, you'll see:
[ERROR] Port 8122 (Backend) is already in use
Port collision detected! Another process is using one or more required ports.
This might be caused by:
- Another Opik instance running from a different worktree
- Stale containers from a previous run
- Other services using the same ports
To resolve:
1. Stop other Opik instances: ./scripts/dev-runner.sh --stop
2. Use a different port offset: export OPIK_PORT_OFFSET=<0-99>
3. Check running processes: lsof -i :8122
Containers are prefixed with the worktree project name:
opik-opik-mysql-1, opik-opik-backend-1opik-feature-xyz-mysql-1, opik-feature-xyz-backend-1Configure the SDK to use your worktree's backend port (shown when the environment starts):
# Configure SDK (use the backend port shown at startup)
export OPIK_URL_OVERRIDE='http://localhost:8080' # or your worktree's port
export OPIK_WORKSPACE='default'
Or edit ~/.opik.config:
[opik]
url_override = http://localhost:8122
workspace = default
All scripts have PowerShell equivalents for Windows developers.
# Build and start all services
.\opik.ps1 --build
# Different profiles
.\opik.ps1 --infra --port-mapping
.\opik.ps1 --backend --port-mapping
.\opik.ps1 --local-be --port-mapping
# Manage services
.\opik.ps1 --verify
.\opik.ps1 --stop
# Full restart
scripts\dev-runner.ps1
# Specific commands
scripts\dev-runner.ps1 --restart
scripts\dev-runner.ps1 --start
scripts\dev-runner.ps1 --stop
scripts\dev-runner.ps1 --verify
# BE-only mode
scripts\dev-runner.ps1 --be-only-restart
# Debug mode
scripts\dev-runner.ps1 --restart --debug
$env:TEMP directory$env:TEMP directoryGet-Content -Wait instead of tail -f for log following$env:USERPROFILE\.opik.config# Build backend only
scripts/dev-runner.sh --build-be
# Build frontend only
scripts/dev-runner.sh --build-fe
# Lint backend
scripts/dev-runner.sh --lint-be
# Lint frontend
scripts/dev-runner.sh --lint-fe
# Run migrations only
scripts/dev-runner.sh --migrate
# This will:
# 1. Start infrastructure if not running
# 2. Build backend if needed
# 3. Run MySQL migrations
# 4. Run ClickHouse migrations
If migrations fail, you may need to clean up:
# Stop all services
scripts/dev-runner.sh --stop # or ./opik.sh --stop
# Remove Opik Docker volumes (WARNING: DATA LOSS - removes Opik databases)
./opik.sh --clean
# Restart
scripts/dev-runner.sh --restart
# Show recent logs (last 20 lines)
scripts/dev-runner.sh --logs
# Follow logs in real-time
tail -f /tmp/opik-backend.log
tail -f /tmp/opik-frontend.log
# On Windows
Get-Content -Wait $env:TEMP\opik-backend.log
Get-Content -Wait $env:TEMP\opik-frontend.log
# View all Opik containers
docker ps --filter "name=opik-"
# View logs from Docker services
docker logs -f opik-backend-1
docker logs -f opik-frontend-1
docker logs -f opik-clickhouse-1
# Execute commands in containers
docker exec -it opik-mysql-1 mysql -u root -p
docker exec -it opik-clickhouse-1 clickhouse-client
# Restart a specific Docker service
docker restart opik-backend-1
# Check Docker is running
docker info
# Check port conflicts (ports shown when environment starts)
lsof -i :5174 # Frontend (default)
lsof -i :8080 # Backend (default)
lsof -i :3306 # MySQL (default)
lsof -i :8123 # ClickHouse (default)
# On Windows
Get-NetTCPConnection -LocalPort 5174
Get-NetTCPConnection -LocalPort 8080
# Clean backend build
cd apps/opik-backend
mvn clean
mvn spotless:apply # Fix formatting issues
mvn clean install
# Clean frontend build
cd apps/opik-frontend
rm -rf node_modules
npm install
npm run lint
# Check MySQL is accessible
docker exec -it opik-mysql-1 mysql -u root -p
# Check ClickHouse is accessible
docker exec -it opik-clickhouse-1 clickhouse-client
# Or via HTTP
echo 'SELECT version()' | curl -H 'X-ClickHouse-User: opik' -H 'X-ClickHouse-Key: opik' 'http://localhost:8123/' -d @-
# Kill stuck backend process
pkill -f "opik-backend.*jar"
# Kill stuck frontend process
pkill -f "vite.*opik-frontend"
# On Windows
Get-Process | Where-Object {$_.Path -like "*opik-backend*"} | Stop-Process -Force
Get-Process | Where-Object {$_.Path -like "*opik-frontend*"} | Stop-Process -Force
# Complete cleanup and restart
scripts/dev-runner.sh --stop
./opik.sh --clean # WARNING: Deletes Opik data
scripts/dev-runner.sh --restart
# 1. Start BE-only mode (fastest for backend work)
scripts/dev-runner.sh --be-only-restart
# 2. Make changes in apps/opik-backend
# 3. Rebuild and restart backend
scripts/dev-runner.sh --build-be
scripts/dev-runner.sh --be-only-start
# 4. Test changes via UI at http://localhost:5173
# 1. Start local process mode
scripts/dev-runner.sh --restart
# 2. Make changes in apps/opik-frontend
# Frontend hot-reloads automatically
# 3. View changes at http://localhost:5174
# 1. Start local process mode
scripts/dev-runner.sh --restart
# 2. Make changes to backend and frontend
# Frontend changes hot-reload
# Backend changes require rebuild:
scripts/dev-runner.sh --build-be
# Backend automatically restarts
# 3. Test at http://localhost:5174
# 1. Start infrastructure only
./opik.sh --infra --port-mapping
# 2. Start backend separately if needed
cd apps/opik-backend
mvn clean install
java -jar target/opik-backend-*.jar server config.yml
# 3. Configure SDK
opik configure --use_local
# 4. Test SDK changes
cd sdks/python
pip install -e .
pytest tests/e2e
# 1. Start full Docker stack
./opik.sh --build
# 2. Run tests against full environment
cd tests_end_to_end
pytest tests/
# 3. Clean up
./opik.sh --stop
--start instead of --restart when dependencies haven't changed--buildAlways run linters before committing:
scripts/dev-runner.sh --lint-be
scripts/dev-runner.sh --lint-fe
Test migrations locally before committing:
scripts/dev-runner.sh --migrate
Clean up regularly to free disk space:
# Clean up Opik Docker resources (WARNING: DATA LOSS - removes Opik databases)
./opik.sh --clean # Removes Opik containers and volumes
# Or clean up dangling Docker containers, networks, images (affects all projects)
docker system prune
Use debug mode for troubleshooting:
scripts/dev-runner.sh --restart --debug
Check service status before reporting issues:
scripts/dev-runner.sh --verify
./opik.sh --verify
Opik provides AI coding rules and configurations for editors like Cursor, Codex, and Claude Code. These are stored in .agents/ and can be synced to your editor of choice using Makefile.
# For Cursor users - creates .cursor symlink to .agents/
make cursor
# For Codex users - creates .codex symlink and generates Codex-compatible AGENTS files
make codex
# For Claude Code users - syncs rules to .claude/ and generates .mcp.json
make claude
make codex keeps .codex linked to .agents and generates AGENTS.override.md plus .agents/generated/codex/rules/*.md so Codex can consume rule content derived from .mdc files.
.agents/
├── rules/ # AI coding rules (.mdc files)
│ ├── *.mdc # Root-level rules (git, clean-code, etc.)
│ ├── apps/ # App-specific rules
│ │ ├── opik-backend/ # Java backend rules
│ │ └── opik-frontend/ # React frontend rules
│ └── sdks/ # SDK-specific rules
├── commands/ # Slash commands
└── mcp.json # MCP server configuration
Install pre-commit hooks to automatically run linting before commits:
# Install hooks
make hooks
# Remove hooks
make hooks-remove