2019-01-12 04:58:27 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016 The Kubernetes Authors.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package stats
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
2019-01-22 20:53:35 +00:00
|
|
|
"k8s.io/klog"
|
|
|
|
|
2019-03-29 00:03:05 +00:00
|
|
|
rsystem "github.com/opencontainers/runc/libcontainer/system"
|
2019-01-22 20:53:35 +00:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2019-03-29 00:03:05 +00:00
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
statsapi "k8s.io/kubernetes/pkg/kubelet/apis/stats/v1alpha1"
|
2019-01-22 20:53:35 +00:00
|
|
|
"k8s.io/kubernetes/pkg/kubelet/util"
|
2019-01-12 04:58:27 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type SummaryProvider interface {
|
|
|
|
// Get provides a new Summary with the stats from Kubelet,
|
|
|
|
// and will update some stats if updateStats is true
|
|
|
|
Get(updateStats bool) (*statsapi.Summary, error)
|
|
|
|
// GetCPUAndMemoryStats provides a new Summary with the CPU and memory stats from Kubelet,
|
|
|
|
GetCPUAndMemoryStats() (*statsapi.Summary, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// summaryProviderImpl implements the SummaryProvider interface.
|
|
|
|
type summaryProviderImpl struct {
|
2019-01-22 20:53:35 +00:00
|
|
|
// kubeletCreationTime is the time at which the summaryProvider was created.
|
|
|
|
kubeletCreationTime metav1.Time
|
|
|
|
// systemBootTime is the time at which the system was started
|
|
|
|
systemBootTime metav1.Time
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
provider StatsProvider
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ SummaryProvider = &summaryProviderImpl{}
|
|
|
|
|
|
|
|
// NewSummaryProvider returns a SummaryProvider using the stats provided by the
|
|
|
|
// specified statsProvider.
|
|
|
|
func NewSummaryProvider(statsProvider StatsProvider) SummaryProvider {
|
2019-01-22 20:53:35 +00:00
|
|
|
kubeletCreationTime := metav1.Now()
|
|
|
|
bootTime, err := util.GetBootTime()
|
|
|
|
if err != nil {
|
|
|
|
// bootTime will be zero if we encounter an error getting the boot time.
|
|
|
|
klog.Warningf("Error getting system boot time. Node metrics will have an incorrect start time: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &summaryProviderImpl{
|
|
|
|
kubeletCreationTime: kubeletCreationTime,
|
|
|
|
systemBootTime: metav1.NewTime(bootTime),
|
|
|
|
provider: statsProvider,
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *summaryProviderImpl) Get(updateStats bool) (*statsapi.Summary, error) {
|
|
|
|
// TODO(timstclair): Consider returning a best-effort response if any of
|
|
|
|
// the following errors occur.
|
|
|
|
node, err := sp.provider.GetNode()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get node info: %v", err)
|
|
|
|
}
|
|
|
|
nodeConfig := sp.provider.GetNodeConfig()
|
|
|
|
rootStats, networkStats, err := sp.provider.GetCgroupStats("/", updateStats)
|
|
|
|
if err != nil {
|
2019-03-29 00:03:05 +00:00
|
|
|
if !rsystem.RunningInUserNS() {
|
|
|
|
return nil, fmt.Errorf("failed to get root cgroup stats: %v", err)
|
|
|
|
}
|
|
|
|
// if we are in userns, cgroups might not be available
|
|
|
|
klog.Errorf("failed to get root cgroup stats: %v", err)
|
|
|
|
rootStats = &statsapi.ContainerStats{}
|
|
|
|
networkStats = &statsapi.NetworkStats{}
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
rootFsStats, err := sp.provider.RootFsStats()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get rootFs stats: %v", err)
|
|
|
|
}
|
|
|
|
imageFsStats, err := sp.provider.ImageFsStats()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get imageFs stats: %v", err)
|
|
|
|
}
|
|
|
|
podStats, err := sp.provider.ListPodStats()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to list pod stats: %v", err)
|
|
|
|
}
|
|
|
|
rlimit, err := sp.provider.RlimitStats()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get rlimit stats: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
nodeStats := statsapi.NodeStats{
|
|
|
|
NodeName: node.Name,
|
|
|
|
CPU: rootStats.CPU,
|
|
|
|
Memory: rootStats.Memory,
|
|
|
|
Network: networkStats,
|
2019-01-22 20:53:35 +00:00
|
|
|
StartTime: sp.systemBootTime,
|
2019-01-12 04:58:27 +00:00
|
|
|
Fs: rootFsStats,
|
|
|
|
Runtime: &statsapi.RuntimeStats{ImageFs: imageFsStats},
|
|
|
|
Rlimit: rlimit,
|
|
|
|
SystemContainers: sp.GetSystemContainersStats(nodeConfig, podStats, updateStats),
|
|
|
|
}
|
|
|
|
summary := statsapi.Summary{
|
|
|
|
Node: nodeStats,
|
|
|
|
Pods: podStats,
|
|
|
|
}
|
|
|
|
return &summary, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sp *summaryProviderImpl) GetCPUAndMemoryStats() (*statsapi.Summary, error) {
|
|
|
|
// TODO(timstclair): Consider returning a best-effort response if any of
|
|
|
|
// the following errors occur.
|
|
|
|
node, err := sp.provider.GetNode()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get node info: %v", err)
|
|
|
|
}
|
|
|
|
nodeConfig := sp.provider.GetNodeConfig()
|
|
|
|
rootStats, err := sp.provider.GetCgroupCPUAndMemoryStats("/", false)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get root cgroup stats: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
podStats, err := sp.provider.ListPodCPUAndMemoryStats()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to list pod stats: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
nodeStats := statsapi.NodeStats{
|
|
|
|
NodeName: node.Name,
|
|
|
|
CPU: rootStats.CPU,
|
|
|
|
Memory: rootStats.Memory,
|
|
|
|
StartTime: rootStats.StartTime,
|
|
|
|
SystemContainers: sp.GetSystemContainersCPUAndMemoryStats(nodeConfig, podStats, false),
|
|
|
|
}
|
|
|
|
summary := statsapi.Summary{
|
|
|
|
Node: nodeStats,
|
|
|
|
Pods: podStats,
|
|
|
|
}
|
|
|
|
return &summary, nil
|
|
|
|
}
|