mirror of https://github.com/k3s-io/k3s
301 lines
8.5 KiB
Go
301 lines
8.5 KiB
Go
// Copyright 2014 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 "raw" containers.
|
|
package raw
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/google/cadvisor/container"
|
|
"github.com/google/cadvisor/container/common"
|
|
"github.com/google/cadvisor/container/libcontainer"
|
|
"github.com/google/cadvisor/fs"
|
|
info "github.com/google/cadvisor/info/v1"
|
|
"github.com/google/cadvisor/machine"
|
|
|
|
cgroupfs "github.com/opencontainers/runc/libcontainer/cgroups/fs"
|
|
"github.com/opencontainers/runc/libcontainer/configs"
|
|
"k8s.io/klog"
|
|
)
|
|
|
|
type rawContainerHandler struct {
|
|
// Name of the container for this handler.
|
|
name string
|
|
machineInfoFactory info.MachineInfoFactory
|
|
|
|
// Absolute path to the cgroup hierarchies of this container.
|
|
// (e.g.: "cpu" -> "/sys/fs/cgroup/cpu/test")
|
|
cgroupPaths map[string]string
|
|
|
|
fsInfo fs.FsInfo
|
|
externalMounts []common.Mount
|
|
includedMetrics container.MetricSet
|
|
|
|
libcontainerHandler *libcontainer.Handler
|
|
}
|
|
|
|
func isRootCgroup(name string) bool {
|
|
return name == "/"
|
|
}
|
|
|
|
func newRawContainerHandler(name string, cgroupSubsystems *libcontainer.CgroupSubsystems, machineInfoFactory info.MachineInfoFactory, fsInfo fs.FsInfo, watcher *common.InotifyWatcher, rootFs string, includedMetrics container.MetricSet) (container.ContainerHandler, error) {
|
|
cgroupPaths := common.MakeCgroupPaths(cgroupSubsystems.MountPoints, name)
|
|
|
|
cHints, err := common.GetContainerHintsFromFile(*common.ArgContainerHints)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Generate the equivalent cgroup manager for this container.
|
|
cgroupManager := &cgroupfs.Manager{
|
|
Cgroups: &configs.Cgroup{
|
|
Name: name,
|
|
},
|
|
Paths: cgroupPaths,
|
|
}
|
|
|
|
var externalMounts []common.Mount
|
|
for _, container := range cHints.AllHosts {
|
|
if name == container.FullName {
|
|
externalMounts = container.Mounts
|
|
break
|
|
}
|
|
}
|
|
|
|
pid := 0
|
|
if isRootCgroup(name) {
|
|
pid = 1
|
|
|
|
// delete pids from cgroup paths because /sys/fs/cgroup/pids/pids.current not exist
|
|
delete(cgroupPaths, "pids")
|
|
}
|
|
|
|
handler := libcontainer.NewHandler(cgroupManager, rootFs, pid, includedMetrics)
|
|
|
|
return &rawContainerHandler{
|
|
name: name,
|
|
machineInfoFactory: machineInfoFactory,
|
|
cgroupPaths: cgroupPaths,
|
|
fsInfo: fsInfo,
|
|
externalMounts: externalMounts,
|
|
includedMetrics: includedMetrics,
|
|
libcontainerHandler: handler,
|
|
}, nil
|
|
}
|
|
|
|
func (self *rawContainerHandler) ContainerReference() (info.ContainerReference, error) {
|
|
// We only know the container by its one name.
|
|
return info.ContainerReference{
|
|
Name: self.name,
|
|
}, nil
|
|
}
|
|
|
|
func (self *rawContainerHandler) GetRootNetworkDevices() ([]info.NetInfo, error) {
|
|
nd := []info.NetInfo{}
|
|
if isRootCgroup(self.name) {
|
|
mi, err := self.machineInfoFactory.GetMachineInfo()
|
|
if err != nil {
|
|
return nd, err
|
|
}
|
|
return mi.NetworkDevices, nil
|
|
}
|
|
return nd, nil
|
|
}
|
|
|
|
// Nothing to start up.
|
|
func (self *rawContainerHandler) Start() {}
|
|
|
|
// Nothing to clean up.
|
|
func (self *rawContainerHandler) Cleanup() {}
|
|
|
|
func (self *rawContainerHandler) GetSpec() (info.ContainerSpec, error) {
|
|
const hasNetwork = false
|
|
hasFilesystem := isRootCgroup(self.name) || len(self.externalMounts) > 0
|
|
spec, err := common.GetSpec(self.cgroupPaths, self.machineInfoFactory, hasNetwork, hasFilesystem)
|
|
if err != nil {
|
|
return spec, err
|
|
}
|
|
|
|
if isRootCgroup(self.name) {
|
|
// Check physical network devices for root container.
|
|
nd, err := self.GetRootNetworkDevices()
|
|
if err != nil {
|
|
return spec, err
|
|
}
|
|
spec.HasNetwork = spec.HasNetwork || len(nd) != 0
|
|
|
|
// Get memory and swap limits of the running machine
|
|
memLimit, err := machine.GetMachineMemoryCapacity()
|
|
if err != nil {
|
|
klog.Warningf("failed to obtain memory limit for machine container")
|
|
spec.HasMemory = false
|
|
} else {
|
|
spec.Memory.Limit = uint64(memLimit)
|
|
// Spec is marked to have memory only if the memory limit is set
|
|
spec.HasMemory = true
|
|
}
|
|
|
|
swapLimit, err := machine.GetMachineSwapCapacity()
|
|
if err != nil {
|
|
klog.Warningf("failed to obtain swap limit for machine container")
|
|
} else {
|
|
spec.Memory.SwapLimit = uint64(swapLimit)
|
|
}
|
|
}
|
|
|
|
return spec, nil
|
|
}
|
|
|
|
func fsToFsStats(fs *fs.Fs) info.FsStats {
|
|
inodes := uint64(0)
|
|
inodesFree := uint64(0)
|
|
hasInodes := fs.InodesFree != nil
|
|
if hasInodes {
|
|
inodes = *fs.Inodes
|
|
inodesFree = *fs.InodesFree
|
|
}
|
|
return info.FsStats{
|
|
Device: fs.Device,
|
|
Type: fs.Type.String(),
|
|
Limit: fs.Capacity,
|
|
Usage: fs.Capacity - fs.Free,
|
|
HasInodes: hasInodes,
|
|
Inodes: inodes,
|
|
InodesFree: inodesFree,
|
|
Available: fs.Available,
|
|
ReadsCompleted: fs.DiskStats.ReadsCompleted,
|
|
ReadsMerged: fs.DiskStats.ReadsMerged,
|
|
SectorsRead: fs.DiskStats.SectorsRead,
|
|
ReadTime: fs.DiskStats.ReadTime,
|
|
WritesCompleted: fs.DiskStats.WritesCompleted,
|
|
WritesMerged: fs.DiskStats.WritesMerged,
|
|
SectorsWritten: fs.DiskStats.SectorsWritten,
|
|
WriteTime: fs.DiskStats.WriteTime,
|
|
IoInProgress: fs.DiskStats.IoInProgress,
|
|
IoTime: fs.DiskStats.IoTime,
|
|
WeightedIoTime: fs.DiskStats.WeightedIoTime,
|
|
}
|
|
}
|
|
|
|
func (self *rawContainerHandler) getFsStats(stats *info.ContainerStats) error {
|
|
var filesystems []fs.Fs
|
|
var err error
|
|
// Get Filesystem information only for the root cgroup.
|
|
if isRootCgroup(self.name) {
|
|
filesystems, err = self.fsInfo.GetGlobalFsInfo()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
} else if self.includedMetrics.Has(container.DiskUsageMetrics) || self.includedMetrics.Has(container.DiskIOMetrics) {
|
|
if len(self.externalMounts) > 0 {
|
|
var mountSet map[string]struct{}
|
|
mountSet = make(map[string]struct{})
|
|
for _, mount := range self.externalMounts {
|
|
mountSet[mount.HostDir] = struct{}{}
|
|
}
|
|
filesystems, err = self.fsInfo.GetFsInfoForPath(mountSet)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
if isRootCgroup(self.name) || self.includedMetrics.Has(container.DiskUsageMetrics) {
|
|
for i := range filesystems {
|
|
fs := filesystems[i]
|
|
stats.Filesystem = append(stats.Filesystem, fsToFsStats(&fs))
|
|
}
|
|
}
|
|
|
|
if isRootCgroup(self.name) || self.includedMetrics.Has(container.DiskIOMetrics) {
|
|
common.AssignDeviceNamesToDiskStats(&fsNamer{fs: filesystems, factory: self.machineInfoFactory}, &stats.DiskIo)
|
|
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (self *rawContainerHandler) GetStats() (*info.ContainerStats, error) {
|
|
if *disableRootCgroupStats && isRootCgroup(self.name) {
|
|
return nil, nil
|
|
}
|
|
stats, err := self.libcontainerHandler.GetStats()
|
|
if err != nil {
|
|
return stats, err
|
|
}
|
|
|
|
// Get filesystem stats.
|
|
err = self.getFsStats(stats)
|
|
if err != nil {
|
|
return stats, err
|
|
}
|
|
|
|
return stats, nil
|
|
}
|
|
|
|
func (self *rawContainerHandler) 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.name)
|
|
}
|
|
return path, nil
|
|
}
|
|
|
|
func (self *rawContainerHandler) GetContainerLabels() map[string]string {
|
|
return map[string]string{}
|
|
}
|
|
|
|
func (self *rawContainerHandler) GetContainerIPAddress() string {
|
|
// the IP address for the raw container corresponds to the system ip address.
|
|
return "127.0.0.1"
|
|
}
|
|
|
|
func (self *rawContainerHandler) ListContainers(listType container.ListType) ([]info.ContainerReference, error) {
|
|
return common.ListContainers(self.name, self.cgroupPaths, listType)
|
|
}
|
|
|
|
func (self *rawContainerHandler) ListProcesses(listType container.ListType) ([]int, error) {
|
|
return self.libcontainerHandler.GetProcesses()
|
|
}
|
|
|
|
func (self *rawContainerHandler) Exists() bool {
|
|
return common.CgroupExists(self.cgroupPaths)
|
|
}
|
|
|
|
func (self *rawContainerHandler) Type() container.ContainerType {
|
|
return container.ContainerTypeRaw
|
|
}
|
|
|
|
type fsNamer struct {
|
|
fs []fs.Fs
|
|
factory info.MachineInfoFactory
|
|
info common.DeviceNamer
|
|
}
|
|
|
|
func (n *fsNamer) DeviceName(major, minor uint64) (string, bool) {
|
|
for _, info := range n.fs {
|
|
if uint64(info.Major) == major && uint64(info.Minor) == minor {
|
|
return info.Device, true
|
|
}
|
|
}
|
|
if n.info == nil {
|
|
mi, err := n.factory.GetMachineInfo()
|
|
if err != nil {
|
|
return "", false
|
|
}
|
|
n.info = (*common.MachineInfoNamer)(mi)
|
|
}
|
|
return n.info.DeviceName(major, minor)
|
|
}
|