KEP-2403: Extend kubelet pod resource assignment endpoint to return allocatable resources
Extend kubelet pod resource assignment endpoint to return allocatable resources
Table of Contents
- Release Signoff Checklist
- Summary
- Motivation
- Proposal
- Design Details
- Production Readiness Review Questionnaire
- Implementation History
- 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
- (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
- e2e Tests for all Beta API Operations (endpoints)
- (R) Ensure GA e2e tests meet requirements for Conformance Tests
- (R) Minimum Two Week Window for GA e2e tests to prove flake free
- (R) Graduation criteria is in place
- (R) all GA Endpoints must be hit by Conformance Tests
- (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
This document presents an addition to the kubelet pod resources endpoint (pod resources API) which allows third party consumers to learn about the compute device allocation, thus, alongside the existing pod resources API endpoint, properly evaluate the node capacity.
Motivation
Goals
- Enable node monitoring agents to know the allocatable compute resources on a node, thus properly calculate the node compute resource utilization.
Non-Goals
- Add new endpoint (like kubelet
/pods)
Proposal
User Stories
Node Feature Discovery
Enable the Node Feature Discovery to expose hardware topology information .
Topology aware scheduling
This interface can be used to track down allocated resources with information about the NUMA topology of the worker node in general way.
This interface can be used to the available resources on the worker node. The kubelet is the best source of information because it manages concrete resources assignment. The information can then be used in NUMA aware scheduling.
Combining the information reported by the List API, which pertains the current allocation, with the information reported by the GetAllocatableResources API, monitoring agent can reliably report the compute device
utilization and availability.
Risks and Mitigations
This API is read-only, which removes a large class of risks. The aspects that we consider below are as follows:
- What are the risks associated with the API service itself?
- What are the risks associated with the data itself?
| Risk | Impact | Mitigation |
|---|---|---|
| Too many requests risk impacting the kubelet performances | High | Implement rate limiting and or passive caching, follow best practices for gRPC resource management. |
| Improper access to the data | Low | Server is listening on a root owned unix socket. This can be limited with proper pod security policies. |
Design Details
Proposed API
We propose to extend the existing pod resources gRPC service of the Kubelet, listening on a unix socket at /var/lib/kubelet/pod-resources/kubelet.sock.
The GRPC Service will expose an additional endpoint:
- ‘GetAllocatableResources`, which returns a single AllocatableResourcesResponse, enabling monitor applications to query for the allocatable set of resources available on the node. This endpoint will return error if the corresponding feature gate is disabled.
NOTE:
GetAllocatableResourcesshould only be used to evaluate allocatable resources on a node. If the goal is to evaluate free/unallocated resources it should be used in conjunction with the List() endpoint. The result obtained byGetAllocatableResourceswould remain the same unless the underlying resources exposed to kubelet change. This happens rarely but when it does (e.g. CPUs onlined/offlined, devices added/removed), client is expected to callGetAlloctableResourcesendpoint.
The extended interface is shown in proto below:
// PodResources is a service provided by the kubelet that provides information about the
// node resources consumed by pods and containers on the node
service PodResources {
rpc List(ListPodResourcesRequest) returns (ListPodResourcesResponse) {}
rpc GetAllocatableResources(AllocatableResourcesRequest) returns (AllocatableResourcesResponse) {}
}
message AllocatableResourcesRequest {}
// AvailableResourcesResponses contains informations about all the devices known by the kubelet
message AllocatableResourcesResponse {
repeated ContainerDevices devices = 1;
repeated int64 cpu_ids = 2;
}
// ListPodResourcesRequest is the request made to the PodResources service
message ListPodResourcesRequest {}
// ListPodResourcesResponse is the response returned by List function
message ListPodResourcesResponse {
repeated PodResources pod_resources = 1;
}
// PodResources contains information about the node resources assigned to a pod
message PodResources {
string name = 1;
string namespace = 2;
repeated ContainerResources containers = 3;
}
// ContainerResources contains information about the resources assigned to a container
message ContainerResources {
string name = 1;
repeated ContainerDevices devices = 2;
repeated int64 cpu_ids = 3;
repeated ContainerMemory memory = 4;
}
// Topology describes hardware topology of the resource
message TopologyInfo {
repeated NUMANode nodes = 1;
}
// NUMA representation of NUMA node
message NUMANode {
int64 ID = 1;
}
// ContainerDevices contains information about the devices assigned to a container
message ContainerDevices {
string resource_name = 1;
repeated string device_ids = 2;
TopologyInfo topology = 3;
}
Test Plan
The implementation PR adds a suite of E2E tests which cover both the existing List endpoint already implemented in the podresources API and
the new proposed GetAllocatableResources API.
Add additional tests to prove that unhealthy devices are skipped as part of GetAllocatable and empty NUMA topology is not returned.
[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
Unit tests
k8s.io/kubernetes/pkg/kubelet/api/podresources:20230530-68.6%
Integration tests
N/A - node local feature covered by e2e test (test/e2e_node)
e2e tests
NodeFeature:PodResources: https://storage.googleapis.com/k8s-triage/index.html?sig=node&test=NodeFeature%3APodResources
Graduation Criteria
Alpha
- Implement the new service API.
- Ensure proper e2e node tests are in place.
Alpha to Beta Graduation
- The new API is consumed by other public software components (e.g. NFD).
- No major bugs reported in the previous cycle.
- Ensure that empty NUMA topology is handled properly.
- Ensure that unhealthy devices are skipped in GetAllocatable.
- External clients are using this capability in their solutions
Topology aware Scheduling is one of the primary use cases of GetAllocatableResource podresource endpoint. As part of this initiative an exporter populates CRs per node to expose the information of resources available per NUMA. Pod Resource API
ListandGetAllocatableResourcesAPI endpoints are used to obtain resource allocation of running pods along with the underlying hardware topology (NUMA) information. Topology aware scheduler can be configured such that users can create custom exporters or use already existing exporters to expose the NodeResourceTopology information as CRs and then Topology aware Scheduler uses this information to make a NUMA aware placement decision leading to the reduction of occurrence of Topology affinity Errors highlighted in the issue here . Examples of two such exporters are:- Node feature Discovery for exposing resource topology information as part of the initiative here: Introducing NFD Topology Updater exposing Resource hardware Topology info through CRs .
- Resource Topology Exporter
Beta to G.A Graduation
- Allowing time for feedback (1 year).
- Risks have been addressed.
- Rate limiting implemented as part of the podresources endpoint GA graduation (KEP 606).
Deprecation
Upgrade / Downgrade Strategy
With gRPC the version is part of the service name. Old versions and new versions should always be served and listened by the kubelet.
To a cluster admin upgrading to the newest API version, means upgrading Kubernetes to a newer version as well as upgrading the monitoring component.
To a vendor changes in the API should always be backwards compatible.
Version Skew Strategy
Kubelet will always be backwards compatible, so going forward existing plugins are not expected to break.
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:
KubeletPodResourcesGetAllocatable. - Components depending on the feature gate: kubelet, 3rd party consumers.
- Feature gate name:
Does enabling the feature change any default behavior?
No
Can the feature be disabled once it has been enabled (i.e. can we roll back the enablement)?
Yes, through feature gate. Once GA, the feature can’t be disabled and is always enabled.
What happens if we reenable the feature if it was previously rolled back?
The API becomes available again. The API is stateless, so no recovery is needed, clients can just consume the data.
Are there any tests for feature enablement/disablement?
An e2e test will demonstrate that when the feature gate is disabled, the API returns the appropriate error code.
Rollout, Upgrade and Rollback Planning
How can a rollout or rollback fail? Can it impact already running workloads?
Kubelet may fail to start. The new API may report inconsistent data, or may cause the kubelet to crash.
What specific metrics should inform a rollback?
pod_resources_endpoint_errors_get_allocatable - but only with feature gate enabled. Otherwise the API will always return a known error, giving a false negative signal.
Were upgrade and rollback tested? Was the upgrade->downgrade->upgrade path tested?
Not Applicable.
Is the rollout accompanied by any deprecations and/or removals of features, APIs, fields of API types, flags, etc.?
No (Not applicable)
Monitoring Requirements
How can an operator determine if the feature is in use by workloads?
- Look at the
pod_resources_endpoint_requests_get_allocatablemetric exposed by the kubelet. - Clients are connected to the podresources unix socket, for example by checking which containers mount the podresources socket path.
How can someone using this feature know that it is working for their instance?
- Events
- Event Reason:
- API .status
- Condition name:
- Other field:
- Other (treat as last resort)
- Look at the
pod_resources_endpoint_requests_get_allocatableandpod_resources_endpoint_errors_get_allocatablemetrics exposed by the kubelet.
- Look at the
What are the reasonable SLOs (Service Level Objectives) for the enhancement?
100% in normal operation. The proposed API exposes in read only mode kubelet internal data, critical for functioning of the kubelet. This data has to be available 100% of the time for the proper functioning of the kubelet, thus is expected to be available 100% of time. The only possible error source is the API calls being throttled by the rate-limiting introduced with the GA graduation of the parent KEP 606.
What are the SLIs (Service Level Indicators) an operator can use to determine the health of the service?
- Metrics
- Metric name:
pod_resources_endpoint_requests_get_allocatablepod_resources_endpoint_errors_get_allocatable
- Components exposing the metric: kubelet
- Metric name:
Are there any missing metrics that would be useful to have to improve observability of this feature?
As part of this feature enhancement, per-API-endpoint resources metrics are being added; to observe this feature the pod_resources_endpoint_requests_get_allocatable metric should be used.
We added the pod_resources_endpoint_errors_get_allocatable metric to report errors. Because the nature of the API (exposing data already used by the kubelet with minimal processing)
the error counter is expected to be stable zero.
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?
No. The feature is not affecting hot code paths in the kubelet, and just give access to cached data already computed by the kubelet for internal bookkeeping.
Will enabling / using this feature result in non-negligible increase of resource usage (CPU, RAM, disk, IO, …) in any components?
Negligible amount of CPU and memory, because the endpoint queries existing data structures inside the kubelet.
Can enabling / using this feature result in resource exhaustion of some node resources (PIDs, sockets, inodes, etc.)?
No, because the endpoint queries existing data structures inside the kubelet.
Troubleshooting
How does this feature react if the API server and/or etcd is unavailable?
No impact, the feature is node-local
What are other known failure modes?
feature gate disabled: the API will always return a well-known error. In normal operation, the API is expected to never return error and always return a valid response, because it utilizes internal kubelet data which is always available. Bugs may lead to the API to return unexpected errors, or to return inconsistent data. Consumers of the API should treat unexpected errors as bugs of this API.
What steps should be taken if SLOs are not being met to determine the problem?
Check the error code to learn if the consumer of the API is being throttle by rate limiting introduced in the parent KEP 606. Check the kubelet logs to learn about resource allocation errors.
Implementation History
- 2021-02-02: KEP extracted from previous iteration
- 2021-02-04: KEP polished, added feature gate, clarified the graduation criteria.
- 2021-02-08: KEP updated adding per-specific-endpoint metrics to the podresources API and clarifying failure modes.
- 2021-09-02: KEP updated to explicitly clarify the behavior of
GetAllocatableResourcesand graduate to Beta in 1.23. - 2021-05-30: KEP updated to the new template and to graduate to GA in 1.28
Alternatives
Add a new endpoint
- Pros:
- No changes to existing APIs
- Cons:
- Requires the client to consume two APIs
- This work nicely fits in the boundaries and purpose of the podresources API
- The changes proposed in this KEP are very low-risk and backward compatible