123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301 |
- // Copyright 2016 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.
- package common
- import (
- "fmt"
- "io/ioutil"
- "os"
- "path"
- "strconv"
- "strings"
- "time"
- "github.com/google/cadvisor/container"
- info "github.com/google/cadvisor/info/v1"
- "github.com/google/cadvisor/utils"
- "github.com/karrick/godirwalk"
- "github.com/pkg/errors"
- "k8s.io/klog"
- )
- func DebugInfo(watches map[string][]string) map[string][]string {
- out := make(map[string][]string)
- lines := make([]string, 0, len(watches))
- for containerName, cgroupWatches := range watches {
- lines = append(lines, fmt.Sprintf("%s:", containerName))
- for _, cg := range cgroupWatches {
- lines = append(lines, fmt.Sprintf("\t%s", cg))
- }
- }
- out["Inotify watches"] = lines
- return out
- }
- func GetSpec(cgroupPaths map[string]string, machineInfoFactory info.MachineInfoFactory, hasNetwork, hasFilesystem bool) (info.ContainerSpec, error) {
- var spec info.ContainerSpec
- // Assume unified hierarchy containers.
- // Get the lowest creation time from all hierarchies as the container creation time.
- now := time.Now()
- lowestTime := now
- for _, cgroupPath := range cgroupPaths {
- // The modified time of the cgroup directory changes whenever a subcontainer is created.
- // eg. /docker will have creation time matching the creation of latest docker container.
- // Use clone_children as a workaround as it isn't usually modified. It is only likely changed
- // immediately after creating a container.
- cgroupPath = path.Join(cgroupPath, "cgroup.clone_children")
- fi, err := os.Stat(cgroupPath)
- if err == nil && fi.ModTime().Before(lowestTime) {
- lowestTime = fi.ModTime()
- }
- }
- if lowestTime != now {
- spec.CreationTime = lowestTime
- }
- // Get machine info.
- mi, err := machineInfoFactory.GetMachineInfo()
- if err != nil {
- return spec, err
- }
- // CPU.
- cpuRoot, ok := cgroupPaths["cpu"]
- if ok {
- if utils.FileExists(cpuRoot) {
- spec.HasCpu = true
- spec.Cpu.Limit = readUInt64(cpuRoot, "cpu.shares")
- spec.Cpu.Period = readUInt64(cpuRoot, "cpu.cfs_period_us")
- quota := readString(cpuRoot, "cpu.cfs_quota_us")
- if quota != "" && quota != "-1" {
- val, err := strconv.ParseUint(quota, 10, 64)
- if err != nil {
- klog.Errorf("GetSpec: Failed to parse CPUQuota from %q: %s", path.Join(cpuRoot, "cpu.cfs_quota_us"), err)
- }
- spec.Cpu.Quota = val
- }
- }
- }
- // Cpu Mask.
- // This will fail for non-unified hierarchies. We'll return the whole machine mask in that case.
- cpusetRoot, ok := cgroupPaths["cpuset"]
- if ok {
- if utils.FileExists(cpusetRoot) {
- spec.HasCpu = true
- mask := readString(cpusetRoot, "cpuset.cpus")
- spec.Cpu.Mask = utils.FixCpuMask(mask, mi.NumCores)
- }
- }
- // Memory
- memoryRoot, ok := cgroupPaths["memory"]
- if ok {
- if utils.FileExists(memoryRoot) {
- spec.HasMemory = true
- spec.Memory.Limit = readUInt64(memoryRoot, "memory.limit_in_bytes")
- spec.Memory.SwapLimit = readUInt64(memoryRoot, "memory.memsw.limit_in_bytes")
- spec.Memory.Reservation = readUInt64(memoryRoot, "memory.soft_limit_in_bytes")
- }
- }
- spec.HasNetwork = hasNetwork
- spec.HasFilesystem = hasFilesystem
- if blkioRoot, ok := cgroupPaths["blkio"]; ok && utils.FileExists(blkioRoot) {
- spec.HasDiskIo = true
- }
- return spec, nil
- }
- func readString(dirpath string, file string) string {
- cgroupFile := path.Join(dirpath, file)
- // Read
- out, err := ioutil.ReadFile(cgroupFile)
- if err != nil {
- // Ignore non-existent files
- if !os.IsNotExist(err) {
- klog.Warningf("readString: Failed to read %q: %s", cgroupFile, err)
- }
- return ""
- }
- return strings.TrimSpace(string(out))
- }
- func readUInt64(dirpath string, file string) uint64 {
- out := readString(dirpath, file)
- if out == "" {
- return 0
- }
- val, err := strconv.ParseUint(out, 10, 64)
- if err != nil {
- klog.Errorf("readUInt64: Failed to parse int %q from file %q: %s", out, path.Join(dirpath, file), err)
- return 0
- }
- return val
- }
- // Lists all directories under "path" and outputs the results as children of "parent".
- func ListDirectories(dirpath string, parent string, recursive bool, output map[string]struct{}) error {
- buf := make([]byte, godirwalk.DefaultScratchBufferSize)
- return listDirectories(dirpath, parent, recursive, output, buf)
- }
- func listDirectories(dirpath string, parent string, recursive bool, output map[string]struct{}, buf []byte) error {
- dirents, err := godirwalk.ReadDirents(dirpath, buf)
- if err != nil {
- // Ignore if this hierarchy does not exist.
- if os.IsNotExist(errors.Cause(err)) {
- err = nil
- }
- return err
- }
- for _, dirent := range dirents {
- // We only grab directories.
- if !dirent.IsDir() {
- continue
- }
- dirname := dirent.Name()
- name := path.Join(parent, dirname)
- output[name] = struct{}{}
- // List subcontainers if asked to.
- if recursive {
- err := listDirectories(path.Join(dirpath, dirname), name, true, output, buf)
- if err != nil {
- return err
- }
- }
- }
- return nil
- }
- func MakeCgroupPaths(mountPoints map[string]string, name string) map[string]string {
- cgroupPaths := make(map[string]string, len(mountPoints))
- for key, val := range mountPoints {
- cgroupPaths[key] = path.Join(val, name)
- }
- return cgroupPaths
- }
- func CgroupExists(cgroupPaths map[string]string) bool {
- // If any cgroup exists, the container is still alive.
- for _, cgroupPath := range cgroupPaths {
- if utils.FileExists(cgroupPath) {
- return true
- }
- }
- return false
- }
- func ListContainers(name string, cgroupPaths map[string]string, listType container.ListType) ([]info.ContainerReference, error) {
- containers := make(map[string]struct{})
- for _, cgroupPath := range cgroupPaths {
- err := ListDirectories(cgroupPath, name, listType == container.ListRecursive, containers)
- if err != nil {
- return nil, err
- }
- }
- // Make into container references.
- ret := make([]info.ContainerReference, 0, len(containers))
- for cont := range containers {
- ret = append(ret, info.ContainerReference{
- Name: cont,
- })
- }
- return ret, nil
- }
- // AssignDeviceNamesToDiskStats assigns the Device field on the provided DiskIoStats by looking up
- // the device major and minor identifiers in the provided device namer.
- func AssignDeviceNamesToDiskStats(namer DeviceNamer, stats *info.DiskIoStats) {
- assignDeviceNamesToPerDiskStats(
- namer,
- stats.IoMerged,
- stats.IoQueued,
- stats.IoServiceBytes,
- stats.IoServiceTime,
- stats.IoServiced,
- stats.IoTime,
- stats.IoWaitTime,
- stats.Sectors,
- )
- }
- // assignDeviceNamesToPerDiskStats looks up device names for the provided stats, caching names
- // if necessary.
- func assignDeviceNamesToPerDiskStats(namer DeviceNamer, diskStats ...[]info.PerDiskStats) {
- devices := make(deviceIdentifierMap)
- for _, stats := range diskStats {
- for i, stat := range stats {
- stats[i].Device = devices.Find(stat.Major, stat.Minor, namer)
- }
- }
- }
- // DeviceNamer returns string names for devices by their major and minor id.
- type DeviceNamer interface {
- // DeviceName returns the name of the device by its major and minor ids, or false if no
- // such device is recognized.
- DeviceName(major, minor uint64) (string, bool)
- }
- type MachineInfoNamer info.MachineInfo
- func (n *MachineInfoNamer) DeviceName(major, minor uint64) (string, bool) {
- for _, info := range n.DiskMap {
- if info.Major == major && info.Minor == minor {
- return "/dev/" + info.Name, true
- }
- }
- for _, info := range n.Filesystems {
- if info.DeviceMajor == major && info.DeviceMinor == minor {
- return info.Device, true
- }
- }
- return "", false
- }
- type deviceIdentifier struct {
- major uint64
- minor uint64
- }
- type deviceIdentifierMap map[deviceIdentifier]string
- // Find locates the device name by device identifier out of from, caching the result as necessary.
- func (m deviceIdentifierMap) Find(major, minor uint64, namer DeviceNamer) string {
- d := deviceIdentifier{major, minor}
- if s, ok := m[d]; ok {
- return s
- }
- s, _ := namer.DeviceName(major, minor)
- m[d] = s
- return s
- }
|