mirror of https://github.com/k3s-io/k3s
320 lines
9.1 KiB
Go
320 lines
9.1 KiB
Go
|
// Copyright 2017 Google Inc. All Rights Reserved.
|
||
|
//
|
||
|
// 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.
|
||
|
|
||
|
// Handler for CRI-O containers.
|
||
|
package crio
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"path"
|
||
|
"path/filepath"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/google/cadvisor/container"
|
||
|
"github.com/google/cadvisor/container/common"
|
||
|
containerlibcontainer "github.com/google/cadvisor/container/libcontainer"
|
||
|
"github.com/google/cadvisor/fs"
|
||
|
info "github.com/google/cadvisor/info/v1"
|
||
|
|
||
|
cgroupfs "github.com/opencontainers/runc/libcontainer/cgroups/fs"
|
||
|
libcontainerconfigs "github.com/opencontainers/runc/libcontainer/configs"
|
||
|
)
|
||
|
|
||
|
type crioContainerHandler struct {
|
||
|
machineInfoFactory info.MachineInfoFactory
|
||
|
|
||
|
// Absolute path to the cgroup hierarchies of this container.
|
||
|
// (e.g.: "cpu" -> "/sys/fs/cgroup/cpu/test")
|
||
|
cgroupPaths map[string]string
|
||
|
|
||
|
// the CRI-O storage driver
|
||
|
storageDriver storageDriver
|
||
|
fsInfo fs.FsInfo
|
||
|
rootfsStorageDir string
|
||
|
|
||
|
// Metadata associated with the container.
|
||
|
envs map[string]string
|
||
|
labels map[string]string
|
||
|
|
||
|
// TODO
|
||
|
// crio version handling...
|
||
|
|
||
|
// Image name used for this container.
|
||
|
image string
|
||
|
|
||
|
// The network mode of the container
|
||
|
// TODO
|
||
|
|
||
|
// Filesystem handler.
|
||
|
fsHandler common.FsHandler
|
||
|
|
||
|
// The IP address of the container
|
||
|
ipAddress string
|
||
|
|
||
|
includedMetrics container.MetricSet
|
||
|
|
||
|
reference info.ContainerReference
|
||
|
|
||
|
libcontainerHandler *containerlibcontainer.Handler
|
||
|
}
|
||
|
|
||
|
var _ container.ContainerHandler = &crioContainerHandler{}
|
||
|
|
||
|
// newCrioContainerHandler returns a new container.ContainerHandler
|
||
|
func newCrioContainerHandler(
|
||
|
client crioClient,
|
||
|
name string,
|
||
|
machineInfoFactory info.MachineInfoFactory,
|
||
|
fsInfo fs.FsInfo,
|
||
|
storageDriver storageDriver,
|
||
|
storageDir string,
|
||
|
cgroupSubsystems *containerlibcontainer.CgroupSubsystems,
|
||
|
inHostNamespace bool,
|
||
|
metadataEnvs []string,
|
||
|
includedMetrics container.MetricSet,
|
||
|
) (container.ContainerHandler, error) {
|
||
|
// Create the cgroup paths.
|
||
|
cgroupPaths := common.MakeCgroupPaths(cgroupSubsystems.MountPoints, name)
|
||
|
|
||
|
// Generate the equivalent cgroup manager for this container.
|
||
|
cgroupManager := &cgroupfs.Manager{
|
||
|
Cgroups: &libcontainerconfigs.Cgroup{
|
||
|
Name: name,
|
||
|
},
|
||
|
Paths: cgroupPaths,
|
||
|
}
|
||
|
|
||
|
rootFs := "/"
|
||
|
if !inHostNamespace {
|
||
|
rootFs = "/rootfs"
|
||
|
storageDir = path.Join(rootFs, storageDir)
|
||
|
}
|
||
|
|
||
|
id := ContainerNameToCrioId(name)
|
||
|
|
||
|
cInfo, err := client.ContainerInfo(id)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
// passed to fs handler below ...
|
||
|
// XXX: this is using the full container logpath, as constructed by the CRI
|
||
|
// /var/log/pods/<pod_uuid>/container_instance.log
|
||
|
// It's not actually a log dir, as the CRI doesn't have per-container dirs
|
||
|
// under /var/log/pods/<pod_uuid>/
|
||
|
// We can't use /var/log/pods/<pod_uuid>/ to count per-container log usage.
|
||
|
// We use the container log file directly.
|
||
|
storageLogDir := cInfo.LogPath
|
||
|
|
||
|
// Determine the rootfs storage dir
|
||
|
rootfsStorageDir := cInfo.Root
|
||
|
// TODO(runcom): CRI-O doesn't strip /merged but we need to in order to
|
||
|
// get device ID from root, otherwise, it's going to error out as overlay
|
||
|
// mounts doesn't have fixed dev ids.
|
||
|
rootfsStorageDir = strings.TrimSuffix(rootfsStorageDir, "/merged")
|
||
|
switch storageDriver {
|
||
|
case overlayStorageDriver, overlay2StorageDriver:
|
||
|
// overlay and overlay2 driver are the same "overlay2" driver so treat
|
||
|
// them the same.
|
||
|
rootfsStorageDir = filepath.Join(rootfsStorageDir, "diff")
|
||
|
}
|
||
|
|
||
|
containerReference := info.ContainerReference{
|
||
|
Id: id,
|
||
|
Name: name,
|
||
|
Aliases: []string{cInfo.Name, id},
|
||
|
Namespace: CrioNamespace,
|
||
|
}
|
||
|
|
||
|
libcontainerHandler := containerlibcontainer.NewHandler(cgroupManager, rootFs, cInfo.Pid, includedMetrics)
|
||
|
|
||
|
// TODO: extract object mother method
|
||
|
handler := &crioContainerHandler{
|
||
|
machineInfoFactory: machineInfoFactory,
|
||
|
cgroupPaths: cgroupPaths,
|
||
|
storageDriver: storageDriver,
|
||
|
fsInfo: fsInfo,
|
||
|
rootfsStorageDir: rootfsStorageDir,
|
||
|
envs: make(map[string]string),
|
||
|
labels: cInfo.Labels,
|
||
|
includedMetrics: includedMetrics,
|
||
|
reference: containerReference,
|
||
|
libcontainerHandler: libcontainerHandler,
|
||
|
}
|
||
|
|
||
|
handler.image = cInfo.Image
|
||
|
// TODO: we wantd to know graph driver DeviceId (dont think this is needed now)
|
||
|
|
||
|
// ignore err and get zero as default, this happens with sandboxes, not sure why...
|
||
|
// kube isn't sending restart count in labels for sandboxes.
|
||
|
restartCount, _ := strconv.Atoi(cInfo.Annotations["io.kubernetes.container.restartCount"])
|
||
|
// Only adds restartcount label if it's greater than 0
|
||
|
if restartCount > 0 {
|
||
|
handler.labels["restartcount"] = strconv.Itoa(restartCount)
|
||
|
}
|
||
|
|
||
|
handler.ipAddress = cInfo.IP
|
||
|
|
||
|
// we optionally collect disk usage metrics
|
||
|
if includedMetrics.Has(container.DiskUsageMetrics) {
|
||
|
handler.fsHandler = common.NewFsHandler(common.DefaultPeriod, rootfsStorageDir, storageLogDir, fsInfo)
|
||
|
}
|
||
|
// TODO for env vars we wanted to show from container.Config.Env from whitelist
|
||
|
//for _, exposedEnv := range metadataEnvs {
|
||
|
//klog.V(4).Infof("TODO env whitelist: %v", exposedEnv)
|
||
|
//}
|
||
|
|
||
|
return handler, nil
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) Start() {
|
||
|
if self.fsHandler != nil {
|
||
|
self.fsHandler.Start()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) Cleanup() {
|
||
|
if self.fsHandler != nil {
|
||
|
self.fsHandler.Stop()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) ContainerReference() (info.ContainerReference, error) {
|
||
|
return self.reference, nil
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) needNet() bool {
|
||
|
if self.includedMetrics.Has(container.NetworkUsageMetrics) {
|
||
|
return self.labels["io.kubernetes.container.name"] == "POD"
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) GetSpec() (info.ContainerSpec, error) {
|
||
|
hasFilesystem := self.includedMetrics.Has(container.DiskUsageMetrics)
|
||
|
spec, err := common.GetSpec(self.cgroupPaths, self.machineInfoFactory, self.needNet(), hasFilesystem)
|
||
|
|
||
|
spec.Labels = self.labels
|
||
|
spec.Envs = self.envs
|
||
|
spec.Image = self.image
|
||
|
|
||
|
return spec, err
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) getFsStats(stats *info.ContainerStats) error {
|
||
|
mi, err := self.machineInfoFactory.GetMachineInfo()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if self.includedMetrics.Has(container.DiskIOMetrics) {
|
||
|
common.AssignDeviceNamesToDiskStats((*common.MachineInfoNamer)(mi), &stats.DiskIo)
|
||
|
}
|
||
|
|
||
|
if !self.includedMetrics.Has(container.DiskUsageMetrics) {
|
||
|
return nil
|
||
|
}
|
||
|
var device string
|
||
|
switch self.storageDriver {
|
||
|
case overlay2StorageDriver, overlayStorageDriver:
|
||
|
deviceInfo, err := self.fsInfo.GetDirFsDevice(self.rootfsStorageDir)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("unable to determine device info for dir: %v: %v", self.rootfsStorageDir, err)
|
||
|
}
|
||
|
device = deviceInfo.Device
|
||
|
default:
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
limit uint64
|
||
|
fsType string
|
||
|
)
|
||
|
|
||
|
// crio does not impose any filesystem limits for containers. So use capacity as limit.
|
||
|
for _, fs := range mi.Filesystems {
|
||
|
if fs.Device == device {
|
||
|
limit = fs.Capacity
|
||
|
fsType = fs.Type
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fsStat := info.FsStats{Device: device, Type: fsType, Limit: limit}
|
||
|
usage := self.fsHandler.Usage()
|
||
|
fsStat.BaseUsage = usage.BaseUsageBytes
|
||
|
fsStat.Usage = usage.TotalUsageBytes
|
||
|
fsStat.Inodes = usage.InodeUsage
|
||
|
|
||
|
stats.Filesystem = append(stats.Filesystem, fsStat)
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) GetStats() (*info.ContainerStats, error) {
|
||
|
stats, err := self.libcontainerHandler.GetStats()
|
||
|
if err != nil {
|
||
|
return stats, err
|
||
|
}
|
||
|
// Clean up stats for containers that don't have their own network - this
|
||
|
// includes containers running in Kubernetes pods that use the network of the
|
||
|
// infrastructure container. This stops metrics being reported multiple times
|
||
|
// for each container in a pod.
|
||
|
if !self.needNet() {
|
||
|
stats.Network = info.NetworkStats{}
|
||
|
}
|
||
|
|
||
|
// Get filesystem stats.
|
||
|
err = self.getFsStats(stats)
|
||
|
if err != nil {
|
||
|
return stats, err
|
||
|
}
|
||
|
|
||
|
return stats, nil
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) ListContainers(listType container.ListType) ([]info.ContainerReference, error) {
|
||
|
// No-op for Docker driver.
|
||
|
return []info.ContainerReference{}, nil
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) GetCgroupPath(resource string) (string, error) {
|
||
|
path, ok := self.cgroupPaths[resource]
|
||
|
if !ok {
|
||
|
return "", fmt.Errorf("could not find path for resource %q for container %q\n", resource, self.reference.Name)
|
||
|
}
|
||
|
return path, nil
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) GetContainerLabels() map[string]string {
|
||
|
return self.labels
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) GetContainerIPAddress() string {
|
||
|
return self.ipAddress
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) ListProcesses(listType container.ListType) ([]int, error) {
|
||
|
return self.libcontainerHandler.GetProcesses()
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) Exists() bool {
|
||
|
return common.CgroupExists(self.cgroupPaths)
|
||
|
}
|
||
|
|
||
|
func (self *crioContainerHandler) Type() container.ContainerType {
|
||
|
return container.ContainerTypeCrio
|
||
|
}
|