opentelemetry-skill
o11y-dev/opentelemetry-skillExpert OpenTelemetry guidance for collector configuration, pipeline design, and production telemetry instrumentation. Use when configuring collectors, designing pipelines, instrumenting applications, implementing sampling, managing cardinality, securing telemetry, writing OTTL transformations, or setting up AI coding agent observability (Claude Code, Codex, Gemini CLI, GitHub Copilot).
SKILL.md
name: opentelemetry-skill description: "Expert OpenTelemetry guidance for collector configuration, pipeline design, and production telemetry instrumentation. Use when configuring collectors, designing pipelines, instrumenting applications, implementing sampling, managing cardinality, securing telemetry, writing OTTL transformations, or setting up AI coding agent observability (Claude Code, Codex, Gemini CLI, GitHub Copilot)." license: Apache-2.0 metadata: author: o11y.dev version: 1.4.1
OpenTelemetry Skill
Core Principles
Use these defaults:
-
Stability over Features: Check otelcol-contrib stability (Alpha/Beta/Stable) and warn on non-stable components in production.
-
Convention over Configuration: Prefer OpenTelemetry Semantic Conventions over custom attribute names.
-
Protocol Unification: Default to OTLP gRPC (4317); use OTLP HTTP (4318) when gRPC is blocked by the agent, proxy, browser, or backend.
-
Deterministic Routing Keys: Use stable routing keys for load-balancing exporters (
traceIDfor tail sampling,tenant_idorclusterfor tenant/shard routing). Normalize non-string attributes first. -
Safety First: Prefer collector stability (memory limiters, persistent queues, backpressure) over completeness. Dropping data is better than crashing the collector.
-
Cardinality Awareness: High-cardinality attributes (>100 unique values) must not be metric dimensions; use traces or logs instead.
-
Security by Default: Redact PII, enable TLS for cross-network communication, and authenticate all collector endpoints.
-
Cross-Field Consistency: Treat collector reviews as systems reviews. Check processor order, memory limits, replica strategy, routing, metric temporality, queue storage, PDB/HPA settings, and OTTL types together before calling a config safe.
Pre-Flight Checklist
Before generating config/code, confirm these. If unknown, ask first:
- Signal volume — High traffic (>10k RPS) or low volume? Drives sampling/scaling. → sampling.md, collector.md
- Cardinality risk — Any unbounded metric attributes (user/request/session IDs)? Move to traces/logs. → instrumentation.md
- Resiliency — Is restart/outage data loss acceptable? If no, use
file_storage+ persistent queues. → collector.md - Trust boundaries — Any public-network hops? Require TLS + mTLS. → security.md
- Deployment target — Kubernetes, EC2, Lambda, or containers? → architecture.md
Eval-Critical Response Minimums
When user requests match these patterns, include these points explicitly:
- Collector setup: include
memory_limiter; keep it first in each pipelineprocessorslist; explain OOM-prevention rationale. - Metric dimension request for
user_id: refuse; explain time-series explosion risk; suggest traces and bounded metric dimensions. - Kubernetes tail sampling: Gateway (Deployment) tier,
loadbalancingwithrouting_key: traceID, Headless Service (clusterIP: None), error+10% policies, Beta stability caution. - Claude Code telemetry: include
CLAUDE_CODE_ENABLE_TELEMETRY=1,OTEL_EXPORTER_OTLP_METRICS_TEMPORALITY_PREFERENCE=cumulative,~/.claude/settings.jsonpersistence;OTEL_LOG_USER_PROMPTS/OTEL_LOG_TOOL_DETAILSdefault tofalse— warn against enabling in shared/production environments without PII controls; avoidsession.idas a metric dimension. - AI agent tool-call tracing: for agents using
gen_ai.*traces, name execute-tool spans with the actual tool name (for examplebashorsearch_code) and preservegen_ai.tool.name; do not use a genericexecute_toolspan name.
Existing Configuration Review Mode
When the user provides an existing collector config, Helm values file, or Kubernetes manifest, audit it for internal contradictions before proposing edits.
Check these interactions together:
- Memory vs pod limit —
limit_mibmust leave headroom for the Go runtime and buffers. - Stateful processing vs scaling —
tail_sampling,spanmetrics, andservicegraphneed sticky routing above one replica. - Durability vs outage tolerance — disabled retries and no persistent queue mean data loss on backend failure.
- Deployment mode vs exposure —
hostPortfits DaemonSet/node-local patterns, not scaled gateway Deployments. - Rollout settings — review
replicaCount, HPAminReplicas, PodDisruptionBudget, and rolling updates together. - OTTL/filter correctness — keep attribute types consistent and prefer current semantic convention keys.
- Metric temporality and state —
deltatocumulative/cumulativetodeltaneed source/backend/restart checks. - Queue storage backend —
file_storageneeds local locking-safe storage; RWX, EFS, and NFS are unsafe defaults.
Progressive Disclosure: Context Triggers
Load detailed reference documentation only when the user's request matches a trigger. This keeps context lean.
| Trigger keywords | Load | Key topics |
|---|---|---|
| Kubernetes, Helm, values.yaml, audit, review, DaemonSet, Sidecar, Gateway, Scaling, Load Balancing | architecture.md | DaemonSet vs Gateway vs Sidecar, Target Allocator, HPA, rollout consistency |
| Pipeline, Receiver, Processor, Exporter, Queue, Batch, Memory, Extensions, existing config | collector.md | Processor ordering, memory_limiter, file_storage, config audit heuristics, temporality/state audits, stability levels |
| SDK, Instrumentation, Spans, Attributes, Semantic Conventions, Cardinality | instrumentation.md | Auto vs manual, SemConv, cardinality Rule of 100 |
| Sampling, Cost, Volume, Head Sampling, Tail Sampling, Probabilistic | sampling.md | Head/tail sampling, sticky sessions, sampling math |
| Security, PII, GDPR, Redaction, TLS, Authentication, Credentials | security.md | PII redaction, mTLS, RBAC, extension exposure risks |
| Monitor the collector, Health, Alerts, Self-monitoring, Collector metrics | monitoring.md | otelcol_* metrics, dashboards, alert rules |
| Lambda, Azure Functions, GCP Functions, Serverless, FaaS, Mobile, Browser | platforms.md | FaaS patterns, Lambda extension layer, client-side apps |
| OTTL, Transform, Transformation, Modify, Filter attributes, Parse, Extract | ottl.md | OTTL syntax, context types, built-in functions, error handling |
| Connector, spanmetrics, servicegraph, routing connector, failover connector | connectors.md | R.E.D. metrics, service graph, routing, failover, stickiness |
| Claude Code, Codex, Gemini CLI, Copilot, AI agent, coding agent, MCP | ai-agents.md | Agent OTel support matrix, unified collector config, GenAI SemConv |
| validate, dry-run, startup error, pipeline error, dropped data, queue full, recovery | validation.md | Config validation commands, live checks, symptom→cause→fix recovery guidance |
| playbook, production playbook, blog, 2025 blog, 2026 blog, real world | playbooks.md | Production patterns from opentelemetry.io blogs |
| anti-pattern, common mistake, what to avoid, pitfall | anti-patterns.md | Full annotated anti-pattern catalogue: pipeline, metrics, Kubernetes, AI agents, OTTL |
Production Baseline Configuration
Use this copy-paste-ready baseline unless the user wants something different:
extensions:
health_check:
endpoint: "0.0.0.0:13133"
file_storage/queue:
directory: /var/lib/otelcol/queue
timeout: 10s
compaction:
on_start: true
on_rebound: false
receivers:
otlp:
protocols:
grpc:
endpoint: "0.0.0.0:4317"
http:
endpoint: "0.0.0.0:4318"
processors:
memory_limiter:
check_interval: 1s
limit_percentage: 80
spike_limit_percentage: 20
batch:
timeout: 10s
send_batch_size: 1024
exporters:
otlp:
endpoint: "your-backend:4317"
sending_queue:
enabled: true
storage: file_storage/queue
num_consumers: 4
queue_size: 1024
retry_on_failure:
enabled: true
initial_interval: 1s
max_interval: 30s
max_elapsed_time: 300s
# otlphttp: # HTTP exporter — use when backend requires HTTP
# endpoint: "https://your-backend:4318"
# sending_queue: { enabled: true, storage: file_storage/queue }
service:
extensions: [health_check, file_storage/queue]
pipelines:
traces:
receivers: [otlp]
processors: [memory_limiter, batch]
exporters: [otlp]
metrics:
receivers: [otlp]
processors: [memory_limiter, batch]
exporters: [otlp]
logs:
receivers: [otlp]
processors: [memory_limiter, batch]
exporters: [otlp]
Key defaults:
memory_limitermust be first in every processor chain.batchreduces exporter network calls.file_storagepreserves queues across restarts only on the same host/volume. In Kubernetes, back/var/lib/otelcol/queuewith aReadWriteOnceblock-backed PVC, not RWX/network storage.health_checkbinds tolocalhost(not0.0.0.0) in shared networks.- Prefer OTLP gRPC (port 4317) for receivers and exporters. Fall back to OTLP HTTP (port 4318) when gRPC is unavailable.
Validation & Error Recovery
Always include at least one validation checkpoint: otelcol validate --config <path>.
For container dry-run commands, live pipeline checks, and symptom→cause→fix recovery, load validation.md.
Anti-Patterns to Avoid
❌ Placing memory_limiter anywhere except first in the processor chain
❌ Using high-cardinality attributes (user_id, trace_id) as metric dimensions
❌ Exposing pprof (1777), zpages (55679) on 0.0.0.0 in production
❌ Using tail_sampling without sticky session load balancing (loadbalancing exporter)
❌ Omitting batch processor (causes excessive network calls)
❌ Calling a config "fine" because it parses, without checking memory limits, sticky routing, exporter durability, and rollout settings together
See anti-patterns.md for the full annotated catalog.
Version and Compatibility
- Use compatibility.md for fast-moving version floors and AI agent support details.
- Keep
SKILL.mdfocused on routing logic, guardrails, and production defaults rather than inline release tracking.