Usage Telemetry and API Lifecycle Governance
Apicurio Registry provides usage telemetry capabilities that track which client applications fetch which schema versions. This enables API lifecycle governance features including dead schema detection, consumer version drift alerts, and deprecation readiness analysis.
Overview
Usage telemetry answers key governance questions:
-
Is anyone actually using this schema? Active/Stale/Dead classification based on actual fetch patterns.
-
Who uses which version? Consumer version heatmap showing adoption across client applications.
-
Is it safe to deprecate this version? Deprecation readiness reports listing affected consumers.
-
Are consumers falling behind? Version drift alerts when clients use outdated versions.
Usage tracking is server-side and zero-configuration for the client. The registry records usage events automatically when it serves schema content. Clients opt in by setting an HTTP header that identifies the application:
-
X-Registry-Client-Id— identifies the client application (e.g.,payments-svc). Must contain only letters, digits, dots, hyphens, and underscores. Maximum 256 characters. -
X-Registry-Operation— optional, the operation type:SERIALIZEorDESERIALIZE.
If no X-Registry-Client-Id header is present, the request is not tracked.
Both globalId and contentId based schema fetches are tracked. Clients configured with IdOption.contentId are fully supported.
Referenced schemas (shared types, base schemas) automatically inherit liveness from the schemas that reference them. If UserEvent v3 is Active and references SharedTypes v1, then SharedTypes v1 is also considered Active.
Usage telemetry is an experimental feature. It requires apicurio.features.experimental.enabled=true in addition to the telemetry-specific configuration below.
|
Configuration
Server Configuration
| Property | Type | Default | Description |
|---|---|---|---|
|
boolean |
|
Master switch for usage telemetry. Must be |
|
int |
|
A schema version fetched within this many days is classified as Active. |
|
int |
|
A schema version not fetched for this many days is classified as Stale. |
|
int |
|
A schema version not fetched for this many days (or never fetched) is classified as Dead. |
|
int |
|
Number of versions behind latest before a consumer is flagged with a drift alert. |
|
int |
|
Number of days to retain raw usage events. Older events are cleaned up automatically. |
|
duration |
|
How often the usage metrics are refreshed and old events are cleaned up. Uses Quarkus scheduler expression syntax (e.g., |
Client Configuration
| Property | Type | Default | Description |
|---|---|---|---|
|
string |
(empty) |
Identifier for this client application. When set, the SerDes SDK sends |
Example: Enabling Telemetry
Server (application.properties):
apicurio.features.experimental.enabled=true
apicurio.usage.telemetry.enabled=true
Usage telemetry is an experimental feature. You must enable experimental features first with apicurio.features.experimental.enabled=true.
|
Kafka SerDes Client:
apicurio.registry.url=http://registry:8080/apis/registry/v3
apicurio.registry.usage-telemetry.client-id=payments-svc
That’s it. No additional endpoints to call, no buffers to configure.
REST API Endpoints
All usage endpoints are under /admin/usage/ and require apicurio.usage.telemetry.enabled=true. Endpoints return HTTP 409 (Conflict) when telemetry is not enabled.
Get Usage Summary
GET /admin/usage/summary
Returns global Active/Stale/Dead counts across all tracked schema versions.
{
"active": 12,
"stale": 3,
"dead": 5
}
Get Artifact Usage Metrics
GET /admin/usage/artifacts/{groupId}/{artifactId}
Returns per-version usage metrics for a specific artifact, including fetch counts, client lists, and classification.
{
"groupId": "default",
"artifactId": "UserEvent",
"versions": [
{
"version": "3.0.0",
"globalId": 42,
"totalFetches": 1523,
"uniqueClients": 3,
"firstFetchedOn": 1710000000000,
"lastFetchedOn": 1714400000000,
"clients": ["payments-svc", "analytics", "fraud-detector"],
"classification": "ACTIVE"
}
]
}
Get Consumer Version Heatmap
GET /admin/usage/artifacts/{groupId}/{artifactId}/heatmap
Returns a consumer version adoption grid showing which clients use which versions, with version drift detection.
{
"groupId": "default",
"artifactId": "UserEvent",
"versions": ["1.0.0", "2.0.0", "3.0.0"],
"consumers": [
{
"clientId": "payments-svc",
"versions": {"3.0.0": 523},
"versionsBehind": 0,
"driftAlert": false
},
{
"clientId": "analytics",
"versions": {"1.0.0": 102},
"versionsBehind": 2,
"driftAlert": true
}
]
}
Get Deprecation Readiness
GET /admin/usage/artifacts/{groupId}/{artifactId}/versions/{version}/deprecation-readiness
Returns which consumers are actively using a specific version and whether it is safe to deprecate.
{
"groupId": "default",
"artifactId": "UserEvent",
"version": "1.0.0",
"activeConsumers": [
{
"clientId": "analytics",
"lastFetched": 1714400000000,
"fetchCount": 102
}
],
"safeToDeprecate": false
}
Deprecation Workflow
Apicurio Registry supports a version lifecycle state machine for controlled deprecation:
ENABLED ──────▶ DEPRECATED ──────▶ SUNSET ──────▶ (delete)
▲ │ │
│ ▼ │
└──────── DISABLED │
◀───────────────────────────────────┘
-
ENABLED — Version is active and available for use.
-
DEPRECATED — Version still works but consumers should migrate. A warning is logged on access.
-
SUNSET — Migration deadline has passed. The version will be removed soon. Requires transitioning through DEPRECATED first.
-
DISABLED — Version is blocked from use.
Transition a version to SUNSET via the API:
# First deprecate
curl -X PUT http://localhost:8080/apis/registry/v3/groups/default/artifacts/UserEvent/versions/1.0.0/state \
-H "Content-Type: application/json" \
-d '{"state":"DEPRECATED"}'
# Then sunset
curl -X PUT http://localhost:8080/apis/registry/v3/groups/default/artifacts/UserEvent/versions/1.0.0/state \
-H "Content-Type: application/json" \
-d '{"state":"SUNSET"}'
Use the deprecation readiness endpoint before transitioning to ensure no active consumers are affected.
UI Features
When usage telemetry is enabled, the registry UI displays:
-
Dashboard widget — Shows Active/Stale/Dead counts with colored labels (green/orange/red).
-
Version badges — Each version in the artifact versions table shows its usage classification badge.
-
Usage tab — The artifact detail page has a "Usage" tab with:
-
Consumer version heatmap table with color-coded fetch counts.
-
Version drift alerts for consumers using outdated versions.
-
Deprecation readiness panel (click a version column header to check).
-
-
SUNSET badge — Versions in the SUNSET state display a purple badge.
OpenTelemetry Metrics Export
When OpenTelemetry is enabled (quarkus.otel.sdk.disabled=false), usage classification counts are exported as metrics:
| Metric | Description |
|---|---|
|
Number of schema versions classified as Active. |
|
Number of schema versions classified as Stale. |
|
Number of schema versions classified as Dead. |
These gauges are updated periodically (configured via apicurio.usage.aggregation.every). When OTel is disabled (the default), no overhead.
Privacy and Data Collection
-
Opt-in only — Disabled by default on both server and client.
-
No message content — Only schema coordinates and client identifier are recorded.
-
No user identity — The
clientIdis a configured application name, not a user or personal identifier. -
Client-controlled — Each client application decides whether to set the
X-Registry-Client-Idheader. -
Zero overhead when disabled — No headers sent, no events recorded, no database writes.
-
Automatic cleanup — Events older than the retention period are deleted automatically.
Demo Script
A demo script is provided to populate realistic usage data for testing:
# Start the registry with telemetry enabled
./mvnw quarkus:dev -pl :apicurio-registry-app \
-Dapicurio.usage.telemetry.enabled=true
# In another terminal, run the demo
./examples/usage-telemetry-demo/populate-usage-data.sh
The script creates three artifacts (UserEvent, OrderEvent, PaymentEvent) with multiple versions and simulates usage from five different client applications by fetching schemas with X-Registry-Client-Id headers.
