mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-10-31 18:28:13 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			457 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			457 lines
		
	
	
		
			14 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 Docker containers.
 | |
| package docker
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io/ioutil"
 | |
| 	"path"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/google/cadvisor/container"
 | |
| 	"github.com/google/cadvisor/container/common"
 | |
| 	containerlibcontainer "github.com/google/cadvisor/container/libcontainer"
 | |
| 	"github.com/google/cadvisor/devicemapper"
 | |
| 	"github.com/google/cadvisor/fs"
 | |
| 	info "github.com/google/cadvisor/info/v1"
 | |
| 	dockerutil "github.com/google/cadvisor/utils/docker"
 | |
| 
 | |
| 	docker "github.com/docker/engine-api/client"
 | |
| 	dockercontainer "github.com/docker/engine-api/types/container"
 | |
| 	"github.com/golang/glog"
 | |
| 	"github.com/opencontainers/runc/libcontainer/cgroups"
 | |
| 	cgroupfs "github.com/opencontainers/runc/libcontainer/cgroups/fs"
 | |
| 	libcontainerconfigs "github.com/opencontainers/runc/libcontainer/configs"
 | |
| 	"golang.org/x/net/context"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	// The read write layers exist here.
 | |
| 	aufsRWLayer = "diff"
 | |
| 	// Path to the directory where docker stores log files if the json logging driver is enabled.
 | |
| 	pathToContainersDir = "containers"
 | |
| )
 | |
| 
 | |
| type dockerContainerHandler struct {
 | |
| 	client             *docker.Client
 | |
| 	name               string
 | |
| 	id                 string
 | |
| 	aliases            []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
 | |
| 
 | |
| 	// Manager of this container's cgroups.
 | |
| 	cgroupManager cgroups.Manager
 | |
| 
 | |
| 	// the docker storage driver
 | |
| 	storageDriver    storageDriver
 | |
| 	fsInfo           fs.FsInfo
 | |
| 	rootfsStorageDir string
 | |
| 
 | |
| 	// devicemapper state
 | |
| 
 | |
| 	// the devicemapper poolname
 | |
| 	poolName string
 | |
| 	// the devicemapper device id for the container
 | |
| 	deviceID string
 | |
| 
 | |
| 	// Time at which this container was created.
 | |
| 	creationTime time.Time
 | |
| 
 | |
| 	// Metadata associated with the container.
 | |
| 	labels map[string]string
 | |
| 	envs   map[string]string
 | |
| 
 | |
| 	// The container PID used to switch namespaces as required
 | |
| 	pid int
 | |
| 
 | |
| 	// Image name used for this container.
 | |
| 	image string
 | |
| 
 | |
| 	// The host root FS to read
 | |
| 	rootFs string
 | |
| 
 | |
| 	// The network mode of the container
 | |
| 	networkMode dockercontainer.NetworkMode
 | |
| 
 | |
| 	// Filesystem handler.
 | |
| 	fsHandler common.FsHandler
 | |
| 
 | |
| 	// The IP address of the container
 | |
| 	ipAddress string
 | |
| 
 | |
| 	ignoreMetrics container.MetricSet
 | |
| 
 | |
| 	// thin pool watcher
 | |
| 	thinPoolWatcher *devicemapper.ThinPoolWatcher
 | |
| }
 | |
| 
 | |
| var _ container.ContainerHandler = &dockerContainerHandler{}
 | |
| 
 | |
| func getRwLayerID(containerID, storageDir string, sd storageDriver, dockerVersion []int) (string, error) {
 | |
| 	const (
 | |
| 		// Docker version >=1.10.0 have a randomized ID for the root fs of a container.
 | |
| 		randomizedRWLayerMinorVersion = 10
 | |
| 		rwLayerIDFile                 = "mount-id"
 | |
| 	)
 | |
| 	if (dockerVersion[0] <= 1) && (dockerVersion[1] < randomizedRWLayerMinorVersion) {
 | |
| 		return containerID, nil
 | |
| 	}
 | |
| 
 | |
| 	bytes, err := ioutil.ReadFile(path.Join(storageDir, "image", string(sd), "layerdb", "mounts", containerID, rwLayerIDFile))
 | |
| 	if err != nil {
 | |
| 		return "", fmt.Errorf("failed to identify the read-write layer ID for container %q. - %v", containerID, err)
 | |
| 	}
 | |
| 	return string(bytes), err
 | |
| }
 | |
