KEP-2305: Dynamic Cardinality Enforcement
KEP-2305: Dynamic Cardinality Enforcement
- Release Signoff Checklist
- Summary
- Motivation
- Proposal
- Design Details
- Production Readiness Review Questionnaire
- Implementation History
- Drawbacks
- Alternatives
Release Signoff Checklist
Items marked with (R) are required prior to targeting to a milestone / release.
- (R) Enhancement issue in release milestone, which links to KEP dir in kubernetes/enhancements (not the initial KEP PR)
- (R) KEP approvers have approved the KEP status as
implementable - (R) Design details are appropriately documented
- (R) Test plan is in place, giving consideration to SIG Architecture and SIG Testing input
- (R) Graduation criteria is in place
- (R) Production readiness review completed
- (R) Production readiness review approved
- “Implementation History” section is up-to-date for milestone
- User-facing documentation has been created in kubernetes/website , for publication to kubernetes.io
- Supporting documentation—e.g., additional design documents, links to mailing list discussions/SIG meetings, relevant PRs/issues, release notes
Summary
Background for this KEP is that metrics with unbounded dimensions can cause memory issues in the components they instrument. Our proposed solution for this is we want to be able to dynamically configure an allowlist of label values for a metric at runtime.
Motivation
Having metrics turn into memory leaks is a problem, but what is even a bigger problem is when we can’t fix these issues without re-releasing the entire Kubernetes binary.
Historically we have approached these issues in various ways and were not consistent. A few approaches:
In another case, we opt to delete the entire metric because it basically can’t be used in a meaningfully correct way.
Recently we’ve opted to both (1) wholesale delete a metric label and (2) retroactively introduce a set of acceptable values for a metric label .
Fixing these issues is currently a manual process, both laborious and time-consuming. We don’t have a standard prescription for resolving this class of issue.
Goals
This KEP proposes a possible solution for this, with the primary goal being to enable binding a metric dimension to a known set of values in such a way that is not coupled to code releases of Kubernetes.
Non-Goals
We will expose the machinery and tools to bind a metric’s labels to a discrete set of values. The allowlist will be ingested via a new-added component metric flag.
It is not a goal to define the allowlist for each Kubernetes component metrics.
Proposal
We should be able to dynamically configure an allowlist of label values for a metric.
Risks and Mitigations
None, we are actually providing a mitigation to cardinality explosion by limiting the number of allowed values for a label.
Design Details
The simple solution to this problem would be for each metric added to keep the unbounded dimensions in mind and prevent it from happening. SIG instrumentation has already explicitly stated this in our instrumentation guidelines: which says that “one should know a comprehensive list of all possible values for a label at instrumentation time.” . The problem is more complicated. First, SIG Instrumentation doesn’t have a way to validate adherence to SIG instrumentation guidelines outside of reviewing PRs with instrumentation changes. Not only is this highly manual, and error-prone, we do not have a terrific existing procedure for ensuring SIG Instrumentation is tagged on relevant PRs. Even if we do have such a mechanism, it isn’t a fully sufficient solution because:
- metrics changes can be seemingly innocuous, even to the most diligent of code reviewers (i.e these issues are hard to catch)
- metrics cardinality issues exist latently; in other words, they’re all over the place in Kubernetes already (so even if we could prevent 100% of these occurrence from now on, that wouldn’t guarantee Kubernetes is free of these classes of issues).
Instead, the proposed solution is we will be able to dynamically configure an allowlist of label values for a metric. By dynamically configure, we mean configure an allowlist at runtime rather than during the build/compile step. And by at runtime, we mean, more specifically, during the boot sequence for a Kubernetes component (and we mean daemons here, not CLI tools like kubectl).
Brief aside: a Kubernetes component (which is a daemon) is an executable, which can be launched from the command line manually if desired. Components take a number of start-up configuration flags, which are passed into the component to modify execution paths (if curious, check out the large amount of flags we have on the kubelet . Our design is going to add a flag that applies to all components to ingest the allowlist.
Our design is also based on the premise that metrics can be uniquely identified (i.e. by their metric descriptor). Fortunately for us, this is actually a built in constraint of prometheus clients (which is how we instrument the Kubernetes components). This metric ID is resolvable to a unique string (this is pretty evident when looking at a raw prometheus client endpoint).
We want to provide a data structure which basically maps a unique metric ID and one of it’s labels to a bounded set of values.
Purely for illustration, take the following metric:
[
{
"metricFamily": "some_metric"
}
]
We want to be able to express this metric and target a label:
[
{
"metricFamily": "some_metric",
"label": "label_too_many_values",
}
]
And we want to express a set of expected values:
[
{
"metricFamily": "some_metric",
"label": "label_too_many_values",
"labelValueAllowlist": [
"1",
"2",
"3"
]
}
]
Since we already have an interception layer built into our Kubernetes monitoring stack (from the metrics stability effort), we can leverage existing wrappers to provide a global entrypoint for intercepting and enforcing rules for individual metrics.
As a result metric data will not be invalid, just data will be more coarse when the metric type is a counter. In the case of a gauge type metric the data will be invalid but the number of series would be bound. One exception to our treatment of labels will be histograms (specifically the label which denotes buckets). If we explicitly declare a whitelist of acceptable values for histogram buckets, then we will simply omit buckets which are not in that list, since we can preserve data fidelity for histogram buckets by bucket omission (for everything except the catch-all bucket).
Following is an example used to demonstrate the proposed solution.
Let’s say we have some client code somewhere which instantiates our metric from above:
SomeMetric = metrics.NewCounterVec(
&metrics.CounterOpts{
Name: "some_metric",
Help: "alalalala",
StabilityLevel: metrics.ALPHA,
},
[]string{"label_too_many_values"},
)
And some other code which increments our metric:
var label_value string
label_value = "1"
SomeMetric.WithLabelValues(label_value).Inc()
label_value = "2"
SomeMetric.WithLabelValues(label_value).Inc()
label_value = "3"
SomeMetric.WithLabelValues(label_value).Inc()
In this case we would expect our raw prometheus metric output to look like this:
# HELP some_metric alalalala
# TYPE some_metric counter
some_metric{label_too_many_values="1"} 1
some_metric{label_too_many_values="2"} 1
some_metric{label_too_many_values="3"} 1
This would not change. What would change is if we encounter label values outside our explicit allowlist of values. So if where to encounter this malicious piece of code:
var label_value string
for i=0;i<1000000;i++ {
label_value = strconv.Itoa(i+4) // bring us out of our expected range
SomeMetric.WithLabelValues(label_value).Inc()
}
Then in existing Kubernetes components, we would have a terrible memory leak, since we would effectively create a million metrics (one per unique label value). If we curled our metrics endpoint it would thus look something like this:
# HELP some_metric alalalala
# TYPE some_metric counter
some_metric{label_too_many_values="1"} 1
some_metric{label_too_many_values="2"} 1
some_metric{label_too_many_values="3"} 1
some_metric{label_too_many_values="4"} 1
... //
... //
... // zillion metrics here
some_metric{label_too_many_values="1000003"} 1
In total we would have more than a million of series for one metric, which is known to cause a memory problem.
With our cardinality enforcer in place, we would expect the output to look like this:
# HELP some_metric alalalala
# TYPE some_metric counter
some_metric{label_too_many_values="1"} 1
some_metric{label_too_many_values="2"} 1
some_metric{label_too_many_values="3"} 1
some_metric{label_too_many_values="unexpected"} 1000000
We can effect this change by registering a allowlist to our prometheus registries during the boot-up sequence. Since we intercept metric registration events and have a wrapper around each of the primitive prometheus metric types, we can effectively add some code that looks like this (disclaimer: the below is pseudocode for demonstration):
const (
unexpectedLabel = "unexpected"
)
func (v *CounterVec) WithLabelValues(lvs ...string) CounterMetric {
if !v.IsCreated() {
return noop // return no-op counter
}
// INJECTED CODE HERE
newLabels = make([]string, len(lvs))
for i, l := range lvs {
// do we have an allowlist on any of the labels for this metric?
if metricLabelAllowlist, ok := MetricsLabelAllowlist[BuildFQName(v.CounterOpts.Namespace, v.CounterOpts.Subsystem, v.CounterOpts.Name)]; ok {
// do we have an allowlist for this label on this metric?
if allowlist, ok := metricLabelAllowlist[v.originalLabels[i]]; ok {
if allowllist.Has(l) {
newLabels[i] = l
continue
}
}
}
newLabels[i] = unexpectedLabel
}
return v.CounterVec.WithLabelValues(newLabels...)
// END INJECTED CODE
}
This design allows us to optionally adopt the idea about simplifying the interface for component owners, who can then opt to just specify a metric and label pair without having to specify an allowlist. The good part of this idea is it simplifies how a component owner can implement our cardinality enforcing helpers without having to necessary plumb through complicated maps. This would make it considerably easier to feed this data in through the command line since it can be done like this:
$ kube-apiserver --accepted-metric-labels "some_metric=label_too_many_values"
This would then be interpreted by our machinery as this:
[
{
"metric-id": "some_metric",
"label": "label_too_many_values",
"labelValueAllowlist": []
}
]
Test Plan
[x] I/we understand the owners of the involved components may require updates to existing tests to make this code solid enough prior to committing the changes necessary to implement this enhancement.
Prerequisite testing updates
N/A
Unit tests
For Alpha, unit test to .
staging/src/k8s.io/component-base/metrics/counter_test.go:3/3/2021-verify that the metric label will be set to "unexpected" for counters if the metric encounters label values outside our explicit allowlist of valuesstaging/src/k8s.io/component-base/metrics/gauge_test.go:4/3/21-verify that the metric label will be set to "unexpected" for gauges if the metric encounters label values outside our explicit allowlist of valuesstaging/src/k8s.io/component-base/metrics/histogram_test.go:4/3/21-verify that the metric label will be set to "unexpected" for histograms if the metric encounters label values outside our explicit allowlist of valuesstaging/src/k8s.io/component-base/metrics/summary_test.go:4/3/21-verify that the metric label will be set to "unexpected" for summaries if the metric encounters label values outside our explicit allowlist of values
Integration tests
Unit tests suffice for the testing of this feature.
e2e tests
Unit tests suffice for the testing of this feature.
Graduation Criteria
Alpha
- Feature implemented behind a feature flag
- The allowlist of metrics can be configured via the exposed flag and the unit test is passed.
Beta
- The allowlist can be configured from a manifest.
GA
- Allow pattern-matching for labels in the allowlist.
Upgrade / Downgrade strategy
N/A
Version Skew Strategy
N/A
Production Readiness Review Questionnaire
Feature Enablement and Rollback
How can this feature be enabled / disabled in a live cluster?
- Feature gate (also fill in values in
kep.yaml)- Feature gate name: MetricCardinalityEnforcement
- Components depending on the feature gate: All components that emit metrics, i.e. (at the time of writing),
- cmd/kube-apiserver
- cmd/kube-controller-manager
- cmd/kubelet
- pkg/kubelet/metrics
- pkg/kubelet/prober
- pkg/kubelet/server
- pkg/proxy/metrics
- cmd/kube-scheduler
- pkg/volume/util
Does enabling the feature change any default behavior?
Any change of default behavior may be surprising to users or break existing automations, so be extremely careful here. Using this feature requires restarting the component with the allowlist flag enabled. Once enabled, the metric label will be set to “unexpected” if the metric encounters label values outside our explicit allowlist of values.
Can the feature be disabled once it has been enabled (i.e. can we roll back the enablement)?
Also set disable-supported to true or false in kep.yaml.
Describe the consequences on existing workloads (e.g., if this is a runtime
feature, can it break the existing applications?).
Yes, disabling the feature gate can revert it back to existing behavior
What happens if we reenable the feature if it was previously rolled back?
The enable-disable-enable process will not cause problem. But it may be problematic during the rolled back period with the unbounded metrics value. Note that metrics are a memory-only construct and do not persist, but re-generated across restarts.
Are there any tests for feature enablement/disablement?
Using unit tests to cover the combination cases w/wo feature and w/wo allowlist.
Rollout, Upgrade and Rollback Planning
How can a rollout or rollback fail? Can it impact already running workloads?
Try to be as paranoid as possible - e.g., what if some components will restart mid-rollout? Using this feature requires restarting the component with the allowlist flag enabled.
What specific metrics should inform a rollback?
None.
Were upgrade and rollback tested? Was the upgrade->downgrade->upgrade path tested?
Describe manual testing that was done and the outcomes. Longer term, we may want to require automated upgrade/rollback tests, but we are missing a bunch of machinery and tooling and can’t do that now. In alpha, we can do some manual tests on enable/disable allowlist flag and enable/disable feature gate.
Is the rollout accompanied by any deprecations and/or removals of features, APIs, fields of API types, flags, etc.?
A component metric flag for ingesting allowlist to be added.
Monitoring Requirements
How can an operator determine if the feature is in use by workloads?
The out-of-bound data will be recorded with label “unexpected” rather than the specific value.
How can someone using this feature know that it is working for their instance?
The out-of-bound data will be recorded with label “unexpected” rather than the specific value.
What are the reasonable SLOs (Service Level Objectives) for the enhancement?
None.
What are the SLIs (Service Level Indicators) an operator can use to determine the health of the service?
None.
Are there any missing metrics that would be useful to have to improve observability of this feature?
cardinality_enforcement_unexpected_categorizations_total: Increments whenever any metric falls into the “unexpected” case (i.e., goes out of the defined bounds).
Dependencies
Does this feature depend on any specific services running in the cluster?
No.
Scalability
Will enabling / using this feature result in any new API calls?
No.
Will enabling / using this feature result in introducing new API types?
No.
Will enabling / using this feature result in any new calls to the cloud provider?
No.
Will enabling / using this feature result in increasing size or count of the existing API objects?
No.
Will enabling / using this feature result in increasing time taken by any operations covered by existing SLIs/SLOs?
Checking metrics label against allowlist may increase the metric recording time.
Will enabling / using this feature result in non-negligible increase of resource usage (CPU, RAM, disk, IO, …) in any components?
No.
Can enabling / using this feature result in resource exhaustion of some node resources (PIDs, sockets, inodes, etc.)?**
No.
Troubleshooting
How does this feature react if the API server and/or etcd is unavailable?
No additional impact comparing to what already exists.
What are other known failure modes?
None.
What steps should be taken if SLOs are not being met to determine the problem?
None.
Implementation History
2020-04-15: KEP opened
2020-05-19: KEP marked implementable
2021-02-08: Graduate to alpha
2023-05-07: Graduate to beta
2024-01-20: Graduate to stable
Drawbacks
None have been identified.
Alternatives
Manually keep track of all the values a label can have while adding a new label to a metric. And have a rigorous review process for PRs that add new metrcs/metric labels. But this approach is manual and error prone.