OpenTelemetry Collector deployment, instrumentation (Java/Python/Node.js/.NET/Go), and OTTL pipeline transforms for Coralogix — coralogix exporter config, Helm chart selection, Kubernetes topology, ECS/EKS/GKE deployments, SDK setup, APM transactions, and OTTL cardinality/PII/routing.
98
97%
Does it follow best practices?
Impact
99%
1.13xAverage score across 81 eval scenarios
Advisory
Suggest reviewing before use
OpenTelemetry Transformation Language (OTTL) transforms, filters, and routes telemetry inside an OTel Collector pipeline without modifying application code. Load this skill when writing or debugging OTTL statements in the transform processor, filter processor, or routing connector.
| Use case | What to do |
|---|---|
| Change values or fields conditionally | transform processor with the correct context |
| Drop telemetry entirely (match = drop) | filter processor |
| Set static resource attributes everywhere | resource processor — simpler than OTTL |
| Copy a resource field down to spans or logs | transform with the correct context |
| Route telemetry to different pipelines | routing connector |
| Reduce metric or trace cardinality | transform with keep_keys / delete_matching_keys → references/cardinality.md |
| Extract histogram _sum/_count into standalone metrics, then drop the histogram | transform with extract_sum_metric / extract_count_metric in context: metric, then filter to drop the original |
| Redact or pseudonymize PII | transform with SHA256 / replace_all_patterns → references/redaction.md |
| Debug no data, DNS, receiver/exporter issues, or pipeline wiring | Not an OTTL problem — say so before going further |
Reference material for each topic lives under references/ and is listed in the References footer at the bottom of this file. When the question involves a specific processor, context, or function you have not recently reviewed — or when the user pastes a Collector error — consult the matching reference file before answering. One or two targeted reads beat guessing from memory.
context: determines what attributes means. In context: resource, attributes["k"] is a resource attribute. In context: log or context: span, attributes["k"] is the record-level attribute — use resource.attributes["k"] to reach the resource. Wrong context = silent nil, no error. → references/contexts.md
| Signal | Valid contexts |
|---|---|
| logs | resource, scope, log |
| traces | resource, scope, span, spanevent |
| metrics | resource, scope, metric, datapoint |
Metric-level edits (name, description, unit) belong in context: metric; per-series label/attribute edits belong in context: datapoint. Mixing the two in one block means one of them silently no-ops. → references/contexts.md
Log event timestamps can be changed in context: log by setting time with Time(...)
or time_unix_nano with an integer nanosecond value. Span status checks belong in
context: span; prefer STATUS_CODE_ERROR, STATUS_CODE_OK, and STATUS_CODE_UNSET
over raw numeric comparisons. → references/transformations.md
| Mode | Behavior |
|---|---|
propagate (default) | Any OTTL runtime error halts the pipeline — causes data loss |
ignore | Log the error, skip the statement, continue the pipeline — use in production |
silent | Skip the statement and suppress error logging |
Set error_mode explicitly on every transform and filter processor — the default
(propagate) halts the whole pipeline on the first runtime error (missing optional
field, wrong type, indexing a nil), silently dropping every subsequent record. The fix
for a pipeline that "goes silent after one failure" is almost always the missing
error_mode key:
processors:
transform:
error_mode: ignore # log, skip the statement, continue the pipeline
log_statements:
- context: log
statements:
- set(attributes["env"], resource.attributes["deployment.environment"])Map the Collector message to a root cause before proposing a fix:
| Collector message | Root cause | First action |
|---|---|---|
INVALID_ARGUMENT | Type mismatch, invalid function input, or invalid path for the active context | Add nil/type guards; confirm active context |
... cannot be indexed | Indexing a non-map value — string or empty body | Add IsMap(body) guard before body indexing |
segment "..." is not a valid path | Wrong context or field not available in the chosen context | Switch to the correct context; check path reference |
one or more paths were modified to include their context prefix | Bare attributes[...] where explicit prefixes are required | Rewrite with resource.attributes, datapoint.attributes, etc. |
statement has invalid syntax: ... invalid quoted string | YAML + OTTL quoting collision — the string was consumed by the YAML parser before reaching OTTL | Use YAML single quotes outside and OTTL double quotes inside → references/processors.md |
| Statement loads but has no visible effect | Condition never matches, wrong signal block, or processor in the wrong pipeline stage | Surface debug attributes to prove matching; verify pipeline placement |
Full annotated example of a transform + filter pipeline (log/trace/metric
statements, error_mode, conditions: [IsMap(body)], filter-before-transform
ordering) lives in references/processors.md.
IsMap, IsString, != nil).attributes vs resource.attributes).conditions: semantics or tail sampling policy ordering.When the body arrives as a raw JSON string (IsMap(body) is false), guard with
IsString(body) and parse with ParseJSON. Prefer IsString(body) over
not IsMap(body) — it is the affirmative check and avoids matching empty/nil
bodies.
- context: log
conditions:
- IsString(body) # body is a JSON string (affirmative guard)
statements:
# Promote every top-level JSON field into attributes
- merge_maps(attributes, ParseJSON(body), "insert")
# Or lift specific fields only:
- set(attributes["user_id"], ParseJSON(body)["user_id"]) where ParseJSON(body)["user_id"] != nil
- set(attributes["request_id"], ParseJSON(body)["request_id"]) where ParseJSON(body)["request_id"] != nilParseJSON is a Converter — it returns a value but has no side effect of its own,
so it must be wrapped in an Editor (set, merge_maps). A standalone
ParseJSON(body) line loads without errors and does nothing. → references/transformations.md
- context: datapoint
statements:
- keep_keys(attributes, ["service.name", "http.route", "http.response.status_code"])
- delete_matching_keys(attributes, "^k8s\\.pod\\.uid$")Prefer keep_keys (allowlist) over many delete_key calls (blocklist). → references/cardinality.md
Exporters that pick a destination from attributes — most notably the Coralogix exporter's
application_name_attributes and subsystem_name_attributes — read from resource
attributes, not log-record or span attributes. If the source value lives on the record,
copy it up to the resource from context: log (or context: span) with
set(resource.attributes[...], attributes[...]). Don't rename the field in the
application and don't change the exporter config.
Full pattern (both attribute pairs, error_mode, and pipeline ordering) is in
→ references/contexts.md.
To keep _sum and _count for average latency calculations while dropping raw bucket data:
processors:
transform:
error_mode: ignore
metric_statements:
- context: metric
conditions:
- type == METRIC_DATA_TYPE_HISTOGRAM
statements:
- extract_sum_metric(true) # creates <name>_sum as a new Sum metric; true = monotonic
- extract_count_metric(true) # creates <name>_count as a new Counter metric
filter:
error_mode: ignore
metrics:
metric:
- 'type == METRIC_DATA_TYPE_HISTOGRAM and name == "http.server.request.duration"'extract_sum_metric(monotonic) and extract_count_metric(monotonic) are OTTL Editor functions that run in context: metric and append new standalone metrics to the pipeline output — the original histogram is still present until the filter processor drops it. The transform processor must come before filter in the pipeline so the new metrics exist before the histogram is removed. Pass true for cumulative/monotonic counters, false for delta.
- context: log
statements:
- set(attributes["user.id"], SHA256(attributes["user.id"])) where attributes["user.id"] != nil
- replace_all_patterns(attributes, "value", "(?i)bearer\\s+[a-z0-9._-]+", "bearer ***")SHA256 preserves correlation without exposing the raw identifier. replace_all_patterns redacts across every attribute value without listing each key. → references/redaction.md
error_mode: ignore explicitly. The default propagate causes data loss on any runtime error.conditions: to scope a block. Statements run when any listed condition matches (OR semantics) — cheaper than a where clause on every statement. For strict AND, combine with a and b or use per-statement where.IsMap(body) before map access, IsString(body)
before string operations, where attributes["x"] != nil before reading
optional fields. An unguarded indexing into the wrong type raises
INVALID_ARGUMENT and (with the default error_mode) halts the pipeline."5" or 5, branch with IsString / IsInt and use Int(...) or
Double(...) before > / < comparisons.conditions: is OR, not AND. For strict AND, combine into one boolean
(a and b) or add where on each statement.keep_keys over many delete_key calls. An allowlist is shorter and self-documenting.OTTL is not the fix for: receiver connectivity, exporter connectivity, DNS or Kubernetes service discovery, load balancing, gateway reachability, pipeline wiring mistakes, or telemetry that never reaches the processor. Say so before going further.
Scope fences:
The OTTL function list evolves each Collector release. This skill covers patterns and common misuses — for current function signatures, consult the upstream docs in the References footer below.
transform, filter, and routing connector configurationParseJSON, fallback chainskeep_keys, delete_matching_keys, replace_patternSHA256 pseudonymization, token/card/auth header redactionUpstream:
evals
scenario-1
scenario-2
scenario-3
scenario-4
scenario-5
scenario-6
scenario-7
scenario-8
scenario-9
scenario-10
scenario-11
scenario-12
scenario-13
scenario-14
scenario-15
scenario-16
scenario-17
scenario-18
scenario-19
scenario-20
scenario-21
scenario-22
scenario-23
scenario-24
scenario-25
scenario-26
scenario-27
scenario-28
scenario-29
scenario-30
scenario-31
scenario-32
scenario-33
scenario-34
scenario-35
scenario-36
scenario-37
scenario-38
scenario-39
scenario-40
scenario-41
scenario-42
scenario-43
scenario-44
scenario-45
scenario-46
scenario-47
scenario-48
scenario-49
scenario-50
scenario-51
scenario-52
scenario-53
scenario-54
scenario-55
scenario-56
scenario-57
scenario-58
scenario-59
scenario-60
scenario-61
scenario-62
scenario-63
scenario-64
scenario-65
scenario-66
scenario-67
scenario-68
scenario-69
scenario-70
scenario-71
scenario-72
scenario-73
scenario-74
scenario-75
scenario-76
scenario-77
scenario-78
scenario-79
scenario-80
scenario-81
skills
opentelemetry
opentelemetry-collector
references
opentelemetry-instrumentation
opentelemetry-ottl