| 
 | |
| // newDockerContainerHandler returns a new container.ContainerHandler
 | |
| func newDockerContainerHandler(
 | |
| 	client *docker.Client,
 | |
| 	name string,
 | |
| 	machineInfoFactory info.MachineInfoFactory,
 | |
| 	fsInfo fs.FsInfo,
 | |
| 	storageDriver storageDriver,
 | |
| 	storageDir string,
 | |
| 	cgroupSubsystems *containerlibcontainer.CgroupSubsystems,
 | |
| 	inHostNamespace bool,
 | |
| 	metadataEnvs []string,
 | |
| 	dockerVersion []int,
 | |
| 	ignoreMetrics container.MetricSet,
 | |
| 	thinPoolWatcher *devicemapper.ThinPoolWatcher,
 | |
| ) (container.ContainerHandler, error) {
 | |
| 	// Create the cgroup paths.
 | |
| 	cgroupPaths := make(map[string]string, len(cgroupSubsystems.MountPoints))
 | |
| 	for key, val := range cgroupSubsystems.MountPoints {
 | |
| 		cgroupPaths[key] = path.Join(val, 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 := ContainerNameToDockerId(name)
 | |
| 
 | |
| 	// Add the Containers dir where the log files are stored.
 | |
| 	// FIXME: Give `otherStorageDir` a more descriptive name.
 | |
| 	otherStorageDir := path.Join(storageDir, pathToContainersDir, id)
 | |
| 
 | |
| 	rwLayerID, err := getRwLayerID(id, storageDir, storageDriver, dockerVersion)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	// Determine the rootfs storage dir OR the pool name to determine the device
 | |
| 	var (
 | |
| 		rootfsStorageDir string
 | |
| 		poolName         string
 | |
| 	)
 | |
| 	switch storageDriver {
 | |
| 	case aufsStorageDriver:
 | |
| 		rootfsStorageDir = path.Join(storageDir, string(aufsStorageDriver), aufsRWLayer, rwLayerID)
 | |
| 	case overlayStorageDriver:
 | |
| 		rootfsStorageDir = path.Join(storageDir, string(overlayStorageDriver), rwLayerID)
 | |
| 	case devicemapperStorageDriver:
 | |
| 		status, err := Status()
 | |
| 		if err != nil {
 | |
| 			return nil, fmt.Errorf("unable to determine docker status: %v", err)
 | |
| 		}
 | |
| 
 | |
| 		poolName = status.DriverStatus[dockerutil.DriverStatusPoolName]
 | |
| 	}
 | |
| 
 | |
| 	// TODO: extract object mother method
 | |
| 	handler := &dockerContainerHandler{
 | |
| 		id:                 id,
 | |
| 		client:             client,
 | |
| 		name:               name,
 | |
| 		machineInfoFactory: machineInfoFactory,
 | |
| 		cgroupPaths:        cgroupPaths,
 | |
| 		cgroupManager:      cgroupManager,
 | |
| 		storageDriver:      storageDriver,
 | |
| 		fsInfo:             fsInfo,
 | |
| 		rootFs:             rootFs,
 | |
| 		poolName:           poolName,
 | |
| 		rootfsStorageDir:   rootfsStorageDir,
 | |
| 		envs:               make(map[string]string),
 | |
| 		ignoreMetrics:      ignoreMetrics,
 | |
| 		thinPoolWatcher:    thinPoolWatcher,
 | |
| 	}
 | |
| 
 | |
| 	// We assume that if Inspect fails then the container is not known to docker.
 | |
| 	ctnr, err := client.ContainerInspect(context.Background(), id)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to inspect container %q: %v", id, err)
 | |
| 	}
 | |
| 	// Timestamp returned by Docker is in time.RFC3339Nano format.
 | |
| 	handler.creationTime, err = time.Parse(time.RFC3339Nano, ctnr.Created)
 | |
| 	if err != nil {
 | |
| 		// This should not happen, report the error just in case
 | |
| 		return nil, fmt.Errorf("failed to parse the create timestamp %q for container %q: %v", ctnr.Created, id, err)
 | |
| 	}
 | |
| 	handler.pid = ctnr.State.Pid
 | |
| 
 | |
| 	// Add the name and bare ID as aliases of the container.
 | |
| 	handler.aliases = append(handler.aliases, strings.TrimPrefix(ctnr.Name, "/"), id)
 | |
| 	handler.labels = ctnr.Config.Labels
 | |
| 	handler.image = ctnr.Config.Image
 | |
| 	handler.networkMode = ctnr.HostConfig.NetworkMode
 | |
| 	handler.deviceID = ctnr.GraphDriver.Data["DeviceId"]
 | |
| 
 | |
| 	// Obtain the IP address for the contianer.
 | |
| 	// If the NetworkMode starts with 'container:' then we need to use the IP address of the container specified.
 | |
| 	// This happens in cases such as kubernetes where the containers doesn't have an IP address itself and we need to use the pod's address
 | |
| 	ipAddress := ctnr.NetworkSettings.IPAddress
 | |
| 	networkMode := string(ctnr.HostConfig.NetworkMode)
 | |
| 	if ipAddress == "" && strings.HasPrefix(networkMode, "container:") {
 | |
| 		containerId := strings.TrimPrefix(networkMode, "container:")
 | |
| 		c, err := client.ContainerInspect(context.Background(), containerId)
 | |
| 		if err != nil {
 | |
| 			return nil, fmt.Errorf("failed to inspect container %q: %v", id, err)
 | |
| 		}
 | |
| 		ipAddress = c.NetworkSettings.IPAddress
 | |
| 	}
 | |
| 
 | |
| 	handler.ipAddress = ipAddress
 | |
| 
 | |
| 	if !ignoreMetrics.Has(container.DiskUsageMetrics) {
 | |
| 		handler.fsHandler = &dockerFsHandler{
 | |
| 			fsHandler:       common.NewFsHandler(common.DefaultPeriod, rootfsStorageDir, otherStorageDir, fsInfo),
 | |
| 			thinPoolWatcher: thinPoolWatcher,
 | |
| 			deviceID:        handler.deviceID,
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// split env vars to get metadata map.
 | |
| 	for _, exposedEnv := range metadataEnvs {
 | |
| 		for _, envVar := range ctnr.Config.Env {
 | |
| 			if envVar != "" {
 | |
| 				splits := strings.SplitN(envVar, "=", 2)
 | |
| 				if len(splits) == 2 && splits[0] == exposedEnv {
 | |
| 					handler.envs[strings.ToLower(exposedEnv)] = splits[1]
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return handler, nil
 | |
| }
 | |
| 
 | |
| // dockerFsHandler is a composite FsHandler implementation the incorporates
 | |
| // the common fs handler and a devicemapper ThinPoolWatcher.
 | |
| type dockerFsHandler struct {
 | |
| 	fsHandler common.FsHandler
 | |
| 
 | |
| 	// thinPoolWatcher is the devicemapper thin pool watcher
 | |
| 	thinPoolWatcher *devicemapper.ThinPoolWatcher
 | |
| 	// deviceID is the id of the container's fs device
 | |
| 	deviceID string
 | |
| }
 | |
| 
 | |
| var _ common.FsHandler = &dockerFsHandler{}
 | |
| 
 | |
| func (h *dockerFsHandler) Start() {
 | |
| 	h.fsHandler.Start()
 | |
| }
 | |
| 
 | |
| func (h *dockerFsHandler) Stop() {
 | |
| 	h.fsHandler.Stop()
 | |
| }
 | |
| 
 | |
| func (h *dockerFsHandler) Usage() common.FsUsage {
 | |
| 	usage := h.fsHandler.Usage()
 | |
| 
 | |
| 	// When devicemapper is the storage driver, the base usage of the container comes from the thin pool.
 | |
| 	// We still need the result of the fsHandler for any extra storage associated with the container.
 | |
| 	// To correctly factor in the thin pool usage, we should:
 | |
| 	// * Usage the thin pool usage as the base usage
 | |
| 	// * Calculate the overall usage by adding the overall usage from the fs handler to the thin pool usage
 | |
| 	if h.thinPoolWatcher != nil {
 | |
| 		thinPoolUsage, err := h.thinPoolWatcher.GetUsage(h.deviceID)
 | |
| 		if err != nil {
 | |
| 			// TODO: ideally we should keep track of how many times we failed to get the usage for this
 | |
| 			// device vs how many refreshes of the cache there have been, and display an error e.g. if we've
 | |
| 			// had at least 1 refresh and we still can't find the device.
 | |
| 			glog.V(5).Infof("unable to get fs usage from thin pool for device %s: %v", h.deviceID, err)
 | |
| 		} else {
 | |
| 			usage.BaseUsageBytes = thinPoolUsage
 | |
| 			usage.TotalUsageBytes += thinPoolUsage
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return usage
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) Start() {
 | |
| 	if self.fsHandler != nil {
 | |
| 		self.fsHandler.Start()
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) Cleanup() {
 | |
| 	if self.fsHandler != nil {
 | |
| 		self.fsHandler.Stop()
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) ContainerReference() (info.ContainerReference, error) {
 | |
| 	return info.ContainerReference{
 | |
| 		Id:        self.id,
 | |
| 		Name:      self.name,
 | |
| 		Aliases:   self.aliases,
 | |
| 		Namespace: DockerNamespace,
 | |
| 		Labels:    self.labels,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) needNet() bool {
 | |
| 	if !self.ignoreMetrics.Has(container.NetworkUsageMetrics) {
 | |
| 		return !self.networkMode.IsContainer()
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) GetSpec() (info.ContainerSpec, error) {
 | |
| 	hasFilesystem := !self.ignoreMetrics.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 *dockerContainerHandler) getFsStats(stats *info.ContainerStats) error {
 | |
| 	if self.ignoreMetrics.Has(container.DiskUsageMetrics) {
 | |
| 		return nil
 | |
| 	}
 | |
| 	var device string
 | |
| 	switch self.storageDriver {
 | |
| 	case devicemapperStorageDriver:
 | |
| 		// Device has to be the pool name to correlate with the device name as
 | |
| 		// set in the machine info filesystems.
 | |
| 		device = self.poolName
 | |
| 	case aufsStorageDriver, overlayStorageDriver, zfsStorageDriver:
 | |
| 		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
 | |
| 	}
 | |
| 
 | |
| 	mi, err := self.machineInfoFactory.GetMachineInfo()
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	var (
 | |
| 		limit  uint64
 | |
| 		fsType string
 | |
| 	)
 | |
| 
 | |
| 	// Docker 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
 | |
| }
 | |
| 
 | |
| // TODO(vmarmol): Get from libcontainer API instead of cgroup manager when we don't have to support older Dockers.
 | |
| func (self *dockerContainerHandler) GetStats() (*info.ContainerStats, error) {
 | |
| 	stats, err := containerlibcontainer.GetStats(self.cgroupManager, self.rootFs, self.pid, self.ignoreMetrics)
 | |
| 	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 *dockerContainerHandler) ListContainers(listType container.ListType) ([]info.ContainerReference, error) {
 | |
| 	// No-op for Docker driver.
 | |
| 	return []info.ContainerReference{}, nil
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) 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 *dockerContainerHandler) GetContainerLabels() map[string]string {
 | |
| 	return self.labels
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) GetContainerIPAddress() string {
 | |
| 	return self.ipAddress
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) ListProcesses(listType container.ListType) ([]int, error) {
 | |
| 	return containerlibcontainer.GetProcesses(self.cgroupManager)
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) Exists() bool {
 | |
| 	return common.CgroupExists(self.cgroupPaths)
 | |
| }
 | |
| 
 | |
| func (self *dockerContainerHandler) Type() container.ContainerType {
 | |
| 	return container.ContainerTypeDocker
 | |
| }
 | 
